diff options
author | Thomas White <taw@physics.org> | 2013-07-29 14:10:52 +0200 |
---|---|---|
committer | Thomas White <taw@physics.org> | 2013-07-31 17:09:35 +0200 |
commit | a8b646119996c297fbdb7e044c776d7bf4fef21a (patch) | |
tree | 46a1fe51cec3828f495d9a811580d72ab36ae2c7 /src/hrs-scaling.c | |
parent | fc488f0bb181eb0969d67caa4efea2c773717d5b (diff) |
partialator: New convergence criterion for scaling
Instead of stopping iteration when the absolute value of any scaling factor changes by more
than a certain (small) amount, calculate the ratios of the scaling factors to their
previous values, and stop when no scaling factor changes by more than 1% compared to the
mean ratio.
This method is robust against "drifting" of the scale factors when the partiality
estimates are poor.
Diffstat (limited to 'src/hrs-scaling.c')
-rw-r--r-- | src/hrs-scaling.c | 52 |
1 files changed, 44 insertions, 8 deletions
diff --git a/src/hrs-scaling.c b/src/hrs-scaling.c index 47c4301e..8f0cd8f5 100644 --- a/src/hrs-scaling.c +++ b/src/hrs-scaling.c @@ -486,6 +486,32 @@ static void calculate_esds(Crystal **crystals, int n, RefList *full, } +static double max_outlier_change(Crystal **crystals, int n, double *old_osfs) +{ + double rdtot = 0.0; + double rdmax = 0.0; + double rdmean; + int j; + + for ( j=0; j<n; j++ ) { + double r; + r = crystal_get_osf(crystals[j]) / old_osfs[j]; + rdtot += r; + } + rdmean = rdtot / n; + for ( j=0; j<n; j++ ) { + double r; + r = crystal_get_osf(crystals[j]) / old_osfs[j]; + old_osfs[j] = crystal_get_osf(crystals[j]); + if ( fabs(r-rdmean) > rdmax ) { + rdmax = fabs(r-rdmean); + } + } + + return rdmax; +} + + /* Scale the stack of images */ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, int n_threads, int noscale, PartialityModel pmodel, @@ -494,6 +520,11 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, int i; double max_corr; RefList *full = NULL; + double *old_osfs; + double rdval; + + old_osfs = malloc(n*sizeof(double)); + if ( old_osfs == NULL ) return NULL; for ( i=0; i<n; i++ ) crystal_set_osf(crystals[i], 1.0); @@ -501,6 +532,7 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, full = lsq_intensities(crystals, n, n_threads, pmodel); calculate_esds(crystals, n, full, n_threads, min_redundancy, pmodel); + free(old_osfs); return full; } @@ -509,11 +541,14 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, full = lsq_intensities(crystals, n, n_threads, pmodel); } + for ( i=0; i<n; i++ ) { + old_osfs[i] = crystal_get_osf(crystals[i]); + } + /* Iterate */ i = 0; do { - int j; RefList *reference; /* Refine against reference or current "full" estimates */ @@ -525,12 +560,12 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, max_corr = iterate_scale(crystals, n, reference, n_threads, pmodel); - STATUS("Scaling iteration %2i: max correction = %5.2f\n", - i+1, max_corr); - for ( j=0; j<10; j++ ) { - printf(" %5.2f", crystal_get_osf(crystals[j])); - } - printf("\n"); + + rdval = max_outlier_change(crystals, n, old_osfs); + + STATUS("Scaling iteration %2i: max correction = %5.2f " + "dev of worst outlier = %5.2f\n", + i+1, max_corr, rdval); /* No reference -> generate list for next iteration */ if ( gref == NULL ) { @@ -540,7 +575,7 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, i++; - } while ( (max_corr > 0.01) && (i < MAX_CYCLES) ); + } while ( (rdval > 0.01) && (i < MAX_CYCLES) ); if ( i == MAX_CYCLES ) { ERROR("Warning: Scaling did not converge.\n"); @@ -552,5 +587,6 @@ RefList *scale_intensities(Crystal **crystals, int n, RefList *gref, calculate_esds(crystals, n, full, n_threads, min_redundancy, pmodel); + free(old_osfs); return full; } |