1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139 140 141 142 143 144 145 146 147 148 149 150 151 152 153 154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177 178 179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196 197 198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271 272 273 274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298 299 300 301 302 303 304 305 306 307 308 309 310 311
|
/*******************************************************************************
*
* McStas, neutron ray-tracing package
* Copyright (C) 1997-2017, All rights reserved
* DTU Physics, Kgs. Lyngby, Denmark
* Component: MCPL_output_noacc
*
* %I
* Written by: Erik B Knudsen
* Date: Mar 2016
* Origin: DTU Physics
*
* Detector-like component that writes neutron state parameters into an mcpl-format
* binary, virtual-source neutron file.
*
* %D
* Detector-like component that writes neutron state parameters into an mcpl-format
* binary, virtual-source neutron file.
*
* MCPL is short for Monte Carlo Particle List, and is a new format for sharing events
* between e.g. MCNP(X), Geant4 and McStas.
*
* When used with MPI, the component will output #MPI nodes individual MCPL files that
* can be merged using the mcpltool.
*
* MCPL_output allows a few flags to tweak the output files:
* 1. If use_polarisation is unset (default) the polarisation vector will not be stored (saving space)
* 2. If doubleprec is unset (default) data will be stored as 32 bit floating points, effectively cutting the output file size in half.
* 3. Extra information may be attached to each ray in the form of a userflag, a user-defined variable wich is packed into 32 bits. If
* the user variable does not fit in 32 bits the value will be truncated and likely garbage. If more than one variable is to be attached to
* each neutron this must be packed into the 32 bits.
*
* These features are set this way to keep file sizes as manageable as possible.
*
* %BUGS
*
* %P
* INPUT PARAMETERS
*
* filename: [str] Name of neutron file to write. If not given, the component name will be used.
* verbose: [1] If 1) Print summary information for created MCPL file. 2) Also print summary of first 10 particles information stored in the MCPL file. >2) Also print information for first 10 particles as they are being stored by McStas.
* polarisationuse: [1] Enable storing the polarisation state of the neutron.
* doubleprec: [1] Use double precision storage
* userflag: [1] Extra variable to attach to each neutron. The value of this variable will be packed into a 32 bit integer.
* userflagcomment: [str] String variable to describe the userflag. If this string is empty (the default) no userflags will be stored.
* merge_mpi: [1] Flag to indicate if output should be merged in case of MPI
* keep_mpi_unmerged: [1] Flag to indicate if original unmerged mcpl-files should be kept (or deleted).
* buffermax: [1] Maximal number of events to save ( <= MAXINT), GPU/OpenACC only
* %E
*******************************************************************************/
DEFINE COMPONENT MCPL_output_noacc
SETTING PARAMETERS (int polarisationuse=0, int doubleprec=0, verbose=0, string userflag="",
string filename=0, string userflagcomment="", merge_mpi=1, keep_mpi_unmerged=0, buffermax=0)
DEPENDENCY "-Wl,-rpath,CMD(mcpl-config --show libdir) -LCMD(mcpl-config --show libdir) -lmcpl -ICMD(mcpl-config --show includedir)"
NOACC
SHARE
%{
#include <mcpl.h>
#include <sys/stat.h>
int mcpl_file_exist (char *filename)
{
struct stat buffer;
return (stat (filename, &buffer) == 0);
}
%}
DECLARE
%{
mcpl_outfile_t outputfile;
mcpl_particle_t *particle;
mcpl_particle_t Particle;
int userflagenabled;
int captured;
%}
INITIALIZE
%{
char extension[128]="";
char *myfilename;
// Use instance name for base output if no input was given
if (!strcmp(filename,"\0")) sprintf(filename, "%s", NAME_CURRENT_COMP);
#if defined (USE_MPI)
/* In case of MPI, simply redefine the filename used by each node */
MPI_MASTER(fprintf(stdout, "Message(%s): You are using MCPL_output with MPI, hence your will get %i filenames %s_node_#i as output.\n",NAME_CURRENT_COMP,mpi_node_count,filename); );
sprintf(extension,"node_%i.mcpl",mpi_node_rank);
#else
sprintf(extension,"mcpl");
#endif
/*add output dir (if applicable) to the output filename and add extension if */
myfilename=mcfull_file(filename,extension);
char line[256];
outputfile = mcpl_create_outfile(myfilename);
/*reset filename to be whatever mcpl actually calls it. It may have added .mcpl*/
snprintf(myfilename,strlen(myfilename)+5,"%s",mcpl_outfile_filename(outputfile));
snprintf(line,255,"%s %s %s",MCCODE_NAME,MCCODE_VERSION,instrument_name);
mcpl_hdr_set_srcname(outputfile,line);
mcpl_enable_universal_pdgcode(outputfile,2112);/*all particles are neutrons*/
snprintf(line,255,"Output by COMPONENT: %s",NAME_CURRENT_COMP);
mcpl_hdr_add_comment(outputfile,line);
/*also add the instrument file and the command line as blobs*/
FILE *fp;
if( (fp=fopen(instrument_source,"rb"))!=NULL){
unsigned char *buffer;
int size,status;
/*find the file size by seeking to end, "tell" the position, and then go back again*/
fseek(fp, 0L, SEEK_END);
size = ftell(fp); // get current file pointer
fseek(fp, 0L, SEEK_SET); // seek back to beginning of file
if ( size && (buffer=malloc(size))!=NULL){
if (size!=(fread(buffer,1,size,fp))){
fprintf(stderr,"\nWarning (%s): Source instrument file not read cleanly\n", NAME_CURRENT_COMP);
}
mcpl_hdr_add_data(outputfile, "mccode_instr_file", size, buffer);
free(buffer);
}
fclose(fp);
} else {
fprintf(stderr,"\nWarning (%s): Source instrument file (%s) not found, hence not embedded.\n", NAME_CURRENT_COMP, instrument_source);
}
int ii;
char clr[2048],*clrp;
clrp=clr;
clrp+=snprintf(clrp,2048,"%s",instrument_exe);
char Parameters[CHAR_BUF_LENGTH];
for (ii=0;ii<numipar;ii++){
(*mcinputtypes[mcinputtable[ii].type].printer)(Parameters, mcinputtable[ii].par);
clrp+=snprintf(clrp,2048-(clrp-clr)," %s=%s",mcinputtable[ii].name, Parameters);
}
*(clrp)='\0';
mcpl_hdr_add_data(outputfile, "mccode_cmd_line" , strlen(clr), clr);
if (polarisationuse) {
mcpl_enable_polarisation(outputfile);
}
if (doubleprec){
mcpl_enable_doubleprec(outputfile);
}
#if defined (USE_MPI)
MPI_MASTER(
#endif
if (verbose==1) {
printf("MCPL_output verbose mode: after generating the mcpl-file it will be reread and a summary printed.\n");
}
#if defined (USE_MPI)
);
#endif
/*Add comments on what the orientation and position of this component is.*/
/*Include the instrument file itself as a binary blob in the mcpl file*/
userflagenabled=0;
/*Have the option of including a user-flag like they do at Loki.*/
if (strlen(userflagcomment)!=0){
mcpl_enable_userflags(outputfile);
userflagenabled=1;
/*Don't add the comment if it's empty*/
if(userflagcomment && strlen(userflagcomment)){
snprintf(line,255,"userflags: %s",userflagcomment);
mcpl_hdr_add_comment(outputfile,line);
}
}
if (myfilename){
free(myfilename);
}
/*pointer to the single particle storage area*/
particle=&Particle;
%}
TRACE
%{
double uvar;
int fail;
double nrm;
/*positions are in cm*/
particle->position[0]=x*100;
particle->position[1]=y*100;
particle->position[2]=z*100;
if(polarisationuse){
particle->polarisation[0]=sx;
particle->polarisation[1]=sy;
particle->polarisation[2]=sz;
}
nrm =sqrt(vx*vx + vy*vy + vz*vz);
/*ekin is in MeV*/
particle->ekin = VS2E*nrm*nrm/1e9;
particle->direction[0] = vx/nrm;
particle->direction[1] = vy/nrm;
particle->direction[2] = vz/nrm;
/*time in ms:*/
particle->time = t*1e3;
/*weight in unspecified units:*/
particle->weight = p;
/*if specified also add the userflags*/
if(userflagenabled){
uvar = particle_getvar(_particle,userflag,&fail); if(fail) uvar=0;
particle->userflags = (uint32_t) uvar;
}
#if defined (USE_MPI)
MPI_MASTER(
#endif
if (verbose==3 && mcrun_num<10) {
printf("id=%ld\tpdg=2112\tekin=%g MeV\tx=%g cm\ty=%g cm\tz=%g cm\tux=%g\tuy=%g\tuz=%g\tt=%g ms\tweight=%g\tpolx=%g\tpoly=%g\tpolz=%g\n",
mcrun_num, particle->ekin, particle->position[0], particle->position[1], particle->position[2],
particle->direction[0], particle->direction[1], particle->direction[2], particle->time, particle->weight,
particle->polarisation[0], particle->polarisation[1], particle->polarisation[2]);
}
#if defined (USE_MPI)
);
#endif
mcpl_add_particle(outputfile,particle);
SCATTER;
%}
SAVE
%{
#ifdef USE_MPI
if (merge_mpi && mpi_node_count > 1) {
mcpl_close_outfile(outputfile);
} else {
mcpl_closeandgzip_outfile(outputfile);
}
#else
mcpl_closeandgzip_outfile(outputfile);
#endif
%}
FINALLY
%{
#ifdef USE_MPI
MPI_Barrier(MPI_COMM_WORLD);
MPI_MASTER(
/* Only attempt merge if requested and meaningful */
if (merge_mpi && mpi_node_count > 1) {
char **mpi_node_files;
char *merge_outfilename;
char extension[128]="mcpl";
int j;
mcpl_outfile_t merge_outfile;
merge_outfilename=mcfull_file(filename,extension);
mpi_node_files=(char **) calloc(mpi_node_count,sizeof(char *));
for (j=0;j<mpi_node_count;j++){
sprintf(extension,"node_%i.mcpl",j);
mpi_node_files[j]=mcfull_file(filename,extension);
}
/*now do the merge through the call to mcpl_merge_files*/
merge_outfile = mcpl_merge_files(merge_outfilename,mpi_node_count,(const char **) mpi_node_files);
mcpl_closeandgzip_outfile(merge_outfile);
/*remove the original unmerged files if wanted*/
if(!keep_mpi_unmerged){
int status=0;
for (j=0;j<mpi_node_count;j++){
status+=remove(mpi_node_files[j]);
}
if (status){
fprintf(stderr,"Warning (%s): Could not remove one or more unmerged files.\n",NAME_CURRENT_COMP);
}
}
/*free the string storage*/
free(merge_outfilename);
for (j=0;j<mpi_node_count;j++){
free(mpi_node_files[j]);
}
free(mpi_node_files);
}
);
#endif
%}
MCDISPLAY
%{
double t,dt;
int i;
multiline(5, 0.2,0.2,0.0, -0.2,0.2,0.0, -0.2,-0.2,0.0, 0.2,-0.2,0.0, 0.2,0.2,0.0);
/*M*/
multiline(5,-0.085,-0.085,0.0, -0.085,0.085,0.0, -0.045,-0.085,0.0, -0.005,0.085,0.0, -0.005,-0.085,0.0);
/*O*/
dt=2*M_PI/32;
t=0;
for (i=0;i<32;i++){
line(0.04*cos(t)+0.045,0.08*sin(t),0, 0.04*cos(t+dt)+0.045,0.08*sin(t+dt),0);
t+=dt;
}
%}
END
|