Loading [MathJax]/extensions/tex2jax.js
ParaMonte MATLAB 3.0.0
Parallel Monte Carlo and Machine Learning Library
See the latest version documentation.
All Data Structures Files Functions Variables Pages
run.m
Go to the documentation of this file.
1%> \brief
2%> Perform the basic runtime checks for the sampler and return nothing.
3%>
4%> \param[inout] self : The input/output parent object of class [pm.sampling.Sampler](@ref Sampler)
5%> which is **implicitly** passed to this dynamic method (not by the user).<br>
6%> \param[in] getLogFunc : The input MATLAB function handle or anonymous (lambda) function
7%> containing the implementation of the objective function to be sampled.<br>
8%> This user-specified function must have the following interface,<br>
9%> \code{.m}
10%> function logFunc = getLogFunc(state)
11%> ...
12%> end<br>
13%> \endcode
14%> where,<br>
15%> <ol>
16%> <li> the input argument ``state`` is a vector of type MATLAB ``double``
17%> of size ``ndim`` representing a single point from within the ``ndim``
18%> dimensional domain of the mathematical object function to be explored.<br>
19%> <li> the output argument `logFunc` is a scalar of the same type as the
20%> input ``state`` containing the natural logarithm of the objective
21%> function at the specified input ``state`` within its domain.<br>
22%> </ol>
23%> \param[in] ndim : The input scalar positive-valued whole-number representing the number of dimensions
24%> of the domain of the user-specified objective function in the input ``getLogFunc()``.<br>
25%>
26%> \interface{run}
27%> \code{.m}
28%>
29%> sampler = pm.sampling.Sampler();
30%> sampleList = sampler.run(getLogFunc, ndim);
31%>
32%> \endcode
33%>
34%> \final{run}
35%>
36%> \author
37%> \JoshuaOsborne, May 21 2024, 12:38 AM, University of Texas at Arlington<br>
38%> \FatemehBagheri, May 20 2024, 1:25 PM, NASA Goddard Space Flight Center (GSFC), Washington, D.C.<br>
39%> \AmirShahmoradi, May 16 2016, 9:03 AM, Oden Institute for Computational Engineering and Sciences (ICES), UT Austin<br>
40function run(self, getLogFunc, ndim)
41
42 %%%%
43 %%%% Sanitize ``sampler.silent``.
44 %%%%
45
46 if ~pm.introspection.verified(self.silent, "logical", 1)
47 help("pm.sampling.Sampler.silent");
48 disp("self.silent =");
49 disp(self.silent);
50 error ( newline ...
51 + "The sampler attribute ``silent`` must be a scalar of type ``logical``." + newline ...
52 + "For more information, see the documentation displayed above." + newline ...
53 + newline ...
54 );
55 end
56
57 %%%%
58 %%%% Sanitize parallelism method to set reporting permission.
59 %%%%
60
61 % global mpiname;
62 % ismember('mpiname', who('global'));
63 % if pm.array.len(self.mpiname) == 0 && ~isempty(mpiname) && pm.introspection.verified(mpiname, "string", 1) % MPI enabled by a global definition of ``mpiname``.
64 % self.mpiname = mpiname;
65 % end
66
67 if ~pm.introspection.verified(self.mpiname, "string", 1)
68
69 %%%%
70 %%%% Sanitize ``mpiname``.
71 %%%%
72
73 help("pm.sampling.Sampler.mpiname");
74 disp("mpiname =");
75 disp(self.mpiname);
76 error ( newline ...
77 + "The sampler attribute ``mpiname`` must be a scalar of type ``char`` or ``string``." + newline ...
78 + "For more information, see the documentation displayed above." + newline ...
79 + newline ...
80 );
81
82 elseif pm.array.len(self.mpiname) == 0
83
84 %%%%
85 %%%% Detect potential MPI launcher.
86 %%%%
87
88 [mpiname, nproc, ~] = pm.lib.mpi.runtime.detect();
89 if pm.array.len(mpiname) > 0 && nproc > 1
90 self.mpiname = mpiname;
91 end
92
93 end
94
95 %%%%
96 %%%% First detect potential use of MPI, then check for thread parallelism.
97 %%%%
98
99 if 0 < pm.array.len(self.mpiname) % MPI enabled.
100
101 self.silent = true; % Otherwise, we keep the default value of self.silent.
102 self.partype = string(pm.lib.mpi.name(self.mpiname));
103
104 elseif ~isempty(self.spec.parallelismNumThread)
105
106 %%%%
107 %%%% Sanitize ``self.spec.parallelismNumThread``.
108 %%%%
109
110 % The following separate conditions are crucial to remain separate.
111 failed = ~pm.introspection.verified(self.spec.parallelismNumThread, "integer", 1);
112 if ~failed
113 failed = self.spec.parallelismNumThread < 0;
114 end
115 if ~failed
116 failed = ~pm.matlab.has.parallel;
117 end
118 if ~failed
119 self.partype = "openmp";
120 if self.spec.parallelismNumThread == 0
121 %%%% Negative threads count indicates to the ParaMonte sampler that the user set `parallelismNumThread` to zero.
122 self.spec.parallelismNumThread = -maxNumCompThreads();
123 end
124 else
125 help("pm.sampling.Sampler.run");
126 error ( newline ...
127 + "The simulation specification ``parallelismNumThread`` must be" + newline ...
128 + "a scalar non-negative integer or whole-number representing the number" + newline ...
129 + "of processor threads to use for a thread-parallel simulation." + newline ...
130 + "Specifying ``0`` as the value of ``parallelismNumThread`` will" + newline ...
131 + "lead to using all available CPU processes for thread-parallelism." + newline ...
132 + "Ideally, a number less than the maximum available number" + newline ...
133 + "of threads should be specified for ``parallelismNumThread`` as using" + newline ...
134 + "all available threads exclusively for a simulation will" + newline ...
135 + "slow down all open system applications including MATLAB." + newline ...
136 + "Specifying this option requires the MATLAB parallel toolbox." + newline ...
137 + "If missing or specified as empty `[]`, the simulation will run in serial mode." + newline ...
138 + "You have specified:" + newline ...
139 + newline ...
140 + pm.io.tab() + "self.spec.parallelismNumThread = " + string(self.parallelismNumThread) + newline ...
141 + newline ...
142 + "Does your MATLAB have Parallel Computing Toolbox?" + newline ...
143 + newline ...
144 + pm.io.tab() + "pm.matlab.has.parallel() = " + string(pm.matlab.has.parallel()) + newline ...
145 + newline ...
146 + "For more information, see the documentation displayed above." + newline ...
147 + newline ...
148 );
149 end
150
151 end
152
153 %%%%
154 %%%% Sanitize ``getLogFunc``.
155 %%%%
156
157 if ~isa(getLogFunc, "function_handle")
158 help("pm.sampling.Sampler.run");
159 error ( newline ...
160 + "The input argument getLogFunc must be a callable function." + newline ...
161 + "It represents the user's objective function to be sampled," + newline ...
162 + "which must take a single input argument of type numpy" + newline ...
163 + "float64 array of length ndim and must return the" + newline ...
164 + "natural logarithm of the objective function." + newline ...
165 + newline ...
166 + "For more information, see the documentation displayed above." + newline ...
167 + newline ...
168 );
169 end
170
171 %%%%
172 %%%% Sanitize ``ndim``.
173 %%%%
174
175 failed = ~pm.introspection.verified(ndim, "integer", 1);
176 if ~failed
177 failed = ndim < 1;
178 end
179 if ~failed
180 self.ndim = ndim;
181 else
182 help("pm.sampling.Sampler.run");
183 disp("ndim = ");
184 disp(ndim);
185 error ( newline ...
186 + "The input argument ``ndim`` must be a scalar positive integer (or whole-number)." + newline ...
187 + "The input argument ``ndim`` represents the number of dimensions of the domain" + newline ...
188 + "of the user-specified objective function ``getLogFunc()``." + newline ...
189 + "For more information, see the documentation displayed above." + newline ...
190 + newline ...
191 );
192 end
193
194 %%%%
195 %%%% Sanitize ``input`` specifications/file string.
196 %%%%
197
198 if ~pm.introspection.verified(self.input, "string", 1)
199 help("pm.sampling.Sampler.input");
200 disp("self.input = ");
201 disp(self.input);
202 error ( newline ...
203 + "The specified sampler attribute ``input`` must be scalar MATLAB string." + newline ...
204 + "For more information, see the documentation displayed above." + newline ...
205 + newline ...
206 );
207 else
208 self.input = string(self.input);
209 if self.input ~= ""
210 if isfile(self.input)
211 self.nml = pm.sys.path.abs(self.input, "lean");
212 else
213 self.nml = self.input; % Could still be a valid namelist.
214 end
215 if ~self.silent
216 warning ( newline ...
217 + "User-specified input namelist file detected: " + newline ...
218 + newline ...
219 + pm.io.tab() + """" + self.input + """" + newline ...
220 + newline ...
221 + "All simulation specifications will be read from the input file." + newline ...
222 + "All simulation specifications in the ``spec`` component of the sampler object will be ignored." + newline ...
223 + newline ...
224 );
225 end
226 else
227 %self.nml = [convertStringsToChars(self.getInputFile())];
228 self.nml = "&" + self.method + " " + self.spec.getEntriesNML(ndim) + " " + "/" + newline;
229 end
230 end
231
232 %%%%
233 %%%% Sanitize ``checked``.
234 %%%%
235
236 if ~isempty(self.checked)
237 if ~pm.introspection.verified(self.checked, "logical", 1)
238 help("pm.sampling.Sampler.checked");
239 disp("self.checked =");
240 disp(self.checked);
241 error ( newline ...
242 + "The specified sampler attribute ``checked`` must be a scalar MATLAB logical." + newline ...
243 + "For more information, see the documentation displayed above." + newline ...
244 + newline ...
245 );
246 elseif ~self.checked
247 chktypes = "nocheck";
248 else
249 chktypes = "checked";
250 end
251 else
252 chktypes = ["nocheck", "checked"];
253 end
254
255 %%%%
256 %%%% Setup the ParaMonter sampler library name.
257 %%%%
258
259 libspecs = [pm.os.namel(), pm.sys.arch(), self.libtype, self.memtype, self.partype];
260 mexdirs = pm.lib.path.mexdir(self.mexname, libspecs);
261
262 % We will choose the checking mode, compiler suite,
263 % and build mode based on the availability below.
264
265 if self.bldtype == ""
266 bldtypes = pm.lib.bldtypes();
267 else
268 bldtypes = lower(self.bldtype);
269 end
270 if self.clstype == ""
271 clstypes = pm.lib.clstypes();
272 else
273 clstypes = lower(self.clstype);
274 end
275 failed = isempty(mexdirs);
276 if ~failed
277 failed = true;
278 for ichk = 1 : length(chktypes)
279 chktype = filesep + chktypes(ichk);
280 for icls = 1 : length(clstypes)
281 clstype = filesep + clstypes(icls);
282 for ibld = 1 : length(bldtypes)
283 bldtype = filesep + bldtypes(ibld) + filesep;
284 for imex = 1 : length(mexdirs)
285 if contains(mexdirs(imex), clstype) ...
286 && contains(mexdirs(imex), bldtype) ...
287 && contains(mexdirs(imex), chktype)
288 mexdir = mexdirs(imex);
289 if self.bldtype == ""
290 self.bldtype = bldtypes(ibld);
291 end
292 if self.clstype == ""
293 self.clstype = clstypes(icls);
294 end
295 failed = false;
296 break;
297 end
298 end
299 if ~failed
300 break;
301 end
302 end
303 if ~failed
304 break;
305 end
306 end
307 if ~failed
308 break;
309 end
310 end
311 end
312
313 if failed
314 help("pm.sampling.Sampler");
315 disp("libspecs");
316 disp( libspecs );
317 disp("bldtypes");
318 disp( bldtypes );
319 disp("clstypes");
320 disp( clstypes );
321 error ( newline ...
322 + "There are no MEX libraries associated with the configurations displayed above:" + newline ...
323 + "Either the user has compromised internal structure of the ParaMonte library" + newline ...
324 + "or the user has tempered with hidden attributes of the ParaMonte sampler." + newline ...
325 + "If you believe neither is the case, please report this error at:" + newline ...
326 + newline ...
327 + pm.io.tab() + pm.web.href(self.weblinks.github.issues.url) + newline ...
328 + newline ...
329 + "for a quick resolution." + newline ...
330 + newline ...
331 );
332 end
333
334 %%%%
335 %%%% Add the identified MEX path to the MATLAB path list, only temporarily.
336 %%%%
337
338 pm.lib.path.clean();
339 self.matpath = path;
340 path(self.matpath, mexdir);
341 munlock(self.mexname);
342 clear(self.mexname);
343
344 if ~self.silent && ~(pm.matlab.iscmd || pm.os.is.win)
345 disp( newline ...
346 + "Also check the shell terminal (from which you opened MATLAB)" + newline ...
347 + "for potential realtime simulation progress and report." + newline ...
348 + newline ...
349 );
350 end
351
352 if self.clstype == "gnu"
353 setenv('GFORTRAN_STDIN_UNIT' , '5');
354 setenv('GFORTRAN_STDOUT_UNIT', '6');
355 setenv('GFORTRAN_STDERR_UNIT', '0');
356 end
357
358 %%%%
359 %%%% Set up the MEX file to call.
360 %%%%
361
362 if self.partype == "openmp"
363 mexcall = string(self.mexname + "(convertStringsToChars(self.method), @getLogFuncConcurrent, ndim, convertStringsToChars(self.nml))");
364 if ~self.silent
365 delete(gcp("nocreate"));
366 % The following works only in MATLAB 2022b and beyond.
367 if pm.matlab.release() < "2022b"
368 pool = parpool("threads");
369 maxNumCompThreads(abs(self.spec.parallelismNumThread));
370 else
371 pool = parpool("threads", abs(self.spec.parallelismNumThread));
372 end
373 else
374 evalc('delete(gcp("nocreate"))');
375 if pm.matlab.release() < "2022b"
376 evalc('pool = parpool("threads")');
377 evalc('maxNumCompThreads(abs(self.spec.parallelismNumThread))');
378 else
379 evalc('pool = parpool("threads", abs(self.spec.parallelismNumThread))');
380 end
381 end
382 else
383 mexcall = string(self.mexname + "(convertStringsToChars(self.method), @getLogFuncWrapped, ndim, convertStringsToChars(self.nml))");
384 %getLogFuncSpec = functions(getLogFunc);
385 end
386
387 %%%%
388 %%%% Define the ``getLogFunc`` wrapper function for serial/MPI sampling.
389 %%%%
390
391 function logFunc = getLogFuncWrapped(state)
392 logFunc = getLogFunc(state);
393 end
394
395 %%%%
396 %%%% Define the ``getLogFunc`` wrapper function for OpenMP sampling.
397 %%%%
398
399 %getLogFuncConst = parallel.pool.Constant(@(state) getLogFunc(state));
400 function [logFunc, avgTimePerFunCall, avgCommPerFunCall] = getLogFuncConcurrent(state)
401 %state = parallel.pool.Constant(state(2 : end, :));
402 avgCommPerFunCall = tic();
403 avgTimePerFunCall = 0;
404 njob = size(state, 2);
405
406 % %getState = arrayfun(@(ijob) state(:, ijob), 1 : njob, 'UniformOutput', 0);
407 % fout(1 : njob) = parallel.FevalFuture;
408 % start = zeros(njob, 'uint64');
409 % delta = zeros(njob);
410 % for ijob = 1 : njob
411 % %slice = getState(ijob);
412 % start(ijob) = tic();
413 % %fout(ijob) = parfeval(pool, getLogFunc, 1, slice{1});
414 % %fout(ijob) = parfeval(getLogFunc, 1, state(:, ijob));
415 % fout(ijob) = parfeval(pool, getLogFunc, 1, state(:, ijob));
416 % delta(ijob) = toc(start(ijob));
417 % end
418 % avgTimePerFunCall = sum(delta);
419 % logFunc = fetchOutputs(fout);
420
421 %spmd
422 % indx = spmdIndex;
423 % avgTimePerFunCall = tic();
424 % logFunc = getLogFunc(state(:, indx));
425 % avgTimePerFunCall = toc(avgTimePerFunCall);
426 %end
427 %logFunc = [logFunc{:}];
428 %avgTimePerFunCall = mean([avgTimePerFunCall{:}]);
429
430 %%pause
431
432 % \devnote
433 % `parfor` is slightly slower than `parfeval` (abour 20 ms for density function costing .1 ms).
434 % However, the slight penalty diminishes for costly problems and furthermore, `parfor` allows
435 % a reliable method of timing the density function, unlike `parfeval`.
436
437 logFunc = zeros(njob, 1);
438 %getSlice = arrayfun(@(ijob) state(:, ijob), 1 : njob, 'UniformOutput', 0);
439 parfor ijob = 1 : njob
440 start = tic();
441 %logFunc(ijob) = getLogFunc(getSlice{ijob});
442 %logFunc(ijob) = getLogFunc(state(:, ijob));
443 logFunc(ijob) = feval(getLogFunc, state(:, ijob));
444 avgTimePerFunCall = avgTimePerFunCall + toc(start);
445 end
446
447 avgTimePerFunCall = avgTimePerFunCall / njob;
448 avgCommPerFunCall = toc(avgCommPerFunCall) - avgTimePerFunCall;
449 end
450
451 %%%%
452 %%%% Call the MEX sampler.
453 %%%%
454
455 try
456 eval(mexcall);
457 self.finalize();
458 if ~self.silent
459 disp( newline ...
460 + self.getppm() + newline ...
461 + "For more information and examples on the usage, visit:" + newline ...
462 + newline ...
463 + pm.io.tab() + pm.web.href(self.weblinks.docs.generic.url) + newline ...
464 + newline ...
465 );
466 end
467 catch me
468 self.finalize();
469 msg = string(me.identifier) + " : " + string(me.message) + newline;
470 if ismac && strcmpi(me.identifier, 'MATLAB:mex:ErrInvalidMEXFile')
471 msg = msg ...
472 + "This error is most likely due to the ""System Integrity Protection""" + newline ...
473 + "(SIP) of your macOS interfering with the ParaMonte MATLAB MEX files." + newline ...
474 + "You can follow the guidelines in the documentation to resolve this error:" + newline ...
475 + newline ...
476 + pm.io.tab() + pm.web.href(self.weblinks.generic.docs.url + "/troubleshooting/macos-developer-cannot-be-verified/") + newline ...
477 + newline ...
478 + "If the problem persists even after following the guidelines" + newline ...
479 + "in the above page, please report this issue to the developers at:" + newline ...
480 + newline ...
481 + pm.io.tab() + pm.web.href(self.weblinks.github.issues.url) ...
482 + newline ...
483 ;
484 else
485 msg = msg ...
486 + "The " + self.method + " simulation failed. See the error message above." + newline ...
487 + "Also check the contents of the generated output '*_report.txt' files" + newline ...
488 + "if any such files were generated before the simulation crash:" + newline ...
489 + newline ...
490 + pm.io.tab() + self.spec.outputFileName + "*_report.txt" + newline ...
491 + newline ...
492 ;
493 end
494 error(msg);
495 end
496
497end
function name(in vendor)
Return the MPI library name as used in naming the ParaMonte MATLAB shared library directories.
function list()
Return a list of MATLAB strings containing the names of OS platforms supported by the ParaMonte MATLA...
function abs(in path, in style)
Return the Get absolute canonical path of a file or folder.
function arch()
Return the processor architecture name of the current system on which MATLAB is running,...
function bldtypes()
Return a list of MATLAB strings containing the names of all currently possible builds of the ParaMont...
This is the base class for the ParaMonte sampler routines.
Definition: Sampler.m:21
function clean()
Remove all paths that contain the ParaMonte lib directory from the MATLAB path variable.
function clstypes()
Return a list of MATLAB strings containing the names of all supported compiler suites (vendor names) ...
function compiler()
Return a scalar MATLAB logical that is true if and only if the current installation of MATLAB contain...
function detect(in vendor)
Return the MPI image count and current image ID (e.g., MPI rank + 1) and the MPI library name as it a...
function getLogFunc(in point)
function href(in url)
Return an HTML-style decoration of the input URL if the ParaMonte MATLAB library is used in GUI,...
function iscmd()
Return a scalar MATLAB logical true if and only if the the MATLAB binary is being called from the she...
function lib()
Return a scalar MATLAB string containing the path to the lib directory of the ParaMonte library packa...
function mexdir(in mexname, in config)
Return the vector of MATLAB strings containing the directory path(s) containing the specified ParaMon...
function namel()
Return a MATLAB string containing the lower-case name of the current OS.
function nproc(in vendor)
Return the runtime number of MPI processes with which the mpiexec launcher may have been invoked.
function parallel()
Return a scalar MATLAB logical that is true if and only if the current installation of MATLAB contain...
function release(in type)
Return a scalar MATLAB string containing the MATLAB release version, year, or season as requested.
function tab()
Return a scalar MATLAB string containing 4 blank characters equivalent to a tab character.
function verified(in varval, in vartype, in maxlen)
Return true if and only if the input varval conforms with the specified input type vartype and maximu...
function which(in vendor)
Return the a MATLAB string containing the path to the first mpiexec executable binary found in system...
function win()
Return true if the current OS is Windows.