Mercurial > hg > ltpda
view m-toolbox/classes/@matrix/linfitsvd.m @ 44:409a22968d5e default
Add unit tests
author | Daniele Nicolodi <nicolodi@science.unitn.it> |
---|---|
date | Tue, 06 Dec 2011 18:42:11 +0100 |
parents | f0afece42f48 |
children |
line wrap: on
line source
% LINFITSVD Linear fit with singular value decomposition %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % % DESCRIPTION: Linear least square problem with singular value % decomposition % % ALGORITHM: Perform linear identification of the parameters of a % multichannel systems. The results of different experiments on the same % system can be passed as input. The algorithm, thanks to the singular % value decomposition, extract the maximum amount of information from each % single channel and for each experiment. Total information is then % combined to get the final result. % % CALL: pars = linfitsvd(os1,...,osn,pl); % % INPUT: % - osi are vector of system output signals. They must be % Nx1 matrix objects, where N is the output dimension of the % system % % OUTPUT: % - pars: a pest object containing parameter estimation % % <a href="matlab:utils.helper.displayMethodInfo('matrix', 'linfitsvd')">Parameters Description</a> % % VERSION: $Id: linfitsvd.m,v 1.38 2011/04/08 08:56:32 hewitson Exp $ % %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% function varargout = linfitsvd(varargin) %%% LTPDA stufs and get data %%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%%% % Check if this is a call for parameters if utils.helper.isinfocall(varargin{:}) varargout{1} = getInfo(varargin{3}); return end import utils.const.* utils.helper.msg(msg.OMNAME, 'running %s/%s', mfilename('class'), mfilename); % Collect input variable names in_names = cell(size(varargin)); for ii = 1:nargin,in_names{ii} = inputname(ii);end % Collect all ltpdauoh objects [mtxs, mtxs_invars] = utils.helper.collect_objects(varargin(:), 'matrix', in_names); [pl, invars] = utils.helper.collect_objects(varargin(:), 'plist'); inhists = [mtxs(:).hist]; % combine plists pl = parse(pl, getDefaultPlist()); %%% get input parameters % if the model is a matrix of smodels lmod = find(pl, 'dmodel'); % if the model is ssm inNames = find(pl,'InNames'); outNames = find(pl,'OutNames'); % common parameters mod = find(pl,'Model'); fitparams = find(pl,'FitParams'); inputdat = find(pl,'Input'); WF = find(pl,'WhiteningFilter'); nloops = find(pl,'Nloops'); ncut = find(pl,'Ncut'); npad = find(pl,'Npad'); kwnpars = find(pl,'KnownParams'); tol = find(pl,'tol'); fastopt = find(pl,'fast'); setalias = find(pl,'SetAlias'); sThreshold = find(pl,'sThreshold'); diffStep = find(pl,'diffStep'); boundedpars = find(pl,'BoundedParams'); boudary = find(pl,'BoundVals'); % check if there are bounded parameters if ~isempty(boundedpars) boundparams = true; else boundparams = false; end % check the class of the model if isa(mod,'ssm') ssmmod = true; else ssmmod = false; end %%% some sanity checks if ~ssmmod if numel(mtxs)~= numel(inputdat.objs) error('Number of input data vectors must be the same of fit data vectors') end end [fitparams,idx] = sort(fitparams); fitparvals = fitparams; if ssmmod %%% get fit parameters for ii=1:numel(fitparams) fitparvals{ii} = mod.getParameters(plist('names',fitparams{ii})); end if isempty(diffStep) sdiffStep = cell2mat(fitparvals).*0.01; idz = sdiffStep == 0; sdiffStep(idz) = 1e-7; else sdiffStep = diffStep(idx); end else %%% get a single set of parameters totparnames = {}; totparvals = {}; for ii=1:numel(mod.objs) aa = mod.objs(ii).params; cc = mod.objs(ii).values; % get total parameter names [bb,i1,i2]=union(totparnames,aa); totparnames = [totparnames(i1),aa(i2)]; % get total parameter values totparvals = [totparvals(i1),cc(i2)]; end [totparnames,id] = sort(totparnames); totparvals = totparvals(id); %%% get fit parameters [nn,i1,i2] = intersect(totparnames,fitparams); fitparams = totparnames(i1); fitparvals = totparvals(i1); [fitparams,id] = sort(fitparams); fitparvals = fitparvals(id); end if ~ssmmod %%% linearize model with respect to fit parameters if isempty(lmod) lmod = linearize(mod,plist('Params',fitparams,'Sorting',false)); end end if isempty(WF) wfdat = copy(mtxs,1); elseif ~fastopt %%% whitening fit data wfdat = copy(mtxs,1); for ii=1:numel(mtxs) wfdat(ii) = filter(mtxs(ii),WF); end end % decide to pad in any case, assuming the objects have the same length if isempty(npad) npad = length(mtxs(1).objs(1).data.y) - 1; end % set alias if there are if setalias && (~ssmmod && ~isempty(mod.objs(1).aliasNames)) nsecs = mtxs(1).objs(1).data.nsecs; fs = mtxs(1).objs(1).data.fs; plalias = plist('nsecs',nsecs,'npad',npad,'fs',fs); for ii=1:numel(mod.objs) mod.objs(ii).assignalias(mod.objs(ii),plalias); end for jj=1:numel(lmod.objs) for ii=1:numel(lmod.objs{jj}.objs) lmod.objs{jj}.objs(ii).assignalias(lmod.objs{jj}.objs(ii),plalias); end end end % do a copy to add at the output pest outmod = copy(mod,1); % check if the fast option is active if ~ssmmod && fastopt % set length of fft (this should match the operation made in fftfilt_core) nfft = length(mtxs(1).objs(1).data.y) + npad; fs = mtxs(1).objs(1).data.fs; % get fft freqs for current data. type option must match the one used % in fftfilt_core for fft_core fftfreq = utils.math.getfftfreq(nfft,fs,'one'); % calculate freq resp of diagonal elements of WF rWF = getWFresp(WF,fftfreq,fs); % combine symbolic models with rWF mod = joinmodelandfilter(mod,rWF); lmod = joinmodelandfilter(lmod,rWF); WF = []; %%% whitening fit data wfdat = copy(mtxs,1); for ii=1:numel(mtxs) for jj=1:numel(mtxs(ii).objs) wfdat(ii).objs(jj) = fftfilt_core(wfdat(ii).objs(jj),rWF.objs(jj),npad); end end clear rWF end % init storage struct loop_results = struct('a',cell(1),... 'Ca',cell(1),... 'Corra',cell(1),... 'Vu',cell(1),... 'bu',cell(1),... 'Cbu',cell(1),... 'Fbu',cell(1),... 'mse',cell(1),... 'params',cell(1),... 'ppm',cell(1)); % init user interaction variable reply = 'N'; %%% run fit loop % This causes problems on some machines so we remove it for now until we % can investigate further. % fftw('planner', 'exhaustive'); for kk=1:nloops % init index variable xxx = 1; % init data struct exps = struct(); %%% Set fit parameters into model if ssmmod fs = wfdat(1).objs(1).fs; mod.doSetParameters(fitparams,cell2mat(fitparvals)); lmod = parameterDiff(mod,plist('names',fitparams,'values',sdiffStep)); lmod.modifyTimeStep(plist('newtimestep',1/fs)); else % fitparvals are updated at each fit loop if fastopt for ii = 1:numel(mod.objs) mod.objs(ii).objs{2}.setParams(fitparams,fitparvals); end else for ii = 1:numel(mod.objs) mod.objs(ii).setParams(fitparams,fitparvals); end end end %%% run over input data for ii=1:numel(inputdat.objs) if ssmmod %%% extract input if isa(inputdat.objs{ii},'ao') in = inputdat.objs{ii}; elseif isa(inputdat.objs{ii},'matrix') in = inputdat.objs{ii}.objs(:); else error('Unknown Input data type.') end %%% calculates zero order response plsym = plist('AOS VARIABLE NAMES',inNames{ii},... 'RETURN OUTPUTS',outNames{ii},... 'AOS',in); %eo = simulate(lmod,plsym); %zor = matrix(eo,plist('shape',[numel(eo) 1])); zor = simulate(lmod,plsym); % check dimensions if size(zor.objs,1)<size(zor.objs,2) % do transpose zor = transpose(zor); end %%% calculates first order response for jj=1:numel(fitparams) % get output ports names [token, remain] = strtok(outNames{ii},'.'); loutNames = token; for zz=1:numel(token) loutNames{zz} = sprintf('%s_DIFF_%s%s',token{zz},fitparams{jj},remain{zz}); end plsym = plist('AOS VARIABLE NAMES',inNames{ii},... 'RETURN OUTPUTS',loutNames,... 'AOS',in); %eol = simulate(lmod,plsym); %fstor(jj) = matrix(eol,plist('shape',[numel(eol) 1])); fstor(jj) = simulate(lmod,plsym); % check dimensions if size(fstor(jj).objs,1)<size(fstor(jj).objs,2) % do transpose fstor(jj) = transpose(fstor(jj)); end fstor(jj).setName(fitparams{jj}); end else %%% calculates zero order response zor = fftfilt(inputdat.objs{ii},mod,plist('Npad',npad)); %%% calculates first order response for jj=1:numel(lmod.objs) fstor(jj) = fftfilt(inputdat.objs{ii},lmod.objs{jj},plist('Npad',npad)); fstor(jj).setName(lmod.objs{jj}.name); end end if isempty(WF) wzor = zor; wfstor = fstor; else %%% whitening zor wzor = filter(zor,WF); %%% whitening fstor for jj=1:numel(fstor) wfstor(jj) = filter(fstor(jj),WF); wfstor(jj).setName(fstor(jj).name); end end %%% Collect object for the fit procedure for jj=1:numel(wfdat(ii).objs) % get difference between fit data and zero order response tfdat = wfdat(ii).objs(jj) - wzor.objs(jj); % remove whitening filter transient tfdats = tfdat.split(plist('samples',[ncut+1 numel(tfdat.y)])); % insert into exps struct fitdata(xxx,1) = tfdats; % build fit basis for gg=1:numel(fitparams) for hh=1:numel(wfstor) if strcmp(fitparams(gg),wfstor(hh).name) bsel = wfstor(hh).objs(jj); % remove whitening filter transient bsels = bsel.split(plist('samples',[ncut+1 numel(tfdat.y)])); end end bs(gg) = bsels; end % insert basis fitbasis(xxx,:) = bs; % step up xxx xxx = xxx + 1; end %jj=1:numel(wfdat(ii).objs) end %ii=1:numel(inputdat.objs) %%% build input objects [NN,MM] = size(fitbasis); for zz=1:MM H(1,zz) = matrix(fitbasis(:,zz)); end Y = matrix(fitdata); %%% Insert known parameters if ~isempty(kwnpars) kwnparmanes = kwnpars.names; kwnparvals = kwnpars.y; kwnparerrors = kwnpars.dy; % init struct groundexps = struct; for ii=1:numel(kwnparmanes) for jj=1:numel(fitparams) if strcmp(kwnparmanes{ii},fitparams{jj}) groundexps(ii).pos = jj; groundexps(ii).value = kwnparvals(ii); groundexps(ii).err = kwnparerrors(ii); end end end end %%% do fit if ~isempty(kwnpars) && isfield(groundexps,'pos') plfit = plist('KnownParams',groundexps,'sThreshold',sThreshold); [out,a,Ca,Corra,Vu,bu,Cbu,Fbu,mse,dof,ppm] = linlsqsvd(H,Y,plfit); else plfit = plist('sThreshold',sThreshold); [out,a,Ca,Corra,Vu,bu,Cbu,Fbu,mse,dof,ppm] = linlsqsvd(H,Y,plfit); end %%% update parameters values for ii=1:numel(fitparams) fitparvals{ii} = fitparvals{ii} + a(ii); end %%% check for bouded params if boundparams for pp=1:numel(fitparams) for qq=1:numel(boundedpars) if strcmp(fitparams{pp},boundedpars{qq}); % check boudaries bd = boudary{qq}; if fitparvals{pp}<bd(1) fitparvals{pp} = bd(1); elseif fitparvals{pp}>bd(2) fitparvals{pp} = bd(2); end end end end end %%% store intermediate results loop_results(kk).a = a; loop_results(kk).Ca = Ca; loop_results(kk).Corra = Corra; loop_results(kk).Vu = Vu; loop_results(kk).bu = bu; loop_results(kk).Cbu = Cbu; loop_results(kk).Fbu = Fbu; loop_results(kk).mse = mse; loop_results(kk).params = fitparvals; loop_results(kk).ppm = ppm; utils.helper.msg(msg.IMPORTANT, 'loop %d, mse %d\n',kk,mse); % check fit stability and accuracy fitmsg = checkfit(Vu,a); if ~isempty(fitmsg) && ~strcmpi(reply,'Y') % display message utils.helper.msg(msg.IMPORTANT, fitmsg); % decide if stop for cycle reply = input('Do you want to carry on with fit iteration? Y/N [Y]: ', 's'); if isempty(reply) reply = 'Y'; end if strcmpi(reply,'N') break end end % check convergence condvec = (abs(a).^2)./diag(Ca); % parameters a are going to zero during the fit iterations if all(condvec < tol) condmsg = sprintf(['Fit parameters have reached convergence.\n'... 'Fit loop was terminated at iteration %s.\n'],num2str(kk)); utils.helper.msg(msg.IMPORTANT, condmsg); break end end %for kk=1:nloops %%% output data % get minimum mse if ~isempty(fitmsg) val = mse; idx = kk; else mseprog = zeros(numel(loop_results),1); for ii=1:numel(loop_results) mseprog(ii) = loop_results(ii).mse; end [val,idx] = min(abs(mseprog-1)); % get the nearest to 1 value utils.helper.msg(msg.PROC1, 'Output values at minimum mse, mse = %d\n',val); end % output pest object pe = pest(); pe.setY(cell2mat(loop_results(idx).params)); pe.setDy(sqrt(diag(loop_results(idx).Ca))); pe.setCov(loop_results(idx).Ca); pe.setChi2(loop_results(idx).mse); pe.setNames(fitparams); pe.setDof(dof); pe.setModels(outmod); pe.setName(sprintf('linfitsvd(%s)', mod.name)); pe.setProcinfo(plist('loop_results',loop_results)); % set History pe.addHistory(getInfo('None'), pl, [mtxs_invars(:)], [inhists(:)]); varargout{1} = pe; end %-------------------------------------------------------------------------- % check fit accuracy and stability %-------------------------------------------------------------------------- function msg = checkfit(V,aa) if size(V,1)<numel(aa) % The number of parameters combinations is less than the number of fit % parameters. Information cannot be recostructed fit results will be % compromised VV = abs(V).^2; num = numel(aa)-size(V,1); mVV = max(VV); % try to identify non measured params unmparams = []; for jj = 1:num [vl,idx] = min(mVV); unmparams = [unmparams idx]; mVV(idx) = []; end msg = sprintf(['!!! The number of parameters combinations is less than the number of fit parameters. \n' ... 'Information cannot be recostructed and fit results will be compromised. \n'... 'Try to remove parameters %s from the fit parameters list or add information with more experiments !!!\n'],num2str(unmparams)); else unmparams = []; trh1 = 1e-4; % eigenvectors are normalized, therefore square of the rows of V are sum % to one. Each column of V store the coefficients for a given parameter % for the set of eigenvectors for jj = 1:size(V,2) cl = abs(V(:,jj)).^2; if all(cl<trh1) unmparams = [unmparams jj]; end end if ~isempty(unmparams) msg = sprintf(['!!! Parameter/s %s is/are not well measured. \n'... 'Fit accuracy could be impaired. \n'... 'Try to remove such parameters from the fit parameters list !!!\n'],num2str(unmparams)); else msg = ''; end end end %-------------------------------------------------------------------------- % calculate frequency response of diagonal elements of the whitening filter %-------------------------------------------------------------------------- function rsp = getWFresp(wf,f,fs) % run over wf elements obj = wf.objs; [rw,cl] = size(obj); if rw~=cl error('??? Matrix of whitening filter must be square'); end amdl = ao.initObjectWithSize(rw,1); for jj=1:rw % check filter type switch lower(class(obj(jj,jj))) case 'filterbank' % get filter response on given frequencies amdly = utils.math.mtxiirresp(obj(jj,jj).filters,f,fs,obj(jj,jj).type); amdl(jj,1) = ao(fsdata(f, amdly, fs)); case 'miir' % get filter response on given frequencies amdly = utils.math.mtxiirresp(obj(jj,jj),f,fs,[]); amdl(jj,1) = ao(fsdata(f, amdly, fs)); end end rsp = matrix(amdl,plist('shape',[rw 1])); end %-------------------------------------------------------------------------- % Join Symbolic model and whitening filter for fast calculations %-------------------------------------------------------------------------- function jmod = joinmodelandfilter(smod,fil) switch class(smod) case 'matrix' mobj = smod.objs; [nn,mm] = size(mobj); nmobj = collection.initObjectWithSize(nn,mm); for ii=1:nn for jj=1:mm nmobj(ii,jj) = collection(fil.objs(ii,1),mobj(ii,jj)); nmobj(ii,jj).setName(mobj(ii,jj).name); end end jmod = matrix(nmobj, plist('shape',[nn,mm])); jmod.setName(smod.name); case 'collection' matobj = matrix.initObjectWithSize(1,numel(smod.objs)); for kk=1:numel(smod.objs) mobj = smod.objs{kk}.objs; [nn,mm] = size(mobj); nmobj = collection.initObjectWithSize(nn,mm); for ii=1:nn for jj=1:mm nmobj(ii,jj) = collection(fil.objs(ii,1),mobj(ii,jj)); nmobj(ii,jj).setName(mobj(ii,jj).name); end end matobj(kk) = matrix(nmobj); matobj(kk).setName(smod.objs{kk}.name); %smod.objs{kk} = matobj; end jmod = collection(matobj); end end %-------------------------------------------------------------------------- % Get Info Object %-------------------------------------------------------------------------- function ii = getInfo(varargin) if nargin == 1 && strcmpi(varargin{1}, 'None') sets = {}; pl = []; else sets = {'Default'}; pl = getDefaultPlist; end % Build info object ii = minfo(mfilename, 'matrix', 'ltpda', utils.const.categories.sigproc, '$Id: linfitsvd.m,v 1.38 2011/04/08 08:56:32 hewitson Exp $', sets, pl); ii.setArgsmin(1); ii.setOutmin(1); ii.setOutmax(1); ii.setModifier(false); end %-------------------------------------------------------------------------- % Get Default Plist %-------------------------------------------------------------------------- function plout = getDefaultPlist() persistent pl; if exist('pl', 'var')==0 || isempty(pl) pl = buildplist(); end plout = pl; end function pl = buildplist() % General plist for moltichannel fits pl = plist.MCH_FIT_PLIST; % p = param({'FitParams','A cell array with the names of the fit parameters'}, {}); % pl.append(p); p = param({'BoundedParams','A cell array with the names of the bounded fit parameters'}, {}); pl.append(p); p = param({'BoundVals','A cell array with the boundaries values for the bounded fit parameters'}, {}); pl.append(p); % p = param({'Model','System parametric model. A matrix of smodel objects or a ssm object'}, paramValue.EMPTY_DOUBLE); % pl.append(p); p = param({'dModel','Partial derivatives of the system parametric model. A matrix of smodel objects'}, paramValue.EMPTY_DOUBLE); pl.append(p); % p = param({'InNames','A cell array containing cell arrays of the input ports names for each experiment. Used only with ssm models.'}, {}); % pl.append(p); % % p = param({'OutNames','A cell array containing cell arrays of the output ports names for each experiment. Used only with ssm models.'}, {}); % pl.append(p); % p = param({'Input','Collection of input signals'},paramValue.EMPTY_DOUBLE); % pl.append(p); p = param({'WhiteningFilter','The multichannel whitening filter. A matrix object of filters'},paramValue.EMPTY_DOUBLE); pl.append(p); p = param({'Nloops', 'Number of desired iteration loops.'}, paramValue.DOUBLE_VALUE(1)); pl.append(p); p = param({'Ncut', 'Number of bins to be discharged in order to cut whitening filter transients'}, paramValue.DOUBLE_VALUE(100)); pl.append(p); % Number of points for zero padding p = param({'Npad', 'Number of points for zero padding.'}, paramValue.EMPTY_DOUBLE); pl.append(p); p = param({'KnownParams', 'Known Parameters. A pest object containing parameters values, names and errors'}, paramValue.EMPTY_DOUBLE); pl.append(p); p = param({'tol','Convergence threshold for fit parameters'}, paramValue.DOUBLE_VALUE(1)); pl.append(p); p = param({'fast',['Using fast option causes the whitening filter to be applied in frequency domain.'... 'The filter matrix is considered diagonal. The method skip time domain filtering saving some process time'... 'It works only when the imput model is a matrix of smodels']}, paramValue.FALSE_TRUE); pl.append(p); p = param({'SetAlias','Set to true in order to aassign internally the values to the model alias'}, paramValue.FALSE_TRUE); pl.append(p); p = param({'sThreshold',['Fix upper treshold for singular values.'... 'Singular values larger than the value will be ignored.'... 'This correspon to consider only parameters combinations with error lower then the value']},... paramValue.DOUBLE_VALUE(1)); pl.append(p); p = plist({'diffStep','Numerical differentiation step for ssm models'}, paramValue.EMPTY_DOUBLE); pl.append(p); end