-
Notifications
You must be signed in to change notification settings - Fork 60
/
Copy pathlinop_TV.m
282 lines (259 loc) · 9.38 KB
/
linop_TV.m
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
function op = linop_TV( sz, variation, action )
%LINOP_TV 2D Total-Variation (TV) linear operator.
% OP = LINOP_TV( SZ ) returns a handle to a TFOCS linear operator that
% implements the total variation linear operator on an M x N grid;
% that is, to be applied to matrices of size [M,N].
% By default, it expects to operate on M*N x 1 vectors
% but if SZ = {M,N}, then expects to operate on M x N matrices
%
% N.B. In this form, OP does not calculate the TV norm but rather
% a vector of size M*N x 1 such that TV(X) = norm( OP(X,1), 1 ).
%
% TV = LINOP_TV( X ) returns ||X||_TV if X is bigger than 2 x 2
%
% OP = LINOP_TV( SZ, VARIANT )
% if VARIANT is 'regular' or 'reflexive' (default),
% assumes reflexive boundary conditions, so never penalizes
% points on the outer boundary
% (prior to Feb 13 2020, this was incorrectly
% labeled as "Dirichlet" boundary condition)
% if VARIANT is 'Dirichlet'
% assumes zeros on the boundary (Dirichlet boundary conditions)
% (added Feb 2020)
% if VARIANT is 'circular', aka 'periodic'
% assumes circularity, so x(:,N+1) - x(:,N)
% is calculated by x(:,1) - x(:,N) (and similarly
% for row differences) [when SZ={M,N}]
%
% [...] = LINOP_TV(SZ, VARIATION, ACTION )
% if ACTION is 'handle', returns a TFOCS function handle (default)
% if ACTION is 'cvx', returns a function handle suitable for CVX
% (this variant returns the full TV semi-norm, not just the
% associated linear operators)
% if ACTION is 'matrix', returns the explicit TV matrix
% (real part corresponds to horizontal differences,
% imaginary part correspond to vertical differences)
% if ACTION is 'norm', returns an estimate of the spectral norm
% of the the linear TV operator
%
% TODO: check circular case for non-square domains
%
% The linear operators are of the form J kron I and I kron J,
% where I is identity, and J is something like:
% Reflexive:
% -1 1 0
% 0 -1 1
% 0 0 0
% Dirichlet:
% -1 1 0
% 0 -1 1
% 0 0 -1
% Circular
% -1 1 0
% 0 -1 1
% 1 0 -1
error(nargchk(1,3,nargin));
if nargin < 2 || isempty(variation), variation = 'regular'; end
if nargin < 3 || isempty(action), action = 'handle'; end
CALCULATE_TV = false;
if numel(sz) > 4
CALCULATE_TV = true;
X = sz;
sz = size(X);
end
if iscell(sz)
n1 = sz{1};
n2 = sz{2};
else
n1 = sz(1);
n2 = sz(2);
end
% Setup the Total-Variation operators
mat = @(x) reshape(x,n1,n2);
if strcmpi(action,'matrix') || strcmpi(action,'cvx')
switch lower(variation)
case {'regular','reflexive'}
% Update: Feb 13 2020, this is not really "Dirichlet" b.c.,
% but actually "reflexiv" b.c., assuming x_{n+1} = x_n,
% so that x_{n+1} - x_{n} = 0
%{
for n2=3, J looks like
-1 1 0
0 -1 1
0 0 0
%}
e = ones(max(n1,n2),1);
e2 = e;
e2(n2:end) = 0;
J = spdiags([-e2,e], 0:1,n2,n2);
I = eye(n1);
Dh = kron(J,I); % horizontal differences, sparse matrix
% see also blkdiag
e2 = e;
e2(n1:end) = 0;
J = spdiags([-e2,e], 0:1,n1,n1);
I = eye(n2);
Dv = kron(I,J); % vertical differences, sparse matrix
case 'dirichlet'
% Adding this Feb 13 2020
% Assumes x_{n+1} = 0
%{
for n2=3, J looks like
-1 1 0
0 -1 1
0 0 -1
%}
e = ones(max(n1,n2),1);
e2 = e;
J = spdiags([-e2,e], 0:1,n2,n2);
I = eye(n1);
Dh = kron(J,I); % horizontal differences, sparse matrix
e2 = e;
J = spdiags([-e2,e], 0:1,n1,n1);
I = eye(n2);
Dv = kron(I,J); % vertical differences, sparse matrix
case {'circular','periodic'}
% Assumes x_{n+1} = x_{1}
%{
for n2=3, J looks like
-1 1 0
0 -1 1
1 0 -1
%}
e = ones(max(n1,n2),1);
e2 = e;
% e2(n2:end) = 0;
J = spdiags([-e2,e], 0:1,n2,n2);
J(end,1) = 1;
I = eye(n1);
Dh = kron(J,I); % horizontal differences, sparse matrix
% see also blkdiag
e = ones(max(n1,n2),1);
e2 = e;
% e2(n1:end) = 0;
J = spdiags([-e2,e], 0:1,n1,n1);
J(end,1) = 1;
I = eye(n2);
Dv = kron(I,J); % vertical differences, sparse matrix
end
if strcmpi(action,'matrix')
op = Dh + 1i*Dv;
else
% "norms" is a CVX function, but we can over-load it (see sub-function below)
% op = @(X) sum( norms( [Dh*X(:), Dv*X(:)]', 1 ) );
% Feb 13 2020, the above seems buggy! might have meant the 1 to
% mean dimension (but then don't transpose, and it's in the wrong
% spot). Try this instead:
% op = @(X) sum( norms( [Dh*X(:), Dv*X(:)]', 2 ) );
% No, that doesn't work: Disciplined convex programming error: Invalid operation: sqrt( {positive convex} )
% Try with explicit imaginary part
% op = @(X) sum_square_abs( Dh*X(:) + 1i*(Dv*X(:)) ); % no, this squares it
% This is correct:
op = @(X) norm( Dh*X(:) + 1i*(Dv*X(:)), 1 );
end
return;
end
switch lower(variation)
% Convention: Dh and Dv are the forward operators
% diff_h and diff_v are the adjoint/transpose operators
% (similar but not the same)
% Note: adjoint of A kron B is A^* kron B^*
% (and we will convert A + i*B to [A';B'] explicitly later)
case {'regular','reflexive'}
Dh = @(X) vec( [diff(X,1,2), zeros(n1,1)] );
diff_h = @(X) [zeros(n1,1),X(:,1:end-1)] - [X(:,1:end-1),zeros(n1,1) ];
Dv = @(X) vec( [diff(X,1,1); zeros(1,n2)] );
diff_v = @(X) [zeros(1,n2);X(1:end-1,:)] - [X(1:end-1,:);zeros(1,n2) ];
% sometimes diff_v is much slower than diff_h
% We can exploit data locality by working with transposes
diff_v_t = @(Xt) ([zeros(n2,1),Xt(:,1:end-1)] - [Xt(:,1:end-1),zeros(n2,1)])';
case 'dirichlet'
% New, Feb 2020
Dh = @(X) vec( [diff(X,1,2), -X(:,end) ] );
diff_h = @(X) [zeros(n1,1),X(:,1:end-1)] - X;
Dv = @(X) vec( [diff(X,1,1); -X(end,:) ] );
% sometimes diff_v is much slower than diff_h
% We can exploit data locality by working with transposes
diff_v_t = @(Xt) ([zeros(n2,1),Xt(:,1:end-1)] - Xt)';
case {'circular','periodic'}
% For circular version, 2 x 2 case is special.
% error('not yet implemented');
Dh = @(X) vec( [diff(X,1,2), X(:,1) - X(:,end)] );
% diff_h needs to be checked
diff_h = @(X) [X(:,end),X(:,1:end-1)] - X;
% diff_v needs to be checked
Dv = @(X) vec( [diff(X,1,1); X(1,:) - X(end,:) ] );
diff_v = @(X) [X(end,:);X(1:end-1,:)] - X;
diff_v_t = @(Xt) ([Xt(:,end),Xt(:,1:end-1)] - Xt)';
otherwise
error('Bad variation parameter');
end
if iscell(sz)
Dh_transpose = @(X) diff_h(mat(X)) ;
% Dv_transpose = @(X) diff_v(mat(X)) ;
Dv_transpose = @(X) diff_v_t(mat(X)') ; % faster
else
Dh_transpose = @(X) vec( diff_h(mat(X)) );
% Dv_transpose = @(X) vec( diff_v(mat(X)) );
Dv_transpose = @(X) vec( diff_v_t(mat(X)') ); % faster
end
TV = @(x) ( Dh(mat(x)) + 1i*Dv(mat(x)) ); % real to complex
TVt = @(z) ( Dh_transpose(real(z)) + Dv_transpose(imag(z)) );
if CALCULATE_TV
op = norm( TV(X), 1 );
return;
end
if strcmpi(action,'norm')
% to compute max eigenvalue, I use a vector
% that is very likely to be the max eigenvector:
% matrix with every entry alternating -1 and 1
even = @(n) ~( n - 2*round(n/2) ); % returns 1 if even, 0 if odd
Y = zeros( n1 + even(n1), n2 + even(n2) );
nn = numel(Y);
Y(:) = (-1).^(1:nn);
Y = Y(1:n1,1:n2);
op = norm( TV(Y) )/norm(Y(:));
% Nearly equivalent to:
% norm(full( [real(tv); imag(tv)] ) )
% where tv is the matrix form
else
if iscell(sz)
szW = { [n1,n2], [n1*n2,1] };
else
szW = sz(1)*sz(2); % n1 * n2
szW = [szW,szW];
end
op = @(x,mode)linop_tv_r2c(szW,TV,TVt,x,mode);
end
function y = linop_tv_r2c( sz, TV, TVt, x, mode )
switch mode,
case 0, y = sz;
case 1, y = TV( realcheck( x ) );
case 2, y = realcheck( TVt( x ) );
end
function y = realcheck( y )
if ~isreal( y ),
error( 'Unexpected complex value in linear operation.' );
end
function y = norms( x, p, dim )
sx = size(x);
if nargin < 3, dim = 1; end
if nargin < 2 || isempty(p), p = 2; end
if isempty(x) || dim > length(sx) || sx(dim)==1
p = 1;
end
switch p,
case 1,
y = sum( abs( x ), dim );
case 2,
y = sqrt( sum( x .* conj( x ), dim ) );
case Inf,
y = max( abs( x ), [], dim );
case {'Inf','inf'}
y = max( abs( x ), [], dim );
otherwise,
y = sum( abs( x ) .^ p, dim ) .^ ( 1 / p );
end
% TFOCS v1.3 by Stephen Becker, Emmanuel Candes, and Michael Grant.
% Copyright 2013 California Institute of Technology and CVX Research.
% See the file LICENSE for full license information.