-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathlogL_CE_w_grad_2.m
463 lines (409 loc) · 16.9 KB
/
logL_CE_w_grad_2.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
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
% logL_CE_w_grad_2 is the main likelihood function of the MEMToolbox and
% supports classical mixed effect models aswell as sigma-point based
% moment-approximation schemes to the solution of population balance
% equations. The likelihood function allows for the parallel estimation of
% parameters from multiple different experimental setups/conditions
%
% USAGE:
% ======
% [logL,dlogLdxi,ddlogLdxidxi] = logL_CE_w_grad_2(xi,Data,Model,options)
% P = logL_CE_w_grad_2(xi,Data,Model,options,1)
%
% INPUTS:
% =======
% xi ... parameter of for which the logLikelihood value is to be computed
% the ordering must correspond to what is given in Model.sym.xi
% Data ... data for which the logLikelihood is computed this should be a cell
% array (one cell per experiment) with some of the following fields
% .SCTL ... Single-Cell-Time-Lapse; must have the following subfields
% .time ... vector of time-points at which measurements were taken
% .Y ... measurements, 3rd order tensor [time,observable,cells]
% .T ... (optional) events, 3rd order tensor [time,event,cells]
% .SCTLstat ... Single-Cell-Time-Lapse-Statistics; must have the following subfields
% .time ... vector of time-points at which measurements were taken
% .mz ... full state vector of all observables at all times
% .Sigma_mz ... standard deviation of mz
% .Cz ... full state covariance, this includes state covariance and
% cross-correlation
% .Sigma_Cz ... standard deviation of Cz
% .SCSH ... Single-Cell-Snap-Shot; must have the following subfields
% .time ... vector of time-points at which measurements were taken
% .m ... mean of measurements, matrix [time,observable]
% .Sigma_m ... standard deviation of mean, matrix [time,observable]
% .C ... covariance of measurements, 3rd order tensor [time,observable,observable]
% .Sigma_C ... standard deviation of covariance, 3rd order tensor [time,observable,observable]
% .PA ... Population-Average; must have the following subfields
% .time ... vector of time-points at which measurements were taken
% .m ... mean of measurements, matrix [time,observable]
% .Sigma_m ... standard deviation of mean, matrix [time,observable]
% furthermore it must have the following fields
% .condition ... vector of experiemental conditions which is passed to
% the model file
% .name ... string for the name of the experiment
% .measurands ... cell array of labels for every experiment
% Model ... model definition ideally generated via make_model and
% complete_model. must have the following fields
% .type_D ... string specifying the parametrisation of the covariance
% matrix for the random effects. either
% 'diag-matrix-logarithm' for diagonal matrix with log. paramet. or
% 'matrix-logarithm' for full matrix with log. parametrisation
% .integration ... flag indicating whether integration for classical
% mixed effect models via laplace approximation should be applied.
% only applicable for SCTL data.
% .penalty ... flag indicating whether additional penalty terms for
% synchronisation of parameters across experiments should be applied.
% only applicable for SCTL data.
% .prior ... cell array containing prior information for the optimization
% parameters. the length of the cell array must not exceed the
% length of the optimization parameter. the prior is applied when
% there exist fields .mu and .std. This imposes a normal prior with
% mean .mu and variance (.std)^2.
% .exp ... cell array containing specific information about individual
% experiments. must have the following fields
% .PA_post_processing ... (optional, only for PA data) function
% handle for post-processing of PA for e.g. normalization
% .sigma_noise ... function of the mixed effect parameter yielding
% the standard deviation in measurements
% .sigma_time ... function of the mixed effect parameter yielding
% the standard deviation in event time-points
% .noise_model ... string for the noise model for measurements.
% either 'normal' or 'lognormal'
% .parameter_model ... distribution assumption for the random effect.
% either 'normal' or 'lognormal'
% .model ... function handle for simulation of the model with arguments
% t ... time
% phi ... mixed effect parameter
% kappa ... experimental condtion
% option_model ... struct which can carry additional options such
% as the number of required sensitivities
% the function handle should return the following object
% sol ... solution struct with the following fields
% (depending on the value of option_model.sensi)
% for option_model.sensi >= 0
% .status ... >= 0 for successful simulation
% .y ... model output for observable, matrix [time,measurement]
% .root ... (optional) model output for event, matrix [time,event]
% .rootval ... (optional) model output for rootfunction, matrix [time,event]
% for option_model.sensi >= 1
% .sy ... sensitivity for observable, matrix [time,measurement]
% .sroot ... (optional) sensitivity for event, matrix [time,event]
% .srootval ... (optional) sensitivity for rootfunction, matrix [time,event]
% for option_model.sensi >= 2 (optional)
% .s2y ... second order sensitivity for observable, matrix [time,measurement]
% .s2root ... (optional) second order sensitivity for event, matrix [time,event]
% .s2rootval ... (optional) second order sensitivity for rootfunction, matrix [time,event]
% .plot ... function handle for plotting of simulation resutls with arguments
% Data ... data
% Sim ... simulation
% s ... experiment index
% and the following fields which contain figure handles whith mixed
% effect parameter as argument of the respective variable
% .dsigma_noisedphi
% .ddsigma_noisedphidphi
% .dddsigma_noisedphidphidphi
% .ddddsigma_noisedphidphidphidphi (only for .integration==1)
% .dsigma_timedphi
% .ddsigma_timedphidphi
% .dddsigma_timedphidphidphi
% .ddddsigma_timedphidphidphidphi (only for .integration==1)
% and the following fields which contain figure handles whith optimization
% parameter as argument of the respective variable
% .beta ... common effect parameter
% .delta ... parametrization of covariance matrix for random effect
% .dbetadxi
% .ddeltadxi
% .ddbetadxidxdi
% .dddeltadxidxdi
% and the following fields which contain figure handles whith random
% and common effect parameter as argument
% .phi ... mixed effect parameter @(b,beta)
% .dphidbeta
% .dphidb
% .ddphidbetadbeta
% .ddphidbdbeta
% .ddphidbdb
% options ... option struct with the following options
% .tau_update ... minimum number of second which must pass before the
% plots are updated
% .plot ... flag whether the function should plot either
% 0 ... no plots
% 1 ... all plots (default)
% .ms_iter ... number of function call with decreasing likelihood
% value between multistarts for the inner optimisation problem for SCTL
% data, for optimisation this is typically the number of iterations
% between multistarts
% 0 ... multistart every iteration
% X ... multistart every X iterations (default = 10)
% Inf ... Only one multistart in the beginning
% extract_flag ... flag indicating whether the values of random effect
% parameters are to be extracted (only for SCTL data)
% 0 ... no extraction (default)
% 1 ... extraction
%
%
% Outputs:
% ========
% extract_flag == 0
% logL ... logLikelihood value
% dlogLdxi ... gradient of logLikelihood
% ddlogLdxidxdi ... hessian of logLikelihood
% extract_flag == 1
% (SCTL)
% P ... cell array with field
% .SCTL if the corresponding Data cell had a .SCTL field. this has
% field has subfields
% .bhat random effect parameter
% .dbdxi gradient of random effect parameter wrt optimization
% parameter
% .ddbdxidxi hessian of random effect parameter wrt optimization
% parameter
% (otherwise)
% SP.B ... location of sigma-points
%
% 2015/04/14 Fabian Froehlich
function varargout = logL_CE_w_grad_2(varargin)
%% Load old values
persistent tau
persistent P_old
persistent logL_old
persistent xi_old
persistent fp
persistent fl
persistent n_store
if isempty(tau)
tau = clock;
end
%% Initialization
xi = varargin{1};
Data = varargin{2};
Model = varargin{3};
% Options
options.tau_update = 0;
options.plot = 1;
options.ms_iter = 10;
if nargin >= 4
if(isstruct(varargin{4}))
options = setdefault(varargin{4},options);
end
end
if nargin >= 5
extract_flag = varargin{5};
else
extract_flag = false;
end
nderiv = max(nargout-1,0);
% initialise storage
if(isempty(logL_old))
logL_old = -Inf;
xi_old = zeros(size(xi));
n_store = 0;
for s = 1:length(Data)
if isfield(Data{s},'SCTL')
P_old{s}.SCTL.bhat = zeros(length(Model.exp{s}.ind_b),size(Data{s}.SCTL.Y,3));
P_old{s}.SCTL.dbdxi = zeros(length(Model.exp{s}.ind_b),length(xi),size(Data{s}.SCTL.Y,3));
else
P_old{s} = [];
end
end
end
% Plot options
if (etime(clock,tau) > options.tau_update) && (options.plot == 1)
options.plot = 30;
tau = clock;
else
options.plot = 0;
end
%% Evaluation of likelihood function
% Initialization
logL = 0;
if nderiv >= 2
dlogLdxi = zeros(length(xi),1);
if nderiv >= 3
ddlogLdxidxi = zeros(length(xi));
end
end
% definition of possible datatypes
data_type = {'SCTL','SCSH','SCTLstat','PA'};
ms_iter = options.ms_iter;
% Loop: Experiments/Experimental Conditions
for s = 1:length(Data)
%% Assignment of global variables
type_D = Model.type_D;
n_b = length(Model.exp{s}.ind_b);
%% Construct fixed effects and covariance matrix
beta = Model.exp{s}.beta(xi);
delta = Model.exp{s}.delta(xi);
n_beta = length(Model.exp{s}.beta(xi));
[D,~,~,~,~,~] = xi2D(delta,type_D);
% debugging:
% [g,g_fd_f,g_fd_b,g_fd_c] = testGradient(delta,@(x) xi2D(x,type_D),1e-4,1,3)
% [g,g_fd_f,g_fd_b,g_fd_c] = testGradient(delta,@(x) xi2D(x,type_D),1e-4,3,5)
% [g,g_fd_f,g_fd_b,g_fd_c] = testGradient(delta,@(x) xi2D(x,type_D),1e-4,2,4)
% [g,g_fd_f,g_fd_b,g_fd_c] = testGradient(delta,@(x) xi2D(x,type_D),1e-4,4,6)
%% Construction of time vector
t_s = [];
for dtype = 1:length(data_type)
if isfield(Data{s},data_type{dtype})
t_s = union(eval(['Data{s}.' data_type{dtype} '.time']),t_s);
end
end
%% Single cell time-lapse data - Individuals
if isfield(Data{s},'SCTL')
switch(nderiv)
case 0
[P,logL_sc] = logL_SCTL(xi, Model, Data, s, options, P);
case 1
[P,logL_sc,dlogL_scdxi] = logL_SCTL(xi, Model, Data, s, options, P);
case 2
[P,logL_sc,dlogL_scdxi,ddlogL_scdxi2] = logL_SCTL(xi, Model, Data, s, options ,P);
end
logL = logL + sum(bsxfun(@times,Model.SCTLscale,logL_sc),2);
if nderiv <= 2
dlogLdxi = dlogLdxi + sum(bsxfun(@times,Model.SCTLscale,dlogL_scdxi),2);
if nderiv <= 3
ddlogLdxidxi = ddlogLdxidxi + sum(bsxfun(@times,Model.SCTLscale,ddlogL_scdxi2),2);
end
end
end
%% Single cell time-lapse data - Statistics
if isfield(Data{s},'SCTLstat')
% Simulation using sigma points
op_SP.nderiv = nderiv;
op_SP.req = [0,0,0,1,1,1,0];
op_SP.type_D = Model.type_D;
if(extract_flag)
SP = testSigmaPointApp(@(phi) simulateForSP(Model.exp{s}.model,Data{s}.SCTLstat.time,phi,Data{s}.condition),xi,Model.exp{s},op_SP);
else
SP = getSigmaPointApp(@(phi) simulateForSP(Model.exp{s}.model,Data{s}.SCTLstat.time,phi,Data{s}.condition),xi,Model.exp{s},op_SP);
end
% Evaluation of likelihood, likelihood gradient and hessian
% Mean
logL_mz = - 0.5*sum(nansum(((Data{s}.SCTLstat.mz - SP.mz)./Data{s}.SCTLstat.Sigma_mz).^2,1),2);
if nderiv >= 2
dlogL_mzdxi = permute(nansum(bsxfun(@times,(Data{s}.SCTLstat.mz - SP.mz)./Data{s}.SCTLstat.Sigma_mz.^2,SP.dmzdxi),1),[2,1]);
if nderiv >= 3
wdmz_SP = bsxfun(@times,1./Data{s}.SCTLstat.Sigma_mz,SP.dmzdxi);
% wdmz_SP = reshape(wdmz_SP,[numel(SP.mz),size(SP.dmdxizdxi,3)]);
ddlogL_mzdxi2 = -wdmz_SP'*wdmz_SP;
end
end
% Covariance
logL_Cz = - 0.5*sum(nansum(nansum(((Data{s}.SCTLstat.Cz - SP.Cz)./Data{s}.SCTLstat.Sigma_Cz).^2,1),2),3);
if nderiv >= 2
dlogL_Czdxi = squeeze(nansum(nansum(bsxfun(@times,(Data{s}.SCTLstat.Cz - SP.Cz)./Data{s}.SCTLstat.Sigma_Cz.^2,SP.dCzdxi),1),2));
if nderiv >= 3
wdCzdxi = bsxfun(@times,1./Data{s}.SCTLstat.Sigma_Cz,SP.dCzdxi);
wdCzdxi = reshape(wdCzdxi,[numel(SP.Cz),size(SP.dCzdxi,3)]);
ddlogL_Czdxi2 = -wdCzdxi'*wdCzdxi;
end
end
% Summation
logL = logL + logL_mz + logL_Cz;
if nderiv >=2
dlogLdxi = dlogLdxi + dlogL_mzdxi + dlogL_Czdxi;
if nderiv >=3
ddlogLdxidxi = ddlogLdxidxi + ddlogL_mzdxi2 + ddlogL_Czdxi2;
end
end
% Visulization
if options.plot
Sim_SCTLstat.mz = SP.mz;
Sim_SCTLstat.Cz = SP.Cz;
Model.exp{s}.plot(Data{s},Sim_SCTLstat,s);
end
P{s}.SCTLstat.SP = SP;
end
%% Single cell snapshot data
if isfield(Data{s},'SCSH')
switch(nderiv)
case 0
[SP,logL_m,logL_C] = logL_PA(xi, Model, Data, s, options);
case 1
[SP,logL_m,logL_C,dlogL_mdxi,dlogL_Cdxi] = logL_PA(xi, Model, Data, s, options);
case 2
[SP,logL_m,logL_C,dlogL_mdxi,dlogL_Cdxi,ddlogL_mdxi2,ddlogL_Cdxi2] = logL_PA(xi, Model, Data, s, options);
end
% Summation
logL = logL + logL_m + logL_C;
if nderiv >= 2
dlogLdxi = dlogLdxi + dlogL_mdxi + dlogL_Cdxi;
if nderiv >= 3
ddlogLdxidxi = ddlogLdxidxi + ddlogL_mdxi2 + ddlogL_Cdxi2;
end
end
P{s}.SCSH.SP = SP;
end
%% Population average data
if isfield(Data{s},'PA')
switch(nderiv)
case 0
[SP,logL_m] = logL_PA(xi, Model, Data, s, options);
case 1
[SP,logL_m,dlogL_mdxi] = logL_PA(xi, Model, Data, s, options);
case 2
[SP,logL_m,dlogL_mdxi,ddlogL_mdxi2] = logL_PA(xi, Model, Data, s, options);
end
% Summation
logL = logL + logL_m;
if nderiv >= 1
dlogLdxi = dlogLdxi + dlogL_mdxi;
if nderiv >= 2
ddlogLdxidxi = ddlogLdxidxi + ddlogL_mdxi2;
end
end
P{s}.PA.SP = SP;
end
end
% updated stored value
if(logL > logL_old)
logL_old = logL;
P_old = P;
xi_old = xi;
n_store = n_store + 1;
end
%% Output
if extract_flag
varargout{1} = P;
return
end
%% Prior
if isfield(Model,'prior')
if(iscell(Model.prior))
if(length(Model.prior) <= length(xi))
for ixi = 1:length(Model.prior)
if(isfield(Model.prior{ixi},'mu') && isfield(Model.prior{ixi},'std'))
if nderiv >= 1
% One output
logL = logL - 0.5*((xi(ixi)-Model.prior{ixi}.mu)/Model.prior{ixi}.std)^2;
if nderiv >= 2
% Two outputs
dlogLdxi(ixi) = dlogLdxi(ixi) - ((xi(ixi)-Model.prior{ixi}.mu)/Model.prior{ixi}.std^2);
if nderiv >= 3
% Two outputs
ddlogLdxidxi(ixi,ixi) = ddlogLdxidxi(ixi,ixi) - 1/Model.prior{ixi}.std^2;
end
end
end
end
end
else
error('length of Model.prior must agree with length of optimization parameter')
end
else
error('Model.prior must be a cell array')
end
end
%%
if nderiv >= 1
% One output
varargout{1} = logL;
if nderiv >= 2
% Two outputs
varargout{2} = dlogLdxi;
if nderiv >= 3
% Two outputs
varargout{3} = ddlogLdxidxi;
end
end
end
end