diff options
author | Thomas White <taw@physics.org> | 2019-06-27 14:04:06 +0200 |
---|---|---|
committer | Thomas White <taw@physics.org> | 2019-07-17 09:22:23 +0200 |
commit | f9be3d5a2334f0d61d5cd6c96c24dcb1f481927a (patch) | |
tree | 0ef71cdaca68ccb7764804e09e38bfb567b02717 /src | |
parent | aaed4bf95fe7e525f5e9c26a4d5b80b27bde83bb (diff) |
Really Simple Minimiser
Diffstat (limited to 'src')
-rw-r--r-- | src/post-refinement.c | 121 |
1 files changed, 86 insertions, 35 deletions
diff --git a/src/post-refinement.c b/src/post-refinement.c index 3da88752..de656d36 100644 --- a/src/post-refinement.c +++ b/src/post-refinement.c @@ -561,6 +561,68 @@ void write_gridscan(Crystal *cr, const RefList *full, } +static int refine_loop(struct rf_alteration *cur, struct rf_alteration *dirns, + int n_dirns, struct rf_priv *priv, int *total_iter, + FILE *fh) +{ + int n_iter = 0; + int best; + + do { + + struct rf_alteration trials[n_dirns+1]; + double lowest_fom = INFINITY; + double freefom; + int i; + + best = 0; + + trials[0] = *cur; + for ( i=0; i<n_dirns; i++ ) { + trials[i+1] = *cur; + trials[i+1].rot_x += dirns[i].rot_x; + trials[i+1].rot_y += dirns[i].rot_y; + trials[i+1].delta_R += dirns[i].delta_R; + trials[i+1].delta_wave += dirns[i].delta_wave; + } + + for ( i=0; i<n_dirns+1; i++ ) { + double nfom = calc_residual(priv, trials[i], 0); + if ( nfom < lowest_fom ) { + best = i; + lowest_fom = nfom; + } + } + + *cur = trials[best]; + + n_iter++; + (*total_iter)++; + + freefom = calc_residual(priv, *cur, 1); + if ( fh != NULL ) { + fprintf(fh, "%5i %10.8f %10.8f %10.8f %10.8f %e %e\n", + *total_iter, lowest_fom, freefom, + cur->rot_x, cur->rot_y, + crystal_get_profile_radius(priv->cr)+cur->delta_R, + crystal_get_image_const(priv->cr)->lambda+cur->delta_wave); + } + + } while ( (best != 0) && (n_iter < 10) ); + + return 0; +} + + +static void zero_alter(struct rf_alteration *alter) +{ + alter->rot_x = 0.0; + alter->rot_y = 0.0; + alter->delta_R = 0.0; + alter->delta_wave = 0.0; +} + + static void do_pr_refine(Crystal *cr, const RefList *full, PartialityModel pmodel, int serial, int cycle, int write_logs, @@ -569,7 +631,6 @@ static void do_pr_refine(Crystal *cr, const RefList *full, struct rf_priv priv; struct rf_alteration alter; int n_iter = 0; - int status = 0; double fom, freefom; RefList *list; FILE *fh = NULL; @@ -582,10 +643,7 @@ static void do_pr_refine(Crystal *cr, const RefList *full, return; } - alter.rot_x = 0.0; - alter.rot_y = 0.0; - alter.delta_R = 0.0; - alter.delta_wave = 0.0; + zero_alter(&alter); priv.cr = cr; priv.full = full; @@ -620,36 +678,29 @@ static void do_pr_refine(Crystal *cr, const RefList *full, } - do { - n_iter++; - - /* FIXME: Actually minimise */ - - fom = calc_residual(&priv, alter, 0); - freefom = calc_residual(&priv, alter, 1); - - if ( fh != NULL ) { - fprintf(fh, "%5i %10.8f %10.8f %10.8f %10.8f %e %e\n", - n_iter, fom, freefom, - alter.rot_x, alter.rot_y, - crystal_get_profile_radius(cr)+alter.delta_R, - crystal_get_image(cr)->lambda+alter.delta_wave); - } - - } while ( n_iter < 10 ); /* FIXME: Termination criteria */ - - if ( status == 0 ) { - - /* Apply the final shifts */ - apply_parameters(cr, cr, alter); - update_predictions(cr); - calculate_partialities(cr, PMODEL_XSPHERE); - scale_one_crystal(cr, full, scaleflags); - - } else { - ERROR("Bad refinement: crystal %i after %i iterations\n", - serial, n_iter); - } + /* Refine orientation */ + struct rf_alteration dirns[4]; + zero_alter(&dirns[0]); dirns[0].rot_x += 0.1e-3; + zero_alter(&dirns[1]); dirns[1].rot_x -= 0.1e-3; + zero_alter(&dirns[2]); dirns[2].rot_y += 0.1e-3; + zero_alter(&dirns[3]); dirns[3].rot_y -= 0.1e-3; + refine_loop(&alter, dirns, 4, &priv, &n_iter, fh); + + /* Refine profile radius */ + zero_alter(&dirns[0]); dirns[0].delta_R += 1e5; + zero_alter(&dirns[1]); dirns[1].delta_R -= 1e5; + refine_loop(&alter, dirns, 2, &priv, &n_iter, fh); + + /* Refine wavelength */ + zero_alter(&dirns[0]); dirns[0].delta_wave += 2.0e-14; + zero_alter(&dirns[1]); dirns[1].delta_wave -= 2.0e-14; + refine_loop(&alter, dirns, 2, &priv, &n_iter, fh); + + /* Apply the final shifts */ + apply_parameters(cr, cr, alter); + update_predictions(cr); + calculate_partialities(cr, PMODEL_XSPHERE); + scale_one_crystal(cr, full, scaleflags); if ( write_logs ) { write_gridscan(cr, full, cycle, serial, scaleflags); |