aboutsummaryrefslogtreecommitdiff
path: root/libcrystfel
diff options
context:
space:
mode:
Diffstat (limited to 'libcrystfel')
-rw-r--r--libcrystfel/src/integration.c259
1 files changed, 181 insertions, 78 deletions
diff --git a/libcrystfel/src/integration.c b/libcrystfel/src/integration.c
index e8b50930..304c41ae 100644
--- a/libcrystfel/src/integration.c
+++ b/libcrystfel/src/integration.c
@@ -165,6 +165,11 @@ struct intcontext
double **reference_profiles;
double **reference_den;
int *n_profiles_in_reference;
+
+ /* For summation integration only */
+ int ir_inn;
+ int ir_mid;
+ int ir_out;
};
@@ -466,6 +471,47 @@ static int init_intcontext(struct intcontext *ic)
}
+static void setup_ring_masks(struct intcontext *ic,
+ double ir_inn, double ir_mid, double ir_out)
+{
+ double lim_sq, out_lim_sq, mid_lim_sq;
+ int p, q;
+
+ lim_sq = pow(ir_inn, 2.0);
+ mid_lim_sq = pow(ir_mid, 2.0);
+ out_lim_sq = pow(ir_out, 2.0);
+
+ for ( p=0; p<ic->w; p++ ) {
+ for ( q=0; q<ic->w; q++ ) {
+
+ int rsq;
+
+ rsq = (p-ic->halfw)*(p-ic->halfw) + (q-ic->halfw)*(q-ic->halfw);
+
+ if ( rsq > out_lim_sq ) {
+ /* Outside outer radius */
+ ic->bm[p + ic->w*q] = BM_IG;
+ } else {
+
+ if ( rsq >= mid_lim_sq ) {
+ /* Inside outer radius, outside middle radius */
+ ic->bm[p + ic->w*q] = BM_BG;
+ } else if ( rsq <= lim_sq ) {
+ /* Inside inner radius */
+ ic->bm[p + ic->w*q] = BM_PK;
+ } else {
+ /* Outside inner radius, inside middle radius */
+ ic->bm[p + ic->w*q] = BM_IG;
+ }
+
+ }
+
+ }
+ }
+
+}
+
+
static struct peak_box *add_box(struct intcontext *ic)
{
int idx;
@@ -1165,7 +1211,7 @@ static void estimate_resolution(RefList *reflections, Crystal *cr,
static void integrate_refine(Crystal *cr, struct image *image, int use_closer,
double min_snr,
double ir_inn, double ir_mid, double ir_out,
- int integrate_saturated, int **bgMasks)
+ int integrate_saturated)
{
RefList *reflections;
UnitCell *cell;
@@ -1197,15 +1243,91 @@ static void integrate_refine(Crystal *cr, struct image *image, int use_closer,
}
-static void integrate_rings(Crystal *cr, struct image *image, int use_closer,
- double min_snr,
+static void integrate_box(struct intcontext *ic, struct peak_box *bx,
+ double *intensity, double *sigma, int *saturated)
+{
+ double pk_total;
+ int pk_counts;
+ double fsct, ssct;
+ double bg_tot;
+ double bg_tot_sq;
+ int bg_counts;
+ double bg_mean, bg_var;
+ double var;
+ double aduph;
+ int p, q;
+
+ if ( saturated != NULL ) *saturated = 0;
+
+ aduph = bx->p->adu_per_eV * ph_lambda_to_eV(ic->image->lambda);
+
+ /* Measure the background */
+ bg_tot = 0.0;
+ bg_tot_sq = 0.0;
+ bg_counts = 0;
+ for ( p=0; p<ic->w; p++ ) {
+ for ( q=0; q<ic->w; q++ ) {
+
+ double bi = boxi(ic, bx, p, q);
+ if ( bx->bm[p + ic->w*q] != BM_BG ) continue;
+
+ if ( (saturated != NULL) && (bi > bx->p->max_adu) ) {
+ *saturated = 1;
+ }
+
+ bg_tot += bi;
+ bg_tot_sq += pow(bi, 2.0);
+ bg_counts++;
+
+ }
+ }
+
+ bg_mean = bg_tot / bg_counts;
+ bg_var = (bg_tot_sq/bg_counts) - pow(bg_mean, 2.0);
+
+ /* Measure the peak */
+ pk_total = 0.0;
+ pk_counts = 0;
+ fsct = 0.0; ssct = 0.0;
+
+ for ( p=0; p<ic->w; p++ ) {
+ for ( q=0; q<ic->w; q++ ) {
+
+ double bi = boxi(ic, bx, p, q);
+ if ( bx->bm[p + ic->w*q] != BM_PK ) continue;
+
+ if ( (saturated != NULL) && (bi > bx->p->max_adu) ) {
+ *saturated = 1;
+ }
+
+ pk_counts++;
+ pk_total += (bi - bg_mean);
+ fsct += (bi-bg_mean)*p;
+ ssct += (bi-bg_mean)*q;
+
+ }
+ }
+
+// *pfs = ((double)fsct / pk_total) + 0.5;
+// *pss = ((double)ssct / pk_total) + 0.5;
+
+ var = pk_counts * bg_var;
+ var += aduph * pk_total;
+
+ if ( intensity != NULL ) *intensity = pk_total;
+ if ( sigma != NULL ) *sigma = sqrt(var);
+}
+
+
+static void integrate_rings(Crystal *cr, struct image *image, double min_snr,
double ir_inn, double ir_mid, double ir_out,
- int integrate_saturated, int **bgMasks)
+ int integrate_saturated)
{
RefList *list;
Reflection *refl;
RefListIterator *iter;
UnitCell *cell;
+ struct intcontext ic;
int n_saturated = 0;
double limit = 0.0;
@@ -1216,73 +1338,76 @@ static void integrate_rings(Crystal *cr, struct image *image, int use_closer,
cell = crystal_get_cell(cr);
+ ic.halfw = ir_out;
+ ic.image = image;
+ ic.k = 1.0/image->lambda;
+ ic.cell = cell;
+ ic.ir_inn = ir_inn;
+ ic.ir_mid = ir_mid;
+ ic.ir_out = ir_out;
+ if ( init_intcontext(&ic) ) {
+ ERROR("Failed to initialise integration.\n");
+ return;
+ }
+ setup_ring_masks(&ic, ir_inn, ir_mid, ir_out);
+
for ( refl = first_refl(list, &iter);
refl != NULL;
refl = next_refl(refl, iter) )
{
- double fs, ss, intensity;
- double d;
- int idx;
- double sigma, snr;
double pfs, pss;
- int r;
signed int h, k, l;
+ struct peak_box *bx;
+ int pn;
struct panel *p;
- int pnum, j, found;
+ int fid_fs, fid_ss; /* Center coordinates, rounded,
+ * in overall data block */
+ int cfs, css; /* Corner coordinates */
+ double intensity;
+ double sigma, snr;
int saturated;
double one_over_d;
+ int r;
- get_detector_pos(refl, &pfs, &pss);
- get_indices(refl, &h, &k, &l);
-
- /* Is there a really close feature which was detected? */
- if ( use_closer ) {
-
- struct imagefeature *f;
-
- if ( image->features != NULL ) {
- f = image_feature_closest(image->features,
- pfs, pss, &d, &idx);
- } else {
- f = NULL;
- }
-
- /* FIXME: Horrible hardcoded value */
- if ( (f != NULL) && (d < 10.0) ) {
-
- double exe;
+ set_redundancy(refl, 0);
- exe = get_excitation_error(refl);
+ get_detector_pos(refl, &pfs, &pss);
+ fid_fs = lrint(pfs);
+ fid_ss = lrint(pss);
+ pn = find_panel_number(image->det, fid_fs, fid_ss);
+ p = &image->det->panels[pn];
- pfs = f->fs;
- pss = f->ss;
+ cfs = (fid_fs-p->min_fs) - ic.halfw;
+ css = (fid_ss-p->min_ss) - ic.halfw;
- set_detector_pos(refl, exe, pfs, pss);
+ if ( (cfs + ic.w >= p->w) || (css + ic.w >= p->h ) ) {
+ continue;
+ }
+ if ( (cfs < 0) || (css < 0 ) ) continue;
- }
+ bx = add_box(&ic);
+ bx->refl = refl;
+ bx->cfs = cfs;
+ bx->css = css;
+ bx->p = p;
+ bx->pn = pn;
+ if ( check_box(&ic, bx) ) {
+ delete_box(&ic, bx);
+ continue;
}
- p = find_panel(image->det, pfs, pss);
- if ( p == NULL ) continue; /* Next peak */
- found = 0;
- for ( j=0; j<image->det->n_panels; j++ ) {
- if ( &image->det->panels[j] == p ) {
- pnum = j;
- found = 1;
- break;
- }
- }
- if ( !found ) {
- ERROR("Couldn't find panel %p in list.\n", p);
- return;
- }
+ get_indices(refl, &h, &k, &l);
+// if ( (h==-1) && (k==14) && (l==13) ) {
+// bx->verbose = 1;
+// }
- r = integrate_peak(image, pfs, pss, &fs, &ss,
- &intensity, &sigma, ir_inn, ir_mid, ir_out,
- bgMasks[pnum], &saturated);
+ if ( bx->verbose ) show_peak_box(&ic, bx);
- if ( !r && saturated ) {
+ integrate_box(&ic, bx, &intensity, &sigma, &saturated);
+
+ r = 0;
+ if ( saturated ) {
n_saturated++;
if ( !integrate_saturated ) r = 1;
}
@@ -1322,23 +1447,6 @@ void integrate_all(struct image *image, IntegrationMethod meth,
int integrate_saturated)
{
int i;
- int **bgMasks;
-
- /* Make background masks for all panels */
- bgMasks = calloc(image->det->n_panels, sizeof(int *));
- if ( bgMasks == NULL ) {
- ERROR("Couldn't create list of background masks.\n");
- return;
- }
- for ( i=0; i<image->det->n_panels; i++ ) {
- int *mask;
- mask = make_BgMask(image, &image->det->panels[i], ir_inn);
- if ( mask == NULL ) {
- ERROR("Couldn't create background mask.\n");
- return;
- }
- bgMasks[i] = mask;
- }
for ( i=0; i<image->n_crystals; i++ ) {
@@ -1348,15 +1456,15 @@ void integrate_all(struct image *image, IntegrationMethod meth,
return;
case INTEGRATION_RINGS :
- integrate_rings(image->crystals[i], image, use_closer,
- min_snr, ir_inn, ir_mid, ir_out,
- integrate_saturated, bgMasks);
+ integrate_rings(image->crystals[i], image, min_snr,
+ ir_inn, ir_mid, ir_out,
+ integrate_saturated);
return;
case INTEGRATION_REFINE :
integrate_refine(image->crystals[i], image, use_closer,
min_snr, ir_inn, ir_mid, ir_out,
- integrate_saturated, bgMasks);
+ integrate_saturated);
return;
default :
@@ -1366,11 +1474,6 @@ void integrate_all(struct image *image, IntegrationMethod meth,
}
}
-
- for ( i=0; i<image->det->n_panels; i++ ) {
- free(bgMasks[i]);
- }
- free(bgMasks);
}