diff options
Diffstat (limited to 'src/diffraction.c')
-rw-r--r-- | src/diffraction.c | 105 |
1 files changed, 93 insertions, 12 deletions
diff --git a/src/diffraction.c b/src/diffraction.c index a4a03b64..e02d80f4 100644 --- a/src/diffraction.c +++ b/src/diffraction.c @@ -20,6 +20,7 @@ #include "utils.h" #include "cell.h" #include "ewald.h" +#include "diffraction.h" static double lattice_factor(struct threevec q, double ax, double ay, double az, @@ -28,31 +29,28 @@ static double lattice_factor(struct threevec q, double ax, double ay, double az, { struct threevec Udotq; double f1, f2, f3; - int na = 64; - int nb = 64; - int nc = 64; + int na = 8; + int nb = 8; + int nc = 8; Udotq.u = (ax*q.u + ay*q.v + az*q.w)/2.0; Udotq.v = (bx*q.u + by*q.v + bz*q.w)/2.0; Udotq.w = (cx*q.u + cy*q.v + cz*q.w)/2.0; if ( na > 1 ) { - f1 = sin(2.0*M_PI*(double)na*Udotq.u) - / sin(2.0*M_PI*Udotq.u); + f1 = sin(2.0*M_PI*(double)na*Udotq.u) / sin(2.0*M_PI*Udotq.u); } else { f1 = 1.0; } if ( nb > 1 ) { - f2 = sin(2.0*M_PI*(double)nb*Udotq.v) - / sin(2.0*M_PI*Udotq.v); + f2 = sin(2.0*M_PI*(double)nb*Udotq.v) / sin(2.0*M_PI*Udotq.v); } else { f2 = 1.0; } if ( nc > 1 ) { - f3 = sin(2.0*M_PI*(double)nc*Udotq.w) - / sin(2.0*M_PI*Udotq.w); + f3 = sin(2.0*M_PI*(double)nc*Udotq.w) / sin(2.0*M_PI*Udotq.w); } else { f3 = 1.0; } @@ -62,7 +60,7 @@ static double lattice_factor(struct threevec q, double ax, double ay, double az, /* Look up f1 and f2 for this atom at this energy (in J/photon) */ -static double complex get_sfac(const char *n, double en) +static double complex get_f1f2(const char *n, double en) { FILE *fh; char filename[64]; @@ -98,12 +96,15 @@ static double complex get_sfac(const char *n, double en) abort(); } + /* Find the first energy greater than the required value */ if ( E < en ) { + /* Store old values ready for interpolation*/ last_E = E; last_f1 = f1; last_f2 = f2; } else { + /* Perform (linear) interpolation */ float f; float actual_f1, actual_f2; @@ -128,6 +129,79 @@ static double complex get_sfac(const char *n, double en) } +/* s = sin(theta)/lambda */ +static double get_waas_kirf(const char *n, double s) +{ + FILE *fh; + char *rval; + double f; + float a1, a2, a3, a4, a5, c, b1, b2, b3, b4, b5; + double s2; + + fh = fopen("scattering-factors/f0_WaasKirf.dat", "r"); + if ( fh == NULL ) { + fprintf(stderr, "Couldn't open f0_WaasKirf.dat\n"); + return 0.0; + } + + do { + + int r; + char line[1024]; + char sp[1024]; + int Z; + + rval = fgets(line, 1023, fh); + + if ( (line[0] != '#') || (line[1] != 'S') ) continue; + + r = sscanf(line, "#S %i %s", &Z, sp); + if ( (r != 2) || (strcmp(sp, n) != 0) ) continue; + + /* Skip two lines */ + fgets(line, 1023, fh); + fgets(line, 1023, fh); + + /* Read scattering coefficients */ + rval = fgets(line, 1023, fh); + r = sscanf(line, " %f %f %f %f %f %f %f %f %f %f %f", + &a1, &a2, &a3, &a4, &a5, &c, &b1, &b2, &b3, &b4, &b5); + if ( r != 11 ) { + fprintf(stderr, "Couldn't read scattering factors\n"); + return 0.0; + } + + break; + + } while ( rval != NULL ); + + fclose(fh); + + s2 = pow(s/1e10, 2.0); /* s2 is s squared in Angstroms squared */ + f = c + a1*exp(-b1*s2) + a2*exp(-b2*s2) + a3*exp(-b3*s2) + + a4*exp(-b4*s2) + a5*exp(-b5*s2); + + return f; +} + + +/* Get complex scattering factors for element 'n' at energy 'en' (J/photon), + * at resolution 's' = sin(theta)/lambda (in m^-1) */ +static double complex get_sfac(const char *n, double s, double en) +{ + double complex f1f2; + double fq, fq0; + + f1f2 = get_f1f2(n, en); + fq = get_waas_kirf(n, s); + fq0 = get_waas_kirf(n, 0.0); + + return fq - fq0 + f1f2; +} + + +/* Return structure factor for molecule 'mol' at energy en' (J/photon) at + * scattering vector 'q' */ static double complex molecule_factor(struct molecule *mol, struct threevec q, double en) { @@ -135,7 +209,8 @@ static double complex molecule_factor(struct molecule *mol, struct threevec q, double F = 0.0; double s; - s = modulus(q.u, q.v, q.w); + /* s = sin(theta)/lambda = 1/2d = (1/d)/2.0 */ + s = modulus(q.u, q.v, q.w) / 2.0; for ( i=0; i<mol->n_species; i++ ) { @@ -151,17 +226,23 @@ static double complex molecule_factor(struct molecule *mol, struct threevec q, double ph; ph= q.u*spec->x[j] + q.v*spec->y[j] + q.w*spec->z[j]; + + /* Conversion from revolutions to radians is required */ contrib += cos(2.0*M_PI*ph) + I*sin(2.0*M_PI*ph); + } - sfac = get_sfac(spec->species, en); + sfac = get_sfac(spec->species, s, en); F += sfac * contrib * exp(-2.0 * spec->B[j] * s); + } return F; } +/* Load PDB file into a memory format suitable for efficient(ish) structure + * factor calculation */ static struct molecule *load_molecule() { struct molecule *mol; |