/* * pattern_sim.c * * Simulate diffraction patterns from small crystals * * (c) 2006-2010 Thomas White * * Part of CrystFEL - crystallography with a FEL * */ #ifdef HAVE_CONFIG_H #include #endif #include #include #include #include #include #include #include "image.h" #include "diffraction.h" #include "diffraction-gpu.h" #include "cell.h" #include "utils.h" #include "hdf5-file.h" #include "detector.h" #include "peaks.h" #include "sfac.h" #include "reflections.h" #include "beam-parameters.h" static void show_help(const char *s) { printf("Syntax: %s [options]\n\n", s); printf( "Simulate diffraction patterns from small crystals probed with femtosecond\n" "pulses of X-rays from a free electron laser.\n" "\n" " -h, --help Display this help message.\n" " -p, --pdb= PDB file from which to get the unit cell.\n" " (The actual Bragg intensities come from the\n" " intensities file)\n" " --simulation-details Show technical details of the simulation.\n" " --gpu Use the GPU to speed up the calculation.\n" " -g, --geometry= Get detector geometry from file.\n" " -b, --beam= Get beam parameters from file.\n" "\n" " --near-bragg Output h,k,l,I near Bragg conditions.\n" " -n, --number= Generate N images. Default 1.\n" " --no-images Do not output any HDF5 files.\n" " -o, --output= Output HDF5 filename. Default: sim.h5.\n" " If you choose to simulate more than one pattern,\n" " the filename given will be postfixed with a\n" " hyphen, the image number and '.h5'. In this\n" " case, the default value is 'sim', such that the\n" " files produced are sim-1.h5, sim-2.h5 and so on.\n" " -r, --random-orientation Use a randomly generated orientation\n" " (a new orientation will be used for each image).\n" " --powder= Write a summed pattern of all images simulated by\n" " this invocation as the given filename.\n" " -i, --intensities= Specify file containing reflection intensities\n" " (and phases) to use.\n" " -t, --gradients= Use for the calculation of shape\n" " transform intensities. Choose from:\n" " mosaic : Take the intensity of the nearest\n" " Bragg position. This is the\n" " fastest method and the only one\n" " supported on the GPU.\n" " interpolate : Interpolate trilinearly between\n" " six adjacent Bragg intensities.\n" " This method has intermediate\n" " accuracy.\n" " phased : As 'interpolate', but take phase\n" " values into account. This is the\n" " most accurate method, but the\n" " slowest.\n" "\n" "By default, the simulation aims to be as accurate as possible. For greater\n" "speed, or for testing, you can choose to disable certain things using the\n" "following options.\n" "\n" " --no-water Do not simulate water background.\n" " --no-noise Do not calculate Poisson noise.\n" ); } static void show_details() { printf( "This program simulates diffraction patterns from small crystals illuminated\n" "with femtosecond X-ray pulses from a free electron laser.\n" "\n" "The lattice transform from the specified number of unit cells is calculated\n" "using the closed-form solution for a truncated lattice faceted on the\n" "(001), (010) and (100) planes:\n" "\n" "I_latt(q) = sin^2(pi*na*q.a)/sin^2(pi*q.a)\n" " * sin^2(pi*nb*q.b)/sin^2(pi*q.b)\n" " * sin^2(pi*nc*q.c)/sin^2(pi*q.c)\n" "\n" "na = number of unit cells in 'a' direction (likewise nb, nc)\n" " q = reciprocal vector (1/d convention, not 2pi/d)\n" "\n" "This is multiplied by a model of the underlying molecular transform, I_mol(q).\n" "This can be approximated to varying levels of accuracy by the methods given by\n" "the '--gradients' option.\n" "\n" "Intensity from water is added according to the first term of equation 5\n" "from Phys Chem Chem Phys 2003 (5) 1981--1991. This simulates the\n" "coherent, elastic part of the diffuse scattering from the water jet only.\n" "\n" "Expected intensities at the CCD are then calculated using:\n" "\n" "I(q) = I0 * r^2 * I_latt(q) * I_mol(q) * S\n" "\n" "I0 = number of photons per unit area in the incident beam\n" " r = Thomson radius\n" " S = solid angle of corresponding pixel\n" "\n" "Polarisation is not currently included in pattern_sim, although it is included\n" "in the analysis of Bragg peaks by 'indexamajig'.\n" "\n" "Poisson counts are generated from the expected intensities using Knuth's\n" "algorithm. When the intensity is sufficiently high that Knuth's algorithm\n" "would result in machine precision problems, a normal distribution with\n" "standard deviation sqrt(I) is used instead.\n" ); } static struct quaternion read_quaternion() { do { int r; float w, x, y, z; char line[1024]; char *rval; printf("Please input quaternion: w x y z\n"); rval = fgets(line, 1023, stdin); if ( rval == NULL ) return invalid_quaternion(); chomp(line); r = sscanf(line, "%f %f %f %f", &w, &x, &y, &z); if ( r == 4 ) { struct quaternion quat; quat.w = w; quat.x = x; quat.y = y; quat.z = z; return quat; } else { ERROR("Invalid rotation '%s'\n", line); } } while ( 1 ); } int main(int argc, char *argv[]) { int c; struct image image; struct gpu_context *gctx = NULL; double *powder; char *intfile = NULL; double *intensities; double *phases; int config_simdetails = 0; int config_nearbragg = 0; int config_randomquat = 0; int config_noimages = 0; int config_nowater = 0; int config_nonoise = 0; int config_nosfac = 0; int config_gpu = 0; char *powder_fn = NULL; char *filename = NULL; char *grad_str = NULL; char *outfile = NULL; char *geometry = NULL; char *beamfile = NULL; GradientMethod grad; int ndone = 0; /* Number of simulations done (images or not) */ int number = 1; /* Number used for filename of image */ int n_images = 1; /* Generate one image by default */ int done = 0; UnitCell *cell; /* Long options */ const struct option longopts[] = { {"help", 0, NULL, 'h'}, {"simulation-details", 0, &config_simdetails, 1}, {"gpu", 0, &config_gpu, 1}, {"near-bragg", 0, &config_nearbragg, 1}, {"random-orientation", 0, NULL, 'r'}, {"number", 1, NULL, 'n'}, {"no-images", 0, &config_noimages, 1}, {"no-water", 0, &config_nowater, 1}, {"no-noise", 0, &config_nonoise, 1}, {"intensities", 1, NULL, 'i'}, {"powder", 1, NULL, 'w'}, {"gradients", 1, NULL, 't'}, {"pdb", 1, NULL, 'p'}, {"output", 1, NULL, 'o'}, {"geometry", 1, NULL, 'g'}, {"beam", 1, NULL, 'b'}, {0, 0, NULL, 0} }; /* Short options */ while ((c = getopt_long(argc, argv, "hrn:i:t:p:o:g:b:", longopts, NULL)) != -1) { switch (c) { case 'h' : show_help(argv[0]); return 0; case 'r' : config_randomquat = 1; break; case 'n' : n_images = atoi(optarg); break; case 'i' : intfile = strdup(optarg); break; case 't' : grad_str = strdup(optarg); break; case 'p' : filename = strdup(optarg); break; case 'o' : outfile = strdup(optarg); break; case 'w' : powder_fn = strdup(optarg); break; case 'g' : geometry = strdup(optarg); break; case 'b' : beamfile = strdup(optarg); break; case 0 : break; default : return 1; } } if ( filename == NULL ) { filename = strdup("molecule.pdb"); } if ( outfile == NULL ) { if ( n_images == 1 ) { outfile = strdup("sim.h5"); } else { outfile = strdup("sim"); } } if ( config_simdetails ) { show_details(); return 0; } if ( (!config_nowater) && config_gpu ) { ERROR("Cannot simulate water scattering on the GPU.\n"); ERROR("Please try again with the --no-water option.\n"); return 1; } if ( grad_str == NULL ) { STATUS("You didn't specify a gradient calculation method, so" " I'm using the 'mosaic' method, which is fastest.\n"); grad = GRADIENT_MOSAIC; } else if ( strcmp(grad_str, "mosaic") == 0 ) { grad = GRADIENT_MOSAIC; } else if ( strcmp(grad_str, "interpolate") == 0) { grad = GRADIENT_INTERPOLATE; } else if ( strcmp(grad_str, "phased") == 0) { grad = GRADIENT_PHASED; } else { ERROR("Unrecognised gradient method '%s'\n", grad_str); return 1; } free(grad_str); if ( config_gpu && (grad != GRADIENT_MOSAIC) ) { ERROR("Only the mosaic method can be used for gradients when" "calculating on the GPU.\n"); return 1; } if ( geometry == NULL ) { ERROR("You need to specify a geometry file with --geometry\n"); return 1; } if ( beamfile == NULL ) { ERROR("You need to specify a beam parameter file" " with --beam\n"); return 1; } if ( intfile == NULL ) { /* Gentle reminder */ STATUS("You didn't specify the file containing the "); STATUS("reflection intensities (with --intensities).\n"); STATUS("I'll simulate a flat intensity distribution.\n"); intensities = NULL; phases = NULL; } else { ReflItemList *items; if ( grad == GRADIENT_PHASED ) { phases = new_list_phase(); } else { phases = NULL; } intensities = new_list_intensity(); phases = new_list_phase(); items = read_reflections(intfile, intensities, phases, NULL, NULL); free(intfile); delete_items(items); } image.det = get_detector_geometry(geometry); if ( image.det == NULL ) { ERROR("Failed to read detector geometry from '%s'\n", geometry); return 1; } free(geometry); image.beam = get_beam_parameters(beamfile); if ( image.beam == NULL ) { ERROR("Failed to read beam parameters from '%s'\n", beamfile); return 1; } free(beamfile); /* Define image parameters */ image.width = image.det->max_x + 1; image.height = image.det->max_y + 1; image.lambda = ph_en_to_lambda(eV_to_J(image.beam->photon_energy)); cell = load_cell_from_pdb(filename); if ( cell == NULL ) { exit(1); } image.filename = NULL; image.features = NULL; image.flags = NULL; image.f0 = 1.0; image.f0_available = 1; powder = calloc(image.width*image.height, sizeof(*powder)); /* Splurge a few useful numbers */ STATUS("Wavelength is %f nm\n", image.lambda/1.0e-9); do { int na, nb, nc; double a, b, c, d; //na = 8*random()/RAND_MAX + 4; //nb = 8*random()/RAND_MAX + 4; //nc = 16*random()/RAND_MAX + 30; na = 24; nb = 24; nc = 40; /* Read quaternion from stdin */ if ( config_randomquat ) { image.orientation = random_quaternion(); } else { image.orientation = read_quaternion(); } STATUS("Orientation is %5.3f %5.3f %5.3f %5.3f\n", image.orientation.w, image.orientation.x, image.orientation.y, image.orientation.z); if ( !quaternion_valid(image.orientation) ) { ERROR("Orientation modulus is not zero!\n"); return 1; } /* Ensure no residual information */ image.data = NULL; image.twotheta = NULL; cell_get_parameters(cell, &a, &b, &c, &d, &d, &d); STATUS("Particle size = %i x %i x %i (=%5.2f x %5.2f x %5.2f nm)\n", na, nb, nc, na*a/1.0e-9, nb*b/1.0e-9, nc*c/1.0e-9); if ( config_gpu ) { if ( gctx == NULL ) { gctx = setup_gpu(config_nosfac, &image, intensities); } get_diffraction_gpu(gctx, &image, na, nb, nc, cell); } else { get_diffraction(&image, na, nb, nc, intensities, phases, cell, !config_nowater, grad); } if ( image.data == NULL ) { ERROR("Diffraction calculation failed.\n"); goto skip; } record_image(&image, !config_nonoise); if ( config_nearbragg ) { find_projected_peaks(&image, cell, 0, 0.1); output_intensities(&image, cell, NULL, 0, 1, 0, stdout, 0, 0.1); } if ( powder_fn != NULL ) { int x, y, w; w = image.width; for ( x=0; x= n_images) ) done = 1; } while ( !done ); if ( gctx != NULL ) { cleanup_gpu(gctx); } free(image.det->panels); free(image.det); free(image.beam); free(powder); cell_free(cell); free(intensities); free(outfile); free(filename); return 0; }