diff options
Diffstat (limited to 'src/pattern_sim.c')
-rw-r--r-- | src/pattern_sim.c | 235 |
1 files changed, 73 insertions, 162 deletions
diff --git a/src/pattern_sim.c b/src/pattern_sim.c index 916257e3..4d74589b 100644 --- a/src/pattern_sim.c +++ b/src/pattern_sim.c @@ -46,12 +46,12 @@ #include <cell.h> #include <cell-utils.h> #include <utils.h> -#include <detector.h> #include <peaks.h> #include <symmetry.h> #include <reflist.h> #include <reflist-utils.h> #include <stream.h> +#include <datatemplate.h> #include "diffraction.h" #include "diffraction-gpu.h" @@ -117,7 +117,8 @@ static void show_help(const char *s) } -static void record_panel(struct panel *p, float *dp, int do_poisson, +static void record_panel(struct detgeom_panel *p, float *dp, + int do_poisson, gsl_rng *rng, double ph_per_e, double background, double lambda, int *n_neg1, int *n_inf1, int *n_nan1, @@ -144,16 +145,16 @@ static void record_panel(struct panel *p, float *dp, int do_poisson, if ( isnan(intensity) ) (*n_nan1)++; /* Area of one pixel */ - pix_area = pow(1.0/p->res, 2.0); - Lsq = pow(p->clen, 2.0); + pix_area = pow(p->pixel_pitch, 2.0); + Lsq = pow(p->cnz*p->pixel_pitch, 2.0); /* Calculate distance from crystal to pixel */ xs = fs*p->fsx + ss*p->ssx; ys = ss*p->fsy + ss*p->ssy; - rx = (xs + p->cnx) / p->res; - ry = (ys + p->cny) / p->res; + rx = (xs + p->cnx) * p->pixel_pitch; + ry = (ys + p->cny) * p->pixel_pitch; dsq = pow(rx, 2.0) + pow(ry, 2.0); - twotheta = atan2(sqrt(dsq), p->clen); + twotheta = atan2(sqrt(dsq), p->cnz*p->pixel_pitch); /* Area of pixel as seen from crystal (approximate) */ proj_area = pix_area * cos(twotheta); @@ -216,11 +217,9 @@ void record_image(struct image *image, int do_poisson, double background, "Total energy = %5.3f microJ\n", nphotons, energy_density/1e7, total_energy*1e6); - fill_in_adu(image); + for ( pn=0; pn<image->detgeom->n_panels; pn++ ) { - for ( pn=0; pn<image->det->n_panels; pn++ ) { - - record_panel(&image->det->panels[pn], image->dp[pn], + record_panel(&image->detgeom->panels[pn], image->dp[pn], do_poisson, rng, ph_per_e, background, image->lambda, &n_neg1, &n_inf1, &n_nan1, @@ -364,28 +363,8 @@ static int random_ncells(double len, double min_m, double max_m) } -static void fixup_geom(struct detector *det) -{ - int i; - - for ( i=0; i<det->n_panels; i++ ) { - det->panels[i].clen += det->panels[i].coffset; - } -} - - -static int geom_contains_references(struct detector *det) -{ - int i; - - for ( i=0; i<det->n_panels; i++ ) { - if ( det->panels[i].clen_from != NULL ) return 1; - } - return 0; -} - - -static void add_metadata(const char *filename, struct quaternion q, +static void add_metadata(const char *filename, + struct quaternion q, UnitCell *cell) { hid_t fh, gh, sh, dh; @@ -488,9 +467,10 @@ static void add_metadata(const char *filename, struct quaternion q, int main(int argc, char *argv[]) { int c; - struct image image; + struct image *image; + DataTemplate *dtempl; struct gpu_context *gctx = NULL; - struct image powder; + struct image *powder; char *intfile = NULL; double *intensities; char *rval; @@ -536,8 +516,6 @@ int main(int argc, char *argv[]) double photon_energy = 9000.0; double sase_spike_width = 2e6; /* m^-1 */ double twocol_sep = 8e6; /* m^-1 */ - struct beam_params beam; - int i; /* Long options */ const struct option longopts[] = { @@ -802,7 +780,7 @@ int main(int argc, char *argv[]) ERROR("You cannot use -r and --template together.\n"); return 1; } - st = open_stream_for_read(template_file); + st = stream_open_for_read(template_file); if ( st == NULL ) { ERROR("Failed to open stream.\n"); return 1; @@ -836,26 +814,13 @@ int main(int argc, char *argv[]) ERROR("You need to specify a geometry file with --geometry\n"); return 1; } - image.beam = &beam; - image.det = get_detector_geometry(geometry, image.beam); - if ( image.det == NULL ) { - ERROR("Failed to read detector geometry from '%s'\n", geometry); + dtempl = data_template_new_from_file(geometry); + if ( dtempl == NULL ) { + ERROR("Failed to read detector geometry from '%s'\n", + geometry); return 1; } free(geometry); - if ( (beam.photon_energy > 0.0) && (beam.photon_energy_from == NULL) ) { - ERROR("WARNING: An explicit photon energy was found in the " - "geometry file. It will be ignored!\n"); - ERROR("The value given on the command line " - "(with --photon-energy) will be used instead.\n"); - } - if ( geom_contains_references(image.det) ) { - ERROR("Geometry file contains a reference to an HDF5 location" - " for the camera length. Change it to a numerical value " - " and try again.\n"); - return 1; - } - fixup_geom(image.det); if ( spectrum_str == NULL ) { STATUS("You didn't specify a spectrum type, so" @@ -952,28 +917,6 @@ int main(int argc, char *argv[]) } - image.lambda = ph_en_to_lambda(eV_to_J(photon_energy)); - image.bw = bandwidth; - - /* Initialise stuff */ - image.filename = NULL; - image.features = NULL; - image.bad = NULL; - image.dp = malloc(image.det->n_panels*sizeof(float *)); - if ( image.dp == NULL ) { - ERROR("Failed to allocate data\n"); - return 1; - } - for ( i=0; i<image.det->n_panels; i++ ) { - image.dp[i] = calloc(image.det->panels[i].w * - image.det->panels[i].h, - sizeof(float)); - if ( image.dp[i] == NULL ) { - ERROR("Failed to allocate data\n"); - return 1; - } - } - rng = gsl_rng_alloc(gsl_rng_mt19937); if ( config_random ) { FILE *fh; @@ -987,30 +930,16 @@ int main(int argc, char *argv[]) fclose(fh); } - powder.det = image.det; - powder.beam = NULL; - powder.lambda = 0.0; - powder.spectrum = NULL; - powder.dp = malloc(image.det->n_panels*sizeof(float *)); - if ( powder.dp == NULL ) { - ERROR("Failed to allocate powder data\n"); - return 1; - } - for ( i=0; i<image.det->n_panels; i++ ) { - powder.dp[i] = calloc(image.det->panels[i].w * - image.det->panels[i].h, - sizeof(float)); - if ( powder.dp[i] == NULL ) { - ERROR("Failed to allocate powder data\n"); - return 1; - } - } + image = image_create_for_simulation(dtempl); + powder = image_create_for_simulation(dtempl); /* Splurge a few useful numbers */ STATUS("Simulation parameters:\n"); - STATUS(" Photon energy: %.2f eV (wavelength %.5f A)\n", - photon_energy, image.lambda*1e10); - STATUS(" Number of photons: %.0f (%.2f mJ)\n", nphotons, + STATUS(" Wavelength: %.5f A (photon energy %.2f eV)\n", + image->lambda*1e10, + ph_lambda_to_eV(image->lambda)); + STATUS(" Number of photons: %.0f (%.2f mJ)\n", + nphotons, eV_to_J(photon_energy)*nphotons*1e3); STATUS(" Beam divergence: not simulated\n"); STATUS(" Beam radius: %.2f microns\n", @@ -1021,22 +950,22 @@ int main(int argc, char *argv[]) case SPECTRUM_TOPHAT: STATUS(" X-ray spectrum: top hat, " - "width %.5f %%\n", image.bw*100.0); + "width %.5f %%\n", image->bw*100.0); break; case SPECTRUM_GAUSSIAN: STATUS(" X-ray spectrum: Gaussian, " - "bandwidth %.5f %%\n", image.bw*100.0); + "bandwidth %.5f %%\n", image->bw*100.0); break; case SPECTRUM_SASE: STATUS(" X-ray spectrum: SASE, " - "bandwidth %.5f %%\n", image.bw*100.0); + "bandwidth %.5f %%\n", image->bw*100.0); break; case SPECTRUM_TWOCOLOUR: STATUS(" X-ray spectrum: two colour, " - "separation %.5f %%\n", image.bw*100.0); + "separation %.5f %%\n", image->bw*100.0); break; case SPECTRUM_FROMFILE: @@ -1066,12 +995,13 @@ int main(int argc, char *argv[]) int err = 0; int pi; - for ( pi=0; pi<image.det->n_panels; pi++ ) { + /* Reset image data to zero for new pattern */ + for ( pi=0; pi<image->detgeom->n_panels; pi++ ) { long j; - long np = image.det->panels[pi].w - * image.det->panels[pi].h; + long np = image->detgeom->panels[pi].w + * image->detgeom->panels[pi].h; for ( j=0; j<np; j++ ) { - image.dp[pi][j] = 0.0; + image->dp[pi][j] = 0.0; } } @@ -1115,70 +1045,54 @@ int main(int argc, char *argv[]) } else { - struct image image; - int i; + struct image *templ_image; Crystal *cr; - image.det = NULL; - /* Get data from next chunk */ - if ( read_chunk(st, &image) ) break; + templ_image = stream_read_chunk(st, dtempl, + STREAM_CRYSTALS); + if ( templ_image == NULL ) break; + if ( templ_image->n_crystals == 0 ) continue; - free(image.filename); - image_feature_list_free(image.features); + cr = templ_image->crystals[0]; + cell = cell_new_from_cell(crystal_get_cell(cr)); - if ( image.n_crystals == 0 ) continue; - - cr = image.crystals[0]; - cell = crystal_get_cell(cr); - - if ( image.n_crystals > 1 ) { + if ( templ_image->n_crystals > 1 ) { ERROR("Using the first crystal only.\n"); } - for ( i=1; i<image.n_crystals; i++ ) { - - Crystal *cr = image.crystals[i]; - cell = crystal_get_cell(cr); - - reflist_free(crystal_get_reflections(cr)); - cell_free(crystal_get_cell(cr)); - crystal_free(cr); - - } - - free(image.crystals); + image_free(templ_image); } switch ( spectrum_type ) { case SPECTRUM_TOPHAT : - image.spectrum = spectrum_generate_tophat(image.lambda, - image.bw); + image->spectrum = spectrum_generate_tophat(image->lambda, + image->bw); break; case SPECTRUM_GAUSSIAN : - image.spectrum = spectrum_generate_gaussian(image.lambda, - image.bw); + image->spectrum = spectrum_generate_gaussian(image->lambda, + image->bw); break; case SPECTRUM_SASE : - image.spectrum = spectrum_generate_sase(image.lambda, - image.bw, - sase_spike_width, - rng); + image->spectrum = spectrum_generate_sase(image->lambda, + image->bw, + sase_spike_width, + rng); break; - case SPECTRUM_TWOCOLOUR : - image.spectrum = spectrum_generate_twocolour(image.lambda, - image.bw, - twocol_sep); + case SPECTRUM_TWOCOLOUR : + image->spectrum = spectrum_generate_twocolour(image->lambda, + image->bw, + twocol_sep); break; - case SPECTRUM_FROMFILE : - image.spectrum = spectrum_load(spectrum_fn); + case SPECTRUM_FROMFILE : + image->spectrum = spectrum_load(spectrum_fn); break; } @@ -1195,12 +1109,12 @@ int main(int argc, char *argv[]) intensities, flags, sym_str, gpu_dev); } - err = get_diffraction_gpu(gctx, &image, na, nb, nc, + err = get_diffraction_gpu(gctx, image, na, nb, nc, cell, no_fringes, flat, nsamples); } else { - get_diffraction(&image, na, nb, nc, intensities, phases, + get_diffraction(image, na, nb, nc, intensities, phases, flags, cell, grad, sym, no_fringes, flat, nsamples); } @@ -1210,28 +1124,28 @@ int main(int argc, char *argv[]) goto skip; } - record_image(&image, !config_nonoise, background, rng, + record_image(image, !config_nonoise, background, rng, beam_radius, nphotons); if ( powder_fn != NULL ) { int pn; - for ( pn=0; pn<image.det->n_panels; pn++ ) { + for ( pn=0; pn<image->detgeom->n_panels; pn++ ) { size_t w, i; - w = image.det->panels[pn].w - * image.det->panels[pn].h; + w = image->detgeom->panels[pn].w + * image->detgeom->panels[pn].h; for ( i=0; i<w; i++ ) { - powder.dp[pn][i] += (double)image.dp[pn][i]; + powder->dp[pn][i] += (double)image->dp[pn][i]; } } if ( !(ndone % 10) ) { - hdf5_write_image(powder_fn, &powder, NULL); + image_write(powder, dtempl, powder_fn); } } @@ -1249,7 +1163,9 @@ int main(int argc, char *argv[]) number++; /* Write the output file */ - hdf5_write_image(filename, &image, NULL); + image_write(image, dtempl, filename); + + /* Add some pattern_sim-specific metadata */ add_metadata(filename, orientation, cell); } @@ -1264,21 +1180,16 @@ skip: } while ( !done ); if ( powder_fn != NULL ) { - hdf5_write_image(powder_fn, &powder, NULL); + image_write(powder, dtempl, powder_fn); } if ( gctx != NULL ) { cleanup_gpu(gctx); } - for ( i=0; i<image.det->n_panels; i++ ) { - free(image.dp[i]); - free(powder.dp[i]); - } - free(image.dp); - free(powder.dp); + image_free(image); + image_free(powder); free(intfile); - free(image.det); cell_free(input_cell); free(intensities); free(outfile); |