aboutsummaryrefslogtreecommitdiff
path: root/src
diff options
context:
space:
mode:
authorThomas White <taw@physics.org>2011-03-15 15:56:26 +0100
committerThomas White <taw@physics.org>2012-02-22 15:27:20 +0100
commitb8ac0faea1abb2e3033140dcd157160bfa98d251 (patch)
treee62dd9269372375c95936d72c33187a5945edf29 /src
parentbe49b97c792edc33d908965823f73f33b830d557 (diff)
WIP on new stream format in process_hkl
Diffstat (limited to 'src')
-rw-r--r--src/process_hkl.c570
-rw-r--r--src/reflist-utils.c72
-rw-r--r--src/reflist-utils.h30
-rw-r--r--src/reflist.c43
-rw-r--r--src/reflist.h6
-rw-r--r--src/stream.c128
-rw-r--r--src/stream.h4
7 files changed, 356 insertions, 497 deletions
diff --git a/src/process_hkl.c b/src/process_hkl.c
index 8df2dae1..da3ee3f8 100644
--- a/src/process_hkl.c
+++ b/src/process_hkl.c
@@ -3,7 +3,7 @@
*
* Assemble and process FEL Bragg intensities
*
- * (c) 2006-2010 Thomas White <taw@physics.org>
+ * (c) 2006-2011 Thomas White <taw@physics.org>
*
* Part of CrystFEL - crystallography with a FEL
*
@@ -24,10 +24,12 @@
#include "utils.h"
#include "statistics.h"
#include "sfac.h"
-#include "reflections.h"
+#include "reflist-utils.h"
#include "symmetry.h"
#include "stream.h"
#include "beam-parameters.h"
+#include "reflist.h"
+#include "image.h"
/* Number of divisions for intensity histograms */
@@ -43,7 +45,7 @@ static void show_help(const char *s)
" -h, --help Display this help message.\n"
" -i, --input=<filename> Specify input filename (\"-\" for stdin).\n"
" -o, --output=<filename> Specify output filename for merged intensities\n"
-" (don't specify for no output).\n"
+" Default: processed.hkl).\n"
" -p, --pdb=<filename> PDB file to use (default: molecule.pdb).\n"
" -b, --beam=<file> Get beam parameters from file (used for sigmas).\n"
"\n"
@@ -66,13 +68,10 @@ static void show_help(const char *s)
" --scale Scale each pattern for best fit with the current\n"
" model.\n"
" -y, --symmetry=<sym> Merge according to point group <sym>.\n"
-" -a, --input-symmetry=<a> Specify the apparent (input) symmetry.\n"
" --reference=<file> Compare against intensities from <file> when\n"
" scaling or resolving ambiguities.\n"
" The symmetry of the reference list must be the\n"
" same as that given with '-y'.\n"
-" --outstream=<file> Write an annotated version of the input stream\n"
-" to <file>.\n"
);
}
@@ -119,175 +118,55 @@ static void plot_histogram(double *vals, int n)
}
-/* Note "holo" needn't actually be a holohedral point group, if you want to try
- * something strange like resolving from a low-symmetry group into an even
- * lower symmetry one.
- */
-static ReflItemList *get_twin_possibilities(const char *holo, const char *mero)
-{
- ReflItemList *test_items;
- ReflItemList *twins;
- int np;
-
- np = num_general_equivs(holo) / num_general_equivs(mero);
-
- test_items = new_items();
-
- /* Some arbitrarily chosen reflections which can't be special
- * reflections in any point group, i.e. lots of odd numbers,
- * prime numbers and so on. There's probably an analytical
- * way of working these out, but this will do. */
- add_item(test_items, 1, 2, 3);
- add_item(test_items, 3, 7, 13);
- add_item(test_items, 5, 2, 1);
-
- twins = get_twins(test_items, holo, mero);
- delete_items(test_items);
-
- /* Idiot check. Wouldn't be necessary if I could prove that the above
- * set of arbitrarily chosen reflections were always general. */
- if ( num_items(twins) != np ) {
- ERROR("Whoops! Couldn't find all the twinning possiblities.\n");
- abort();
- }
-
- return twins;
-}
-
-
-static int resolve_twin(const double *model, ReflItemList *observed,
- const double *patt, ReflItemList *items,
- ReflItemList *twins, const char *holo, const char *mero)
-{
- int n, i;
- double best_fom = 0.0;
- int best_op = 0;
-
- n = num_items(twins);
-
- for ( i=0; i<n; i++ ) {
-
- int j;
- int op;
- double *trial_ints = new_list_intensity();
- unsigned int *trial_counts = new_list_count();
- double fom;
- ReflItemList *intersection;
-
- op = get_item(twins, i)->op;
-
- for ( j=0; j<num_items(items); j++ ) {
-
- signed int h, k, l;
- struct refl_item *r = get_item(items, j);
-
- get_general_equiv(r->h, r->k, r->l, &h, &k, &l,
- holo, op);
- get_asymm(h, k, l, &h, &k, &l, mero);
-
- set_intensity(trial_ints, h, k, l,
- lookup_intensity(patt, r->h, r->k, r->l));
- set_count(trial_counts, h, k, l, 1);
-
- }
-
- intersection = intersection_items(observed, items);
- fom = stat_pearson_i(trial_ints, model, intersection);
- delete_items(intersection);
-
- free(trial_ints);
- free(trial_counts);
-
- //printf(" %f", fom);
- if ( fom > best_fom ) {
- best_fom = fom;
- best_op = op;
- }
-
- }
- //printf("\n");
-
- return best_op;
-}
-
-
-static void merge_pattern(double *model, ReflItemList *observed,
- const double *new, ReflItemList *items,
- unsigned int *model_counts, int mo,
- ReflItemList *twins,
- const char *holo, const char *mero,
- ReflItemList *reference, const double *reference_i,
- double *hist_vals,
- signed int hist_h, signed int hist_k,
- signed int hist_l, int *hist_n, double *devs,
- double *means, FILE *outfh)
+static void merge_pattern(RefList *model, RefList *new, int max_only,
+ const char *sym,
+ double *hist_vals, signed int hist_h,
+ signed int hist_k, signed int hist_l, int *hist_n)
{
- int i;
- int twin;
- ReflItemList *sym_items = new_items();
-
- if ( twins != NULL ) {
- if ( reference != NULL ) {
- twin = resolve_twin(reference_i, reference, new, items,
- twins, holo, mero);
- } else {
- twin = resolve_twin(model, observed, new, items,
- twins, holo, mero);
- }
- } else {
- twin = 0;
- }
-
- if ( outfh != NULL ) {
- fprintf(outfh, "twin=%i\n", twin);
- }
+ Reflection *refl;
+ RefListIterator *iter;
- for ( i=0; i<num_items(items); i++ ) {
+ for ( refl = first_refl(new, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
double intensity;
- signed int hs, ks, ls;
signed int h, k, l;
- struct refl_item *item;
-
- item = get_item(items, i);
+ Reflection *model_version;
+ double model_int;
- hs = item->h;
- ks = item->k;
- ls = item->l;
-
- /* Transform into correct side of the twin law.
- * "twin" is always zero if no de-twinning is performed. */
- get_general_equiv(hs, ks, ls, &h, &k, &l, holo, twin);
+ get_indices(refl, &h, &k, &l);
+ model_version = find_refl(model, h, k, l);
+ if ( model_version == NULL ) {
+ model_version = add_refl(model, h, k, l);
+ }
/* Put into the asymmetric unit for the target group */
- get_asymm(h, k, l, &h, &k, &l, mero);
+ get_asymm(h, k, l, &h, &k, &l, sym);
/* Read the intensity from the original location
* (i.e. before screwing around with symmetry) */
- intensity = lookup_intensity(new, hs, ks, ls);
+ intensity = get_intensity(refl);
+
+ /* Get the current model intensity */
+ model_int = get_intensity(model_version);
/* User asked for max only? */
- if ( !mo ) {
- integrate_intensity(model, h, k, l, intensity);
+ if ( !max_only ) {
+ set_int(model_version, model_int + intensity);
} else {
- if ( intensity > lookup_intensity(model, h, k, l) ) {
- set_intensity(model, h, k, l, intensity);
+ if ( intensity > get_intensity(model_version) ) {
+ set_int(model_version, intensity);
}
}
- if ( devs != NULL ) {
- double m;
- m = lookup_intensity(means, h, k, l);
- integrate_intensity(devs, h, k, l,
- pow(intensity-m, 2.0));
- }
-
- if ( !find_item(sym_items, h, k, l) ) {
- add_item(sym_items, h, k, l);
- }
+ double dev = get_sum_squared_dev(model_version);
+ set_sum_squared_dev(model_version,
+ dev + pow(intensity-model_int, 2.0));
- /* Increase count */
- integrate_count(model_counts, h, k, l, 1);
+ /* Increase redundancy */
+ int cur_redundancy = get_redundancy(model_version);
+ set_redundancy(model_version, cur_redundancy+1);
if ( hist_vals != NULL ) {
int p = *hist_n;
@@ -298,11 +177,6 @@ static void merge_pattern(double *model, ReflItemList *observed,
}
}
-
- /* Dump the reflections in this pattern into the overall list */
- union_items(observed, sym_items);
-
- delete_items(sym_items);
}
@@ -314,34 +188,36 @@ enum {
};
-static void scale_intensities(const double *model, ReflItemList *model_items,
- double *new_pattern, ReflItemList *new_items,
- double f0, int f0_valid, const char *sym)
+static void scale_intensities(RefList *model, RefList *new, const char *sym)
{
double s;
double top = 0.0;
double bot = 0.0;
- unsigned int i;
const int scaling = SCALE_INTPERBRAGG;
+ Reflection *refl;
+ RefListIterator *iter;
+ Reflection *model_version;
- for ( i=0; i<num_items(new_items); i++ ) {
+ for ( refl = first_refl(new, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
double i1, i2;
- struct refl_item *it;
signed int hu, ku, lu;
+ signed int h, k, l;
- /* Get the next item in the list of new reflections */
- it = get_item(new_items, i);
+ get_indices(refl, &h, &k, &l);
switch ( scaling ) {
case SCALE_TWOPASS :
- /* Find the (only) partner in the model */
- find_unique_equiv(model_items, it->h, it->k, it->l, sym,
- &hu, &ku, &lu);
+ model_version = find_refl(model, h, k, l);
+ if ( model_version == NULL ) continue;
+
+ get_asymm(h, k, l, &hu, &ku, &lu, sym);
- i1 = lookup_intensity(model, hu, ku, lu);
- i2 = lookup_intensity(new_pattern, it->h, it->k, it->l);
+ i1 = get_intensity(model_version);
+ i2 = get_intensity(refl);
/* Calculate LSQ estimate of scaling factor */
top += i1 * i2;
@@ -352,7 +228,7 @@ static void scale_intensities(const double *model, ReflItemList *model_items,
case SCALE_CONSTINT :
/* Sum up the intensity in the pattern */
- i2 = lookup_intensity(new_pattern, it->h, it->k, it->l);
+ i2 = get_intensity(refl);
top += i2;
break;
@@ -360,7 +236,7 @@ static void scale_intensities(const double *model, ReflItemList *model_items,
case SCALE_INTPERBRAGG :
/* Sum up the intensity in the pattern */
- i2 = lookup_intensity(new_pattern, it->h, it->k, it->l);
+ i2 = get_intensity(refl);
top += i2;
bot += 1.0;
@@ -381,162 +257,108 @@ static void scale_intensities(const double *model, ReflItemList *model_items,
s = 1000.0 / (top/bot);
break;
}
- //if ( f0_valid ) printf("%f %f\n", s, f0);
/* Multiply the new pattern up by "s" */
- for ( i=0; i<LIST_SIZE; i++ ) {
- new_pattern[i] *= s;
+ for ( refl = first_refl(new, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
+
+ double intensity = get_intensity(refl);
+ set_int(refl, intensity*s);
+
}
}
-static void merge_all(FILE *fh, double **pmodel, ReflItemList **pobserved,
- unsigned int **pcounts,
+static void merge_all(FILE *fh, RefList *model,
int config_maxonly, int config_scale, int config_sum,
int config_startafter, int config_stopafter,
- ReflItemList *twins, const char *holo, const char *mero,
+ const char *sym,
int n_total_patterns,
- ReflItemList *reference, double *reference_i,
- double *hist_vals,
- signed int hist_h, signed int hist_k, signed int hist_l,
- int *hist_i, double *devs, double *means, FILE *outfh)
+ double *hist_vals, signed int hist_h,
+ signed int hist_k, signed int hist_l,
+ int *hist_i)
{
- char *rval;
- float f0;
- int n_nof0 = 0;
- int f0_valid = 0;
+ int rval;
int n_patterns = 0;
- double *new_pattern = new_list_intensity();
- ReflItemList *items = new_items();
- ReflItemList *observed = new_items();
- double *model = new_list_intensity();
- unsigned int *counts = new_list_count();
- int i;
-
- if ( config_startafter != 0 ) {
-
- do {
-
- char line[1024];
-
- rval = fgets(line, 1023, fh);
- if ( (strncmp(line, "Reflections from indexing", 25)
- == 0)
- || (strncmp(line, "New pattern", 11) == 0) ) {
- n_patterns++;
- }
-
- if ( n_patterns == config_startafter ) break;
-
- } while ( rval != NULL );
+ int n_used = 0;
+ Reflection *refl;
+ RefListIterator *iter;
+ if ( skip_some_files(fh, config_startafter) ) {
+ ERROR("Failed to skip first %i files.\n", config_startafter);
+ return;
}
do {
- char line[1024];
- signed int h, k, l;
- float intensity;
- int r;
+ struct image image;
- rval = fgets(line, 1023, fh);
- if ( ((strncmp(line, "Reflections from indexing", 25) == 0)
- || (rval == NULL)) && (num_items(items)>0) ) {
+ /* Get data from next chunk */
+ rval = read_chunk(fh, &image);
+ if ( rval ) continue;
- /* Assume a default I0 if we don't have one by now */
- if ( config_scale && !f0_valid ) {
- n_nof0++;
- f0 = 1.0;
- }
+ n_patterns++;
+
+ if ( (image.reflections != NULL) && (image.indexed_cell) ) {
/* Scale if requested */
if ( config_scale ) {
- if ( reference == NULL ) {
- scale_intensities(model, observed,
- new_pattern, items,
- f0, f0_valid, mero);
- } else {
- scale_intensities(reference_i,
- reference,
- new_pattern, items,
- f0, f0_valid, mero);
- }
+ scale_intensities(model, image.reflections,
+ sym);
}
- /* Start of second or later pattern */
- merge_pattern(model, observed, new_pattern, items,
- counts, config_maxonly,
- twins, holo, mero,
- reference, reference_i,
- hist_vals, hist_h, hist_k, hist_l,
- hist_i, devs, means, outfh);
-
- n_patterns++;
- if ( n_patterns == config_stopafter ) {
- progress_bar(n_total_patterns, n_total_patterns,
- "Merging");
- break;
- }
- progress_bar(n_patterns, n_total_patterns, "Merging");
+ merge_pattern(model, image.reflections, config_maxonly,
+ sym, hist_vals, hist_h, hist_k, hist_l,
+ hist_i);
- /* Reset for the next pattern */
- clear_items(items);
-
- f0_valid = 0;
+ n_used++;
}
- if ( outfh != NULL ) {
- fprintf(outfh, "%s", line);
- }
+ reflist_free(image.reflections);
+ image_feature_list_free(image.features);
+ cell_free(image.indexed_cell);
- if ( strncmp(line, "f0 = ", 5) == 0 ) {
- r = sscanf(line, "f0 = %f", &f0);
- if ( r != 1 ) {
- f0 = 1.0;
- f0_valid = 0;
- continue;
- }
- f0_valid = 1;
- }
+ progress_bar(n_patterns, n_total_patterns-config_startafter,
+ "Merging");
- r = sscanf(line, "%i %i %i %f", &h, &k, &l, &intensity);
- if ( r != 4 ) continue;
+ } while ( rval == 0 );
- /* Not interested in the central beam */
- if ( (h==0) && (k==0) && (l==0) ) continue;
+ /* Divide by counts to get mean intensity if necessary */
+ if ( !config_sum && !config_maxonly ) {
- /* The same raw indices (before mapping into the asymmetric
- * unit should not turn up twice in one pattern. */
- if ( find_item(items, h, k, l) != 0 ) {
- ERROR("More than one measurement for %i %i %i in"
- " pattern number %i\n", h, k, l, n_patterns);
- }
- set_intensity(new_pattern, h, k, l, intensity);
+ Reflection *refl;
+ RefListIterator *iter;
- /* NB: This list contains raw indices, before working out
- * where they belong in the asymmetric unit. */
- add_item(items, h, k, l);
+ for ( refl = first_refl(model, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
- } while ( rval != NULL );
+ double intensity = get_intensity(refl);
+ double sum_squared_dev = get_sum_squared_dev(refl);
+ int red = get_redundancy(refl);
- delete_items(items);
- free(new_pattern);
+ set_int(refl, intensity / (double)red);
+ set_sum_squared_dev(refl, sum_squared_dev/(double)red);
- /* Calculate mean intensity if necessary */
- if ( !config_sum && !config_maxonly ) {
- for ( i=0; i<IDIM*IDIM*IDIM; i++ ) {
- if ( counts[i] > 0 ) {
- model[i] /= (double)counts[i];
- }
}
+
}
- *pmodel = model;
- *pcounts = counts;
- *pobserved = observed;
+ /* Calculate ESDs */
+ for ( refl = first_refl(model, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
+
+ double sum_squared_dev = get_sum_squared_dev(refl);
+ int red = get_redundancy(refl);
+
+ set_esd_intensity(refl, sum_squared_dev/(double)red);
+
+ }
- STATUS("%i patterns had no f0 valid value.\n", n_nof0);
+ STATUS("%i of the patterns could be used.\n", n_used);
}
@@ -546,35 +368,22 @@ int main(int argc, char *argv[])
char *filename = NULL;
char *output = NULL;
FILE *fh;
- double *model;
- unsigned int *counts;
+ RefList *model;
UnitCell *cell = NULL;
int config_maxonly = 0;
int config_startafter = 0;
int config_stopafter = 0;
int config_sum = 0;
int config_scale = 0;
- int config_rmerge = 0;
unsigned int n_total_patterns;
char *sym = NULL;
- char *in_sym = NULL;
char *pdb = NULL;
- ReflItemList *twins;
- ReflItemList *observed;
- int i;
char *histo = NULL;
signed int hist_h, hist_k, hist_l;
double *hist_vals = NULL;
int hist_i;
- char *outstream = NULL;
- char *reference = NULL;
- ReflItemList *reference_items;
- double *reference_i;
- FILE *outfh = NULL;
struct beam_params *beam = NULL;
int space_for_hist = 0;
- double *devs;
- double *esds;
/* Long options */
const struct option longopts[] = {
@@ -588,18 +397,14 @@ int main(int argc, char *argv[])
{"sum", 0, &config_sum, 1},
{"scale", 0, &config_scale, 1},
{"symmetry", 1, NULL, 'y'},
- {"input-symmetry", 1, NULL, 'a'},
{"pdb", 1, NULL, 'p'},
{"histogram", 1, NULL, 'g'},
- {"rmerge", 0, &config_rmerge, 1},
- {"outstream", 1, NULL, 2},
- {"reference", 1, NULL, 'r'},
{"beam", 1, NULL, 'b'},
{0, 0, NULL, 0}
};
/* Short options */
- while ((c = getopt_long(argc, argv, "hi:e:r:o:p:y:g:f:a:b:",
+ while ((c = getopt_long(argc, argv, "hi:e:o:p:y:g:f:b:",
longopts, NULL)) != -1) {
switch (c) {
@@ -635,18 +440,6 @@ int main(int argc, char *argv[])
histo = strdup(optarg);
break;
- case 'r' :
- reference = strdup(optarg);
- break;
-
- case 2 :
- outstream = strdup(optarg);
- break;
-
- case 'a' :
- in_sym = strdup(optarg);
- break;
-
case 'b' :
beam = get_beam_parameters(optarg);
if ( beam == NULL ) {
@@ -665,17 +458,15 @@ int main(int argc, char *argv[])
}
- if ( config_sum && config_rmerge ) {
- ERROR("Options --sum and --rmerge do not make sense"
- " together.\n");
- return 1;
- }
-
if ( filename == NULL ) {
ERROR("Please specify filename using the -i option\n");
return 1;
}
+ if ( output == NULL ) {
+ output = strdup("processed.hkl");
+ }
+
if ( pdb == NULL ) {
pdb = strdup("molecule.pdb");
}
@@ -685,35 +476,6 @@ int main(int argc, char *argv[])
if ( sym == NULL ) sym = strdup("1");
- /* Show useful symmetry information */
- if ( (sym != NULL) && (in_sym != NULL) ) {
-
- int np = num_general_equivs(in_sym) / num_general_equivs(sym);
- if ( np > 1 ) {
-
- STATUS("Resolving point group %s into %s "
- "(%i possibilities)\n",
- in_sym, sym, np);
- /* Get the list of twin/Bijvoet possibilities */
- twins = get_twin_possibilities(in_sym, sym);
- STATUS("Twin operation indices from %s are:", in_sym);
- for ( i=0; i<num_items(twins); i++ ) {
- STATUS(" %i", get_item(twins, i)->op);
- }
- STATUS("\n");
-
- } else {
- STATUS("No resolution required to get from %s to %s\n",
- in_sym, sym);
- twins = NULL;
- }
-
- } else {
- STATUS("No twin resolution requested (use -a otherwise).\n");
- twins = NULL;
- in_sym = strdup(sym);
- }
-
/* Open the data stream */
if ( strcmp(filename, "-") == 0 ) {
fh = stdin;
@@ -726,33 +488,13 @@ int main(int argc, char *argv[])
return 1;
}
- /* Read the reference reflections */
- if ( reference != NULL ) {
- reference_i = new_list_intensity();
- reference_items = read_reflections(reference, reference_i,
- NULL, NULL, NULL);
- if ( reference_items == NULL ) {
- ERROR("Couldn't read '%s'\n", reference);
- return 1;
- }
- } else {
- reference_items = NULL;
- reference_i = NULL;
- }
-
- if ( outstream != NULL ) {
- outfh = fopen(outstream, "w");
- if ( outfh == NULL ) {
- ERROR("Couldn't open '%s'\n", outstream);
- return 1;
- }
- }
-
/* Count the number of patterns in the file */
n_total_patterns = count_patterns(fh);
STATUS("There are %i patterns to process\n", n_total_patterns);
rewind(fh);
+ model = reflist_new();
+
if ( histo != NULL ) {
int r;
r = sscanf(histo, "%i,%i,%i", &hist_h, &hist_k, &hist_l);
@@ -771,13 +513,10 @@ int main(int argc, char *argv[])
}
hist_i = 0;
- merge_all(fh, &model, &observed, &counts,
- config_maxonly, config_scale, config_sum,
+ merge_all(fh, model, config_maxonly, config_scale, config_sum,
config_startafter, config_stopafter,
- twins, in_sym, sym, n_total_patterns,
- reference_items, reference_i,
- hist_vals, hist_h, hist_k, hist_l, &hist_i, NULL, NULL,
- outfh);
+ sym, n_total_patterns,
+ hist_vals, hist_h, hist_k, hist_l, &hist_i);
rewind(fh);
if ( space_for_hist && (hist_i >= space_for_hist) ) {
ERROR("Histogram array was too small!\n");
@@ -790,60 +529,17 @@ int main(int argc, char *argv[])
}
STATUS("Extra pass to calculate ESDs...\n");
- devs = new_list_intensity();
- esds = new_list_sigma();
rewind(fh);
- merge_all(fh, &model, &observed, &counts,
- config_maxonly, config_scale, 0,
- config_startafter, config_stopafter,
- twins, in_sym, sym,
- n_total_patterns, reference_items, reference_i,
- NULL, 0, 0, 0, NULL, devs, model, NULL);
+ merge_all(fh, model, config_maxonly, config_scale, 0,
+ config_startafter, config_stopafter, sym, n_total_patterns,
+ NULL, 0, 0, 0, NULL);
- for ( i=0; i<num_items(observed); i++ ) {
-
- struct refl_item *it;
- signed int h, k, l;
- double dev, esd;
- unsigned int count;
-
- it = get_item(observed, i);
- h = it->h; k = it->k, l = it->l;
-
- dev = lookup_intensity(devs, h, k, l);
- count = lookup_count(counts, h, k, l);
-
- if ( count < 2 ) {
- /* If we have only one measurement, the error is 100% */
- esd = lookup_sigma(model, h, k, l);
- } else {
- esd = sqrt(dev) / (double)count;
- }
- set_sigma(esds, h, k, l, esd);
-
- }
-
- if ( output != NULL ) {
-
- double adu_per_photon;
-
- if ( beam == NULL ) {
- adu_per_photon = 167.0;
- STATUS("No beam parameters file provided (use -b), "
- "so I'm assuming 167.0 ADU per photon.\n");
- } else {
- adu_per_photon = beam->adu_per_photon;
- }
-
- write_reflections(output, observed, model, esds,
- NULL, counts, cell);
-
- }
+ write_reflist(output, model, cell);
+ cell_free(cell);
fclose(fh);
free(sym);
- free(model);
- free(counts);
+ reflist_free(model);
free(output);
if ( cell != NULL ) cell_free(cell);
diff --git a/src/reflist-utils.c b/src/reflist-utils.c
new file mode 100644
index 00000000..33d373b8
--- /dev/null
+++ b/src/reflist-utils.c
@@ -0,0 +1,72 @@
+/*
+ * reflist-utils.c
+ *
+ * Utilities to complement the core reflist.c
+ *
+ * (c) 2006-2011 Thomas White <taw@physics.org>
+ *
+ * Part of CrystFEL - crystallography with a FEL
+ *
+ */
+
+
+#include <stdio.h>
+
+
+#include "reflist.h"
+#include "cell.h"
+
+
+void write_reflections_to_file(FILE *fh, RefList *list, UnitCell *cell)
+{
+ Reflection *refl;
+ RefListIterator *iter;
+
+ fprintf(fh, " h k l I phase sigma(I) "
+ " 1/d(nm^-1) counts fs/px ss/px\n");
+
+ for ( refl = first_refl(list, &iter);
+ refl != NULL;
+ refl = next_refl(refl, iter) ) {
+
+ signed int h, k, l;
+ double intensity, esd_i, s;
+ double fs, ss;
+
+ get_indices(refl, &h, &k, &l);
+ get_detector_pos(refl, &fs, &ss);
+ intensity = get_intensity(refl);
+ esd_i = 0.0; /* FIXME! */
+ s = 0.0; /* FIXME! */
+
+ /* h, k, l, I, sigma(I), s */
+ fprintf(fh,
+ "%3i %3i %3i %10.2f %s %10.2f %10.2f %7i %6.1f %6.1f\n",
+ h, k, l, intensity, " -", esd_i, s/1.0e9, 1,
+ fs, ss);
+
+ }
+}
+
+
+int write_reflist(const char *filename, RefList *list, UnitCell *cell)
+{
+ FILE *fh;
+
+ if ( filename == NULL ) {
+ fh = stdout;
+ } else {
+ fh = fopen(filename, "w");
+ }
+
+ if ( fh == NULL ) {
+ ERROR("Couldn't open output file '%s'.\n", filename);
+ return 1;
+ }
+
+ write_reflections_to_file(fh, list, cell);
+
+ fclose(fh);
+
+ return 0;
+}
diff --git a/src/reflist-utils.h b/src/reflist-utils.h
new file mode 100644
index 00000000..d5c1e655
--- /dev/null
+++ b/src/reflist-utils.h
@@ -0,0 +1,30 @@
+/*
+ * reflist-utils.h
+ *
+ * Utilities to complement the core reflist.c
+ *
+ * (c) 2006-2011 Thomas White <taw@physics.org>
+ *
+ * Part of CrystFEL - crystallography with a FEL
+ *
+ */
+
+#ifdef HAVE_CONFIG_H
+#include <config.h>
+#endif
+
+#ifndef REFLIST_UTILS_H
+#define REFLIST_UTILS_H
+
+
+#include "reflist.h"
+#include "cell.h"
+
+
+extern void write_reflections_to_file(FILE *fh, RefList *list, UnitCell *cell);
+
+extern int write_reflist(const char *filename, RefList *list,
+ UnitCell *cell);
+
+
+#endif /* REFLIST_UTILS_H */
diff --git a/src/reflist.c b/src/reflist.c
index 3a53330c..3181b418 100644
--- a/src/reflist.c
+++ b/src/reflist.c
@@ -45,6 +45,13 @@ struct _refldata {
/* Intensity */
double intensity;
double esd_i;
+
+ /* Redundancy */
+ int redundancy;
+
+ /* Total squared difference between all estimates of this reflection
+ * and the estimated mean value */
+ double sum_squared_dev;
};
@@ -225,6 +232,24 @@ int get_scalable(Reflection *refl)
}
+int get_redundancy(Reflection *refl)
+{
+ return refl->data.redundancy;
+}
+
+
+double get_sum_squared_dev(Reflection *refl)
+{
+ return refl->data.sum_squared_dev;
+}
+
+
+double get_esd_intensity(Reflection *refl)
+{
+ return refl->data.esd_i;
+}
+
+
/********************************** Setters ***********************************/
void set_detector_pos(Reflection *refl, double exerr, double x, double y)
@@ -258,6 +283,24 @@ void set_scalable(Reflection *refl, int scalable)
}
+void set_redundancy(Reflection *refl, int red)
+{
+ refl->data.redundancy = red;
+}
+
+
+void set_sum_squared_dev(Reflection *refl, double dev)
+{
+ refl->data.sum_squared_dev = dev;
+}
+
+
+void set_esd_intensity(Reflection *refl, double esd)
+{
+ refl->data.esd_i = esd;
+}
+
+
/********************************* Insertion **********************************/
static void insert_node(Reflection *head, Reflection *new)
diff --git a/src/reflist.h b/src/reflist.h
index e25a16ad..f6d01ecf 100644
--- a/src/reflist.h
+++ b/src/reflist.h
@@ -41,6 +41,9 @@ extern double get_intensity(Reflection *refl);
extern void get_partial(Reflection *refl, double *r1, double *r2, double *p,
int *clamp_low, int *clamp_high);
extern int get_scalable(Reflection *refl);
+extern int get_redundancy(Reflection *refl);
+extern double get_sum_squared_dev(Reflection *refl);
+extern double get_esd_intensity(Reflection *refl);
/* Set */
extern void set_detector_pos(Reflection *refl, double exerr,
@@ -49,6 +52,9 @@ extern void set_partial(Reflection *refl, double r1, double r2, double p,
double clamp_low, double clamp_high);
extern void set_int(Reflection *refl, double intensity);
extern void set_scalable(Reflection *refl, int scalable);
+extern void set_redundancy(Reflection *refl, int red);
+extern void set_sum_squared_dev(Reflection *refl, double dev);
+extern void set_esd_intensity(Reflection *refl, double esd);
/* Insertion */
extern Reflection *add_refl(RefList *list, INDICES);
diff --git a/src/stream.c b/src/stream.c
index 2a75216d..4b2bea9f 100644
--- a/src/stream.c
+++ b/src/stream.c
@@ -23,6 +23,7 @@
#include "image.h"
#include "stream.h"
#include "reflist.h"
+#include "reflist-utils.h"
#define CHUNK_START_MARKER "----- Begin chunk -----"
@@ -32,7 +33,6 @@
#define REFLECTION_START_MARKER "Reflections measured after indexing"
#define REFLECTION_END_MARKER "End of reflections"
-
static void exclusive(const char *a, const char *b)
{
ERROR("The stream options '%s' and '%s' are mutually exclusive.\n",
@@ -103,10 +103,9 @@ int count_patterns(FILE *fh)
rval = fgets(line, 1023, fh);
if ( rval == NULL ) continue;
- if ( (strncmp(line, "Reflections from indexing", 25) == 0)
- || (strncmp(line, "New pattern", 11) == 0) ) {
- n_total_patterns++;
- }
+ chomp(line);
+ if ( strcmp(line, CHUNK_END_MARKER) == 0 ) n_total_patterns++;
+
} while ( rval != NULL );
return n_total_patterns;
@@ -147,6 +146,7 @@ static UnitCell *read_orientation_matrix(FILE *fh)
static int read_reflections(FILE *fh, struct image *image)
{
char *rval = NULL;
+ int first = 1;
image->reflections = reflist_new();
@@ -164,17 +164,20 @@ static int read_reflections(FILE *fh, struct image *image)
if ( rval == NULL ) continue;
chomp(line);
- if ( strcmp(line, PEAK_LIST_END_MARKER) == 0 ) return 0;
+ if ( strcmp(line, REFLECTION_END_MARKER) == 0 ) return 0;
r = sscanf(line, "%i %i %i %f %s %f %f %i %f %f",
&h, &k, &l, &intensity, phs, &sigma, &res, &cts,
&fs, &ss);
- if ( r != 10 ) return 1;
-
- refl = add_refl(image->reflections, h, k, l);
- set_int(refl, intensity);
- set_detector_pos(refl, fs, ss, 0.0);
- /* FIXME: Set ESD */
+ if ( (r != 10) && (!first) ) return 1;
+
+ first = 0;
+ if ( r == 10 ) {
+ refl = add_refl(image->reflections, h, k, l);
+ set_int(refl, intensity);
+ set_detector_pos(refl, fs, ss, 0.0);
+ set_esd_intensity(refl, sigma);
+ }
} while ( rval != NULL );
@@ -184,45 +187,10 @@ static int read_reflections(FILE *fh, struct image *image)
}
-static void write_reflections(struct image *image, FILE *ofh)
-{
- Reflection *refl;
- RefListIterator *iter;
-
- fprintf(ofh, REFLECTION_START_MARKER"\n");
- /* FIXME: Unify this with write_reflections() over in reflections.c */
- fprintf(ofh, " h k l I phase sigma(I) "
- " 1/d(nm^-1) counts fs/px ss/px\n");
-
- for ( refl = first_refl(image->reflections, &iter);
- refl != NULL;
- refl = next_refl(refl, iter) ) {
-
- signed int h, k, l;
- double intensity, esd_i, s;
- double fs, ss;
-
- get_indices(refl, &h, &k, &l);
- get_detector_pos(refl, &fs, &ss);
- intensity = get_intensity(refl);
- esd_i = 0.0; /* FIXME! */
- s = 0.0; /* FIXME! */
-
- /* h, k, l, I, sigma(I), s */
- fprintf(ofh,
- "%3i %3i %3i %10.2f %s %10.2f %10.2f %7i %6.1f %6.1f\n",
- h, k, l, intensity, " -", esd_i, s/1.0e9, 1,
- fs, ss);
-
- }
-
- fprintf(ofh, REFLECTION_END_MARKER"\n");
-}
-
-
static int read_peaks(FILE *fh, struct image *image)
{
char *rval = NULL;
+ int first = 1;
image->features = image_feature_list_new();
@@ -239,9 +207,17 @@ static int read_peaks(FILE *fh, struct image *image)
if ( strcmp(line, PEAK_LIST_END_MARKER) == 0 ) return 0;
r = sscanf(line, "%f %f %f %f", &x, &y, &d, &intensity);
- if ( r != 4 ) return 1;
+ if ( (r != 4) && (!first) ) {
+ ERROR("Failed to parse peak list line.\n");
+ ERROR("The failed line was: '%s'\n", line);
+ return 1;
+ }
- image_add_feature(image->features, x, y, image, 1.0, NULL);
+ first = 0;
+ if ( r == 4 ) {
+ image_add_feature(image->features, x, y,
+ image, 1.0, NULL);
+ }
} while ( rval != NULL );
@@ -255,7 +231,7 @@ static void write_peaks(struct image *image, FILE *ofh)
int i;
fprintf(ofh, PEAK_LIST_START_MARKER"\n");
- fprintf(ofh, " fs/px ss/px (1/d)/nm^-1 Intensity\n");
+ fprintf(ofh, " fs/px ss/px (1/d)/nm^-1 Intensity\n");
for ( i=0; i<image_feature_count(image->features); i++ ) {
@@ -269,7 +245,7 @@ static void write_peaks(struct image *image, FILE *ofh)
r = get_q(image, f->fs, f->ss, NULL, 1.0/image->lambda);
q = modulus(r.u, r.v, r.w);
- fprintf(ofh, "%8.3f %8.3f %8.3f %12.3f\n",
+ fprintf(ofh, "%6.1f %6.1f %10.2f %10.2f\n",
f->fs, f->ss, q/1.0e9, f->intensity);
}
@@ -330,8 +306,12 @@ void write_chunk(FILE *ofh, struct image *i, int f)
}
if ( (f & STREAM_PIXELS) || (f & STREAM_INTEGRATED) ) {
+
fprintf(ofh, "\n");
- write_reflections(i, ofh);
+ fprintf(ofh, REFLECTION_START_MARKER"\n");
+ write_reflections_to_file(ofh, i->reflections, i->indexed_cell);
+ fprintf(ofh, REFLECTION_END_MARKER"\n");
+
}
fprintf(ofh, CHUNK_END_MARKER"\n\n");
@@ -374,10 +354,11 @@ int read_chunk(FILE *fh, struct image *image)
if ( find_start_of_chunk(fh) ) return 1;
image->i0_available = 0;
- if ( image->features != NULL ) {
- image_feature_list_free(image->features);
- image->features = NULL;
- }
+ image->i0 = 1.0;
+ image->lambda = -1.0;
+ image->features = NULL;
+ image->reflections = NULL;
+ image->indexed_cell = NULL;
do {
@@ -426,16 +407,22 @@ int read_chunk(FILE *fh, struct image *image)
}
if ( strcmp(line, PEAK_LIST_START_MARKER) == 0 ) {
- if ( read_peaks(fh, image) ) return 1;
+ if ( read_peaks(fh, image) ) {
+ ERROR("Failed while reading peaks\n");
+ return 1;
+ }
}
if ( strcmp(line, REFLECTION_START_MARKER) == 0 ) {
- if ( read_reflections(fh, image) ) return 1;
+ if ( read_reflections(fh, image) ) {
+ ERROR("Failed while reading reflections\n");
+ return 1;
+ }
}
} while ( strcmp(line, CHUNK_END_MARKER) != 0 );
- if ( have_filename && have_cell && have_ev ) return 0;
+ if ( have_filename && have_ev ) return 0;
ERROR("Incomplete chunk found in input file.\n");
return 1;
@@ -496,3 +483,24 @@ int find_chunk(FILE *fh, UnitCell **cell, char **filename, double *ev)
return 1;
}
+
+
+int skip_some_files(FILE *fh, int n)
+{
+ char *rval = NULL;
+ int n_patterns = 0;
+
+ do {
+
+ char line[1024];
+
+ if ( n_patterns == n ) return 0;
+
+ rval = fgets(line, 1023, fh);
+ if ( rval == NULL ) continue;
+ if ( strcmp(line, CHUNK_END_MARKER) == 0 ) n_patterns++;
+
+ } while ( rval != NULL );
+
+ return 1;
+}
diff --git a/src/stream.h b/src/stream.h
index 3869e515..64bd9529 100644
--- a/src/stream.h
+++ b/src/stream.h
@@ -38,4 +38,8 @@ extern void write_chunk(FILE *ofh, struct image *image, int flags);
extern int parse_stream_flags(const char *a);
+extern int read_chunk(FILE *fh, struct image *image);
+
+extern int skip_some_files(FILE *fh, int n);
+
#endif /* STREAM_H */