Special Functions¶
The C code follows the C99 standard, with the usual math functions, as defined in OpenCL. These are automatically available for your models.
Standard functions and constants include:
- M_PI, M_PI_2, M_PI_4, M_SQRT1_2, M_E:
π, π/2, π/4, 1/√2 and Euler’s constant e
- exp, log, pow(x,y), expm1, log1p, sqrt, cbrt:
Power functions ex, lnx, xy, ex−1, ln1+x, √x, 3√x. The functions expm1(x) and log1p(x) are accurate across all x, including x very close to zero.
- sin, cos, tan, asin, acos, atan:
Trigonometry functions and inverses, operating on radians.
- sinh, cosh, tanh, asinh, acosh, atanh:
Hyperbolic trigonometry functions.
- atan2(y,x):
Angle from the x-axis to the point (x,y), which is equal to tan−1(y/x) corrected for quadrant. That is, if x and y are both negative, then atan2(y,x) returns a value in quadrant III where atan(y/x) would return a value in quadrant I. Similarly for quadrants II and IV when x and y have opposite sign.
- fabs(x), fmin(x,y), fmax(x,y), trunc, rint:
Floating point functions. rint(x) returns the nearest integer.
- NAN:
NaN, Not a Number, 0/0. Use isnan(x) to test for NaN. Note that you cannot use
x == NAN
to test for NaN values since that will always return false. NAN does not equal NAN! The alternative,x != x
may fail if the compiler optimizes the test away.- INFINITY:
∞,1/0. Use isinf(x) to test for infinity, or isfinite(x) to test for finite and not NaN.
- erf, erfc, tgamma, lgamma: do not use
Special functions that should be part of the standard, but are missing or inaccurate on some platforms. Use sas_erf, sas_erfc, sas_gamma and sas_lgamma instead (see below).
Some non-standard constants and functions are also provided:
- M_PI_180, M_4PI_3:
π180, 4π3
- SINCOS(x, s, c):
Macro which sets s=sin(x) and c=cos(x). The variables c and s must be declared first.
- square(x):
x2
- cube(x):
x3
- clip(a, a_min, a_max):
min, or NaN if a is NaN.
- sas_sinx_x(x):
\sin(x)/x, with limit \sin(0)/0 = 1.
- powr(x, y):
x^y for x \ge 0; this is faster than general x^y on some GPUs.
- pown(x, n):
x^n for n integer; this is faster than general x^n on some GPUs.
- FLOAT_SIZE:
The number of bytes in a floating point value. Even though all variables are declared double, they may be converted to single precision float before running. If your algorithm depends on precision (which is not uncommon for numerical algorithms), use the following:
#if FLOAT_SIZE>4 ... code for double precision ... #else ... code for single precision ... #endif- SAS_DOUBLE:
A replacement for
double
so that the declared variable will stay double precision; this should generally not be used since some graphics cards do not support double precision. There is no provision for forcing a constant to stay double precision.
The following special functions and scattering calculations are defined in
sasmodels/models/lib.
These functions have been tuned to be fast and numerically stable down
to q=0 even in single precision. In some cases they work around bugs
which appear on some platforms but not others, so use them where needed.
Add the files listed in source = ["lib/file.c", ...]
to your model.py
file in the order given, otherwise these functions will not be available.
- polevl(x, c, n):
Polynomial evaluation p(x) = \sum_{i=0}^n c_i x^i using Horner’s method so it is faster and more accurate.
c = \{c_n, c_{n-1}, \ldots, c_0 \} is the table of coefficients, sorted from highest to lowest.
source = ["lib/polevl.c", ...]
(link to code)- p1evl(x, c, n):
Evaluation of normalized polynomial p(x) = x^n + \sum_{i=0}^{n-1} c_i x^i using Horner’s method so it is faster and more accurate.
c = \{c_{n-1}, c_{n-2} \ldots, c_0 \} is the table of coefficients, sorted from highest to lowest.
source = ["lib/polevl.c", ...]
(polevl.c)- sas_gamma(x):
Gamma function sas_gamma(x) = \Gamma(x).
The standard math function, tgamma(x), is unstable for x < 1 on some platforms.
source = ["lib/sas_gamma.c", ...]
(sas_gamma.c)- sas_gammaln(x):
log gamma function sas_gammaln(x) = \log \Gamma(|x|).
The standard math function, lgamma(x), is incorrect for single precision on some platforms.
source = ["lib/sas_gammainc.c", ...]
(sas_gammainc.c)- sas_gammainc(a, x), sas_gammaincc(a, x):
Incomplete gamma function sas_gammainc(a, x) = \int_0^x t^{a-1}e^{-t}\,dt / \Gamma(a) and complementary incomplete gamma function sas_gammaincc(a, x) = \int_x^\infty t^{a-1}e^{-t}\,dt / \Gamma(a)
source = ["lib/sas_gammainc.c", ...]
(sas_gammainc.c)- sas_erf(x), sas_erfc(x):
Error function sas_erf(x) = \frac{2}{\sqrt\pi}\int_0^x e^{-t^2}\,dt and complementary error function sas_erfc(x) = \frac{2}{\sqrt\pi}\int_x^{\infty} e^{-t^2}\,dt.
The standard math functions erf(x) and erfc(x) are slower and broken on some platforms.
source = ["lib/polevl.c", "lib/sas_erf.c", ...]
(sas_erf.c)- sas_J0(x):
Bessel function of the first kind sas_J0(x)=J_0(x) where J_0(x) = \frac{1}{\pi}\int_0^\pi \cos(x\sin(\tau))\,d\tau.
The standard math function j0(x) is not available on all platforms.
source = ["lib/polevl.c", "lib/sas_J0.c", ...]
(sas_J0.c)- sas_J1(x):
Bessel function of the first kind sas_J1(x)=J_1(x) where J_1(x) = \frac{1}{\pi}\int_0^\pi \cos(\tau - x\sin(\tau))\,d\tau.
The standard math function j1(x) is not available on all platforms.
source = ["lib/polevl.c", "lib/sas_J1.c", ...]
(sas_J1.c)- sas_JN(n, x):
Bessel function of the first kind and integer order n, sas_JN(n, x) =J_n(x) where J_n(x) = \frac{1}{\pi}\int_0^\pi \cos(n\tau - x\sin(\tau))\,d\tau. If n = 0 or 1, it uses sas_J0(x) or sas_J1(x), respectively.
Warning: JN(n,x) can be very inaccurate (0.1%) for x not in [0.1, 100].
The standard math function jn(n, x) is not available on all platforms.
source = ["lib/polevl.c", "lib/sas_J0.c", "lib/sas_J1.c", "lib/sas_JN.c", ...]
(sas_JN.c)- sas_Si(x):
Sine integral Si(x) = \int_0^x \tfrac{\sin t}{t}\,dt.
Warning: Si(x) can be very inaccurate (0.1%) for x in [0.1, 100].
This function uses Taylor series for small and large arguments:
For large arguments use the following Taylor series,
\text{Si}(x) \sim \frac{\pi}{2} - \frac{\cos(x)}{x}\left(1 - \frac{2!}{x^2} + \frac{4!}{x^4} - \frac{6!}{x^6} \right) - \frac{\sin(x)}{x}\left(\frac{1}{x} - \frac{3!}{x^3} + \frac{5!}{x^5} - \frac{7!}{x^7}\right)For small arguments,
\text{Si}(x) \sim x - \frac{x^3}{3\times 3!} + \frac{x^5}{5 \times 5!} - \frac{x^7}{7 \times 7!} + \frac{x^9}{9\times 9!} - \frac{x^{11}}{11\times 11!}
source = ["lib/Si.c", ...]
(Si.c)- sas_3j1x_x(x):
Spherical Bessel form sph_j1c(x) = 3 j_1(x)/x = 3 (\sin(x) - x \cos(x))/x^3, with a limiting value of 1 at x=0, where j_1(x) is the spherical Bessel function of the first kind and first order.
This function uses a Taylor series for small x for numerical accuracy.
source = ["lib/sas_3j1x_x.c", ...]
(sas_3j1x_x.c)- sas_2J1x_x(x):
Bessel form sas_J1c(x) = 2 J_1(x)/x, with a limiting value of 1 at x=0, where J_1(x) is the Bessel function of first kind and first order.
source = ["lib/polevl.c", "lib/sas_J1.c", ...]
(sas_J1.c)- Gauss76Z[i], Gauss76Wt[i]:
Points z_i and weights w_i for 76-point Gaussian quadrature, respectively, computing \int_{-1}^1 f(z)\,dz \approx \sum_{i=1}^{76} w_i\,f(z_i).
Similar arrays are available in
gauss20.c
for 20-point quadrature and ingauss150.c
for 150-point quadrature. The macrosGAUSS_N
,GAUSS_Z
andGAUSS_W
are defined so that you can change the order of the integration by selecting an different source without touching the C code.
source = ["lib/gauss76.c", ...]
(gauss76.c)
Complex numbers¶
A small complex number library is available using source = ["lib/cl_complex.h", ...]
.
Numbers are defined as type cdouble, which can be passed to and returned from
functions. Operations are as follows:
declare: cdouble xdefine: cplx(real, imag)x.real: creal(x)x.imag: cimag(x)1j: Ireal + x: radd(real, x)real - x: rsub(real, x)real * x: rmul(real, x)real / x: rdiv(real, x)x + real: radd(real, x)x - real: radd(-real, x)x * real: rmul(real, x)x / real: rmul(1.0/real, x)x + y: cadd(x, y)x - y: csub(x, y)x * y: cmul(x, y)x / y: cdiv(x, y)-x: cneg(x)abs(x): cabs(x)angle(x): carg(x)special functions:csqrt, cexp, cpow, clog, clog10csin, ccos, ctancsinh, ccosh, ctanh
Python Functions¶
To ease the transition between python and C models, the Special Functions
available in C models are reproduced in Python with the same names. All of
them are available for import from sasmodels.special
. For example, you
can include the spherical form factor sas_3j1x_x
using
from sasmodels.special import sas_3j1x_x
in your python model. The
python functions operate on both scalars and arrays.
Unlike C models where the constant FLOAT_SIZE
is 4 or 8 depending on
whether the model is single or double precision,
sasmodels.special.FLOAT_SIZE
is fixed at 8. Python models should check
input.dtype == np.float32
or input.dtype == np.float64
if
needed.
SAS_DOUBLE
is for C type definitions; it is not available in Python.
sas_j1 = (sin(x) - x cos(x))/x**2
is an additional function that
is not available for C.
For Gauss-Lobatto integration points use
from sasmodels.special import gauss76 as gauss
, then access them
with attributes gauss.n
, gauss.z
and gauss.w
. This is
equivalent to the macros GAUSS_N
, GAUSS_Z
and GAUSS_W
in C models.
The complex number support functions aren’t needed in Python, and haven’t been implemented.
WARNING: The python functions do not yet have the same accuracy as the corresponding C functions.