source: sasmodels/sasmodels/core.py @ 2e66ef5

core_shell_microgelscostrafo411magnetic_modelticket-1257-vesicle-productticket_1156ticket_1265_superballticket_822_more_unit_tests
Last change on this file since 2e66ef5 was 2e66ef5, checked in by Paul Kienzle <pkienzle@…>, 7 years ago

add a short scripting guide; start in on developer docs

  • Property mode set to 100644
File size: 10.3 KB
Line 
1"""
2Core model handling routines.
3"""
4from __future__ import print_function
5
6__all__ = [
7    "list_models", "load_model", "load_model_info",
8    "build_model", "precompile_dlls",
9    ]
10
11import os
12from os.path import basename, dirname, join as joinpath
13from glob import glob
14
15import numpy as np # type: ignore
16
17from . import generate
18from . import modelinfo
19from . import product
20from . import mixture
21from . import kernelpy
22from . import kerneldll
23
24if os.environ.get("SAS_OPENCL", "").lower() == "none":
25    HAVE_OPENCL = False
26else:
27    try:
28        from . import kernelcl
29        HAVE_OPENCL = True
30    except Exception:
31        HAVE_OPENCL = False
32
33try:
34    from typing import List, Union, Optional, Any
35    from .kernel import KernelModel
36    from .modelinfo import ModelInfo
37except ImportError:
38    pass
39
40# TODO: refactor composite model support
41# The current load_model_info/build_model does not reuse existing model
42# definitions when loading a composite model, instead reloading and
43# rebuilding the kernel for each component model in the expression.  This
44# is fine in a scripting environment where the model is built when the script
45# starts and is thrown away when the script ends, but may not be the best
46# solution in a long-lived application.  This affects the following functions:
47#
48#    load_model
49#    load_model_info
50#    build_model
51
52KINDS = ("all", "py", "c", "double", "single", "opencl", "1d", "2d",
53         "nonmagnetic", "magnetic")
54def list_models(kind=None):
55    # type: (str) -> List[str]
56    """
57    Return the list of available models on the model path.
58
59    *kind* can be one of the following:
60
61        * all: all models
62        * py: python models only
63        * c: compiled models only
64        * single: models which support single precision
65        * double: models which require double precision
66        * opencl: controls if OpenCL is supperessed
67        * 1d: models which are 1D only, or 2D using abs(q)
68        * 2d: models which can be 2D
69        * magnetic: models with an sld
70        * nommagnetic: models without an sld
71
72    For multiple conditions, combine with plus.  For example, *c+single+2d*
73    would return all oriented models implemented in C which can be computed
74    accurately with single precision arithmetic.
75    """
76    if kind and any(k not in KINDS for k in kind.split('+')):
77        raise ValueError("kind not in " + ", ".join(KINDS))
78    files = sorted(glob(joinpath(generate.MODEL_PATH, "[a-zA-Z]*.py")))
79    available_models = [basename(f)[:-3] for f in files]
80    if kind and '+' in kind:
81        all_kinds = kind.split('+')
82        condition = lambda name: all(_matches(name, k) for k in all_kinds)
83    else:
84        condition = lambda name: _matches(name, kind)
85    selected = [name for name in available_models if condition(name)]
86
87    return selected
88
89def _matches(name, kind):
90    if kind is None or kind == "all":
91        return True
92    info = load_model_info(name)
93    pars = info.parameters.kernel_parameters
94    if kind == "py" and callable(info.Iq):
95        return True
96    elif kind == "c" and not callable(info.Iq):
97        return True
98    elif kind == "double" and not info.single:
99        return True
100    elif kind == "single" and info.single:
101        return True
102    elif kind == "opencl" and info.opencl:
103        return True
104    elif kind == "2d" and any(p.type == 'orientation' for p in pars):
105        return True
106    elif kind == "1d" and all(p.type != 'orientation' for p in pars):
107        return True
108    elif kind == "magnetic" and any(p.type == 'sld' for p in pars):
109        return True
110    elif kind == "nonmagnetic" and any(p.type != 'sld' for p in pars):
111        return True
112    return False
113
114def load_model(model_name, dtype=None, platform='ocl'):
115    # type: (str, str, str) -> KernelModel
116    """
117    Load model info and build model.
118
119    *model_name* is the name of the model, or perhaps a model expression
120    such as sphere*hardsphere or sphere+cylinder.
121
122    *dtype* and *platform* are given by :func:`build_model`.
123    """
124    return build_model(load_model_info(model_name),
125                       dtype=dtype, platform=platform)
126
127
128def load_model_info(model_name):
129    # type: (str) -> modelinfo.ModelInfo
130    """
131    Load a model definition given the model name.
132
133    *model_name* is the name of the model, or perhaps a model expression
134    such as sphere*hardsphere or sphere+cylinder.
135
136    This returns a handle to the module defining the model.  This can be
137    used with functions in generate to build the docs or extract model info.
138    """
139    parts = model_name.split('+')
140    if len(parts) > 1:
141        model_info_list = [load_model_info(p) for p in parts]
142        return mixture.make_mixture_info(model_info_list)
143
144    parts = model_name.split('*')
145    if len(parts) > 1:
146        if len(parts) > 2:
147            raise ValueError("use P*S to apply structure factor S to model P")
148        P_info, Q_info = [load_model_info(p) for p in parts]
149        return product.make_product_info(P_info, Q_info)
150
151    kernel_module = generate.load_kernel_module(model_name)
152    return modelinfo.make_model_info(kernel_module)
153
154
155def build_model(model_info, dtype=None, platform="ocl"):
156    # type: (modelinfo.ModelInfo, str, str) -> KernelModel
157    """
158    Prepare the model for the default execution platform.
159
160    This will return an OpenCL model, a DLL model or a python model depending
161    on the model and the computing platform.
162
163    *model_info* is the model definition structure returned from
164    :func:`load_model_info`.
165
166    *dtype* indicates whether the model should use single or double precision
167    for the calculation.  Choices are 'single', 'double', 'quad', 'half',
168    or 'fast'.  If *dtype* ends with '!', then force the use of the DLL rather
169    than OpenCL for the calculation.
170
171    *platform* should be "dll" to force the dll to be used for C models,
172    otherwise it uses the default "ocl".
173    """
174    composition = model_info.composition
175    if composition is not None:
176        composition_type, parts = composition
177        models = [build_model(p, dtype=dtype, platform=platform) for p in parts]
178        if composition_type == 'mixture':
179            return mixture.MixtureModel(model_info, models)
180        elif composition_type == 'product':
181            P, S = models
182            return product.ProductModel(model_info, P, S)
183        else:
184            raise ValueError('unknown mixture type %s'%composition_type)
185
186    # If it is a python model, return it immediately
187    if callable(model_info.Iq):
188        return kernelpy.PyModel(model_info)
189
190    numpy_dtype, fast, platform = parse_dtype(model_info, dtype, platform)
191
192    source = generate.make_source(model_info)
193    if platform == "dll":
194        #print("building dll", numpy_dtype)
195        return kerneldll.load_dll(source['dll'], model_info, numpy_dtype)
196    else:
197        #print("building ocl", numpy_dtype)
198        return kernelcl.GpuModel(source, model_info, numpy_dtype, fast=fast)
199
200def precompile_dlls(path, dtype="double"):
201    # type: (str, str) -> List[str]
202    """
203    Precompile the dlls for all builtin models, returning a list of dll paths.
204
205    *path* is the directory in which to save the dlls.  It will be created if
206    it does not already exist.
207
208    This can be used when build the windows distribution of sasmodels
209    which may be missing the OpenCL driver and the dll compiler.
210    """
211    numpy_dtype = np.dtype(dtype)
212    if not os.path.exists(path):
213        os.makedirs(path)
214    compiled_dlls = []
215    for model_name in list_models():
216        model_info = load_model_info(model_name)
217        if not callable(model_info.Iq):
218            source = generate.make_source(model_info)['dll']
219            old_path = kerneldll.DLL_PATH
220            try:
221                kerneldll.DLL_PATH = path
222                dll = kerneldll.make_dll(source, model_info, dtype=numpy_dtype)
223            finally:
224                kerneldll.DLL_PATH = old_path
225            compiled_dlls.append(dll)
226    return compiled_dlls
227
228def parse_dtype(model_info, dtype=None, platform=None):
229    # type: (ModelInfo, str, str) -> (np.dtype, bool, str)
230    """
231    Interpret dtype string, returning np.dtype and fast flag.
232
233    Possible types include 'half', 'single', 'double' and 'quad'.  If the
234    type is 'fast', then this is equivalent to dtype 'single' but using
235    fast native functions rather than those with the precision level guaranteed
236    by the OpenCL standard.
237
238    Platform preference can be specfied ("ocl" vs "dll"), with the default
239    being OpenCL if it is availabe.  If the dtype name ends with '!' then
240    platform is forced to be DLL rather than OpenCL.
241
242    This routine ignores the preferences within the model definition.  This
243    is by design.  It allows us to test models in single precision even when
244    we have flagged them as requiring double precision so we can easily check
245    the performance on different platforms without having to change the model
246    definition.
247    """
248    # Assign default platform, overriding ocl with dll if OpenCL is unavailable
249    # If opencl=False OpenCL is switched off
250
251    if platform is None:
252        platform = "ocl"
253    if platform == "ocl" and not HAVE_OPENCL or not model_info.opencl:
254        platform = "dll"
255
256    # Check if type indicates dll regardless of which platform is given
257    if dtype is not None and dtype.endswith('!'):
258        platform = "dll"
259        dtype = dtype[:-1]
260
261    # Convert special type names "half", "fast", and "quad"
262    fast = (dtype == "fast")
263    if fast:
264        dtype = "single"
265    elif dtype == "quad":
266        dtype = "longdouble"
267    elif dtype == "half":
268        dtype = "float16"
269
270    # Convert dtype string to numpy dtype.
271    if dtype is None:
272        numpy_dtype = (generate.F32 if platform == "ocl" and model_info.single
273                       else generate.F64)
274    else:
275        numpy_dtype = np.dtype(dtype)
276
277    # Make sure that the type is supported by opencl, otherwise use dll
278    if platform == "ocl":
279        env = kernelcl.environment()
280        if not env.has_type(numpy_dtype):
281            platform = "dll"
282            if dtype is None:
283                numpy_dtype = generate.F64
284
285    return numpy_dtype, fast, platform
286
287def list_models_main():
288    # type: () -> None
289    """
290    Run list_models as a main program.  See :func:`list_models` for the
291    kinds of models that can be requested on the command line.
292    """
293    import sys
294    kind = sys.argv[1] if len(sys.argv) > 1 else "all"
295    print("\n".join(list_models(kind)))
296
297if __name__ == "__main__":
298    list_models_main()
Note: See TracBrowser for help on using the repository browser.