-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathmaskit.m
345 lines (305 loc) · 13.4 KB
/
maskit.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
function varargout=maskit(v,p,scl,w,opt)
% [v,I,w,vw,scl,cr]=MASKIT(v,p,scl,w,opt)
%
% Arbitrary masking of a gridded spatial field.
%
% INPUT:
%
% v A vector that is the unwrapping of a matrix
% p A single parameter structure with AT LEAST this:
% NyNx number of samples in the y and x directions
% mask an index matrix with a mask, OR:
% a region name that will be scaled to fit
% scl A scale between 0 and 1 with the fractional area that the
% bounding box of the domain occupies; due to strict
% inequality at maximum always leaves one pixel rim; and
% not being used if an actual mask is input instead of a name
% w A second vector that is the unwrapping of a second matrix
% opt (Only for demo2) A field structure for optimization flags:
% algo the optimization algorithm
% ifinv ordered inversion flags for [s2 nu rho], e.g. [1 0 1]
%
% OUTPUT:
%
% v The first input with the samples inside I retained, rest set to NaN
% I The mask, unrolled into a vector, note, this is an "anti-mask"
% w The second input but everything outside the mask set to NaN
% vw The merged field, second field w sucked into the masked first field v
% scl A scale [0 to 1] expressing the non-missing data fraction
% cr The colum,row index coordinates of the masking curve
%
% EXAMPLE:
%
% [~,I]=maskit('demo1'); % To get a quick mask to look at
% maskit('demo1') % A geographical region masking a single field
% maskit('demo1','england') % 'amazon', 'orinoco', for geographical variability
% maskit('demo2') % A geographical region merging two fields, with estimation
% maskit('demo2','england') % 'amazon', 'orinoco', for geographical variability
% maskit('demo3',[],rand) % Illustrating the nomenclature
%
% Last modified by owalbert-princeton.edu, 09/03/2024
% Last modified by fjsimons-at-alum.mit.edu, 06/11/2024
% The default is the demo, for once
defval('v','demo1')
defstruct('opt',{'algo','ifinv'},{'unc',[1 1 1]});
if ~isstr(v)
% Get the mask by name
if isstr(p.mask)
XY=eval(p.mask);
% Scale indicates survivorship (in symmetry with MUCKIT)
% Default scale so bounding-box-area-proportionality is scl
defval('scl',0.8)
% Scale the mask, remember for the curve the Y goes up, for the image, down
cr(:,1)= scale(XY(:,1),[1 p.NyNx(2)]+[1 -1]*(1-sqrt(scl))/2*(p.NyNx(2)-1));
cr(:,2)=p.NyNx(1)-scale(XY(:,2),[1 p.NyNx(1)]+[1 -1]*(1-sqrt(scl))/2*(p.NyNx(1)-1))+1;
% The underlying image grid
[X,Y]=meshgrid(1:p.NyNx(2),1:p.NyNx(1));
% Determine the mask
I=inpolygon(X,Y,cr(:,1),cr(:,2));
else
% You already have it
cr=NaN;
I=p.mask;
end
% Apply the mask to v, w and merge into vw as desired
if nargin<4
[w,vw]=deal(NaN);
elseif nargout>3
% Apply the mask to the second field
w(~I(:))=NaN;
vw=NaN;
if nargout>4
% Construct a new field with the SECOND field inside the mask and the FIRST outside
vw=nan(size(v));
% avoid complaints about indices needing to
% be logicals in the case mask uses zeros
vw(~~I(:))=w(~~I(:));
vw(~I(:)) =v(~I(:));
end
end
% Apply the mask to the first field
v(~I(:))=NaN;
% Note that you must not feed fields with NaN into MLEOSL - the taper I will
% take of the selectivity
elseif strcmp(v,'demo1')
% Capture the second input
defval('p','france')
defp=p; clear p
% Now proceed with a fresh copy
p.mask=defp;
p.quart=0; p.blurs=Inf; p.kiso=NaN; clc;
[Hx,th,p]=simulosl([],p,1);
% Capture the third input, default is none, which defaults inside
defval('scl',[])
[Hm,I,~,~,scl,cr]=maskit(Hx,p,scl);
clf
ah(1)=subplot(121); plotit(Hx,p,cr,th)
ah(2)=subplot(122); plotit(Hm,p,cr,[])
% Create right type of output even though there is non
v=Hm; [w,vw]=deal(NaN);
elseif strcmp(v,'demo2')
% Capture the second input
defval('p','france')
defp=p; clear p
% Capture the fifth input
defstruct('opt',{'algo','ifinv'},{[],[]});
% Change the below if you like
opt.ifinv=[1 1 1];
% Now proceed with a fresh copy
p.mask=defp;
% Simulate using circulant embedding, no taper
p.quart=0; p.blurs=Inf; p.kiso=NaN; clc;
% Something manageable without overdoing it
p.NyNx=[188 233]+randi(20,[1 2]);
% Something larger without overdoing it, check weirdness
% If you want to generate the same hash you need to keep the same dimensions
% p.NyNx=[196 243];
% p.NyNx=[193 247];
p.NyNx=[193 236];
% Do all the tests or not
xver=0;
% Decide on the length of the pause between displays, if any,
if xver==1
pz=2;
else
pz=0.1;
end
% Define the parameters for the two fields
th1=[1.15 1.15 23000];
th2=[2.90 2.75 11300];
% Needing to be very explicit in order for parallel computing to work with in the loop
p.dydx=[10000 10000];
p.taper=0; p.nugget=0;
% Number of processors, must agree with your machine
NumWorkers=8;
% Number of identical experiments to run experiments
N=1*NumWorkers;
% Save some output for later? Make new filename hash from all relevant input
fname=hash([struct2array(orderfields(p)) struct2array(orderfields(opt)) th1 th2 N],'SHA-1');
% You need to have an environmental variable file structure set up
fnams=fullfile(getenv('IFILES'),'HASHES',sprintf('%s_%s.mat','MASKIT',fname));
% Run the experiment anew
if ~exist(fnams,'file') % || 1==1
% Initialize the pool of workers
if isempty(gcp('nocreate')); pnw=parpool(NumWorkers); end
% Should we preallocate?
% Run the experiment!
parfor index=1:N
clc; disp(sprintf('\n Simulating all fields \n'))
% Simulate first field
Hx=simulosl(th1,p,1);
% Simulate second field
Gx=simulosl(th2,p,1);
% How much should the masked area occupy?
% Don't SET this here or PARFOR won't work
scl=[];
% Now do the masking and the merging
[Hm,I,Gm,HG,scl,cr]=maskit(Hx,p,scl,Gx);
% So HG is the mixed field
v=Hm; w=Gm; vw=HG;
% As appropriate you'll force the use of BLUROSY in MATERNOSP in LOGLIOSL in MLEOSL
% This is confusing PARFOR but works with FOR
% p.blurs=-1;
% Make a close initial guess?
thini1(index,:)=th1+(-1).^randi(2,[1 3]).*th1/1000;
thini2(index,:)=th2+(-1).^randi(2,[1 3]).*th2/1000;
% Don't perturb the thini for the one you don't want to move; could
% have also have stuck in th1 and th2 into the aguess slot of
% MLEOSL, whence they would be appropriately (or not) perturbed
if opt.ifinv==[1 0 1]
% This didn't work
%thini1(index,2)=th1(2);
%thini2(index,2)=th2(2);
% But this did
thini1(index,:)=th1+(-1).^randi(2,[1 3]).*[th1(1) 0 th1(3)]/1000;
thini2(index,:)=th2+(-1).^randi(2,[1 3]).*[th2(1) 0 th2(3)]/1000;
end
% Recover the parameters of the full original fields without any masking
% The relabeling is to make PARFOR work, with FOR they could all just be p
p1=p; p1.taper=0;
pause(pz); clc; disp(sprintf('\n Estimating first whole field \n'))
[thhat4(index,:),~,~,scl4(index,:)]=mleosl(Hx,thini1(index,:),p1,opt.algo,[],[],opt.ifinv,xver);
pause(pz); clc ; disp(sprintf('\n Estimating second whole field \n'))
[thhat5(index,:),~,~,scl5(index,:)]=mleosl(Gx,thini2(index,:),p1,opt.algo,[],[],opt.ifinv,xver);
% Now recover the parameters of the mixed field but only in the region ~I
% Perform the optimization on the complement which should look like the FIRST field
% The relabeling is to make PARFOR work, with FOR they could all just be p
p2=p; p2.taper=~I;
pause(pz); clc; disp(sprintf('\n Estimating first partial field \n'))
[thhat1(index,:),~,~,scl1(index,:)]=mleosl(HG,thini1(index,:),p2,opt.algo,[],[],opt.ifinv,xver);
% Now recover the parameters of the insert which should look like the SECOND field
% The relabeling is to make PARFOR work, with FOR they could all just be p
p3=p; p3.taper=I;
pause(pz); clc; disp(sprintf('\n Estimating second partial field \n'))
[thhat2(index,:),~,~,scl2(index,:)]=mleosl(HG,thini2(index,:),p3,opt.algo,[],[],opt.ifinv,xver);
% Now recover the parameters of the merged field without knowing of the partition
% The relabeling is to make PARFOR work, with FOR they could all just be p
p4=p; p4.taper=0;
pause(pz); clc; disp(sprintf('\n Estimating whole mixed field \n'))
% Force the inversion of all parameters in this particular case
[thhat3(index,:),~,~,scl3(index,:)]=mleosl(HG,[], p4,opt.algo,[],[],[1 1 1],xver);
% Pause so you can watch live (big pause if xver==0)
pause(pz)
% This is unnecessary for PARFOR but works with FOR
% Reset to the original values
%p.blurs=Inf; p.taper=0;
end
% Save all the output that you'll need later, may add more later
save(fnams,'th1','th2','p','thhat1','thhat2','scl1','scl2',...
'thhat3','scl3','thhat4','scl4','thhat5','scl5','thini1','thini2','N')
else
% Load all the saved output
load(fnams)
end
% And now look at the statistics of the recovery on screen
str0='%8.2f %5.2f %6.0f';
str1=sprintf('%s %s\n',str0,str0);
str2=sprintf('%s\n',str0);
disp(sprintf('\nFirst partial | Second partial\n'))
disp(sprintf(str1,[thhat1.*scl1 thhat2.*scl2]'))
disp(sprintf('\nFirst whole | Second whole\n'))
disp(sprintf(str1,[thhat4.*scl4 thhat5.*scl5]'))
disp(sprintf('\nMixed whole\n'))
disp(sprintf(str2,[thhat3.*scl3]'))
% Save the figures bjootifooly
% Remake one sampled field just to make the visual
[Hx,~,~,~,~,Sb1]=simulosl(th1,p,1);
[Gx,~,~,~,~,Sb2]=simulosl(th2,p,1);
[~,~,~,HG,scl,cr]=maskit(Hx,p,[],Gx);
% This was confusing PARFOR so we took it out of the loop
if xver==1
% Very explicit comparison like BLUROSY
p.blurs=-1; Sbar1=blurosy(th1,p,1);
p.blurs=-1; Sbar2=blurosy(th2,p,1);
% Are any of them real bad?
if max(Sb1./Sbar1)>20; warning('You will want to review this code') ; end
if max(Sb2./Sbar2)>20; warning('You will want to review this code') ; end
% Reset to the original value
p.blurs=Inf;
end
% Make a visual for good measure
figure(3)
clf
ah(1)=subplot(221); plotit(Hx,p,cr,th1)
ah(3)=subplot(223); plotit(Gx,p,cr,th2)
ah(4)=subplot(224); plotit(HG,p,cr,[])
movev(ah(4),0.25)
figdisp(sprintf('%s_1',pref(sprintf('%s_%s.mat','MASKIT',fname))),[],[],2)
% Then plot the estimates using MLEPLOS, remember second field is the inserted one
% So this is the ANTI region
figure(1)
mleplos(thhat1.*scl1,th1,[],[],[],[],[],p,sprintf('anti %s',p.mask),[],opt.ifinv)
figure(1)
figdisp(sprintf('%s_2a',pref(sprintf('%s_%s.mat','MASKIT',fname))),[],[],2)
clf
figure(2)
figdisp(sprintf('%s_2b',pref(sprintf('%s_%s.mat','MASKIT',fname))),[],[],2)
clf
% And this is the SELECTED region
figure(1)
mleplos(thhat2.*scl2,th2,[],[],[],[],[],p,p.mask,[],opt.ifinv)
pause(1)
figure(1)
figdisp(sprintf('%s_3a',pref(sprintf('%s_%s.mat','MASKIT',fname))),[],[],2)
clf
figure(2)
figdisp(sprintf('%s_3b',pref(sprintf('%s_%s.mat','MASKIT',fname))),[],[],2)
% With PARFOR none of the once-used are available out of the loop but in
% this demo2 you don't want any output anyway, so put in empties
[v,I,w,vw,scl,cr]=deal(NaN);
elseif strcmp(v,'demo3')
th1=[1.15 1.15 23000];
th2=[2.90 2.75 11300];
p.mask='france';
% Something manageable without overdoing it
p.NyNx=[188 233]+randi(20,[1 2]);
% Capture the third input, default is none, which defaults inside
defval('scl',[])
% Remake one sampled field just to make the visual
[Hx,~,p,~,~,Sb1]=simulosl(th1,p);
[Gx,~,~,~,~,Sb2]=simulosl(th2,p);
[v,I,w,vw,scl,cr]=maskit(Hx,p,scl,Gx);
% Make a visual for good measure
clf
ah(1)=subplot(221); plotit(v,p,cr,sprintf('%i %% v',round(100*scl)))
ah(3)=subplot(223); plotit(w,p,cr,sprintf('%i %% w',round(100*scl)))
ah(4)=subplot(224); plotit(vw,p,cr,'vw')
movev(ah(4),0.25)
end
% Variable output
varns={v,I,w,vw,scl,cr};
varargout=varns(1:nargout);
function plotit(v,p,cr,th)
% Should use th(1) instead of halverange, shouldn't I...
imagefnan([1 1],p.NyNx([2 1]),v2s(v,p),[],halverange(v,80,NaN)); axis ij image
hold on; twoplot(cr,'Color','k'); hold off; longticks; grid on
xticks([1 round(p.NyNx(2)/2) p.NyNx(2)]); yticks([1 round(p.NyNx(1)/2) p.NyNx(1)])
if ~isempty(th)
if isstr(th)
t=title(th);
else
t=title(sprintf('%i x %i | %s = [%g %g %gx]',p.NyNx(1),p.NyNx(2),'\theta',...
th./[1 1 sqrt(prod(p.dydx))]));
end
movev(t,-p.NyNx(1)/20)
end