diff options
author | Thomas White <taw@physics.org> | 2018-05-03 14:35:32 +0200 |
---|---|---|
committer | Thomas White <taw@physics.org> | 2018-05-07 10:08:02 +0200 |
commit | d1a2c71235611ad878fa9bc705c6eff1a04b3600 (patch) | |
tree | 5a05d4abcc6b980ec67c0236d1a0c4be1412f539 | |
parent | 5790b06b2e0080c48e1e9a33eb0b43914f2b5824 (diff) |
Simplify scaling
-rw-r--r-- | src/scaling.c | 350 | ||||
-rw-r--r-- | src/scaling.h | 8 | ||||
-rw-r--r-- | tests/linear_scale_check.c | 17 |
3 files changed, 92 insertions, 283 deletions
diff --git a/src/scaling.c b/src/scaling.c index 92f618ae..29b0c2b4 100644 --- a/src/scaling.c +++ b/src/scaling.c @@ -48,195 +48,38 @@ #include "scaling.h" -/* Apply the given shift to the 'k'th parameter of 'image'. */ -static void apply_shift(Crystal *cr, int k, double shift) +struct scale_args { - double t; - - switch ( k ) { - - case GPARAM_BFAC : - t = crystal_get_Bfac(cr); - t += shift; - crystal_set_Bfac(cr, t); - break; - - case GPARAM_OSF : - t = -log(crystal_get_osf(cr)); - t += shift; - crystal_set_osf(cr, exp(-t)); - break; - - default : - ERROR("No shift defined for parameter %i\n", k); - abort(); - - } -} + RefList *full; + Crystal *crystal; +}; -/* Perform one cycle of scaling of 'cr' against 'full' */ -static double scale_iterate(Crystal *cr, const RefList *full, int *nr) +struct queue_args { - gsl_matrix *M; - gsl_vector *v; - gsl_vector *shifts; - int param; - Reflection *refl; - RefListIterator *iter; - RefList *reflections; - double max_shift; - int nref = 0; - int num_params = 0; - enum gparam rv[32]; - double G, B; - - *nr = 0; - - rv[num_params++] = GPARAM_OSF; - rv[num_params++] = GPARAM_BFAC; - - M = gsl_matrix_calloc(num_params, num_params); - v = gsl_vector_calloc(num_params); - - reflections = crystal_get_reflections(cr); - G = crystal_get_osf(cr); - B = crystal_get_Bfac(cr); - - /* Scaling terms */ - for ( refl = first_refl(reflections, &iter); - refl != NULL; - refl = next_refl(refl, iter) ) - { - signed int ha, ka, la; - double I_full, delta_I, esd; - double w; - double I_partial; - int k; - double p, L, s; - double fx; - Reflection *match; - double gradients[num_params]; - - /* If reflection is free-flagged, don't use it here */ - if ( get_flag(refl) ) continue; - - /* Find the full version */ - get_indices(refl, &ha, &ka, &la); - match = find_refl(full, ha, ka, la); - if ( match == NULL ) continue; - - /* Merged intensitty */ - I_full = get_intensity(match); - - /* Actual measurement of this reflection from this pattern */ - I_partial = get_intensity(refl); - esd = get_esd_intensity(refl); - p = get_partiality(refl); - - /* Scale only using strong reflections */ - if ( I_partial <= 3.0*esd ) continue; /* Also because of log */ - if ( get_redundancy(match) < 2 ) continue; - if ( I_full <= 0 ) continue; /* Because log */ - if ( p <= 0.0 ) continue; /* Because of log */ - - L = get_lorentz(refl); - s = resolution(crystal_get_cell(cr), ha, ka, la); - - /* Calculate the weight for this reflection */ - w = 1.0; - - /* Calculate all gradients for this reflection */ - for ( k=0; k<num_params; k++ ) { - - if ( rv[k] == GPARAM_OSF ) { - gradients[k] = 1.0; - } else if ( rv[k] == GPARAM_BFAC ) { - gradients[k] = -s*s; - } else { - ERROR("Unrecognised scaling gradient.\n"); - abort(); - } - } - - for ( k=0; k<num_params; k++ ) { - - int g; - double v_c, v_curr; - - for ( g=0; g<num_params; g++ ) { - - double M_c, M_curr; - - /* Matrix is symmetric */ - if ( g > k ) continue; - - M_c = w * gradients[g] * gradients[k]; - - M_curr = gsl_matrix_get(M, k, g); - gsl_matrix_set(M, k, g, M_curr + M_c); - gsl_matrix_set(M, g, k, M_curr + M_c); - - } - - fx = -log(G) + log(p) - log(L) - B*s*s + log(I_full); - delta_I = log(I_partial) - fx; - v_c = w * delta_I * gradients[k]; - v_curr = gsl_vector_get(v, k); - gsl_vector_set(v, k, v_curr + v_c); - - } - - nref++; - } - - *nr = nref; - - if ( nref < num_params ) { - crystal_set_user_flag(cr, PRFLAG_FEWREFL); - gsl_matrix_free(M); - gsl_vector_free(v); - return 0.0; - } - - max_shift = 0.0; - shifts = solve_svd(v, M, NULL, 0); - if ( shifts != NULL ) { - - for ( param=0; param<num_params; param++ ) { - double shift = gsl_vector_get(shifts, param); - apply_shift(cr, rv[param], shift); - if ( fabs(shift) > max_shift ) max_shift = fabs(shift); - } - - } else { - crystal_set_user_flag(cr, PRFLAG_SOLVEFAIL); - } - - gsl_matrix_free(M); - gsl_vector_free(v); - gsl_vector_free(shifts); - - return max_shift; -} + int n_started; + int n_done; + Crystal **crystals; + int n_crystals; + struct scale_args task_defaults; +}; -static void do_scale_refine(Crystal *cr, const RefList *full, int *nr) +static void scale_crystal(void *task, int id) { + struct scale_args *pargs = task; int i, done; double old_dev; - old_dev = log_residual(cr, full, 0, NULL, NULL); + old_dev = log_residual(pargs->crystal, pargs->full, 0, NULL, NULL); i = 0; done = 0; do { double dev; - - scale_iterate(cr, full, nr); - - dev = log_residual(cr, full, 0, 0, NULL); + scale_one_crystal(pargs->crystal, pargs->full, 0); + dev = log_residual(pargs->crystal, pargs->full, 0, 0, NULL); if ( fabs(dev - old_dev) < dev*0.01 ) done = 1; i++; @@ -246,32 +89,6 @@ static void do_scale_refine(Crystal *cr, const RefList *full, int *nr) } -struct scale_args -{ - RefList *full; - Crystal *crystal; - int n_reflections; -}; - - -struct queue_args -{ - int n_started; - int n_done; - Crystal **crystals; - int n_crystals; - long long int n_reflections; - struct scale_args task_defaults; -}; - - -static void scale_crystal(void *task, int id) -{ - struct scale_args *pargs = task; - do_scale_refine(pargs->crystal, pargs->full, &pargs->n_reflections); -} - - static void *get_crystal(void *vqargs) { struct scale_args *task; @@ -291,11 +108,7 @@ static void *get_crystal(void *vqargs) static void done_crystal(void *vqargs, void *task) { struct queue_args *qa = vqargs; - struct scale_args *wargs = task; - qa->n_done++; - qa->n_reflections += wargs->n_reflections; - progress_bar(qa->n_done, qa->n_crystals, "Scaling"); free(task); } @@ -353,11 +166,8 @@ void scale_all(Crystal **crystals, int n_crystals, int nthreads, int no_Bscale) qargs.task_defaults.full = full; qargs.n_started = 0; qargs.n_done = 0; - qargs.n_reflections = 0; run_threads(nthreads, scale_crystal, get_crystal, done_crystal, &qargs, n_crystals, 0, 0, 0); - STATUS("%lli reflections went into the scaling.\n", - qargs.n_reflections); new_res = total_log_r(crystals, n_crystals, full, &ninc); STATUS("Log residual went from %e to %e, %i crystals\n", @@ -382,13 +192,11 @@ void scale_all(Crystal **crystals, int n_crystals, int nthreads, int no_Bscale) } -/* Calculates G and B, by which list2 should be multiplied to fit list1 */ -int scale_one(const RefList *list1, const RefList *list2, int flags, - double *G, double *B) +/* Calculates G and B, by which cr's reflections should be multiplied to fit reference */ +int scale_one_crystal(Crystal *cr, const RefList *listR, int flags) { int complain_loudly = 0; - const Reflection *refl1; - const Reflection *refl2; + const Reflection *reflS; RefListIterator *iter; int max_n = 256; int n = 0; @@ -396,20 +204,24 @@ int scale_one(const RefList *list1, const RefList *list2, int flags, double *y; double *w; int r; - double cov11; - double sumsq; - int n_esd1 = 0; - int n_esd2 = 0; - int n_ih1 = 0; - int n_ih2 = 0; - int n_nan1 = 0; - int n_nan2 = 0; - int n_inf1 = 0; - int n_inf2 = 0; + double cov00, cov01, cov11, chisq; + int n_esdS = 0; + int n_esdR = 0; + int n_ihS = 0; + int n_ihR = 0; + int n_nanS = 0; + int n_nanR = 0; + int n_infS = 0; + int n_infR = 0; int n_part = 0; int n_nom = 0; + RefList *listS = crystal_get_reflections(cr); + UnitCell *cell = crystal_get_cell(cr); + double G, B; - *B = 0.0; /* FIXME */ + assert(cell != NULL); + assert(listR != NULL); + assert(listS != NULL); x = malloc(max_n*sizeof(double)); w = malloc(max_n*sizeof(double)); @@ -420,38 +232,40 @@ int scale_one(const RefList *list1, const RefList *list2, int flags, } int nb = 0; - for ( refl2 = first_refl_const(list2, &iter); - refl2 != NULL; - refl2 = next_refl_const(refl2, iter) ) + for ( reflS = first_refl_const(listS, &iter); + reflS != NULL; + reflS = next_refl_const(reflS, iter) ) { signed int h, k, l; - double Ih1, Ih2; - double esd1, esd2; + const Reflection *reflR; + double IhR, IhS, esdS, pS, LS; + double s; + nb++; - get_indices(refl2, &h, &k, &l); - refl1 = find_refl(list1, h, k, l); - if ( refl1 == NULL ) { + get_indices(reflS, &h, &k, &l); + reflR = find_refl(listR, h, k, l); + if ( reflR == NULL ) { n_nom++; continue; } - Ih1 = get_intensity(refl1); - Ih2 = get_intensity(refl2); - esd1 = get_esd_intensity(refl1); - esd2 = get_esd_intensity(refl2); + s = resolution(cell, h, k, l); + + IhR = get_intensity(reflR); + IhS = get_intensity(reflS); + esdS = get_esd_intensity(reflS); + pS = get_partiality(reflS); + LS = get_lorentz(reflS); /* Problem cases in approximate descending order of severity */ - if ( isnan(Ih1) ) { n_nan1++; continue; } - if ( isinf(Ih1) ) { n_inf1++; continue; } - if ( isnan(Ih2) ) { n_nan2++; continue; } - if ( isinf(Ih2) ) { n_inf2++; continue; } - if ( get_partiality(refl2) < 0.3 ) { n_part++; continue; } - //if ( Ih1 <= 0.0 ) { n_ih1++; continue; } - if ( Ih2 <= 0.0 ) { n_ih2++; continue; } - //if ( Ih1 <= 3.0*esd1 ) { n_esd1++; continue; } - if ( Ih2 <= 3.0*esd2 ) { n_esd2++; continue; } - //if ( get_redundancy(refl1) < 2 ) continue; + if ( isnan(IhR) ) { n_nanR++; continue; } + if ( isinf(IhR) ) { n_infR++; continue; } + if ( isnan(IhS) ) { n_nanS++; continue; } + if ( isinf(IhS) ) { n_infS++; continue; } + if ( pS < 0.3 ) { n_part++; continue; } + if ( IhS <= 0.0 ) { n_ihS++; continue; } + if ( IhS <= 3.0*esdS ) { n_esdS++; continue; } if ( n == max_n ) { max_n *= 2; @@ -464,9 +278,9 @@ int scale_one(const RefList *list1, const RefList *list2, int flags, } } - x[n] = Ih2 / get_partiality(refl2); - y[n] = Ih1; - w[n] = get_partiality(refl2); + x[n] = s*s; + y[n] = log(IhR) - log(pS) + log(LS) - log(IhS); + w[n] = pS; n++; } @@ -474,62 +288,48 @@ int scale_one(const RefList *list1, const RefList *list2, int flags, if ( n < 2 ) { if ( complain_loudly ) { ERROR("Not enough reflections for scaling (had %i, but %i remain)\n", nb, n); - if ( n_esd1 ) ERROR("%i reference reflection esd\n", n_esd1); - if ( n_esd2 ) ERROR("%i subject reflection esd\n", n_esd2); - if ( n_ih1 ) ERROR("%i reference reflection intensity\n", n_ih1); - if ( n_ih2 ) ERROR("%i subject reflection intensity\n", n_ih2); - if ( n_nan1 ) ERROR("%i reference reflection nan\n", n_nan1); - if ( n_nan2 ) ERROR("%i subject reflection nan\n", n_nan2); - if ( n_inf1 ) ERROR("%i reference reflection inf\n", n_inf1); - if ( n_inf2 ) ERROR("%i subject reflection inf\n", n_inf2); + if ( n_esdR ) ERROR("%i reference reflection esd\n", n_esdR); + if ( n_esdS ) ERROR("%i subject reflection esd\n", n_esdS); + if ( n_ihR ) ERROR("%i reference reflection intensity\n", n_ihR); + if ( n_ihS ) ERROR("%i subject reflection intensity\n", n_ihS); + if ( n_nanR ) ERROR("%i reference reflection nan\n", n_nanR); + if ( n_nanS ) ERROR("%i subject reflection nan\n", n_nanS); + if ( n_infR ) ERROR("%i reference reflection inf\n", n_infR); + if ( n_infS ) ERROR("%i subject reflection inf\n", n_infS); if ( n_part ) ERROR("%i subject reflection partiality\n", n_part); if ( n_nom ) ERROR("%i no match in reference list\n", n_nom); } - *G = 1.0; return 1; } - r = gsl_fit_wmul(x, 1, w, 1, y, 1, n, G, &cov11, &sumsq); + r = gsl_fit_wlinear(x, 1, w, 1, y, 1, n, &G, &B, &cov00, &cov01, &cov11, &chisq); if ( r ) { ERROR("Scaling failed.\n"); - *G = 1.0; return 1; } - if ( isnan(*G) ) { + if ( isnan(G) ) { if ( complain_loudly ) { ERROR("Scaling gave NaN (%i pairs)\n", n); if ( n < 10 ) { int i; for ( i=0; i<n; i++ ) { - STATUS("%i %e %e %e\n", i, x[i], y[i], w[n]); + STATUS("%3i %e %e %e\n", i, x[i], y[i], w[i]); } } } - *G = 1.0; return 1; } + crystal_set_osf(cr, exp(G)); + crystal_set_Bfac(cr, B); + free(x); free(y); free(w); return 0; } - - -int scale_one_crystal(Crystal *cr, const RefList *list2, int flags) -{ - double G, B; - int r; - - r = scale_one(crystal_get_reflections(cr), list2, flags, &G, &B); - if ( r ) return r; - - crystal_set_osf(cr, G); - crystal_set_Bfac(cr, B); - return 0; -} diff --git a/src/scaling.h b/src/scaling.h index b6958b6c..a4ec6931 100644 --- a/src/scaling.h +++ b/src/scaling.h @@ -40,13 +40,11 @@ enum ScaleFlags { - SCALE_NO_B, /* Don't use Debye-Waller part */ + SCALE_NONE = 0, + SCALE_NO_B = 1<<0, /* Don't use Debye-Waller part */ }; -extern int scale_one(const RefList *list1, const RefList *list2, int flags, - double *G, double *B); - -extern int scale_one_crystal(Crystal *cr, const RefList *list2, int flags); +extern int scale_one_crystal(Crystal *cr, const RefList *reference, int flags); extern void scale_all(Crystal **crystals, int n_crystals, int nthreads, int flags); diff --git a/tests/linear_scale_check.c b/tests/linear_scale_check.c index 5c723849..439cff36 100644 --- a/tests/linear_scale_check.c +++ b/tests/linear_scale_check.c @@ -43,10 +43,11 @@ int main(int argc, char *argv[]) int fail = 0; int i; gsl_rng *rng; + Crystal *cr; RefList *list1; RefList *list2; - double G, B; int r; + UnitCell *cell; list1 = reflist_new(); list2 = reflist_new(); @@ -66,12 +67,22 @@ int main(int argc, char *argv[]) intens = gsl_rng_uniform(rng); /* [0,1) */ set_intensity(refl1, intens); set_partiality(refl1, 1.0); + set_lorentz(refl1, 1.0); set_intensity(refl2, intens*2.0); set_partiality(refl2, 1.0); + set_lorentz(refl2, 1.0); } - r = scale_one(list1, list2, SCALE_NO_B, &G, &B); - STATUS("Scaling result: %i, G = %f\n", r, G); + cr = crystal_new(); + cell = cell_new(); + cell_set_parameters(cell, 50e-10, 50e-10, 50e-10, + deg2rad(90), deg2rad(90), deg2rad(90)); + crystal_set_reflections(cr, list1); + crystal_set_cell(cr, cell); + + r = scale_one_crystal(cr, list2, SCALE_NO_B); + STATUS("Scaling result: %i, G = %f, B = %e\n", r, + crystal_get_osf(cr), crystal_get_Bfac(cr)); return fail; } |