2%> Perform the basic runtime checks
for the sampler and
return nothing.
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>
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>
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>
29%> sampler = pm.sampling.Sampler();
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>
43 %%%% Sanitize ``sampler.silent``.
46 if ~pm.introspection.verified(self.silent,
"logical", 1)
47 help(
"pm.sampling.Sampler.silent");
48 disp(
"self.silent =");
51 +
"The sampler attribute ``silent`` must be a scalar of type ``logical``." + newline ...
52 +
"For more information, see the documentation displayed above." + newline ...
58 %%%% Sanitize parallelism method to set reporting permission.
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;
67 if ~pm.introspection.verified(self.mpiname,
"string", 1)
70 %%%% Sanitize ``mpiname``.
73 help(
"pm.sampling.Sampler.mpiname");
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 ...
82 elseif pm.array.len(self.mpiname) == 0
85 %%%% Detect potential MPI launcher.
88 [mpiname,
nproc, ~] = pm.lib.mpi.runtime.detect();
89 if pm.array.len(mpiname) > 0 &&
nproc > 1
90 self.mpiname = mpiname;
96 %%%% First
detect potential use of MPI, then check
for thread parallelism.
99 if 0 < pm.array.len(self.mpiname) % MPI enabled.
101 self.silent =
true; % Otherwise, we keep the
default value of self.silent.
102 self.partype = string(pm.lib.mpi.name(self.mpiname));
104 elseif ~isempty(self.spec.parallelismNumThread)
107 %%%% Sanitize ``self.spec.parallelismNumThread``.
110 % The following separate conditions are crucial to remain separate.
111 failed = ~pm.introspection.verified(self.spec.parallelismNumThread,
"integer", 1);
113 failed = self.spec.parallelismNumThread < 0;
116 failed = ~pm.matlab.has.parallel;
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();
125 help(
"pm.sampling.Sampler.run");
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 ...
140 + pm.io.tab() +
"self.spec.parallelismNumThread = " +
string(self.parallelismNumThread) + newline ...
142 +
"Does your MATLAB have Parallel Computing Toolbox?" + newline ...
144 + pm.io.tab() +
"pm.matlab.has.parallel() = " +
string(pm.matlab.has.parallel()) + newline ...
146 +
"For more information, see the documentation displayed above." + newline ...
158 help("pm.sampling.
Sampler.run");
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 ...
166 + "For more information, see the documentation displayed above." + newline ...
172 %%%% Sanitize ``ndim``.
175 failed = ~pm.introspection.
verified(ndim, "integer", 1);
182 help("pm.sampling.
Sampler.run");
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 ...
195 %%%% Sanitize ``input`` specifications/file
string.
198 if ~pm.introspection.
verified(self.input, "
string", 1)
199 help("pm.sampling.
Sampler.input");
200 disp("self.input = ");
203 + "The specified sampler attribute ``input`` must be scalar MATLAB
string." + newline ...
204 + "For more information, see the documentation displayed above." + newline ...
208 self.input =
string(self.input);
210 if isfile(self.input)
211 self.nml = pm.sys.path.
abs(self.input, "lean");
213 self.nml = self.input; % Could still be a valid namelist.
216 warning ( newline ...
217 + "User-specified input namelist file detected: " + newline ...
219 + pm.io.
tab() + """" + self.input + """" + 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 ...
227 %self.nml = [convertStringsToChars(self.getInputFile())];
228 self.nml = "&" + self.method + " " + self.spec.getEntriesNML(ndim) + " " + "/" + newline;
233 %%%% Sanitize ``checked``.
236 if ~isempty(self.checked)
237 if ~pm.introspection.
verified(self.checked, "logical", 1)
238 help("pm.sampling.
Sampler.checked");
239 disp("self.checked =");
242 + "The specified sampler attribute ``checked`` must be a scalar MATLAB logical." + newline ...
243 + "For more information, see the documentation displayed above." + newline ...
247 chktypes = "nocheck";
249 chktypes = "checked";
252 chktypes = ["nocheck", "checked"];
256 %%%% Setup the ParaMonter sampler library
name.
259 libspecs = [pm.os.
namel(), pm.sys.
arch(), self.libtype, self.memtype, self.partype];
260 mexdirs = pm.
lib.path.
mexdir(self.mexname, libspecs);
262 % We will choose the checking mode,
compiler suite,
263 % and build mode based on the availability below.
265 if self.bldtype == ""
270 if self.clstype == ""
275 failed = isempty(mexdirs);
278 for ichk = 1 : length(chktypes)
279 chktype = filesep + chktypes(ichk);
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)
289 if self.bldtype == ""
292 if self.clstype == ""
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 ...
327 + pm.io.
tab() + pm.web.
href(self.
weblinks.github.issues.url) + newline ...
329 + "for a quick resolution." + newline ...
335 %%%% Add the identified MEX path to the MATLAB path
list, only temporarily.
340 path(self.matpath,
mexdir);
341 munlock(self.mexname);
344 if ~self.silent && ~(pm.matlab.
iscmd || pm.os.is.
win)
346 + "Also check the shell terminal (from
which you opened MATLAB)" + newline ...
347 + "for potential realtime simulation progress and report." + newline ...
352 if self.clstype == "gnu"
353 setenv('GFORTRAN_STDIN_UNIT' , '5');
354 setenv('GFORTRAN_STDOUT_UNIT', '6');
355 setenv('GFORTRAN_STDERR_UNIT', '0');
359 %%%% Set up the MEX file to call.
362 if self.partype == "openmp"
363 mexcall =
string(self.mexname + "(convertStringsToChars(self.method), @getLogFuncConcurrent, ndim, convertStringsToChars(self.nml))");
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));
371 pool = parpool("threads",
abs(self.spec.parallelismNumThread));
374 evalc('delete(gcp("nocreate"))');
375 if pm.matlab.
release() < "2022b"
376 evalc('pool = parpool("threads")');
377 evalc('maxNumCompThreads(
abs(self.spec.parallelismNumThread))');
379 evalc('pool = parpool("threads",
abs(self.spec.parallelismNumThread))');
383 mexcall =
string(self.mexname + "(convertStringsToChars(self.method), @getLogFuncWrapped, ndim, convertStringsToChars(self.nml))");
388 %%%% Define the ``
getLogFunc`` wrapper function for serial/MPI sampling.
391 function logFunc = getLogFuncWrapped(state)
396 %%%% Define the ``
getLogFunc`` wrapper function for OpenMP sampling.
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);
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));
418 % avgTimePerFunCall = sum(delta);
419 % logFunc = fetchOutputs(fout);
423 % avgTimePerFunCall = tic();
425 % avgTimePerFunCall = toc(avgTimePerFunCall);
427 %logFunc = [logFunc{:}];
428 %avgTimePerFunCall = mean([avgTimePerFunCall{:}]);
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`.
437 logFunc = zeros(njob, 1);
438 %getSlice = arrayfun(@(ijob) state(:, ijob), 1 : njob,
'UniformOutput', 0);
439 parfor ijob = 1 : njob
443 logFunc(ijob) = feval(
getLogFunc, state(:, ijob));
444 avgTimePerFunCall = avgTimePerFunCall + toc(start);
447 avgTimePerFunCall = avgTimePerFunCall / njob;
448 avgCommPerFunCall = toc(avgCommPerFunCall) - avgTimePerFunCall;
452 %%%% Call the MEX sampler.
460 + self.getppm() + newline ...
461 +
"For more information and examples on the usage, visit:" + newline ...
463 + pm.io.tab() + pm.web.href(self.weblinks.docs.generic.url) + newline ...
469 msg = string(me.identifier) +
" : " + string(me.message) + newline;
470 if ismac && strcmpi(me.identifier,
'MATLAB:mex:ErrInvalidMEXFile')
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 ...
476 + pm.io.
tab() + pm.web.
href(self.
weblinks.generic.docs.url + "/troubleshooting/macos-developer-cannot-be-
verified/") + 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 ...
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 ...
490 + pm.io.
tab() + self.spec.outputFileName + "*_report.txt" + newline ...
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.
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 weblinks()
Return a structure containing tree of weblinks for the ParaMonte MATLAB library source file and docum...
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.