function [Bm,Am,FIR]=parfiltid(in,out,p,NFIR); % PARFILTID - System identification in the form of second-order parallel filters for a given % pole set. % [Bm,Am,FIRcoeff]=parfiltid(INPUT,OUTPUT,P,NFIR); identifies the second-order sections % [Bm,Am] and the coefficients of the FIR part (FIRcoeff) for a given % pole set P. The parameters are set such that, when the INPUT signal is % filtered by the parallel filter, it gives an output which is the closest % to the OUTPUT vector in the LS sense. The number of taps in the parallel % FIR filter is set by NFIR. The default is NFIR=1, in this case FIRcoeff % is a simple gain. The only difference from the PARFILTDES function is that % now the input can be arbitrary, and not just a unit pulse as for filter design. % % The Bm and Am matrices are containing the [b0 b1]' and [1 a0 a1]' % coefficients for the different sections in their columns. For example, % Bm(:,3) gives the [b0 b1]' parameters of the third second-order % section. These can be used by the filter command separatelly (e.g., by % y=filter(Bm(:,3),Am(:,3),x), or by the PARFILT command. % % Note that this function does not support pole multiplicity, so P should % contain each pole only once. % % More details about the parallel filter can be found in % % http://www.mit.bme.hu/~bank/parfilt % % Balazs Bank, "Perceptually Motivated Audio Equalization Using Fixed-Pole Parallel % Second-Order Filters", IEEE Signal Processing Letters, 2008. % http://www.acoustics.hut.fi/go/spl08-parfilt % % Balazs Bank, "Direct Design of Parallel Second-order Filters for % Instrument Body Modeling", International Computer Music Conference, % Copenhagen, Denmark, Aug. 2007. % http://www.acoustics.hut.fi/go/icmc07-parfilt % % C. Balazs Bank, Helsinki University of Technology, 2007. if nargin==3, NFIR=1; end; p=p(find(p)); %we don't want to have any poles in the origin - for that we have the parallel FIR part for k=1:length(p), %making the filter stable by flipping the poles into the unit cirle if abs(p(k))>1 p(k)=1/conj(p(k)); end; end; p=cplxpair(p); %order it to complex pole pairs + real ones afterwards %in order to have second-order sections only (i.e., no first order) pnum=length(p); %number of poles ppnum=2*floor(pnum/2); %the even part of pnum ODD=0; if pnum>ppnum, %if pnum is odd ODD=1; end; in=in(:); %making column vectors out=out(:); OUTL=length(out); INL=length(in); %making input the same length as the output if INL>OUTL, in=in(1:OUTL); end; if INL0, FIR=par(pnum+1:pnum+NFIR); end;