/* * detector.c * * Detector properties * * Copyright © 2012-2014 Deutsches Elektronen-Synchrotron DESY, * a research centre of the Helmholtz Association. * Copyright © 2012 Richard Kirian * * Authors: * 2009-2014 Thomas White * 2014 Valerio Mariani * 2014 Kenneth Beyerlein * 2011 Andrew Aquila * 2011 Richard Kirian * * 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 . * */ #define _ISOC99_SOURCE #define _GNU_SOURCE #include #include #include #include #include #include #include "image.h" #include "utils.h" #include "detector.h" #include "hdf5-file.h" /** * SECTION:detector * @short_description: Detector geometry * @title: Detector * @section_id: * @see_also: * @include: "detector.h" * @Image: * * This structure represents the detector geometry */ static int atob(const char *a) { if ( strcasecmp(a, "true") == 0 ) return 1; if ( strcasecmp(a, "false") == 0 ) return 0; return atoi(a); } static int assplode_algebraic(const char *a_orig, char ***pbits) { int len, i; int nexp; char **bits; char *a; int idx, istr; len = strlen(a_orig); /* Add plus at start if no sign there already */ if ( (a_orig[0] != '+') && (a_orig[0] != '-') ) { len += 1; a = malloc(len+1); snprintf(a, len+1, "+%s", a_orig); a[len] = '\0'; } else { a = strdup(a_orig); } /* Count the expressions */ nexp = 0; for ( i=0; i= 0 ) bits[idx][istr] = '\0'; idx++; bits[idx] = malloc(len+1); istr = 0; } if ( !isdigit(ch) && (ch != '.') && (ch != 'x') && (ch != 'y') && (ch != '+') && (ch != '-') ) { ERROR("Invalid character '%C' found.\n", ch); return 0; } assert(idx >= 0); bits[idx][istr++] = ch; } if ( idx >= 0 ) bits[idx][istr] = '\0'; *pbits = bits; free(a); return nexp; } /* Parses the scan directions (accounting for possible rotation) * Assumes all white spaces have been already removed */ static int dir_conv(const char *a, double *sx, double *sy) { int n; char **bits; int i; *sx = 0.0; *sy = 0.0; n = assplode_algebraic(a, &bits); if ( n == 0 ) { ERROR("Invalid direction '%s'\n", a); return 1; } for ( i=0; i %5.2fx + %5.2fy\n", a, *sx, *sy); return 0; } struct rvec get_q_for_panel(struct panel *p, double fs, double ss, double *ttp, double k) { struct rvec q; double twotheta, r, az; double rx, ry; double xs, ys; /* Convert xs and ys, which are in fast scan/slow scan coordinates, * to x and y */ xs = fs*p->fsx + ss*p->ssx; ys = fs*p->fsy + ss*p->ssy; rx = (xs + p->cnx) / p->res; ry = (ys + p->cny) / p->res; /* Calculate q-vector for this sub-pixel */ r = sqrt(pow(rx, 2.0) + pow(ry, 2.0)); twotheta = atan2(r, p->clen); az = atan2(ry, rx); if ( ttp != NULL ) *ttp = twotheta; q.u = k * sin(twotheta)*cos(az); q.v = k * sin(twotheta)*sin(az); q.w = k * (cos(twotheta) - 1.0); return q; } struct rvec get_q(struct image *image, double fs, double ss, double *ttp, double k) { struct panel *p; const unsigned int fsi = fs; const unsigned int ssi = ss; /* Explicit rounding */ /* Determine which panel to use */ p = find_panel(image->det, fsi, ssi); assert(p != NULL); return get_q_for_panel(p, fs-(double)p->min_fs, ss-(double)p->min_ss, ttp, k); } int in_bad_region(struct detector *det, double fs, double ss) { double rx, ry; struct panel *p; double xs, ys; int i; /* Determine which panel to use */ p = find_panel(det, fs, ss); /* No panel found -> definitely bad! */ if ( p == NULL ) return 1; /* Convert xs and ys, which are in fast scan/slow scan coordinates, * to x and y */ xs = (fs-(double)p->min_fs)*p->fsx + (ss-(double)p->min_ss)*p->ssx; ys = (fs-(double)p->min_fs)*p->fsy + (ss-(double)p->min_ss)*p->ssy; rx = xs + p->cnx; ry = ys + p->cny; for ( i=0; in_bad; i++ ) { struct badregion *b = &det->bad[i]; if ( b->is_fsss ) { if ( fs < b->min_fs ) continue; if ( fs > b->max_fs ) continue; if ( ss < b->min_ss ) continue; if ( ss > b->max_ss ) continue; } else { if ( rx < b->min_x ) continue; if ( rx > b->max_x ) continue; if ( ry < b->min_y ) continue; if ( ry > b->max_y ) continue; } return 1; } return 0; } double get_tt(struct image *image, double fs, double ss, int *err) { double r, rx, ry; struct panel *p; double xs, ys; *err = 0; p = find_panel(image->det, fs, ss); if ( p == NULL ) { *err = 1; return 0.0; } /* Convert xs and ys, which are in fast scan/slow scan coordinates, * to x and y */ xs = (fs-p->min_fs)*p->fsx + (ss-p->min_ss)*p->ssx; ys = (fs-p->min_fs)*p->fsy + (ss-p->min_ss)*p->ssy; rx = (xs + p->cnx) / p->res; ry = (ys + p->cny) / p->res; r = sqrt(pow(rx, 2.0) + pow(ry, 2.0)); return atan2(r, p->clen); } void record_image(struct image *image, int do_poisson, double background, gsl_rng *rng, double beam_radius, double nphotons) { int x, y; double total_energy, energy_density; double ph_per_e; double area; double max_tt = 0.0; int n_inf1 = 0; int n_neg1 = 0; int n_nan1 = 0; int n_inf2 = 0; int n_neg2 = 0; int n_nan2 = 0; /* How many photons are scattered per electron? */ area = M_PI*pow(beam_radius, 2.0); total_energy = nphotons * ph_lambda_to_en(image->lambda); energy_density = total_energy / area; ph_per_e = (nphotons /area) * pow(THOMSON_LENGTH, 2.0); STATUS("Fluence = %8.2e photons, " "Energy density = %5.3f kJ/cm^2, " "Total energy = %5.3f microJ\n", nphotons, energy_density/1e7, total_energy*1e6); for ( x=0; xwidth; x++ ) { for ( y=0; yheight; y++ ) { double counts; double cf; double intensity, sa; double pix_area, Lsq; double xs, ys, rx, ry; double dsq, proj_area; float dval; struct panel *p; intensity = (double)image->data[x + image->width*y]; if ( isinf(intensity) ) n_inf1++; if ( intensity < 0.0 ) n_neg1++; if ( isnan(intensity) ) n_nan1++; p = find_panel(image->det, x, y); /* Area of one pixel */ pix_area = pow(1.0/p->res, 2.0); Lsq = pow(p->clen, 2.0); /* Area of pixel as seen from crystal (approximate) */ proj_area = pix_area * cos(image->twotheta[x + image->width*y]); /* Calculate distance from crystal to pixel */ xs = (x-p->min_fs)*p->fsx + (y-p->min_ss)*p->ssx; ys = (x-p->min_fs)*p->fsy + (y-p->min_ss)*p->ssy; rx = (xs + p->cnx) / p->res; ry = (ys + p->cny) / p->res; dsq = pow(rx, 2.0) + pow(ry, 2.0); /* Projected area of pixel divided by distance squared */ sa = proj_area / (dsq + Lsq); if ( do_poisson ) { counts = poisson_noise(rng, intensity * ph_per_e * sa); } else { cf = intensity * ph_per_e * sa; counts = cf; } /* Number of photons in pixel */ dval = counts + poisson_noise(rng, background); /* Convert to ADU */ dval *= p->adu_per_eV * ph_lambda_to_eV(image->lambda); /* Saturation */ if ( dval > p->max_adu ) dval = p->max_adu; image->data[x + image->width*y] = dval; /* Sanity checks */ if ( isinf(image->data[x+image->width*y]) ) n_inf2++; if ( isnan(image->data[x+image->width*y]) ) n_nan2++; if ( image->data[x+image->width*y] < 0.0 ) n_neg2++; if ( image->twotheta[x + image->width*y] > max_tt ) { max_tt = image->twotheta[x + image->width*y]; } } progress_bar(x, image->width-1, "Post-processing"); } STATUS("Max 2theta = %.2f deg, min d = %.2f nm\n", rad2deg(max_tt), (image->lambda/(2.0*sin(max_tt/2.0)))/1e-9); double tt_side = image->twotheta[(image->width/2)+image->width*0]; STATUS("At middle of bottom edge: %.2f deg, min d = %.2f nm\n", rad2deg(tt_side), (image->lambda/(2.0*sin(tt_side/2.0)))/1e-9); tt_side = image->twotheta[0+image->width*(image->height/2)]; STATUS("At middle of left edge: %.2f deg, min d = %.2f nm\n", rad2deg(tt_side), (image->lambda/(2.0*sin(tt_side/2.0)))/1e-9); STATUS("Halve the d values to get the voxel size for a synthesis.\n"); if ( n_neg1 + n_inf1 + n_nan1 + n_neg2 + n_inf2 + n_nan2 ) { ERROR("WARNING: The raw calculation produced %i negative" " values, %i infinities and %i NaNs.\n", n_neg1, n_inf1, n_nan1); ERROR("WARNING: After processing, there were %i negative" " values, %i infinities and %i NaNs.\n", n_neg2, n_inf2, n_nan2); } } signed int find_panel_number(struct detector *det, double fs, double ss) { int p; /* Fractional pixel coordinates are allowed to be a little further along * than "== max_{f,s}s" for an integer. */ for ( p=0; pn_panels; p++ ) { if ( (fs >= det->panels[p].min_fs) && (fs < det->panels[p].max_fs+1) && (ss >= det->panels[p].min_ss) && (ss < det->panels[p].max_ss+1) ) return p; } return -1; } struct panel *find_panel(struct detector *det, double fs, double ss) { signed int pn = find_panel_number(det, fs, ss); if ( pn == -1 ) return NULL; return &det->panels[pn]; } /* Like find_panel(), but uses the original panel bounds, i.e. referring to * what's in the HDF5 file */ struct panel *find_orig_panel(struct detector *det, double fs, double ss) { int p; for ( p=0; pn_panels; p++ ) { if ( (fs >= det->panels[p].orig_min_fs) && (fs < det->panels[p].orig_max_fs+1) && (ss >= det->panels[p].orig_min_ss) && (ss < det->panels[p].orig_max_ss+1) ) { return &det->panels[p]; } } return NULL; } void fill_in_values(struct detector *det, struct hdfile *f, struct event* ev) { int i; for ( i=0; in_panels; i++ ) { struct panel *p = &det->panels[i]; if ( p->clen_from != NULL ) { if (det->path_dim !=0 || det->dim_dim !=0 ){ p->clen = get_ev_based_value(f, p->clen_from, ev) * 1.0e-3; } else { p->clen = get_value(f, p->clen_from) * 1.0e-3; } } p->clen += p->coffset; } } static struct panel *new_panel(struct detector *det, const char *name) { struct panel *new; det->n_panels++; det->panels = realloc(det->panels, det->n_panels*sizeof(struct panel)); new = &det->panels[det->n_panels-1]; memcpy(new, &det->defaults, sizeof(struct panel)); strcpy(new->name, name); /* Create a new copy of the camera length location if needed */ if ( new->clen_from != NULL ) { new->clen_from = strdup(new->clen_from); } /* Create a new copy of the data location if needed */ if ( new->data != NULL ) { new->data = strdup(new->data); } /* Create a new copy of the bad pixel mask location */ if ( new->mask != NULL ) { new->mask = strdup(new->mask); } return new; } static struct badregion *new_bad_region(struct detector *det, const char *name) { struct badregion *new; det->n_bad++; det->bad = realloc(det->bad, det->n_bad*sizeof(struct badregion)); new = &det->bad[det->n_bad-1]; new->min_x = NAN; new->max_x = NAN; new->min_y = NAN; new->max_y = NAN; new->min_fs = 0; new->max_fs = 0; new->min_ss = 0; new->max_ss = 0; new->is_fsss = 0; strcpy(new->name, name); return new; } struct panel *find_panel_by_name(struct detector *det, const char *name) { int i; for ( i=0; in_panels; i++ ) { if ( strcmp(det->panels[i].name, name) == 0 ) { return &det->panels[i]; } } return NULL; } static struct badregion *find_bad_region_by_name(struct detector *det, const char *name) { int i; for ( i=0; in_bad; i++ ) { if ( strcmp(det->bad[i].name, name) == 0 ) { return &det->bad[i]; } } return NULL; } static struct rigid_group *find_or_add_rg(struct detector *det, const char *name) { int i; struct rigid_group **new; struct rigid_group *rg; for ( i=0; in_rigid_groups; i++ ) { if ( strcmp(det->rigid_groups[i]->name, name) == 0 ) { return det->rigid_groups[i]; } } new = realloc(det->rigid_groups, (1+det->n_rigid_groups)*sizeof(struct rigid_group *)); if ( new == NULL ) return NULL; det->rigid_groups = new; rg = malloc(sizeof(struct rigid_group)); if ( rg == NULL ) return NULL; det->rigid_groups[det->n_rigid_groups++] = rg; rg->name = strdup(name); rg->panels = NULL; rg->n_panels = 0; rg->have_deltas = 0; return rg; } static void add_to_rigid_group(struct rigid_group *rg, struct panel *p) { struct panel **pn; pn = realloc(rg->panels, (1+rg->n_panels)*sizeof(struct panel *)); if ( pn == NULL ) { ERROR("Couldn't add panel to rigid group.\n"); return; } assert(p->rigid_group == rg); rg->panels = pn; rg->panels[rg->n_panels++] = p; } static void rigid_groups_free(struct detector *det) { int i; if ( det->rigid_groups == NULL ) return; for ( i=0; in_rigid_groups; i++ ) { free(det->rigid_groups[i]->name); free(det->rigid_groups[i]->panels); free(det->rigid_groups[i]); } free(det->rigid_groups); } static void fix_up_rigid_groups(struct detector *det) { int i; for ( i=0; in_panels; i++ ) { struct panel *p = &det->panels[i]; if ( p->rigid_group != NULL ) { add_to_rigid_group(p->rigid_group, p); } } } static int parse_field_for_panel(struct panel *panel, const char *key, const char *val, struct detector *det) { int reject = 0; if ( strcmp(key, "min_fs") == 0 ) { panel->min_fs = atof(val); } else if ( strcmp(key, "max_fs") == 0 ) { panel->max_fs = atof(val); } else if ( strcmp(key, "min_ss") == 0 ) { panel->min_ss = atof(val); } else if ( strcmp(key, "max_ss") == 0 ) { panel->max_ss = atof(val); } else if ( strcmp(key, "corner_x") == 0 ) { panel->cnx = atof(val); } else if ( strcmp(key, "corner_y") == 0 ) { panel->cny = atof(val); } else if ( strcmp(key, "adu_per_eV") == 0 ) { panel->adu_per_eV = atof(val); } else if ( strcmp(key, "rigid_group") == 0 ) { panel->rigid_group = find_or_add_rg(det, val); } else if ( strcmp(key, "clen") == 0 ) { char *end; double v = strtod(val, &end); if ( end == val ) { /* This means "fill in later" */ panel->clen = -1.0; panel->clen_from = strdup(val); } else { panel->clen = v; panel->clen_from = NULL; } } else if ( strcmp(key, "data") == 0 ) { if ( strncmp(val,"/",1) != 0 ) { ERROR("Invalid data location '%s'\n", val); reject = -1; } panel->data = strdup(val); } else if ( strcmp(key, "mask") == 0 ) { if ( strncmp(val,"/",1) != 0 ) { ERROR("Invalid mask location '%s'\n", val); reject = -1; } panel->mask = strdup(val); } else if ( strcmp(key, "coffset") == 0) { panel->coffset = atof(val); } else if ( strcmp(key, "res") == 0 ) { panel->res = atof(val); } else if ( strcmp(key, "max_adu") == 0 ) { panel->max_adu = atof(val); } else if ( strcmp(key, "badrow_direction") == 0 ) { panel->badrow = val[0]; /* First character only */ if ( (panel->badrow != 'x') && (panel->badrow != 'y') && (panel->badrow != 'f') && (panel->badrow != 's') && (panel->badrow != '-') ) { ERROR("badrow_direction must be x, y, f, s or '-'\n"); ERROR("Assuming '-'\n."); panel->badrow = '-'; } if ( panel->badrow == 'x' ) panel->badrow = 'f'; if ( panel->badrow == 'y' ) panel->badrow = 's'; } else if ( strcmp(key, "no_index") == 0 ) { panel->no_index = atob(val); } else if ( strcmp(key, "fs") == 0 ) { if ( dir_conv(val, &panel->fsx, &panel->fsy) != 0 ) { ERROR("Invalid fast scan direction '%s'\n", val); reject = 1; } } else if ( strcmp(key, "ss") == 0 ) { if ( dir_conv(val, &panel->ssx, &panel->ssy) != 0 ) { ERROR("Invalid slow scan direction '%s'\n", val); reject = 1; } } else if ( strncmp(key, "dim", 3) == 0) { if ( panel->dim_structure == NULL ) { panel->dim_structure = initialize_dim_structure(); } set_dim_structure_entry(panel->dim_structure, key, val); } else { ERROR("Unrecognised field '%s'\n", key); } return reject; } static int parse_field_bad(struct badregion *panel, const char *key, const char *val) { int reject = 0; if ( strcmp(key, "min_x") == 0 ) { panel->min_x = atof(val); if ( panel->is_fsss ) { ERROR("You can't mix x/y and fs/ss in a bad region.\n"); } } else if ( strcmp(key, "max_x") == 0 ) { panel->max_x = atof(val); if ( panel->is_fsss ) { ERROR("You can't mix x/y and fs/ss in a bad region.\n"); } } else if ( strcmp(key, "min_y") == 0 ) { panel->min_y = atof(val); if ( panel->is_fsss ) { ERROR("You can't mix x/y and fs/ss in a bad region.\n"); } } else if ( strcmp(key, "max_y") == 0 ) { panel->max_y = atof(val); if ( panel->is_fsss ) { ERROR("You can't mix x/y and fs/ss in a bad region.\n"); } } else if ( strcmp(key, "min_fs") == 0 ) { panel->min_fs = atof(val); panel->is_fsss = 1; } else if ( strcmp(key, "max_fs") == 0 ) { panel->max_fs = atof(val); panel->is_fsss = 1; } else if ( strcmp(key, "min_ss") == 0 ) { panel->min_ss = atof(val); panel->is_fsss = 1; } else if ( strcmp(key, "max_ss") == 0 ) { panel->max_ss = atof(val); panel->is_fsss = 1; } else { ERROR("Unrecognised field '%s'\n", key); } return reject; } static void parse_toplevel(struct detector *det, struct beam_params *beam, const char *key, const char *val) { if ( strcmp(key, "mask_bad") == 0 ) { char *end; double v = strtod(val, &end); if ( end != val ) { det->mask_bad = v; } } else if ( strcmp(key, "mask_good") == 0 ) { char *end; double v = strtod(val, &end); if ( end != val ) { det->mask_good = v; } } else if ( strcmp(key, "coffset") == 0 ) { det->defaults.coffset = atof(val); } else if ( strcmp(key, "photon_energy") == 0 ) { if ( strncmp(val, "/", 1) == 0 ) { beam->photon_energy = 0.0; beam->photon_energy_from = strdup(val); } else { beam->photon_energy = atof(val); beam->photon_energy_from = NULL; } } else if ( strcmp(key, "photon_energy_scale") == 0 ) { if ( beam != NULL ) { beam->photon_energy_scale = atof(val); } } else if ( parse_field_for_panel(&det->defaults, key, val, det) ) { ERROR("Unrecognised top level field '%s'\n", key); } } /* Test if fs,ss in panel "p" is further {out,in} than {*p_max_d,*p_min_d}, and * if so update det->furthest_{out,in}_{panel,fs,ss}. */ static void check_point(struct panel *p, double fs, double ss, double *p_min_d, double *p_max_d, struct detector *det) { double xs, ys, rx, ry, d; xs = fs*p->fsx + ss*p->ssx; ys = fs*p->fsy + ss*p->ssy; rx = (xs + p->cnx) / p->res; ry = (ys + p->cny) / p->res; d = sqrt(pow(rx, 2.0) + pow(ry, 2.0)); if ( d > *p_max_d ) { det->furthest_out_panel = p; det->furthest_out_fs = fs; det->furthest_out_ss = ss; *p_max_d = d; } else if ( d < *p_min_d ) { det->furthest_in_panel = p; det->furthest_in_fs = fs; det->furthest_in_ss = ss; *p_min_d = d; } } static void find_min_max_d(struct detector *det) { double max_d, min_d; int i; min_d = +INFINITY; max_d = 0.0; for ( i=0; in_panels; i++ ) { struct panel *p; double w, h; p = &det->panels[i]; w = p->max_fs - p->min_fs + 1; h = p->max_ss - p->min_ss + 1; check_point(p, 0, 0, &min_d, &max_d, det); check_point(p, w, 0, &min_d, &max_d, det); check_point(p, 0, h, &min_d, &max_d, det); check_point(p, w, h, &min_d, &max_d, det); } } struct detector *get_detector_geometry(const char *filename, struct beam_params *beam) { FILE *fh; struct detector *det; char *rval; char **bits; int i; int reject = 0; int path_dim; int dim_dim; int x, y, max_fs, max_ss; int dim_reject = 0; int dim_dim_reject = 0; fh = fopen(filename, "r"); if ( fh == NULL ) return NULL; det = calloc(1, sizeof(struct detector)); if ( det == NULL ) { fclose(fh); return NULL; } if ( beam != NULL ) { beam->photon_energy = 0.0; beam->photon_energy_from = NULL; beam->photon_energy_scale = 1.0; } det->n_panels = 0; det->panels = NULL; det->n_bad = 0; det->bad = NULL; det->mask_good = 0; det->mask_bad = 0; det->n_rigid_groups = 0; det->rigid_groups = NULL; det->path_dim = 0; det->dim_dim = 0; /* The default defaults... */ det->defaults.min_fs = -1; det->defaults.min_ss = -1; det->defaults.max_fs = -1; det->defaults.max_ss = -1; det->defaults.orig_min_fs = -1; det->defaults.orig_min_ss = -1; det->defaults.orig_max_fs = -1; det->defaults.orig_max_ss = -1; det->defaults.cnx = NAN; det->defaults.cny = NAN; det->defaults.clen = -1.0; det->defaults.coffset = 0.0; det->defaults.res = -1.0; det->defaults.badrow = '-'; det->defaults.no_index = 0; det->defaults.fsx = 1.0; det->defaults.fsy = 0.0; det->defaults.ssx = 0.0; det->defaults.ssy = 1.0; det->defaults.rigid_group = NULL; det->defaults.adu_per_eV = NAN; det->defaults.max_adu = +INFINITY; det->defaults.mask = NULL; det->defaults.data = NULL; det->defaults.dim_structure = NULL; strncpy(det->defaults.name, "", 1023); do { int n1, n2; char **path; char line[1024]; struct badregion *badregion = NULL; struct panel *panel = NULL; char wholeval[1024]; rval = fgets(line, 1023, fh); if ( rval == NULL ) break; chomp(line); if ( line[0] == ';' ) continue; n1 = assplode(line, " \t", &bits, ASSPLODE_NONE); if ( n1 < 3 ) { for ( i=0; in_panels == -1 ) { ERROR("No panel descriptions in geometry file.\n"); fclose(fh); free(det); return NULL; } max_fs = 0; max_ss = 0; path_dim = -1; dim_reject = 0; for ( i=0; in_panels; i++ ) { int panel_dim = 0; char *next_instance; next_instance = det->panels[i].data; while ( next_instance ) { next_instance = strstr(next_instance, "%"); if ( next_instance != NULL ) { next_instance += 1*sizeof(char); panel_dim += 1; } } if ( path_dim == -1 ) { path_dim = panel_dim; } else { if ( panel_dim != path_dim ) { dim_reject = 1; } } } for ( i=0; in_panels; i++ ) { int panel_mask_dim = 0; char *next_instance; if ( det->panels[i].mask != NULL ) { next_instance = det->panels[i].mask; while(next_instance) { next_instance = strstr(next_instance, "%"); if ( next_instance != NULL ) { next_instance += 1*sizeof(char); panel_mask_dim += 1; } } if ( panel_mask_dim != path_dim ) { dim_reject = 1; } } } if ( dim_reject == 1) { ERROR("All panels' data and mask entries must have the same number "\ "of placeholders\n"); reject = 1; } det->path_dim = path_dim; dim_dim_reject = 0; dim_dim = -1; for ( i=0; in_panels; i++ ) { int di; int found_ss = 0; int found_fs = 0; int panel_dim_dim = 0; if ( det->panels[i].dim_structure == NULL ) { det->panels[i].dim_structure = default_dim_structure(); } for ( di=0; dipanels[i].dim_structure->num_dims; di++ ) { if ( det->panels[i].dim_structure->dims[di] == HYSL_UNDEFINED ) { dim_dim_reject = 1; } if ( det->panels[i].dim_structure->dims[di] == HYSL_PLACEHOLDER ) { panel_dim_dim += 1; } if ( det->panels[i].dim_structure->dims[di] == HYSL_SS ) { found_ss += 1; } if ( det->panels[i].dim_structure->dims[di] == HYSL_FS ) { found_fs += 1; } } if ( found_ss != 1 ) { ERROR("Only one slow scan dim coordinate is allowed\n"); dim_dim_reject = 1; } if ( found_fs != 1 ) { ERROR("Only one fast scan dim coordinate is allowed\n"); dim_dim_reject = 1; } if ( panel_dim_dim > 1 ) { ERROR("Maximum one placeholder dim coordinate is allowed\n"); dim_dim_reject = 1; } if ( dim_dim == -1 ) { dim_dim = panel_dim_dim; } else { if ( panel_dim_dim != dim_dim ) { dim_dim_reject = 1; } } } if ( dim_dim_reject == 1) { reject = 1; } det->dim_dim = dim_dim; for ( i=0; in_panels; i++ ) { if ( det->panels[i ].min_fs < 0 ) { ERROR("Please specify the minimum FS coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( det->panels[i].max_fs < 0 ) { ERROR("Please specify the maximum FS coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( det->panels[i].min_ss < 0 ) { ERROR("Please specify the minimum SS coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( det->panels[i].max_ss < 0 ) { ERROR("Please specify the maximum SS coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( isnan(det->panels[i].cnx) ) { ERROR("Please specify the corner X coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( isnan(det->panels[i].cny) ) { ERROR("Please specify the corner Y coordinate for" " panel %s\n", det->panels[i].name); reject = 1; } if ( (det->panels[i].clen < 0.0) && (det->panels[i].clen_from == NULL) ) { ERROR("Please specify the camera length for" " panel %s\n", det->panels[i].name); reject = 1; } if ( det->panels[i].res < 0 ) { ERROR("Please specify the resolution for" " panel %s\n", det->panels[i].name); reject = 1; } if ( isnan(det->panels[i].adu_per_eV) ) { ERROR("Please specify the number of ADU per eV for" " panel %s\n", det->panels[i].name); reject = 1; } /* It's OK if the badrow direction is '0' */ /* It's not a problem if "no_index" is still zero */ /* The default transformation matrix is at least valid */ if ( det->panels[i].max_fs > max_fs ) { max_fs = det->panels[i].max_fs; } if ( det->panels[i].max_ss > max_ss ) { max_ss = det->panels[i].max_ss; } det->panels[i].orig_max_fs = det->panels[i].max_fs; det->panels[i].orig_min_fs = det->panels[i].min_fs; det->panels[i].orig_max_ss = det->panels[i].max_ss; det->panels[i].orig_min_ss = det->panels[i].min_ss; } for ( i=0; in_bad; i++ ) { if ( !det->bad[i].is_fsss && isnan(det->bad[i].min_x) ) { ERROR("Please specify the minimum x coordinate for" " bad region %s\n", det->bad[i].name); reject = 1; } if ( !det->bad[i].is_fsss && isnan(det->bad[i].min_y) ) { ERROR("Please specify the minimum y coordinate for" " bad region %s\n", det->bad[i].name); reject = 1; } if ( !det->bad[i].is_fsss && isnan(det->bad[i].max_x) ) { ERROR("Please specify the maximum x coordinate for" " bad region %s\n", det->bad[i].name); reject = 1; } if ( !det->bad[i].is_fsss && isnan(det->bad[i].max_y) ) { ERROR("Please specify the maximum y coordinate for" " bad region %s\n", det->bad[i].name); reject = 1; } } for ( x=0; x<=max_fs; x++ ) { for ( y=0; y<=max_ss; y++ ) { if ( find_panel(det, x, y) == NULL ) { ERROR("Detector geometry invalid: contains gaps.\n"); reject = 1; goto out; } } } out: det->max_fs = max_fs; det->max_ss = max_ss; free(det->defaults.clen_from); free(det->defaults.data); free(det->defaults.mask); /* Calculate matrix inverses and other stuff */ for ( i=0; in_panels; i++ ) { struct panel *p; double d; p = &det->panels[i]; if ( p->fsx*p->ssy == p->ssx*p->fsy ) { ERROR("Panel %i transformation singular.\n", i); reject = 1; } d = (double)p->fsx*p->ssy - p->ssx*p->fsy; p->xfs = p->ssy / d; p->yfs = -p->ssx / d; p->xss = -p->fsy / d; p->yss = p->fsx / d; p->w = p->max_fs - p->min_fs + 1; p->h = p->max_ss - p->min_ss + 1; if ( p->rigid_group == NULL ) { p->rigid_group = find_or_add_rg(det, p->name); } } /* Fix up rigid group panel pointers now that the panels and RGs have * stopped being realloc()ed */ fix_up_rigid_groups(det); find_min_max_d(det); if ( reject ) return NULL; fclose(fh); return det; } void free_detector_geometry(struct detector *det) { int i; rigid_groups_free(det); for ( i=0; in_panels; i++ ) { free(det->panels[i].clen_from); free_dim_structure(det->panels[i].dim_structure); } free(det->panels); free(det->bad); free(det); } struct detector *copy_geom(const struct detector *in) { struct detector *out; int i; out = malloc(sizeof(struct detector)); memcpy(out, in, sizeof(struct detector)); out->panels = malloc(out->n_panels * sizeof(struct panel)); memcpy(out->panels, in->panels, out->n_panels * sizeof(struct panel)); out->bad = malloc(out->n_bad * sizeof(struct badregion)); memcpy(out->bad, in->bad, out->n_bad * sizeof(struct badregion)); out->n_rigid_groups = 0; out->rigid_groups = NULL; for ( i=0; in_panels; i++ ) { struct panel *p; p = &out->panels[i]; if ( p->clen_from != NULL ) { /* Make a copy of the clen_from fields unique to this * copy of the structure. */ p->clen_from = strdup(p->clen_from); } if ( p->data != NULL ) { /* Make a copy of the data fields unique to this * copy of the structure. */ p->clen_from = strdup(p->clen_from); } if ( p->clen_from != NULL ) { /* Make a copy of the mask fields unique to this * copy of the structure. */ p->clen_from = strdup(p->clen_from); } } for ( i=0; in_panels; i++ ) { struct rigid_group *rg; rg = in->panels[i].rigid_group; if ( rg == NULL ) continue; out->panels[i].rigid_group = find_or_add_rg(out, rg->name); } fix_up_rigid_groups(out); return out; } struct detector *simple_geometry(const struct image *image) { struct detector *geom; geom = calloc(1, sizeof(struct detector)); geom->n_panels = 1; geom->panels = calloc(1, sizeof(struct panel)); geom->panels[0].min_fs = 0; geom->panels[0].max_fs = image->width-1; geom->panels[0].min_ss = 0; geom->panels[0].max_ss = image->height-1; geom->panels[0].orig_min_fs = 0; geom->panels[0].orig_max_fs = image->width-1; geom->panels[0].orig_min_ss = 0; geom->panels[0].orig_max_ss = image->height-1; geom->panels[0].cnx = -image->width / 2.0; geom->panels[0].cny = -image->height / 2.0; geom->panels[0].rigid_group = NULL; geom->panels[0].max_adu = INFINITY; geom->panels[0].orig_min_fs = -1; geom->panels[0].orig_max_fs = -1; geom->panels[0].orig_min_ss = -1; geom->panels[0].orig_max_ss = -1; geom->panels[0].fsx = 1; geom->panels[0].fsy = 0; geom->panels[0].ssx = 0; geom->panels[0].ssy = 1; geom->panels[0].xfs = 1; geom->panels[0].xss = 0; geom->panels[0].yfs = 0; geom->panels[0].yss = 1; geom->panels[0].w = image->width; geom->panels[0].h = image->height; geom->panels[0].mask = NULL; geom->panels[0].data = NULL; find_min_max_d(geom); return geom; } void twod_mapping(double fs, double ss, double *px, double *py, struct panel *p) { double xs, ys; xs = fs*p->fsx + ss*p->ssx; ys = fs*p->fsy + ss*p->ssy; *px = xs + p->cnx; *py = ys + p->cny; } int reverse_2d_mapping(double x, double y, double *pfs, double *pss, struct detector *det) { int i; for ( i=0; in_panels; i++ ) { struct panel *p = &det->panels[i]; double cx, cy, fs, ss; /* Get position relative to corner */ cx = x - p->cnx; cy = y - p->cny; /* Reverse the transformation matrix */ fs = cx*p->xfs + cy*p->yfs; ss = cx*p->xss + cy*p->yss; /* In range? */ if ( fs < 0 ) continue; if ( ss < 0 ) continue; if ( fs > p->max_fs-p->min_fs ) continue; if ( ss > p->max_ss-p->min_ss ) continue; *pfs = fs + p->min_fs; *pss = ss + p->min_ss; return 0; } return 1; } static void check_extents(struct panel p, double *min_x, double *min_y, double *max_x, double *max_y, double fs, double ss) { double xs, ys, rx, ry; xs = fs*p.fsx + ss*p.ssx; ys = fs*p.fsy + ss*p.ssy; rx = xs + p.cnx; ry = ys + p.cny; if ( rx > *max_x ) *max_x = rx; if ( ry > *max_y ) *max_y = ry; if ( rx < *min_x ) *min_x = rx; if ( ry < *min_y ) *min_y = ry; } double largest_q(struct image *image) { struct rvec q; double tt; q = get_q_for_panel(image->det->furthest_out_panel, image->det->furthest_out_fs, image->det->furthest_out_ss, &tt, 1.0/image->lambda); return modulus(q.u, q.v, q.w); } double smallest_q(struct image *image) { struct rvec q; double tt; q = get_q_for_panel(image->det->furthest_in_panel, image->det->furthest_in_fs, image->det->furthest_in_ss, &tt, 1.0/image->lambda); return modulus(q.u, q.v, q.w); } void get_pixel_extents(struct detector *det, double *min_x, double *min_y, double *max_x, double *max_y) { int i; *min_x = 0.0; *max_x = 0.0; *min_y = 0.0; *max_y = 0.0; /* To determine the maximum extents of the detector, put all four * corners of each panel through the transformations and watch for the * biggest */ for ( i=0; in_panels; i++ ) { check_extents(det->panels[i], min_x, min_y, max_x, max_y, 0.0, 0.0); check_extents(det->panels[i], min_x, min_y, max_x, max_y, 0.0, det->panels[i].max_ss-det->panels[i].min_ss+1); check_extents(det->panels[i], min_x, min_y, max_x, max_y, det->panels[i].max_fs-det->panels[i].min_fs+1, 0.0); check_extents(det->panels[i], min_x, min_y, max_x, max_y, det->panels[i].max_fs-det->panels[i].min_fs+1, det->panels[i].max_ss-det->panels[i].min_ss+1); } } int write_detector_geometry(const char *geometry_filename, const char *output_filename, struct detector *det) { FILE *ifh; FILE *fh; if ( geometry_filename == NULL ) return 2; if ( output_filename == NULL ) return 2; if ( det->n_panels < 1 ) return 3; ifh = fopen(geometry_filename, "r"); if ( ifh == NULL ) return 1; fh = fopen(output_filename, "w"); if ( fh == NULL ) return 1; do { char *rval; char line[1024]; int n_bits; char **bits; int i; struct panel *p; rval = fgets(line, 1023, ifh); if ( rval == NULL ) break; n_bits = assplode(line, "/", &bits, ASSPLODE_NONE); if ( n_bits < 2 || n_bits > 2 ) { for ( i=0; iname, p->fsx, p->fsy); } else if ( strncmp(bits[1], "ss = ", 5) == 0) { fprintf(fh, "%s/ss = %+fx %+fy\n", p->name, p->ssx, p->ssy); } else if ( strncmp(bits[1], "corner_x = ", 11) == 0) { fprintf(fh, "%s/corner_x = %g\n", p->name, p->cnx); } else if ( strncmp(bits[1], "corner_y = ", 11) == 0) { fprintf(fh, "%s/corner_y = %g\n", p->name, p->cny); } else { fputs(line, fh); } for ( i=0; idet->n_panels; i++ ) { int fs, ss; struct panel *p = &image->det->panels[i]; for ( ss=0; ssh; ss++ ) { for ( fs=0; fsw; fs++ ) { struct rvec q; double r; q = get_q_for_panel(p, fs, ss, NULL, 1.0/image->lambda); r = modulus(q.u, q.v, q.w); if ( (r >= min) && (r <= max) ) { image->bad[i][fs+p->w*ss] = 1; } } } } } extern int single_panel_data_source(struct detector *det, const char *element) { int pi; char *first_datafrom = NULL; char *curr_datafrom = NULL; if ( det->panels[0].data == NULL ) { if ( element != NULL ) { first_datafrom = strdup(element); } else { first_datafrom = strdup("/data/data"); } } else { first_datafrom = strdup(det->panels[0].data); } for ( pi=1;pin_panels;pi++ ) { if ( det->panels[pi].data == NULL ) { if ( element != NULL ) { curr_datafrom = strdup(element); } else { curr_datafrom = strdup("/data/data"); } } else { curr_datafrom = strdup(det->panels[pi].data); } if ( strcmp(curr_datafrom, first_datafrom) != 0 ) { return 0; } } free(first_datafrom); free(curr_datafrom); return 1; }