-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathfir_ap_cvx.m
305 lines (256 loc) · 10.1 KB
/
fir_ap_cvx.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
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
function [h, status] = fir_ap_cvx(n, f, a, d, obj, Peak, dbg)
% fir_ap_cvx - FIR filter design using cvx
%
% function [h, status] = fir_ap_cvx(n, f, a, d, obj, Peak, dbg)
%
% Design n-tap arbitrary-phase filter that meets multiband frequency
% respones manitude specification.
% Additional function:
% (1) constraint of spikes(peaks) at two ends
% (2) multiobjective function of stopband attenuation and total energy
%
% Based on "FIR Filter Design via Spectral Factorization and Convex
% Optimization" by S.-P. Wu, S. Boyd, and L. Vandenberghe
%
% Inputs: --- similar to cfirpm
% n: number of taps returned
% f: frequency bands (-1->1)
% a: amplitude at band edges
% d: ripple in bands
% obj: parameter to determine trade-off between two objective functions
% Peak: constraint on peak amplitude, especially at ends
% dbg: flag to turn on debugging statements/plots
%
% Outputs:
% h: filter coefficients
% status: 'Solved' or 'Failed';
%
% (c) 2013-2015 The Regents of the University of California
% All Rights Reserved.
% Author: Hong Shang June 2014
if nargin < 4, error('not enough input'); end;
if nargin <= 4, obj = 0; end;
if nargin <= 5, Peak = 1e-3; end;
if nargin <= 6, dbg = 0; end;
% internal parameters for constraint
% positive constraint on spectrum, to make sure S>=0 at all frequency
epsilon = 1e-10;
% Create optimization arrays
f = f * pi; % Scale to +/- pi
oversamp = 15;
m = 2 * n * oversamp;
w = linspace(-pi,pi,m);
w = sort([w f]); % Add explicit samples to w at the edge of each specified band
% Find indices to passbands/stopbands, and fill in upper/lower bounds
idx_band = []; U_band = []; L_band = [];
nband = length(f)/2;
for band = 1:nband,
idx = find( (w >= f(band*2-1)) & (w <= f(band*2)) );
% Get amplitude from linear interpolation on band
idx_band = [idx_band idx];
if (f(band*2-1) == f(band*2))
amp = a(band*2-1);
else
amp = a(band*2-1) + (a(band*2)-a(band*2-1)) * ((w(idx) - f(band*2-1))/(f(band*2)-f(band*2-1)));
end;
U_band = [U_band (amp + d(band))];
L_band = [L_band (amp - d(band))];
end;
% Get transition indices
idx_tmp = ones(1,length(w));
idx_tmp(idx_band) = 0;
idx_tran = find(idx_tmp == 1);
% Add transition band limits to be between the max specification on each
% band and min of (0,min(L_band))
if ~isempty(idx_tran)
U_amp_tran = max(U_band);
U_tran = U_amp_tran*ones(1,length(idx_tran));
L_amp_tran = min(0, min(L_band));
L_tran = L_amp_tran*ones(1,length(idx_tran));
else
U_tran = [];
L_tran = [];
end;
% Update w, idx_band
wband = w(idx_band);
idx_band = [1:length(wband)];
wtran = w(idx_tran);
idx_tran = [1:length(wtran)] + length(wband);
w = [wband(:).' wtran(:).'];
m = size(w,2);
if dbg >= 3,
figure; plot(wband,U_band,'b.'); hold on;
plot(wband,L_band,'b.'); plot(wtran,U_tran,'r.'); plot(wtran,L_tran,'r.');
hold off; title('upper and lower bound on magnitude');
end;
% create matrix A used to compute the power spectrum
A = [ones(m,1), 2*cos(kron(w',[1:n-1])), 2*sin(kron(w',[1:n-1]))];
% Build matrix for upper bound constraints
A_U = [A(idx_band,:); A(idx_tran,:)];
U_b = [U_band U_tran];
U_b = U_b.^2;;
% Build matrices for lower bound constraints
A_L = [A(idx_band, :); A(idx_tran,:)];
L_b = [L_band L_tran];
idx_lb = find( L_b < 0);
L_b(idx_lb) = 0;
L_b = L_b.^2;
% positive constraint
idx_lb = find(L_b < epsilon^2);
L_b(idx_lb) = epsilon^2;
% Combine matrices
A_b = [A_U; -A_L];
b = [U_b -L_b];
% Set linear objective function, add up spectrum energy in transition band
% and/or stopband, to minimize total energy and stopband ripple respectively.
fmin_tran = sum(A(idx_tran,:), 1);
idx_stop = find( sqrt(U_b) < (min(sqrt(U_b))+1e-2) );
if dbg >= 3
hold on;
plot(wtran,0.8*ones(size(wtran)),'g.');
plot(w(idx_stop),0.2*ones(size(w(idx_stop))),'k.');
legend('transition band for objective function 1','stopband for objective function 2' ); hold off;
end
% matrix to pick out r(n-1) for spike constraint
Famp_array = {};
Fi = zeros(1,2*n-1); Fi(1) = 1; Famp_array{1} = Fi;
for i=2:n
Fi = zeros(2,2*n-1);
Fi(1,i) = 1;
Fi(2,n+i-1) = 1;
Famp_array{i} = Fi;
end
% Call minimization routine
if obj >= 0 % multi-objective
% minimize energy at transition bands
% cvx_begin quiet
% variables x(2*n-1) ripple_stop % Peak is actually Peak^2
% minimize fmin_tran*x + obj*ripple_stop %+ obj*Peak
% subject to
% A_b * x <= transpose(b)
% A_U(idx_stop,:) * x <= ripple_stop
% for i=1:n
% norm(Famp_array{i} * x,2) <= (n-i+1)*Peak
% end
% cvx_end
% minimize total energy
cvx_begin quiet
variables x(2*n-1) ripple_stop
minimize x(1) + obj*ripple_stop
subject to
A_b * x <= transpose(b)
A_U(idx_stop,:) * x <= ripple_stop
for i=1:n
norm(Famp_array{i} * x,2) <= (n-i+1)*Peak
end
cvx_end
else
error('invalid input of obj');
end
if ( isequal(cvx_status, 'Solved') | isequal(cvx_status, 'Inaccurate/Solved') )
status = 'Solved';
else
status = 'Failed';
h = [];
return;
end;
% reshape variable x to autocorrelation r
r = [x(1); x(2:n) + sqrt(-1) * x((n+1):(2*n-1))];
r = [conj(r(end:-1:2)); r];
if dbg >= 3
figure;
plot(-(n-1):(n-1), real(r),'r-',-(n-1):(n-1), imag(r),'b-'); legend('real','imag'); title('Autocorrelation sequence');
end
% check calculation of |r(n)|
if dbg >= 3
for i=1:n
r_test(i) = norm(Famp_array{i} * x,2);
end
figure;
plot(-(n-1):(n-1), abs(r),'r-*',0:(n-1), r_test,'b-s', 0:(n-1), [n:-1:1]*Peak,'k-.'); legend('|r(n)|','F*x','Constraint'); title('Autocorrelation magnitude');
end
% mp filter by spectral factorization
h = fmp2(r);
if dbg >= 2
figure;
plot(1:length(h), real(h),'r-*',1:length(h), imag(h),'b-*'); legend('real','imag'); title('filter coefficients');
end
% check zero-pole
if dbg >= 3
Z1 = roots(r);
P1 = zeros(length(Z1),1);
Z2 = roots(h);
P2 = zeros(length(Z2),1);
figure;
subplot(1,2,1); zplane(Z1, P1); title('zero-pole of autocorrelation');
subplot(1,2,2); zplane(Z2, P2); title('zero-pole of filter');
end
% check magnitude constraint
if dbg >= 2
S = A * x; % spectrum, |H(e^(jw))|^2
figure;
[wsort, sidx] = sort(w); plot(w(sidx), U_b(sidx), 'r-',w(sidx), L_b(sidx), 'r-'); hold on;
plot(w(sidx), S(sidx)); title('lower/upper bound with Spectrum'); hold off; set(gca,'xlim',[-pi pi]);
figure; plot_spec(f, a, d); hold on;
[wsort, sidx] = sort(w); plot(w(sidx), sqrt(S(sidx))); title('square root of Spectrum'); hold off; set(gca,'xlim',[-pi pi]);
figure;
wdisp = linspace(-pi,pi,1e4); H = fftshift(fft(h, length(wdisp)));
subplot(2,1,1); plot_spec(f, a, d); hold on; plot(wdisp, abs(H)); title('frequency response magnitude'); hold off; set(gca,'xlim',[-pi pi]);
subplot(2,1,2); plot(wdisp, unwrap(angle(H))); title('frequency response phase'); set(gca,'xlim',[-pi pi]);
% display all in one
figure;
subplot(1,2,1); plot(1:length(h), real(h),'b-',1:length(h), imag(h),'b--','linewidth',2); leg = legend('real','imag'); set(leg,'FontSize',17); ylabel('Filter Coefficients','FontSize',18); set(gca,'FontSize',18); axis tight;
subplot(1,2,2); plot(wdisp/pi, abs(H),'b-','linewidth',1.5); hold on; plot_spec(f/pi, a, d); hold off; ylabel('|H(e^j^w)|','FontSize',18); xlabel('Normalized Frequency','FontSize',18); set(gca,'FontSize',18); set(gca,'xlim',[min(f/pi)-0.02 max(f/pi)+0.02]); set(gca,'ylim',[min(a)-min(d)-0.02 max(a)+max(d)+0.02]);
set(gcf, 'Position', [100,100,800,350], 'PaperPositionMode', 'auto');
hgexport(gcf, ['example_FIR_design_n',num2str(length(h)),'.eps']);
end
return;
end
% fft wrt the center of the array, instead of the first sample
% written by John Pauly, 1992
% (c) Board of Trustees, Leland Stanford Junior University
function y=fftc(x)
y = fftshift(fft(fftshift(x)));
end
% Generate a minimum phase filter by Spectral factorization
% written by John Pauly, 1992
% (c) Board of Trustees, Leland Stanford Junior University
% modified by Hong Shang, withnot adding delta2 as in equal-ripple pm
% filter design
function hmp = fmp2(h)
h = transpose(h(:));
l = length(h);
if rem(l,2) == 0,
disp('filter length must be odd');
return;
end;
lp = 8*exp(ceil(log(l)/log(2))*log(2));
hp = [zeros(1,ceil((lp-l)/2)) h zeros(1,floor((lp-l)/2))];
hpf = fftc(hp);
% hpfs = hpf-min(real(hpf))*1.000001;
% add ripple of stopband for equal-ripple pm filter design, but this is not
% necessary in my case when the spectrum is already positive, which makes
% the spectral factorization result less accurate
hpfs = hpf;
hpfmp = mag2mp(sqrt(abs(hpfs)));
hpmp = ifft(fftshift(conj(hpfmp)));
hmp = hpmp(1:(l+1)/2);
end
% mag2mp - take the magnitude of the fft of a signal, and return the
% fft of the analytic signal.
% as = mag2mp(ms)
% as - fft of analytic signal
% ms - magnitude of analytic signal fft.
% Written by John Pauly, Oct 1989
% (c) Board of Trustees, Leland Stanford Junior University
function [a] = mag2mp(x)
n = length(x);
xl = log(x); % log of mag spectrum
xlf = fft(xl); %
xlfp(1) = xlf(1); % keep DC the same
xlfp(2:(n/2)) = 2*xlf(2:(n/2)); % double positive freqs
xlfp((n/2+1)) = xlf((n/2+1)); % keep half Nyquist the same,too
xlfp((n/2+2):n) = 0*xlf((n/2+2):n); % zero neg freqs
xlaf = ifft(xlfp); %
a = exp(xlaf); % complex exponentiation
end