-
Notifications
You must be signed in to change notification settings - Fork 60
/
Copy pathproj_0.m
227 lines (215 loc) · 7 KB
/
proj_0.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
function op = proj_0(offset,A,algo,R)
%PROJ_0 Projection onto the set {0}
% OP = PROJ_0 returns an implementation of the indicator
% function for the set including only zero.
%
% OP = PROJ_0( c ) returns an implementation of the
% indicator function of the set {c}
% If c is a scalar, this is interpreted as c*1
% where "1" is the all ones object of the appropriate size.
%
% OP = PROJ_0( c , A) returns an implementation of the
% indicator function of the set {x : Ax = c}
% where A is a matrix. Unlike the previous call, c must now
% be a vector of appropriate size, and x must be a matrix
% (no support for multi-dimensional arrays).
% Note:
% This requires computing the Cholesky decomposition of A
% which can be expensive. For large dimensions, consider
% using TFOCS_SCD.m and explicitly passing in the linear
% operator, which will recast the optimization problem to avoid
% requiring this function. For small problems, directly
% calling this function is fine and may lead to faster convergence.
%
% If A is diagonal, then we can exploit that, but we don't
% currently have it implemented. Instead, just rescale c
% since if A=di\ag(d) then { x : Ax=c } = { x : x = c./d }
%
% "A" must be an explicit matrix, not a function handle
%
% OP = PROJ_0( c , A, algo)
% lets the user select between two algorithms (algo=1 or =2)
% Algo. 1 is a bit faster but more prone to numerical roundoff error
% which could be significant if A is extremely ill-conditioned
%
% OP = PROJ_0( c , A, algo, R)
% uses the pre-computed value R (if algo==1, R=chol(A*A'),
% and if algo==2, [~,R] = qr(A',0); ).
% This usage is useful if you have already computed R and want
% to save time. For algo==1, you can also set
% R=alpha to indicate that A*A' = alpha^2*Id
%
% Note: for the set { x : ||Ax-b|| <= eps } where eps > 0, see
% proj_l2.m
%
% See also proj_l2.m, prox_0.m, proj_Rn.m and smooth_constant.m,
% which are the Fenchel conjugates of this function.
if nargin == 0
op = @proj_0_impl;
elseif nargin == 1
op = @(varargin) proj_0_impl_q( offset, varargin{:} );
elseif nargin >= 2 % added June 14 2014
if norm(offset)==0
error('offset term is 0, so solution is always 0. Are you sure this is what you want?');
end
% We have two different algorithms, same asymptoptic complexity
% but Algo. 1 is faster in practice though less numerically robust
% since it forms the matrix A*A' which has larger condition number
% Algo. 2 is a bit more stable but a bit slower
if nargin < 3 || isempty(algo), algo = 1; end
if nargin < 4, R = []; end
if isa(A,'function_handle')
if algo ~= 1
error('If "A" is a function handle, must use algo #1');
end
%sz = A([],0);
% Now, build AAt using implicit2explicit
if isempty(R)
fprintf('Now computing explicit representation of A*A''. Please wait...');
t1 = tic;
AAt_fun = linop_compose( A, linop_adjoint(A) );
AAt = implicit2explicit(AAt_fun);
fprintf('... done (%.1f seconds)\n', toc(t1));
end
else
if size( A, 1 ) ~= size(offset,1)
error('A must have as many rows as c');
end
if algo == 1 && isempty(R)
AAt = A*A';
end
end
if isempty(R)
t2 = tic;
if algo == 1
fprintf('Now computing Cholesky factorization of A. Please wait...');
R = chol(AAt);
clear AAt
elseif algo==2
fprintf('Now computing QR decomposition of A^T. Please wait...');
[~,R] = qr(A',0);
else
error('Bad value for "algo". Should be 1 or 2');
end
fprintf('... done (%.1f seconds)\n', toc(t2));
end
op = @(varargin) proj_0_impl_qA( offset, A, R, algo, varargin{:} );
end
function [ v, x ] = proj_0_impl( x, t )
v = 0;
switch nargin,
case 1,
if nargout == 2,
error( 'This function is not differentiable.' );
elseif any( x(:) ),
v = Inf;
end
case 2,
% "t" variable has no effect
x = 0*x;
otherwise,
error( 'Not enough arguments.' );
end
function [ v, x ] = proj_0_impl_q( c, x, t )
v = 0;
switch nargin,
case 2,
if nargout == 2,
error( 'This function is not differentiable.' );
end
if isscalar(c)
if any( x(:) - c )
v = Inf;
end
elseif any( x(:) - c(:) ),
v = Inf;
end
case 3,
% "t" variable has no effect
if isscalar(c) && ~isscalar(x)
x = c*ones( size(x) );
else
x = c;
end
otherwise,
error( 'Not enough arguments.' );
end
function [ v, x ] = proj_0_impl_qA( b, A, R, algo, x, t )
v = 0;
switch nargin,
case 5,
if nargout == 2,
error( 'This function is not differentiable.' );
end
if isa(A,'function_handle')
if norm( A(x,1) - b )/norm(b) > 1e-10
v = Inf;
end
else
if norm( A*x - b )/norm(b) > 1e-10
v = Inf;
end
end
case 6,
% "t" variable has no effect
if algo==1
if isnumeric(R)&&isscalar(R)
AAtinv = @(x) x/(R^2 );
else
AAtinv = @(x) R\( R'\x );
end
if isa(A,'function_handle')
x = x - A(AAtinv( A(x,1) - b ),2);
else
x = x - A'*AAtinv( A*x - b );
end
elseif algo==2
bb = (R')\b;
AA = @(x) (R')\(A*x); % this is now Q'
AAt= @(x) A'*(R\x);
x = x - AAt( AA(x) - bb );
end
otherwise,
error( 'Not enough arguments.' );
end
function A = implicit2explicit(Afun)
%IMPLICIT2EXPLICIT takes linear function A(x) and builts corresponding matrix
% makes an explicit matrix using the linear function
% in the function handle "Afun", where the domain is R^n
% and the range is in R^m
% Usage: implicit2explicit(Afun)
% where Afun is a TFOCS linear operator
%
% i.e., Afun([],0) returns "sz",
% where "sz" follows the usual TFOCS size conventions
% e.g., { [n1,n2], [m1,m2] } for a function
% from n1 x n2 --> m1 x m2
% and the shortcut convention
% sz = [m,n] for the common case when
% n1 = n and n2 = 1, and m1=m and m2=1
%
% Stephen Becker, [email protected], 2009
% Incorporated into TFOCS June 17 2014
sz = Afun([],0);
if iscell(sz)
nn = sz{1};
mm = sz{2};
n1 = nn(1);
n2 = nn(2);
m = prod(mm);
else
m = sz(1);
n1 = sz(2);
n2 = 1;
end
A = zeros(m,n1*n2);
e = zeros(n1,n2);
vec = @(x) x(:);
for j = 1:(n1*n2)
e(j) = 1;
A(:,j) = vec(Afun(e,1)); % call it in forward mode
e(j) = 0;
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.