qutip/core/cy/math.pyx
#cython: language_level=3
cimport cython
from libc.math cimport (fabs, sinh, cosh, exp, pi, sqrt, cos, sin, copysign)
cdef extern from "<complex>" namespace "std" nogil:
double real(double complex x)
double imag(double complex x)
@cython.boundscheck(False)
@cython.cdivision(True)
cdef double erf(double x):
"""
A Cython version of the erf function from the cdflib in SciPy.
"""
cdef double c = 0.564189583547756
cdef double a[5]
a[:] = [0.771058495001320e-4, -0.133733772997339e-2, 0.323076579225834e-1,
0.479137145607681e-1, 0.128379167095513]
cdef double b[3]
b[:] = [0.301048631703895e-2, 0.538971687740286e-1, .375795757275549]
cdef double p[8]
p[:] = [-1.36864857382717e-7, 5.64195517478974e-1, 7.21175825088309,
4.31622272220567e1, 1.52989285046940e2, 3.39320816734344e2,
4.51918953711873e2, 3.00459261020162e2]
cdef double q[8]
q[:]= [1.0, 1.27827273196294e1, 7.70001529352295e1, 2.77585444743988e2,
6.38980264465631e2, 9.31354094850610e2, 7.90950925327898e2,
3.00459260956983e2]
cdef double r[5]
r[:] = [2.10144126479064, 2.62370141675169e1, 2.13688200555087e1,
4.65807828718470, 2.82094791773523e-1]
cdef double s[4]
s[:] = [9.41537750555460e1, 1.87114811799590e2, 9.90191814623914e1,
1.80124575948747e1]
cdef double ax = fabs(x)
cdef double t, x2, top, bot, erf
if ax <= 0.5:
t = x*x
top = ((((a[0]*t+a[1])*t+a[2])*t+a[3])*t+a[4]) + 1.0
bot = ((b[0]*t+b[1])*t+b[2])*t + 1.0
erf = x * (top/bot)
return erf
elif ax <= 4.0:
x2 = x*x
top = ((((((p[0]*ax+p[1])*ax+p[2])*ax+p[3])*ax+p[4])*ax+p[5])*ax+p[6])*ax + p[7]
bot = ((((((q[0]*ax+q[1])*ax+q[2])*ax+q[3])*ax+q[4])*ax+q[5])*ax+q[6])*ax + q[7]
erf = 0.5 + (0.5-exp(-x2)*top/bot)
if x < 0.0:
erf = -erf
return erf
elif ax < 5.8:
x2 = x*x
t = 1.0/x2
top = (((r[0]*t+r[1])*t+r[2])*t+r[3])*t + r[4]
bot = (((s[0]*t+s[1])*t+s[2])*t+s[3])*t + 1.0
erf = (c-top/(x2*bot))/ax
erf = 0.5 + (0.5-exp(-x2)*erf)
if x < 0.0:
erf = -erf
return erf
else:
erf = copysign(1.0, x)
return erf
# COMPUTATION OF SPECIAL FUNCTIONS
#
# Shanjie Zhang and Jianming Jin
#
# Copyrighted but permission granted to use code in programs.
# Buy their book "Computation of Special Functions", 1996, John Wiley & Sons, Inc.
@cython.cdivision(True)
@cython.boundscheck(False)
cdef double complex zerf(double complex Z):
"""
Parameters
----------
Z : double complex
Input parameter.
X : double
Real part of Z.
Y : double
Imag part of Z.
Returns
-------
erf(z) : double complex
"""
cdef double EPS = 1e-12
cdef double X = real(Z)
cdef double Y = imag(Z)
cdef double X2 = X * X
cdef double ER, R, W, C0, ER0, ERR, ERI, CS, SS, ER1, EI1, ER2, W1
cdef size_t K, N
if X < 3.5:
ER = 1.0
R = 1.0
W = 0.0
for K in range(1, 100):
R = R*X2/(K+0.5)
ER = ER+R
if (fabs(ER-W) < EPS*fabs(ER)):
break
W = ER
C0 = 2.0/sqrt(pi)*X*exp(-X2)
ER0 = C0*ER
else:
ER = 1.0
R=1.0
for K in range(1, 12):
R = -R*(K-0.5)/X2
ER = ER+R
C0 = exp(-X2)/(X*sqrt(pi))
ER0 = 1.0-C0*ER
if Y == 0.0:
ERR = ER0
ERI = 0.0
else:
CS = cos(2.0*X*Y)
SS = sin(2.0*X*Y)
ER1 = exp(-X2)*(1.0-CS)/(2.0*pi*X)
EI1 = exp(-X2)*SS/(2.0*pi*X)
ER2 = 0.0
W1 = 0.0
for N in range(1,100):
ER2 = ER2+exp(-.25*N*N)/(N*N+4.0*X2)*(2.0*X-2.0*X*cosh(N*Y)*CS+N*sinh(N*Y)*SS)
if (fabs((ER2-W1)/ER2) < EPS):
break
W1 = ER2
C0 = 2.0*exp(-X2)/pi
ERR = ER0+ER1+C0*ER2
EI2 = 0.0
W2 = 0.0
for N in range(1,100):
EI2 = EI2+exp(-.25*N*N)/(N*N+4.0*X2)*(2.0*X*cosh(N*Y)*SS+N*sinh(N*Y)*CS)
if (fabs((EI2-W2)/EI2) < EPS):
break
W2 = EI2
ERI = EI1+C0*EI2
return ERR + 1j*ERI