diff options
-rw-r--r-- | Makefile.am | 5 | ||||
-rw-r--r-- | src/predict-refine.c | 430 | ||||
-rw-r--r-- | src/predict-refine.h | 44 | ||||
-rw-r--r-- | src/process_image.c | 60 |
4 files changed, 519 insertions, 20 deletions
diff --git a/Makefile.am b/Makefile.am index 10aafcbf..73d9a89e 100644 --- a/Makefile.am +++ b/Makefile.am @@ -71,7 +71,8 @@ src_process_hkl_SOURCES = src/process_hkl.c src_list_events_SOURCES = src/list_events.c -src_indexamajig_SOURCES = src/indexamajig.c src/im-sandbox.c src/process_image.c +src_indexamajig_SOURCES = src/indexamajig.c src/im-sandbox.c \ + src/process_image.c src/predict-refine.c if BUILD_HDFSEE src_hdfsee_SOURCES = src/hdfsee.c src/dw-hdfsee.c src/hdfsee-render.c @@ -136,7 +137,7 @@ EXTRA_DIST += src/dw-hdfsee.h src/hdfsee.h src/render_hkl.h \ src/cl-utils.h src/hdfsee-render.h src/diffraction.h \ src/diffraction-gpu.h src/pattern_sim.h src/list_tmp.h \ src/im-sandbox.h src/process_image.h src/multihistogram.h \ - src/rejection.h + src/rejection.h src/predict-refine.h crystfeldir = $(datadir)/crystfel crystfel_DATA = data/diffraction.cl data/hdfsee.ui diff --git a/src/predict-refine.c b/src/predict-refine.c new file mode 100644 index 00000000..5d0c37db --- /dev/null +++ b/src/predict-refine.c @@ -0,0 +1,430 @@ +/* + * predict-refine.c + * + * Prediction refinement + * + * Copyright © 2012-2015 Deutsches Elektronen-Synchrotron DESY, + * a research centre of the Helmholtz Association. + * + * Authors: + * 2010-2015 Thomas White <taw@physics.org> + * + * This file is part of CrystFEL. + * + * CrystFEL is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * CrystFEL is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with CrystFEL. If not, see <http://www.gnu.org/licenses/>. + * + */ + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + + +#include <stdlib.h> +#include <assert.h> +#include <gsl/gsl_matrix.h> +#include <gsl/gsl_vector.h> + +#include "image.h" +#include "post-refinement.h" +#include "cell-utils.h" + + +/* Maximum number of iterations of NLSq to do for each image per macrocycle. */ +#define MAX_CYCLES (1) + +/* Weighting of excitation error term (m^-1) compared to position term (m) */ +#define EXC_WEIGHT (2e-11) + +struct reflpeak { + Reflection *refl; + struct imagefeature *peak; + double Ih; /* normalised */ +}; + +static int pair_peaks(ImageFeatureList *flist, UnitCell *cell, RefList *reflist, + struct reflpeak *rps) +{ + int i; + const double min_dist = 0.25; + int n_acc = 0; + int n_notintegrated = 0; + + for ( i=0; i<image_feature_count(flist); i++ ) { + + struct imagefeature *f; + double h, k, l, hd, kd, ld; + + /* Assume all image "features" are genuine peaks */ + f = image_get_feature(flist, i); + if ( f == NULL ) continue; + + double ax, ay, az; + double bx, by, bz; + double cx, cy, cz; + + cell_get_cartesian(cell, + &ax, &ay, &az, &bx, &by, &bz, &cx, &cy, &cz); + + /* Decimal and fractional Miller indices of nearest + * reciprocal lattice point */ + hd = f->rx * ax + f->ry * ay + f->rz * az; + kd = f->rx * bx + f->ry * by + f->rz * bz; + ld = f->rx * cx + f->ry * cy + f->rz * cz; + h = lrint(hd); + k = lrint(kd); + l = lrint(ld); + + /* Check distance */ + if ( (fabs(h - hd) < min_dist) + && (fabs(k - kd) < min_dist) + && (fabs(l - ld) < min_dist) ) + { + Reflection *refl; + + /* Dig out the reflection */ + refl = find_refl(reflist, h, k, l); + if ( refl == NULL ) { + n_notintegrated++; + continue; + } + + rps[n_acc].refl = refl; + rps[n_acc].peak = f; + n_acc++; + } + + } + + return n_acc; +} + + +static void twod_mapping(double fs, double ss, double *px, double *py, + struct detector *det) +{ + double xs, ys; + struct panel *p = find_panel(det, fs, ss); + + xs = fs*p->fsx + ss*p->ssx; + ys = fs*p->fsy + ss*p->ssy; + + *px = xs + p->cnx; + *py = ys + p->cny; +} + + +static double r_gradient(UnitCell *cell, int k, Reflection *refl, + struct image *image) +{ + double azi; + double asx, asy, asz; + double bsx, bsy, bsz; + double csx, csy, csz; + double xl, yl, zl; + signed int hs, ks, ls; + double rlow, rhigh, p; + double philow, phihigh, phi; + double khigh, klow; + double tl, cet, cez; + + get_partial(refl, &rlow, &rhigh, &p); + + get_symmetric_indices(refl, &hs, &ks, &ls); + + cell_get_reciprocal(cell, &asx, &asy, &asz, + &bsx, &bsy, &bsz, + &csx, &csy, &csz); + xl = hs*asx + ks*bsx + ls*csx; + yl = hs*asy + ks*bsy + ls*csy; + zl = hs*asz + ks*bsz + ls*csz; + + /* "low" gives the largest Ewald sphere (wavelength short => k large) + * "high" gives the smallest Ewald sphere (wavelength long => k small) + */ + klow = 1.0/(image->lambda - image->lambda*image->bw/2.0); + khigh = 1.0/(image->lambda + image->lambda*image->bw/2.0); + + tl = sqrt(xl*xl + yl*yl); + + cet = -sin(image->div/2.0) * klow; + cez = -cos(image->div/2.0) * klow; + philow = angle_between_2d(tl-cet, zl-cez, 0.0, 1.0); + + cet = -sin(image->div/2.0) * khigh; + cez = -cos(image->div/2.0) * khigh; + phihigh = angle_between_2d(tl-cet, zl-cez, 0.0, 1.0); + + /* Approximation: philow and phihigh are very similar */ + phi = (philow + phihigh) / 2.0; + + azi = atan2(yl, xl); + + /* For many gradients, just multiply the above number by the gradient + * of excitation error wrt whatever. */ + switch ( k ) { + + /* Cell parameters and orientation */ + case REF_ASX : + return - hs * sin(phi) * cos(azi); + + case REF_BSX : + return - ks * sin(phi) * cos(azi); + + case REF_CSX : + return - ls * sin(phi) * cos(azi); + + case REF_ASY : + return - hs * sin(phi) * sin(azi); + + case REF_BSY : + return - ks * sin(phi) * sin(azi); + + case REF_CSY : + return - ls * sin(phi) * sin(azi); + + case REF_ASZ : + return - hs * cos(phi); + + case REF_BSZ : + return - ks * cos(phi); + + case REF_CSZ : + return - ls * cos(phi); + + } + + ERROR("No gradient defined for parameter %i\n", k); + abort(); +} + + +static double pos_gradient(int param, struct reflpeak *rp, struct detector *det) +{ + signed int h, k, l; + double xpk, ypk, xh, yh; + double fsh, ssh; + + twod_mapping(rp->peak->fs, rp->peak->ss, &xpk, &ypk, det); + get_detector_pos(rp->refl, &fsh, &ssh); + twod_mapping(fsh, ssh, &xh, &yh, det); + get_indices(rp->refl, &h, &k, &l); + + switch ( param ) { + + /* Cell parameters and orientation */ + case REF_ASX : + return 2.0 * h * (xh-xpk); + + case REF_BSX : + return 2.0 * k * (xh-xpk); + + case REF_CSX : + return 2.0 * l * (xh-xpk); + + case REF_ASY : + return 2.0 * h * (yh-ypk); + + case REF_BSY : + return 2.0 * k * (yh-ypk); + + case REF_CSY : + return 2.0 * l * (yh-ypk); + + case REF_ASZ : + return 0.0; + + case REF_BSZ : + return 0.0; + + case REF_CSZ : + return 0.0; + + } + + ERROR("Positional gradient requested for parameter %i?\n", param); + abort(); +} + + +static double r_dev(struct reflpeak *rp) +{ + /* Excitation error term */ + double rlow, rhigh, p; + get_partial(rp->refl, &rlow, &rhigh, &p); + return pow((rlow+rhigh)/2.0, 2.0); +} + + +static double pos_dev(struct reflpeak *rp, struct detector *det) +{ + /* Peak position term */ + double xpk, ypk, xh, yh; + double fsh, ssh; + twod_mapping(rp->peak->fs, rp->peak->ss, &xpk, &ypk, det); + get_detector_pos(rp->refl, &fsh, &ssh); + twod_mapping(fsh, ssh, &xh, &yh, det); + return (xh-xpk)*(xh-xpk) + (yh-ypk)*(yh-ypk); +} + + +static int iterate(struct reflpeak *rps, int n, UnitCell *cell, + struct image *image) +{ + int i; + gsl_matrix *M; + gsl_vector *v; + gsl_vector *shifts; + double asx, asy, asz; + double bsx, bsy, bsz; + double csx, csy, csz; + + /* Number of parameters to refine */ + M = gsl_matrix_calloc(9, 9); + v = gsl_vector_calloc(9); + + for ( i=0; i<n; i++ ) { + + int k; + double gradients[9]; + double w; + + /* Weight for this peak */ + w = rps[i].Ih; + + for ( k=0; k<9; k++ ) { + gradients[k] = r_gradient(cell, k, rps[i].refl, image); + gradients[k] += pos_gradient(k, &rps[i], image->det); + } + + for ( k=0; k<9; k++ ) { + + int g; + double v_c, v_curr; + + for ( g=0; g<9; 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); + + } + + v_c = EXC_WEIGHT * r_dev(&rps[i]); + v_c += pos_dev(&rps[i], image->det); + v_c *= w * gradients[k]; + v_curr = gsl_vector_get(v, k); + gsl_vector_set(v, k, v_curr + v_c); + + } + + } + + show_matrix_eqn(M, v); + shifts = solve_svd(v, M, NULL, 1); + + for ( i=0; i<9; i++ ) { + STATUS("Shift %i = %e\n", i, gsl_vector_get(shifts, i)); + } + + /* Apply shifts */ + cell_get_reciprocal(cell, &asx, &asy, &asz, + &bsx, &bsy, &bsz, + &csx, &csy, &csz); + asx += gsl_vector_get(shifts, 0); + asy += gsl_vector_get(shifts, 1); + asz += gsl_vector_get(shifts, 2); + bsx += gsl_vector_get(shifts, 3); + bsy += gsl_vector_get(shifts, 4); + bsz += gsl_vector_get(shifts, 5); + csx += gsl_vector_get(shifts, 6); + csy += gsl_vector_get(shifts, 7); + csz += gsl_vector_get(shifts, 8); + cell_set_reciprocal(cell, asx, asy, asz, bsx, bsy, bsz, csx, csy, csz); + + gsl_vector_free(shifts); + gsl_matrix_free(M); + gsl_vector_free(v); + + return 0; +} + + +static double residual(struct reflpeak *rps, int n, struct detector *det) +{ + int i; + double res = 0.0; + + for ( i=0; i<n; i++ ) { + res += EXC_WEIGHT * rps[i].Ih * pow(r_dev(&rps[i]), 2.0); + res += pow(pos_dev(&rps[i], det), 2.0); + } + + return res; +} + + +int refine_prediction(struct image *image, Crystal *cr) +{ + int n; + int i; + struct reflpeak *rps; + double max_I; + + rps = malloc(image_feature_count(image->features) + * sizeof(struct reflpeak)); + if ( rps == NULL ) return 1; + + n = pair_peaks(image->features, crystal_get_cell(cr), + crystal_get_reflections(cr), rps); + STATUS("%i peaks\n", n); + if ( n < 10 ) { + ERROR("Too few paired peaks to refine orientation.\n"); + free(rps); + return 1; + } + + /* Normalise the intensities to max 1 */ + max_I = -INFINITY; + for ( i=0; i<n; i++ ) { + double cur_I = rps[i].peak->intensity; + if ( cur_I > max_I ) max_I = cur_I; + } + if ( max_I <= 0.0 ) { + ERROR("All peaks negative?\n"); + free(rps); + return 1; + } + for ( i=0; i<n; i++ ) { + rps[i].Ih = rps[i].peak->intensity / max_I; + } + + /* Refine */ + STATUS("Initial residual = %e\n", residual(rps, n, image->det)); + for ( i=0; i<MAX_CYCLES; i++ ) { + iterate(rps, n, crystal_get_cell(cr), image); + update_partialities(cr, PMODEL_SCSPHERE); + STATUS("Residual after iteration %i = %e\n", + i, residual(rps, n, image->det)); + } + + free(rps); + return 0; +} diff --git a/src/predict-refine.h b/src/predict-refine.h new file mode 100644 index 00000000..9742f9e2 --- /dev/null +++ b/src/predict-refine.h @@ -0,0 +1,44 @@ +/* + * predict-refine.h + * + * Prediction refinement + * + * Copyright © 2012-2015 Deutsches Elektronen-Synchrotron DESY, + * a research centre of the Helmholtz Association. + * + * Authors: + * 2010-2015 Thomas White <taw@physics.org> + * + * This file is part of CrystFEL. + * + * CrystFEL is free software: you can redistribute it and/or modify + * it under the terms of the GNU General Public License as published by + * the Free Software Foundation, either version 3 of the License, or + * (at your option) any later version. + * + * CrystFEL is distributed in the hope that it will be useful, + * but WITHOUT ANY WARRANTY; without even the implied warranty of + * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the + * GNU General Public License for more details. + * + * You should have received a copy of the GNU General Public License + * along with CrystFEL. If not, see <http://www.gnu.org/licenses/>. + * + */ + +#ifndef PREDICT_REFINE_H +#define PREDICT_REFINE_H + + +#ifdef HAVE_CONFIG_H +#include <config.h> +#endif + +#include "crystal.h" + +struct image; + +extern int refine_prediction(struct image *image, Crystal *cr); + + +#endif /* PREDICT_REFINE_H */ diff --git a/src/process_image.c b/src/process_image.c index b4f831e0..7473aab8 100644 --- a/src/process_image.c +++ b/src/process_image.c @@ -50,6 +50,7 @@ #include "reflist-utils.h" #include "process_image.h" #include "integration.h" +#include "predict-refine.h" static int cmpd2(const void *av, const void *bv) @@ -310,37 +311,60 @@ void process_image(const struct index_args *iargs, struct pattern_args *pargs, } } + /* Preliminary integration, needed for refinement */ + integrate_all_4(&image, iargs->int_meth, PMODEL_SCSPHERE, + iargs->push_res, + iargs->ir_inn, iargs->ir_mid, iargs->ir_out, + INTDIAG_NONE, 0, 0, 0, results_pipe); + + /* FIXME: Temporary to monitor R during refinement */ + for ( i=0; i<image.n_crystals; i++ ) { + refine_radius(image.crystals[i], image.features); + } + /* Integrate all the crystals at once - need all the crystals so that * overlaps can be detected. */ if ( iargs->fix_profile_r < 0.0 ) { - integrate_all_4(&image, iargs->int_meth, PMODEL_SCSPHERE, - iargs->push_res, - iargs->ir_inn, iargs->ir_mid, iargs->ir_out, - INTDIAG_NONE, 0, 0, 0, results_pipe); - for ( i=0; i<image.n_crystals; i++ ) { + + if ( refine_prediction(&image, image.crystals[i]) ) { + ERROR("Prediction refinement failed.\n"); + continue; + } + + STATUS("R before: %e", + crystal_get_profile_radius(image.crystals[i])); + + /* Reset the profile radius and estimate again with + * better geometry */ + crystal_set_profile_radius(image.crystals[i], 0.02e9); refine_radius(image.crystals[i], image.features); - reflist_free(crystal_get_reflections(image.crystals[i])); + + STATUS(" after: %e\n", + crystal_get_profile_radius(image.crystals[i])); + } - integrate_all_4(&image, iargs->int_meth, PMODEL_SCSPHERE, - iargs->push_res, - iargs->ir_inn, iargs->ir_mid, iargs->ir_out, - iargs->int_diag, iargs->int_diag_h, - iargs->int_diag_k, iargs->int_diag_l, - results_pipe); } else { - integrate_all_4(&image, iargs->int_meth, PMODEL_SCSPHERE, - iargs->push_res, - iargs->ir_inn, iargs->ir_mid, iargs->ir_out, - iargs->int_diag, iargs->int_diag_h, - iargs->int_diag_k, iargs->int_diag_l, - results_pipe); + for ( i=0; i<image.n_crystals; i++ ) { + refine_prediction(&image, image.crystals[i]); + } } + /* The final, definitive, integration */ + for ( i=0; i<image.n_crystals; i++ ) { + reflist_free(crystal_get_reflections(image.crystals[i])); + } + integrate_all_4(&image, iargs->int_meth, PMODEL_SCSPHERE, + iargs->push_res, + iargs->ir_inn, iargs->ir_mid, iargs->ir_out, + iargs->int_diag, iargs->int_diag_h, + iargs->int_diag_k, iargs->int_diag_l, + results_pipe); + ret = write_chunk(st, &image, hdfile, iargs->stream_peaks, iargs->stream_refls, pargs->filename_p_e->ev); |