Mercurial > hg > ltpda
view m-toolbox/classes/@ao/fftfilt_core.m @ 44:409a22968d5e default
Add unit tests
author | Daniele Nicolodi <nicolodi@science.unitn.it> |
---|---|
date | Tue, 06 Dec 2011 18:42:11 +0100 (2011-12-06) |
parents | f0afece42f48 |
children |
line wrap: on
line source
% FFTFILT_CORE Simple core method which computes the fft filter. %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DESCRIPTION: Simple core method which computes the fft filter. % % CALL: ao = fftfilt_core(ao, filt, Npad) % % INPUTS: ao: Single input analysis object % Npad: Number of bins for zero padding % filt: The filter to apply to the data % smodel - a model to filter with. % mfir - an FIR filter % miir - an IIR filter % tf - an ltpda_tf object. Including: % - pzmodel % - rational % - parfrac % % VERSION: $Id: fftfilt_core.m,v 1.17 2011/05/30 10:42:32 mauro Exp $ % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% function bs = fftfilt_core(varargin) bs = varargin{1}; filt = varargin{2}; Npad = varargin{3}; if nargin == 4 inCondsMdl = varargin{4}; else inCondsMdl = []; end [m, n] = size(bs.data.y); % FFT time-series data fs = bs.data.fs; % zero padding data before fft if isempty(Npad) Npad = length(bs.data.y) - 1; end if n == 1 tdat = ao(tsdata([bs.data.y;zeros(Npad,1)], fs)); else tdat = ao(tsdata([bs.data.y zeros(1,Npad)], fs)); end % get onesided fft ft = fft_core(tdat, 'one'); switch class(filt) case 'smodel' % Evaluate model at the given frequencies amdly = filt.setXvals(ft.data.x).double; amdly = reshape(amdly, size(ft.data.y)); amdl = ao(fsdata(ft.data.x, amdly, fs)); % set units bs.setYunits(simplify(bs.data.yunits .* filt.yunits)); case {'miir', 'mfir', 'pzmodel', 'parfrac', 'rational'} % Check if the frequency of the filter is the same as the frequency % of the AO. if isa (filt, 'ltpda_filter') && fs ~= filt.fs error('### Please use a filter with the same frequency as the AO [%dHz]', fs); end % get filter response on given frequencies amdl = resp(filt, plist('f', ft.data.x)); % set units bs.setYunits(simplify(bs.data.yunits .* filt.ounits ./ filt.iunits)); case 'ao' % check if filter and data have the same shape if size(ft.data.y)~=size(filt.data.y) % reshape amdl = copy(filt,1); amdl.setX(ft.data.x); amdl.setY(reshape(filt.data.y,size(ft.data.x))); amdl.setName(filt.name); else amdl = copy(filt,1); amdl.setName(filt.name); end % set units bs.setYunits(simplify(bs.data.yunits .* amdl.data.yunits)); case 'collection' % run over collection elements amdl = ao(fsdata(ft.data.x, ones(size(ft.data.x)), fs)); for ii=1:numel(filt.objs) switch class(filt.objs{ii}) case 'smodel' % Evaluate model at the given frequencies amdly = filt.objs{ii}.setXvals(ft.data.x).double; amdly = reshape(amdly, size(ft.data.y)); amdl_temp = ao(fsdata(ft.data.x, amdly, fs)); % set units bs.setYunits(simplify(bs.data.yunits .* filt.objs{ii}.yunits)); case {'miir'} % get filter response on given frequencies amdly = utils.math.mtxiirresp(filt.objs{ii},ft.data.x,fs,[]); amdl_temp = ao(fsdata(ft.data.x, amdly, fs)); % set units bs.setYunits(simplify(bs.data.yunits .* filt.objs{ii}.ounits ./ filt.objs{ii}.iunits)); case 'ao' % check if filter and data have the same shape if size(ft.data.y)~=size(filt.objs{ii}.data.y) % reshape amdl_temp = copy(filt.objs{ii},1); amdl_temp.setX(ft.data.x); amdl_temp.setY(reshape(filt.objs{ii}.data.y,size(ft.data.x))); amdl_temp.setName(filt.objs{ii}.name); else amdl_temp = copy(filt.objs{ii},1); amdl_temp.setName(filt.objs{ii}.name); end % set units bs.setYunits(simplify(bs.data.yunits .* amdl_temp.data.yunits)); case 'filterbank' % get filter response on given frequencies amdly = utils.math.mtxiirresp(filt.objs{ii}.filters,ft.data.x,fs,filt.objs{ii}.type); amdl_temp = ao(fsdata(ft.data.x, amdly, fs)); % handle units switch lower(filt.objs{ii}.type) case 'parallel' % set units of the output object bs.setYunits(simplify(bs.data.yunits .* filt.objs{ii}.filters(1).ounits ./ filt.objs{ii}.filters(1).iunits)); case 'series' % get units from the series sunits = filt.objs{ii}.filters(1).ounits ./ filt.objs{ii}.filters(1).iunits; for jj = 2:numel(filt.objs{ii}.filters) sunits = sunits.*filt.objs{ii}.filters(jj).ounits ./ filt.objs{ii}.filters(jj).iunits; end % set units of the output object bs.setYunits(simplify(bs.data.yunits .* sunits)); end end % update response amdl = amdl .* amdl_temp; end otherwise error('### Unknown filter mode.'); end % Add initial conditions if ~isempty(inCondsMdl) && ~isempty(inCondsMdl.expr.s) inCondsMdl.setXvals(amdl.x); inCondsMdl.setXunits(amdl.xunits); inCondsMdl.setYunits(amdl.yunits*ft.yunits); inCondsEval = inCondsMdl.eval; % Multiply by model and take inverse FFT y = ifft_core(ft.*amdl+inCondsEval, 'symmetric'); else % Multiply by model and take inverse FFT y = ifft_core(ft.*amdl, 'symmetric'); end % split and reshape the data if m == 1 bs.setY(y.data.getY(1:n)); else bs.setY(y.data.getY(1:m)); end % clear errors bs.clearErrors; end