Jdu do práce
Former-commit-id: 3be0f5adda5334dae7397d21b6d48835a190632d
This commit is contained in:
parent
2a7015b80b
commit
00f9f5234a
|
@ -39,6 +39,9 @@ typedef enum {
|
||||||
QPMS_NORMALISATION_UNDEF = 0
|
QPMS_NORMALISATION_UNDEF = 0
|
||||||
} qpms_normalisation_t;
|
} qpms_normalisation_t;
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
static inline int qpms_normalisation_t_csphase(qpms_normalisation_t norm) {
|
static inline int qpms_normalisation_t_csphase(qpms_normalisation_t norm) {
|
||||||
return (norm & QPMS_NORMALISATION_T_CSBIT)? -1 : 1;
|
return (norm & QPMS_NORMALISATION_T_CSBIT)? -1 : 1;
|
||||||
}
|
}
|
||||||
|
@ -48,6 +51,46 @@ static inline int qpms_normalisation_t_normonly(qpms_normalisation_t norm) {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// relative to QPMS_NORMALISATION_KRISTENSSON_CS, i.e.
|
||||||
|
// P_l^m[normtype] = P_l^m[Kristensson]
|
||||||
|
static inline double qpms_normalisation_t_factor(qpms_normalisation_t norm, qpms_l_t l, qpms_m_t m) {
|
||||||
|
int csphase = qpms_normalisation_t_csphase(norm);
|
||||||
|
norm = qpms_normalisation_t_normonly(norm);
|
||||||
|
double factor;
|
||||||
|
switch (norm) {
|
||||||
|
case QPMS_NORMALISATION_KRISTENSSON:
|
||||||
|
factor = 1.;
|
||||||
|
break;
|
||||||
|
case QPMS_NORMALISATION_TAYLOR:
|
||||||
|
factor = sqrt(l*(l+1));
|
||||||
|
break;
|
||||||
|
case QPMS_NORMALISATION_XU:
|
||||||
|
factor = sqrt(l*(l+1) * 4 * M_PI / (2*l+1) * exp(lgamma(l+m+1)-lgamma(l-m+1)));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
}
|
||||||
|
factor *= (m%2)?(-csphase):1;
|
||||||
|
return factor;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline double qpms_normalisation_t_factor_abssquare(qpms_normalisation_t norm, qpms_l_t l, qpms_m_t m) {
|
||||||
|
norm = qpms_normalisation_t_normonly(norm);
|
||||||
|
switch (norm) {
|
||||||
|
case QPMS_NORMALISATION_KRISTENSSON:
|
||||||
|
return 1.;
|
||||||
|
break;
|
||||||
|
case QPMS_NORMALISATION_TAYLOR:
|
||||||
|
return l*(l+1);
|
||||||
|
break;
|
||||||
|
case QPMS_NORMALISATION_XU:
|
||||||
|
return l*(l+1) * 4 * M_PI / (2*l+1) * exp(lgamma(l+m+1)-lgamma(l-m+1));
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
assert(0);
|
||||||
|
return NAN;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
|
@ -75,6 +118,11 @@ typedef struct {
|
||||||
double r, theta, phi;
|
double r, theta, phi;
|
||||||
} sph_t;
|
} sph_t;
|
||||||
|
|
||||||
|
typedef struct { // Do I really need this???
|
||||||
|
complex double r;
|
||||||
|
double theta, phi;
|
||||||
|
} csph_t;
|
||||||
|
|
||||||
// complex vector components in local spherical basis
|
// complex vector components in local spherical basis
|
||||||
typedef struct {
|
typedef struct {
|
||||||
complex double rc, thetac, phic;
|
complex double rc, thetac, phic;
|
||||||
|
|
|
@ -27,6 +27,21 @@ static inline cart3_t sph2cart(const sph_t sph) {
|
||||||
return cart;
|
return cart;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static inline cart3_t cart3_add(const cart3_t a, const cart3_t b) {
|
||||||
|
cart3_t res = {a.x+b.x, a.y+b.y, a.z+b.z};
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline csphvec_t csphvec_add(const csphvec_t a, const csphvec_t b) {
|
||||||
|
csphvec_t res = {a.rc + b.rc, a.thetac + b.thetac, a.phic + b.phic};
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
|
static inline csphvec_t scphvec_scale(complex double c, const scphvec_t v) {
|
||||||
|
csphvec_t res = {c * v.rc, c * v.thetac, c * v.phic};
|
||||||
|
return res;
|
||||||
|
}
|
||||||
|
|
||||||
// equivalent to sph_loccart2cart in qpms_p.py
|
// equivalent to sph_loccart2cart in qpms_p.py
|
||||||
static inline ccart3_t csphvec2cart(const csphvec_t sphvec, const sph_t at) {
|
static inline ccart3_t csphvec2cart(const csphvec_t sphvec, const sph_t at) {
|
||||||
ccart3_t res = {0, 0, 0};
|
ccart3_t res = {0, 0, 0};
|
||||||
|
|
176
qpms/vswf.c
176
qpms/vswf.c
|
@ -217,7 +217,7 @@ void qpms_vswfset_sph_pfree(qpms_vswfset_sph_t *w) {
|
||||||
free(w);
|
free(w);
|
||||||
}
|
}
|
||||||
|
|
||||||
qpms_errno_t qpms_vswf_fill(csphvec_t *mgtarget, csphvec_t *eltarget,
|
qpms_errno_t qpms_vswf_fill(csphvec_t *const longtarget, csphvec_t * const mgtarget, csphvec_t * const eltarget,
|
||||||
qpms_l_t lMax, sph_t kr,
|
qpms_l_t lMax, sph_t kr,
|
||||||
qpms_bessel_t btyp, qpms_normalisation_t norm) {
|
qpms_bessel_t btyp, qpms_normalisation_t norm) {
|
||||||
assert(lMax >= 1);
|
assert(lMax >= 1);
|
||||||
|
@ -228,19 +228,32 @@ qpms_errno_t qpms_vswf_fill(csphvec_t *mgtarget, csphvec_t *eltarget,
|
||||||
double const *pleg = pt.leg;
|
double const *pleg = pt.leg;
|
||||||
double const *ppi = pt.pi;
|
double const *ppi = pt.pi;
|
||||||
double const *ptau = pt.tau;
|
double const *ptau = pt.tau;
|
||||||
csphvec_t *pmg = mgtarget, *pel = eltarget;
|
csphvec_t *plong = longtarget, *pmg = mgtarget, *pel = eltarget;
|
||||||
for(qpms_l_t l = 1; l <= lMax; ++l) {
|
for(qpms_l_t l = 1; l <= lMax; ++l) {
|
||||||
complex double besfac = *pbes / kr.r;
|
complex double besfac = *pbes / kr.r;
|
||||||
complex double besderfac = *(pbes-1) - l * besfac;
|
complex double besderfac = *(pbes-1) - l * besfac;
|
||||||
for(qpms_m_t m = -l; m <= l; ++m) {
|
for(qpms_m_t m = -l; m <= l; ++m) {
|
||||||
complex double eimf = cexp(m * kr.phi * I);
|
complex double eimf = cexp(m * kr.phi * I);
|
||||||
pel->rc = l*(l+1) * (*pleg) * besfac * eimf;
|
if (longtarget) {
|
||||||
pel->thetac = *ptau * besderfac * eimf;
|
complex double longfac = sqrt(l*(l+1)) * eimf;
|
||||||
pel->phic = *ppi * besderfac * I * eimf;
|
plong->rc = (besderfac-besfac) * (*pleg) * longfac;
|
||||||
pmg->rc = 0.;
|
plong->thetac = *ptau * besfac * longfac;
|
||||||
pmg->thetac = *ppi * (*pbes) * I * eimf;
|
plong->phic = *ppi * I * besfac * longfac;
|
||||||
pmg->phic = - *ptau * (*pbes) * eimf;
|
++plong;
|
||||||
++pleg; ++ppi; ++ptau; ++pel; ++pmg;
|
}
|
||||||
|
if (eltarget) {
|
||||||
|
pel->rc = l*(l+1) * (*pleg) * besfac * eimf;
|
||||||
|
pel->thetac = *ptau * besderfac * eimf;
|
||||||
|
pel->phic = *ppi * besderfac * I * eimf;
|
||||||
|
++pel;
|
||||||
|
}
|
||||||
|
if (mgtarget) {
|
||||||
|
pmg->rc = 0.;
|
||||||
|
pmg->thetac = *ppi * (*pbes) * I * eimf;
|
||||||
|
pmg->phic = - *ptau * (*pbes) * eimf;
|
||||||
|
++pmg;
|
||||||
|
}
|
||||||
|
++pleg; ++ppi; ++ptau;
|
||||||
}
|
}
|
||||||
++pbes;
|
++pbes;
|
||||||
}
|
}
|
||||||
|
@ -248,3 +261,148 @@ qpms_errno_t qpms_vswf_fill(csphvec_t *mgtarget, csphvec_t *eltarget,
|
||||||
qpms_pitau_free(pt);
|
qpms_pitau_free(pt);
|
||||||
return QPMS_SUCCESS;
|
return QPMS_SUCCESS;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// consistency check: this should give the same results as the above function (up to rounding errors)
|
||||||
|
qpms_errno_t qpms_vswf_fill_alternative(csphvec_t *const longtarget, csphvec_t * const mgtarget, csphvec_t * const eltarget,
|
||||||
|
qpms_l_t lMax, sph_t kr,
|
||||||
|
qpms_bessel_t btyp, qpms_normalisation_t norm) {
|
||||||
|
assert(lMax >= 1);
|
||||||
|
complex double *bessel = malloc((lMax+1)*sizeof(complex double));
|
||||||
|
if(qpms_sph_bessel_fill(btyp, lMax, kr.r, bessel)) abort();
|
||||||
|
complex double const *pbes = bessel + 1; // starting from l = 1
|
||||||
|
|
||||||
|
qpms_y_t nelem = qpms_lMax2nelem(lMax);
|
||||||
|
csphvec_t * const a1 = malloc(3*nelem*sizeof(csphvec_t)), * const a2 = a1 + nelem, * const a3 = a2 + nelem;
|
||||||
|
if(qpms_vecspharm_fill(a1, a2, a3, lMax, kr, norm)) abort();
|
||||||
|
const sphvec_t *p1 = a1, const *p2 = a2, const *p3 = a3;
|
||||||
|
|
||||||
|
csphvec_t *plong = longtarget, *pmg = mgtarget, *pel = eltarget;
|
||||||
|
for(qpms_l_t l = 1; l <= lMax; ++l) {
|
||||||
|
complex double besfac = *pbes / kr.r;
|
||||||
|
complex double besderfac = *(pbes-1) - l * besfac;
|
||||||
|
double sqrtlfac = sqrt(l*(l+1));
|
||||||
|
for(qpms_m_t m = -l; m <= l; ++m) {
|
||||||
|
complex double eimf = cexp(m * kr.phi * I);
|
||||||
|
if (longtarget) {
|
||||||
|
*plong = csphvec_add(csphvec_scale(besderfac-besfac, *p3),
|
||||||
|
csphvec_scale(sqrtlfac * besfac, *p2));
|
||||||
|
++plong;
|
||||||
|
}
|
||||||
|
if (eltarget) {
|
||||||
|
*pel = csphvec_add(csphvec_scale(besderfac, *p2),
|
||||||
|
csphvec_scale(sqrtlfac * besfac, *p3));
|
||||||
|
++pel;
|
||||||
|
}
|
||||||
|
if (mgtarget) {
|
||||||
|
*pmg = csphvec_scale(*pbes, *p1);
|
||||||
|
++pmg;
|
||||||
|
}
|
||||||
|
++p1; ++p2; ++p3;
|
||||||
|
}
|
||||||
|
++pbes;
|
||||||
|
}
|
||||||
|
free(a1);
|
||||||
|
free(bessel);
|
||||||
|
return QPMS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
qpms_errno_t qpms_vecspharm_fill(csphvec_t *const a1target, csphvec_t *const a2target, csphvec_t *const a3target,
|
||||||
|
qpms_l_t lMax, sph_t dir, qpms_normalisation_t norm) {
|
||||||
|
assert(lMax >= 1);
|
||||||
|
qpms_pitau_t pt = qpms_pitau_get(dir.theta, lMax, norm);
|
||||||
|
double const *pleg = pt.leg;
|
||||||
|
double const *ppi = pt.pi;
|
||||||
|
double const *ptau = pt.tau;
|
||||||
|
csphvec_t *p1 = a1target, *p2 = a2target, *p3 = a3target;
|
||||||
|
for(qpms_l_t = 1; l <= lMax; ++l) {
|
||||||
|
for(qpms_m_t m = -l; m <= l; ++m) {
|
||||||
|
complex double eimf = cexp(m * dir.phi * I);
|
||||||
|
if (a1target) {
|
||||||
|
p1->rc = 0;
|
||||||
|
p1->thetac = *ppi * I * eimf;
|
||||||
|
p1->phic = -*ptau * eimf;
|
||||||
|
++p1;
|
||||||
|
}
|
||||||
|
if (a2target) {
|
||||||
|
p2->rc = 0;
|
||||||
|
p2->thetac = *ptau * eimf;
|
||||||
|
p2->phic = *ppi * I * eimf;
|
||||||
|
++p2;
|
||||||
|
}
|
||||||
|
if (a3target) {
|
||||||
|
p3->rc = sqrt(l*(l+1)) * (*pleg) * eimf;
|
||||||
|
p3->thetac = 0;
|
||||||
|
p3->phic = 0;
|
||||||
|
++p3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
++pleg; ++ppi; ++ptau;
|
||||||
|
}
|
||||||
|
qpms_pitau_free(pt);
|
||||||
|
return QPMS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
qpms_errno_t qpms_vecspharm_dual_fill(csphvec_t *const a1target, csphvec_t *const a2target, csphvec_t *const a3target,
|
||||||
|
qpms_l_t lMax, sph_t dir, qpms_normalisation_t norm) {
|
||||||
|
assert(lMax >= 1);
|
||||||
|
qpms_pitau_t pt = qpms_pitau_get(dir.theta, lMax, norm);
|
||||||
|
double const *pleg = pt.leg;
|
||||||
|
double const *ppi = pt.pi;
|
||||||
|
double const *ptau = pt.tau;
|
||||||
|
csphvec_t *p1 = a1target, *p2 = a2target, *p3 = a3target;
|
||||||
|
for(qpms_l_t = 1; l <= lMax; ++l) {
|
||||||
|
for(qpms_m_t m = -l; m <= l; ++m) {
|
||||||
|
double normfac = 1./qpms_normalisation_t_factor_abssquare(norm, l, m); // factor w.r.t. Kristensson
|
||||||
|
complex double eimf = cexp(m * dir.phi * I);
|
||||||
|
if (a1target) {
|
||||||
|
p1->rc = 0;
|
||||||
|
p1->thetac = conj(*ppi * normfac * I * eimf);
|
||||||
|
p1->phic = conj(-*ptau * normfac * eimf);
|
||||||
|
++p1;
|
||||||
|
}
|
||||||
|
if (a2target) {
|
||||||
|
p2->rc = 0;
|
||||||
|
p2->thetac = conj(*ptau * normfac * eimf);
|
||||||
|
p2->phic = conj(*ppi * normfac * I * eimf);
|
||||||
|
++p2;
|
||||||
|
}
|
||||||
|
if (a3target) {
|
||||||
|
p3->rc = conj(sqrt(l*(l+1)) * (*pleg) * normfac * eimf);
|
||||||
|
p3->thetac = 0;
|
||||||
|
p3->phic = 0;
|
||||||
|
++p3;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
++pleg; ++ppi; ++ptau;
|
||||||
|
}
|
||||||
|
qpms_pitau_free(pt);
|
||||||
|
return QPMS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
qpms_errno_t qpms_planewave2vswf_fill_sph(sph_t wavedir, csphvec_t amplitude,
|
||||||
|
complex double *targt_longcoeff, complex double *target_mgcoeff,
|
||||||
|
complex double *target_elcoeff, qpms_l_t lMax, qpms_normalisation_t norm) {
|
||||||
|
abort(); //NI
|
||||||
|
qpms_y_t nelem = qpms_lMax2nelem(lMax);
|
||||||
|
csphvec_t * const dual_A1 = malloc(3*nelem*sizeof(csphvec_t)), *const dual_A2 = dual_A1 + nelem,
|
||||||
|
* const dual_A3 = dual_A2 + nelem;
|
||||||
|
if (QPMS_SUCCESS != qpms_vecspharm_dual_fill(dual_A1, dual_A2, dual_A3, lMax, wavedir, norm))
|
||||||
|
abort();
|
||||||
|
for (qpms_l_t l = 1; l <= lMax; ++l) {
|
||||||
|
...
|
||||||
|
}
|
||||||
|
|
||||||
|
free(dual_A1);
|
||||||
|
return QPMS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
qpms_errno_t qpms_planewave2vswf_fill_cart(cart3_t wavedir /*allow complex k?*/, ccart3_t amplitude,
|
||||||
|
complex double * const longcoeff, complex double * const mgcoeff,
|
||||||
|
complex double * const elcoeff, qpms_normalisation_t norm)
|
||||||
|
{
|
||||||
|
abort(); //NI
|
||||||
|
|
||||||
|
return QPMS_SUCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
21
qpms/vswf.h
21
qpms/vswf.h
|
@ -31,8 +31,27 @@ qpms_errno_t qpms_legendre_deriv_y_get(double **result, double **result_deriv, d
|
||||||
qpms_errno_t qpms_legendre_deriv_y_fill(double *where, double *where_deriv, double x,
|
qpms_errno_t qpms_legendre_deriv_y_fill(double *where, double *where_deriv, double x,
|
||||||
qpms_l_t lMax, gsl_sf_legendre_t lnorm, double csphase);
|
qpms_l_t lMax, gsl_sf_legendre_t lnorm, double csphase);
|
||||||
|
|
||||||
qpms_errno_t qpms_vswf_fill(csphvec_t *resultM, csphvec_t *resultN, qpms_l_t lMax, sph_t kdrj,
|
/* some of the result targets may be NULL */
|
||||||
|
qpms_errno_t qpms_vswf_fill(csphvec_t *resultL, csphvec_t *resultM, csphvec_t *resultN, qpms_l_t lMax, sph_t kdrj,
|
||||||
qpms_bessel_t btyp, qpms_normalisation_t norm);
|
qpms_bessel_t btyp, qpms_normalisation_t norm);
|
||||||
|
// Should give the same results: for consistency checks
|
||||||
|
qpms_errno_t qpms_vswf_fill_alternative(csphvec_t *resultL, csphvec_t *resultM, csphvec_t *resultN, qpms_l_t lMax, sph_t kdrj,
|
||||||
|
qpms_bessel_t btyp, qpms_normalisation_t norm);
|
||||||
|
|
||||||
|
qpms_errno_t qpms_vecspharm_fill(csphvec_t *const a1target, csphvec_t *const a2target, csphvec_t *const a3target,
|
||||||
|
qpms_l_t lMax, sph_t dir, qpms_normalisation_t norm);
|
||||||
|
qpms_errno_t qpms_vecspharm_dual_fill(csphvec_t *const a1target, csphvec_t *const a2target, csphvec_t *const a3target,
|
||||||
|
qpms_l_t lMax, sph_t dir, qpms_normalisation_t norm);
|
||||||
|
|
||||||
|
qpms_errno_t qpms_planewave2vswf_fill_cart(ccart3_t amplitude, ccart3_t wavedir /* real or complex? */,
|
||||||
|
complex double *targt_longcoeff, complex double *target_mgcoeff, complex double *target_elcoeff,
|
||||||
|
qpms_normalisation_t norm);
|
||||||
|
qpms_errno_t qpms_planewave2vswf_fill_cart(sph_t wavedir, csphvec_t amplitude,
|
||||||
|
complex double *targt_longcoeff, complex double *target_mgcoeff, complex double *target_elcoeff,
|
||||||
|
qpms_normalisation_t norm);
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
qpms_vswfset_sph_t *qpms_vswfset_make(qpms_l_t lMax, sph_t kdlj,
|
qpms_vswfset_sph_t *qpms_vswfset_make(qpms_l_t lMax, sph_t kdlj,
|
||||||
qpms_bessel_t btyp, qpms_normalisation_t norm);//NI
|
qpms_bessel_t btyp, qpms_normalisation_t norm);//NI
|
||||||
|
|
Loading…
Reference in New Issue