-
Notifications
You must be signed in to change notification settings - Fork 60
/
smooth_quad.m
159 lines (148 loc) · 5.41 KB
/
smooth_quad.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
function op = smooth_quad( P, q, r, use_eig )
%SMOOTH_QUAD Quadratic function generation.
% FUNC = SMOOTH_QUAD( P, q, r ) returns a function handle that implements
%
% FUNC(X) = 0.5 * TFOCS_DOT( P * x, x ) + TFOCS_DOT( q, x ) + r.
%
% All arguments are optional; the default values are P=I, q=0, r=0. In
% particular, calling FUNC = SMOOTH_QUAD with no arguments yields
%
% FUNC(X) = 0.5 * TFOCS_NORMSQ( X ) = 0.5 * TFOCS_DOT( X, X ).
%
% If supplied, P must be a scalar, square matrix, or symmetric linear
% operator. Furthermore, it must be positive semidefinite (convex) or
% negative semidefinite (concave). TFOCS does not verify operator
% symmetry or definiteness; that is your responsibility.
% If P is a vector, then it assumed this is the diagonal part of P.
% Note: when P is diagonal, this function can compute a proximity
% operator. If P is zero, then smooth_linear should be called instead.
% When P is an explicit matrix, this can also act as a proimity operator
% but it may be slow, since it must invert (I+tP). True use_eig mode (see below)
%
% If P is a scaling matrix, like the identity or a multiple of the identity
% (say, P*x = 5*x), then specifying the scaling factor is sufficient (in
% this example, 5). If P is empty, then P=1 is assumed.
%
% FUNC = SMOOTH_QUAD( P, q, r, use_eig )
% will perform a one-time (but expensive) eigenvalue decomposition
% of P if use_eig=true, which will speed up future iterations. In general,
% this may be significantly faster, especially if you run
% an algorithm for many iterations.
% This mode is only useful when P is a full matrix and when
% smooth_quad is used as a proximity operator; does not affect
% it when used as a smooth operator. New in v 1.3.
%
% See also smooth_linear.m
if nargin == 0,
op = @smooth_quad_simple;
return
end
if isa( P, 'function_handle' ),
sz = P([],0);
if ~isequal( sz{1}, sz{2} ),
error( 'P must be square.' );
end
elseif ~isnumeric( P ),
error( 'P must be a scalar, matrix, or linear operator.' );
elseif isempty(P)
P = 1;
elseif ndims( P ) > 2 || (~isvector(P) && size( P, 1 ) ~= size( P, 2 ) ),
error( 'P must be a square matrix.' );
end
if nargin < 2 || isempty( q ),
q = 0;
elseif numel(P) > 1 && numel(q) > 1 && length(q) ~= size(P,1),
error( 'Dimension mismatch between p and q.' );
end
if nargin < 3 || isempty( r ),
r = 0;
elseif numel(r) > 1 || ~isreal( r ),
error( 'r must be a real scalar.' );
end
if nargin < 4, use_eig = false; end
if isnumeric( P ),
if isvector( P )
if any(P) < 0 && any(P) < 0
error(' P must be convex (minimization) or concave (maximization) but cannot be mixed');
end
P = P(:); % make it a column vector
op = @(varargin)smooth_quad_diag_matrix( P, q, r, varargin{:} );
else
P = 0.5 * ( P + P' );
if use_eig
[V,DD] = safe_eig(P); dd = diag(DD);
else
V = []; dd = [];
end
op = @(varargin)smooth_quad_matrix( P, q, r, V, dd, varargin{:} );
end
else
op = @(varargin)smooth_quad_linop( P, q, r, varargin{:} );
end
function [ v, g ] = smooth_quad_matrix( P, q, r, V, dd, x, t )
switch nargin
case 6,
if size(x,2)>1 && size(q,1)==1
% User meant to supply repmat(q,1,size(x,2)) so we do it for
% them
g = P * x + repmat(q,1,size(x,2));
v = 0.5 * tfocs_dot( x, g + repmat(q,1,size(x,2)) ) + r;
else
g = P * x + q;
v = 0.5 * tfocs_dot( x, g + q ) + r;
end
case 7,
n = length(x);
if ~isempty(V) && ~isempty(dd)
% Use the stored eigenvalue decomposition to improve speed
x = V*( (V'*(x-t*q))./(t*dd+ones(n,1)) );
else
x = (eye(n) + t*P)\ (x-t*q);
end
g = x; % for this case, "g" is NOT the gradient
v = 0.5 * tfocs_dot( P*x + 2*q, x ) + r;
end
%if nargin == 5,
%error( 'Proximity minimization not supported by this function.' );
%end
% Note: we don't support proximity minimization, but there is
% nothing that prevents it theoretically. You would need to know P^-1
% (and ideally calculate it quickly).
% In particular, if P is diagonal, then it is easy.
%g = P * x + q;
%v = 0.5 * tfocs_dot( x, g + q ) + r;
% Jan 10, 2011: this function isn't correct for case 5
% function [ v, g ] = smooth_quad_diag_matrix( p, q, r, x, t )
% switch nargin
% case 4,
% case 5,
% x = (1./(t*p+1)) .* x;
% end
% g = p .* x + q;
% v = 0.5 * tfocs_dot( x, g + q ) + r;
function [ v, g ] = smooth_quad_diag_matrix( p, q, r, x, t )
switch nargin
case 4,
g = p .* x + q;
v = 0.5 * tfocs_dot( x, g + q ) + r;
case 5,
x = (1./(t*p+1)) .* (x-t*q);
g = x; % for this case, "g" is NOT the gradient
v = 0.5 * tfocs_dot( p.*x + 2*q, x ) + r;
end
function [ v, g ] = smooth_quad_linop( P, q, r, x, t )
if nargin == 5,
error( 'Proximity minimization not supported by this function.' );
end
g = P( x, 1 ) + q;
v = 0.5 * tfocs_dot( x, g + q ) + r;
function [ v, x ] = smooth_quad_simple( x, t )
switch nargin,
case 1,
case 2,
x = (1/(t+1)) * x;
end
v = 0.5 * tfocs_normsq( x );
% 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.