aboutsummaryrefslogtreecommitdiff
path: root/libcrystfel/src/geometry.c
diff options
context:
space:
mode:
Diffstat (limited to 'libcrystfel/src/geometry.c')
-rw-r--r--libcrystfel/src/geometry.c86
1 files changed, 80 insertions, 6 deletions
diff --git a/libcrystfel/src/geometry.c b/libcrystfel/src/geometry.c
index 4d25b89f..b99fdd04 100644
--- a/libcrystfel/src/geometry.c
+++ b/libcrystfel/src/geometry.c
@@ -36,6 +36,8 @@
#include <fenv.h>
#include <gsl/gsl_sf_erf.h>
#include <gsl/gsl_linalg.h>
+#include <ctype.h>
+#include <math.h>
#include "utils.h"
#include "cell.h"
@@ -952,7 +954,72 @@ void update_predictions(Crystal *cryst)
}
-void polarisation_correction(RefList *list, UnitCell *cell, struct image *image)
+struct polarisation parse_polarisation(const char *text)
+{
+ struct polarisation p;
+ char angle[14];
+ char deg[14];
+ char frac[14];
+ int i, n;
+
+ if ( strlen(text) > 13 ) goto err;
+
+ if ( strcmp(text, "none") == 0 ) {
+ p.fraction = 0.5;
+ p.angle = 0.0;
+ return p;
+ }
+
+ p.fraction = 1.0;
+
+ i = 0;
+ n = 0;
+ while ( isdigit(text[i]) ) {
+ angle[n++] = text[i++];
+ }
+ angle[n] = '\0';
+
+ n = 0;
+ while ( isalpha(text[i]) ) {
+ deg[n++] = text[i++];
+ }
+ deg[n] = '\0';
+
+ n = 0;
+ while ( isdigit(text[i]) ) {
+ frac[n++] = text[i++];
+ }
+ frac[n] = '\0';
+
+ if ( text[i] != '\0' ) goto err;
+
+ if ( frac[0] != '\0' ) {
+ int nfrac = atoi(frac);
+ if ( nfrac > 100 ) goto err;
+ if ( nfrac < 0 ) goto err;
+ p.fraction = nfrac / 100.0;
+ }
+
+ if ( strcmp(deg, "deg") == 0 ) {
+ p.angle = deg2rad(atoi(angle));
+ } else {
+ if ( angle[0] != '\0' ) goto err;
+ if ( strncmp(deg, "horiz", 5) == 0 ) p.angle = 0.0;
+ if ( strncmp(deg, "vert", 4) == 0 ) p.angle = M_PI_2;
+ }
+
+ return p;
+
+err:
+ ERROR("Invalid polarisation '%s'\n", text);
+ p.fraction = 0.5;
+ p.angle = 0.0;
+ return p;
+}
+
+
+void polarisation_correction(RefList *list, UnitCell *cell, double lambda,
+ struct polarisation p)
{
Reflection *refl;
RefListIterator *iter;
@@ -970,16 +1037,23 @@ void polarisation_correction(RefList *list, UnitCell *cell, struct image *image)
{
double pol;
double intensity;
- double xl, yl;
+ double xl, yl, zl;
+ double phi, tt, kpred;
signed int h, k, l;
- const double P = 1.0; /* degree of polarisation */
+ const double P = p.fraction; /* degree of polarisation */
get_indices(refl, &h, &k, &l);
- xl = (h*asx + k*bsx + l*csx)*image->lambda;
- yl = (h*asy + k*bsy + l*csy)*image->lambda;
+ xl = h*asx + k*bsx + l*csx;
+ yl = h*asy + k*bsy + l*csy;
+ zl = h*asz + k*bsz + l*csz;
+
+ kpred = get_kpred(refl);
+ tt = angle_between(0.0, 0.0, 1.0, xl, yl, zl+kpred);
+ phi = atan2(yl, xl) - p.angle;
- pol = P*(1.0 - xl*xl) + (1.0-P)*(1.0 - yl*yl);
+ pol = P*(1.0 - cos(phi)*cos(phi)*sin(tt)*sin(tt))
+ + (1.0-P)*(1.0 - sin(phi)*sin(phi)*sin(tt)*sin(tt));
intensity = get_intensity(refl);
set_intensity(refl, intensity / pol);