-
Notifications
You must be signed in to change notification settings - Fork 60
/
Copy pathprox_scale.m
111 lines (103 loc) · 4.31 KB
/
prox_scale.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
function op = prox_scale( proxF, scale )
%PROX_SCALE Scaling a proximity/projection function.
% PSCALE = PROX_SCALE( PROXF, s ) is the proximity function formed
% by multiplying the input by the real value SCALE, then calling the
% function PROXF. In other words,
% PSCALE( y ) = PROXF( s * y ).
% In three-argument form, [v,x]=PSCALE(y,t) finds the minimizer of
% PROXF( s * x ) + (0.5/t)*||x-y||_2^2
%
% Why scale the input, and not the output? When the proximity function
% is a norm, the two choices are equivalent when SCALE >= 0, since
% ||s*x|| = abs(s)*||x||. However, for *indicator* functions, scaling
% the output is useless. Scaling the input grows or shrinks the domain
% of PSCALE by choosing SCALE<1 or SCALE>1, respectively.
%
% Note that when constructing our equivalence we assume a particular
% form for the proximity minimization. More generally, PROJ_SCALE will
% preserve equivalence if 0.5*||x-y||^2 is replaced with D(x-y), as
% long as D() satisfies the following homogeneity property:
% D(s*z) = s^2*D(z)
% If this is not the case, you will not be able to use the TFOCS
% internal scaling with this function.
if ~isa( proxF, 'function_handle' ),
error( 'The first argument must be a function handle.' );
elseif ~isa( scale, 'double' )
error( 'The second argument must be a nonzero scalar or matrix');
elseif numel( scale ) == 1 && isreal( scale )
% (we could handle complex, but need to work this out)
if scale == 0,
error( 'The second argument must be a nonzero real scalar.' );
elseif scale == 1,
op = proxF;
else
op = @(varargin)prox_scale_impl( proxF, scale, varargin{:} );
end
else
% added Sept 14, allowing matrix scaling with "scale" = L
% First, we verify that LL^* = nu * Identity
% We do this probabilistically in order to avoid potentially
% huge matrices:
[m,n] = size(scale);
seed = rng;
xTest = randn(m,5);
rng(seed); % try not to affect the random seed
xOut = scale*(scale'*xTest);
if size(xOut,1) ~= m || size(xOut,2) ~= 5
% This shouldn't every happen with matrix inputs,
% but if we later allow functions, then it could happen
% if the function doesn't actually do what we think.
error('Something is wrong with the linear operator');
end
ref = xOut./xTest;
nu = mean(ref(:));
if norm( ref(:) - nu*ones(5*m,1) )/norm( nu*ones(5*m,1) ) > 1e-10
% If we have LL^* = diag(d), this will be compatible
% with some proxes, so issue a warning
if mean( std(ref,[],2) )/norm(ref(:)/5) < 1e-10
nu = mean(ref,2);
warning('TFOCS:proxMustBeSeparable',...
'Matrix is not a tight frame but SS'' is diagonal, so this works if the prox is separable');
else
error('Matrix does not appear to be a tight frame: need S*S'' = nu*I');
end
else
fprintf('Matrix is a tight frame with constant nu = %.2e\n', nu );
end
op = @(varargin)prox_scaleMatrix_impl( proxF, scale, nu, varargin{:} );
end
function varargout = prox_scale_impl( proxF, s, y, t )
no = max(nargout,1);
if nargin < 4 || t == 0,
[ varargout{1:no} ] = proxF( s * y );
else
% For three-input mode, we wish to simulate
% x = argmin_x projectorF(s*x) + (0.5/t)*||x-y||^2
% Setting z = s * x, we have
% z = argmin_z projectorF(z) + (0.5/(s^2*t))*||z-s*y||^2
% Therefore, we have
% [v,z] = projectorF( s * y, t * s^2 ); x = z / s;
[ varargout{1:no} ] = proxF( s * y, abs(s)^2 * t );
end
if no > 1,
varargout{2} = varargout{2} / s;
end
function varargout = prox_scaleMatrix_impl( proxF, L, nu, y, t )
no = max(nargout,1);
Ly = L*y;
if nargin < 5 || t == 0,
[ varargout{1:no} ] = proxF( Ly );
else
% Proposition 11 in Combettes and Pesquet 2007
%"A Douglas?Rachford Splitting Approach to Nonsmooth Convex Variational
% Signal Recovery"
% If L*L'=nu I, then if g(x) = f(L*x),
% prox_g = I + 1/nu L'( prox_{nu f} - I )L
[ varargout{1:no} ] = proxF( Ly, nu * t );
end
if no > 1,
varargout{2} = y + L'*((varargout{2} - Ly)./ nu);
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.