/******************************************************************************* * * McStas, neutron ray-tracing package * Copyright (C) 1997-2006, All rights reserved * Risoe National Laboratory, Roskilde, Denmark * Institut Laue Langevin, Grenoble, France * * Component: Source_gen * * %I * Written by: Emmanuel Farhi, Kim Lefmann * Date: Aug 27, 2001 * Version: $Revision: 1.26 $ * Origin: ILL/Risoe * Release: McStas 1.9.1 * Modified by: EF, Aug 27, 2001 ; can use Energy/wavelength and I1 * Modified by: EF, Sep 18, 2001 ; corrected illumination bug. Add options * Modified by: EF, Oct 30, 2001 ; minor changes. mccompcurname replaced * * Circular/squared neutron source with flat or Maxwellian energy/wavelength * spectrum (possibly spatially gaussian) * * %D * This routine is a neutron source (rectangular or circular), which aims at * a square target centered at the beam (in order to improve MC-acceptance * rate). The angular divergence is then given by the dimensions of the * target. * The neutron energy/wavelength is distributed between Emin=E0-dE and * Emax=E0+dE or Lmin=Lambda0-dLambda and Lmax=Lambda0+dLambda. The I1 may * be either arbitrary (I1=0), or specified in neutrons per steradian per * square cm per Angstrom. A Maxwellian spectra may be selected if you give * the source temperatures (up to 3). Finally, a file with the flux as a * function of the wavelength [lambda(AA) flux(n/s/cm^2/st/AA)] may be used * with the 'flux_file' parameter. Format is 2 columns free text. * The source shape is defined by its radius, or can alternatively be squared * if you specify non-zero h and w parameters. * The beam is spatially uniform, but becomes gaussian if one of the source * dimensions (radius, h or w) is negative, or you set the gaussian flag. * Divergence profiles are triangular. * The source may have a thickness, which will broaden the default zero time * distribution. * For the Maxwellian spectra, the generated intensity is dPhi/dLambda (n/s/AA) * For flat spectra, the generated intensity is Phi (n/s). * * Usage example: * Source_gen(radius=0.1,Lambda0=2.36,dLambda=0.16, T1=20, T2=38, I1=1e13) * Source_gen(h=0.1,w=0.1,Emin=1,Emax=3, I1=1e13, verbose=1, gaussian=1) * EXTEND * %{ * t = rand0max(1e-3); // set time from 0 to 1 ms for TOF instruments. * %} * * Some sources parameters: * PSI cold source T1=150.42,I1=3.67e11 T2=38.74,I2=3.64e11 * T3=14.84,I3=0.95e11 * ILL VCS cold source T1=216.8,I1=1.24e+13,T2=33.9,I2=1.02e+13 * (H1) T3=16.7 ,I3=3.0423e+12 * ILL HCS cold source T1=413.5,I1=10.22e12,T2=145.8,I2=3.44e13 * (H5) T3=40.1 ,I3=2.78e13 * ILL Thermal tube T1=683.7,I1=0.5874e+13,T2=257.7,I2=2.5099e+13 * (H12) T3=16.7 ,I3=1.0343e+12 * ILL Hot source T1=1695,I1=1.74e13,T2=708,I2=3.9e12 * * %VALIDATION * Feb 2005: output cross-checked for 3 Maxwellians against VITESS source * I(lambda), I(hor_div), I(vert_div) identical in shape and absolute values * Validated by: K. Lieutenant * * %P * radius: [m] Radius of circle in (x,y,0) plane where neutrons * are generated. You may also use 'h' and 'w' for a square source * dist: [m] Distance to target along z axis. * xw: [m] Width(x) of target. * If dist=0.0, xw = horz. div in deg * yh: [m] Height(y) of target. * If dist=0.0, yh = vert. div in deg * * Energy or wavelength range must be defined by giving min and max value or average and spread: * Emin: [meV] Minimum energy of neutrons * Emax: [meV] Maximum energy of neutrons * E0: [meV] Mean energy of neutrons. * dE: [meV] Energy spread of neutrons, half width. * Lmin: [AA] Minimum wavelength of neutrons * Lmax: [AA] Maximum wavelength of neutrons * Lambda0: [AA] Mean wavelength of neutrons. * dLambda: [AA] Wavelength spread of neutrons,half width * * Optional parameters: * h: [m] Source y-height, then does not use radius parameter * w: [m] Source x-width, then does not use radius parameter * length: [m] Source z-length, not anymore flat * T1: [K] Temperature of the Maxwellian source, 0=none * I1: [1/(cm**2*st*AA)] Source flux per solid angle, area and Angstrom * T2: [K] Second Maxwellian source Temperature, 0=none * I2: [1/(cm**2*st*AA)] Second Maxwellian Source flux * T3: [K] Third Maxwellian source Temperature, 0=none * I3: [1/(cm**2*st*AA)] Third Maxwellian Source flux * flux_file:[str] name of a file that contains the wavelength distribution of * the flux in [1/(s*cm**2*st*AA)]. Format is [lambda flux] * if a file is given, temperature and intensity values are ignored * gaussian: [0/1] Use gaussian divergence beam, normal distributions * verbose: [0/1] display info about the source. -1 unactivate source. * * Output parameters: * jmax: [1] number of data points in flux distribution array * * %L * The ILL Yellow book * %L * P. Ageron, Nucl. Inst. Meth. A 284 (1989) 197 * %E ******************************************************************************/ DEFINE COMPONENT Source_gen DEFINITION PARAMETERS (flux_file=0) SETTING PARAMETERS (radius=0.0, dist=57.296, xw=0, yh=0, E0=0, dE=0, Lambda0=0, dLambda=0, I1=0, h=0, w=0, gaussian=0, verbose=0, T1=0, Lmin=0,Lmax=0,Emin=0,Emax=0,T2=0,I2=0,T3=0,I3=0,length=0) OUTPUT PARAMETERS (p_in,lambda0, lambda02, L2P,lambda_0,d_lambda,lambda0b, lambda02b, L2Pb,lambda0c, lambda02c, L2Pc, pTable) STATE PARAMETERS (x,y,z,vx,vy,vz,t,s1,s2,p) SHARE %{ %include "read_table-lib" %} DECLARE %{ double p_in; double lambda_0=0,d_lambda=0; double lambda0, lambda02, L2P; /* first Maxwellian source */ double lambda0b, lambda02b, L2Pb; /* second Maxwellian source */ double lambda0c, lambda02c, L2Pc; /* third Maxwellian source */ t_Table pTable; %} INITIALIZE %{ double delta_lambda, source_area, k; if (radius == 0 && h == 0 && w == 0) { fprintf(stderr,"Source_gen: %s: Error: Please precise source geometry (radius, h, w)\n", NAME_CURRENT_COMP); exit(-1); } if (xw*yh == 0) { fprintf(stderr,"Source_gen: %s: Error: Please precise source target (xw, yh)\n", NAME_CURRENT_COMP); exit(-1); } if (Emin < 0 || Emax < 0 || Lmin < 0 || Lmax < 0 || E0 < 0 || dE < 0 || Lambda0 < 0 || dLambda < 0) { fprintf(stderr,"Source_gen: %s: Error: Negative average\n" " or range values for wavelength or energy encountered\n", NAME_CURRENT_COMP); exit(-1); } if ((Emin == 0 && Emax > 0) || (dE > 0 && dE >= E0)) { fprintf(stderr,"Source_gen: %s: Error: minimal energy cannot be less or equal zero\n", NAME_CURRENT_COMP); exit(-1); } if ((Emax >= Emin) && (Emin > 0)) { E0 = (Emax+Emin)/2; dE = (Emax-Emin)/2; } if ((E0 > dE) && (dE >= 0)) { Lmin = sqrt(81.81/(E0+dE)); /* Angstroem */ Lmax = sqrt(81.81/(E0-dE)); } if (Lmax > 0) { Lambda0 = (Lmax+Lmin)/2; dLambda = (Lmax-Lmin)/2; } if ((Lambda0 >= dLambda) && (dLambda > 0)) { lambda_0 = Lambda0; d_lambda = dLambda; } else { fprintf(stderr,"Source_gen: %s: Error: Wavelength range %.3f +/- %.3f AA calculated \n", NAME_CURRENT_COMP, Lambda0, dLambda); fprintf(stderr,"- whole wavelength range must be >= 0 \n"); fprintf(stderr,"- range must be > 0; otherwise intensity gets zero, use other sources in this case \n\n"); exit(-1); } radius = fabs(radius); w=fabs(w); h=fabs(h); I1=fabs(I1); lambda_0=fabs(lambda_0); d_lambda=fabs(d_lambda); xw = fabs(xw); yh=fabs(yh); dist=fabs(dist); if (dist == 0) { fprintf(stderr,"Source_gen: %s: warning: focusing distance is null.\n" " xw and yh interpreted as divergence in deg\n", NAME_CURRENT_COMP); } delta_lambda = 2*d_lambda; Lmin = lambda_0 - d_lambda; /* Angstroem */ Lmax = lambda_0 + d_lambda; if ((I1 > 0 && T1 > 0) || (flux_file && strlen(flux_file) > 0)) { if ((h == 0) || (w == 0)) source_area = radius*radius*PI*1e4; /* circular cm^2 */ else source_area = h*w*1e4; /* square cm^2 */ p_in = source_area*delta_lambda; } else p_in = 1.0/4/PI; /* Small angle approx. */ p_in /= mcget_ncount(); if (flux_file && strlen(flux_file) > 0) { if (Table_Read(&pTable, flux_file, 1) <= 0) /* read 1st block data from file into pTable */ exit(fprintf(stderr, "Source_gen: %s: can not read file %s\n", NAME_CURRENT_COMP, flux_file)); else { /* put table in Log scale */ int i; if (pTable.columns != 2) exit(fprintf(stderr, "Source_gen: %s: Flux file %s should contain 2 columns\n", NAME_CURRENT_COMP, flux_file)); for (i=0; i 0 ? val : 1/FLT_MAX); Table_SetElement(&pTable, i, 1, val); } } } else { k = 1.38066e-23; /* k_B */ if (T1 > 0) { lambda0 = 1.0e10*sqrt(HBAR*HBAR*4.0*PI*PI/2.0/MNEUTRON/k/T1); lambda02 = lambda0*lambda0; L2P = 2*lambda02*lambda02; } else { lambda0 = lambda_0; } if (T2 > 0) { lambda0b = 1.0e10*sqrt(HBAR*HBAR*4.0*PI*PI/2.0/MNEUTRON/k/T2); lambda02b = lambda0b*lambda0b; L2Pb = 2*lambda02b*lambda02b; } else { lambda0b = lambda_0; } if (T3 > 0) { lambda0c = 1.0e10*sqrt(HBAR*HBAR*4.0*PI*PI/2.0/MNEUTRON/k/T3); lambda02c = lambda0c*lambda0c; L2Pc = 2*lambda02c*lambda02c; } else { lambda0c = lambda_0; } } if (verbose) { printf("Source_gen: component %s ", NAME_CURRENT_COMP); if ((h == 0) || (w == 0)) printf("(disk)"); else printf("(square)"); printf("\n spectra "); printf("%.3f to %.3f AA (%.3f to %.3f meV)", Lmin, Lmax, 81.81/Lmax/Lmax, 81.81/Lmin/Lmin); if (gaussian) printf(", gaussian divergence beam"); printf("\n"); if (flux_file && strlen(flux_file) > 0) { printf(" File %s for flux distribution used. Flux is dPhi/dLambda in [n/s/AA]. \n", flux_file); Table_Info(pTable); } else if (T1 && I1) { if (T1 != 0) printf(" T1=%.1f K (%.3f AA)", T1, lambda0); if (T2*I2 != 0) printf(", T2=%.1f K (%.3f AA)", T2, lambda0b); if (T3*I3 != 0) printf(", T3=%.1f K (%.3f AA)", T3, lambda0c); if (T1) printf("\n"); printf(" Flux is dPhi/dLambda in [n/s/AA].\n"); } else { printf(" Flux is Phi in [n/s].\n"); } } else if (verbose == -1) printf("Source_gen: component %s unactivated", NAME_CURRENT_COMP); %} TRACE %{ double theta0,phi0,theta1,phi1,chi,theta,phi,v,r, lambda; double tan_h, tan_v, Maxwell, lambda2, lambda5; if (verbose >= 0) { z=0; if ((h == 0) || (w == 0)) { chi=2*PI*rand01(); /* Choose point on source */ r=sqrt(rand01())*radius; /* with uniform distribution. */ x=r*cos(chi); y=r*sin(chi); } else { x = w*randpm1()/2; /* select point on source (uniform) */ y = h*randpm1()/2; } if (length != 0) z = length*randpm1()/2; if (dist == 0) { theta0 = DEG2RAD*xw; phi0 = DEG2RAD*yh; theta1 = -DEG2RAD*xw; phi1 = -DEG2RAD*yh; } else { theta0= -atan((x-xw/2.0)/dist); /* Angles to aim at target */ phi0 = -atan((y-yh/2.0)/dist); theta1= -atan((x+xw/2.0)/dist); phi1 = -atan((y+yh/2.0)/dist); } /* shot towards target : flat distribution */ if (gaussian) { theta= theta0+(theta1- theta0)*(randnorm()*FWHM2RMS+0.5); phi = phi0 +(phi1 - phi0) *(randnorm()*FWHM2RMS+0.5); } else { theta= theta0+(theta1- theta0)*rand01(); phi = phi0 +(phi1 - phi0) *rand01(); } /* Assume linear distribution */ lambda = lambda_0+d_lambda*randpm1(); if (lambda <= 0) ABSORB; v = K2V*(2*PI/lambda); p = p_in * fabs((theta1 - theta0)*(phi1 - phi0)); if (flux_file && strlen(flux_file) > 0) { p *= exp(Table_Value(pTable, lambda, 1)); } else if (T1 > 0 && I1 > 0) { lambda2 = lambda*lambda; lambda5 = lambda2*lambda2*lambda; Maxwell = I1 * L2P/lambda5 * exp(-lambda02/lambda2); /* 1/AA */ if ((T2 > 0) && (I2 > 0)) { Maxwell += I2 * L2Pb/lambda5 * exp(-lambda02b/lambda2); } if ((T3 > 0) && (I3 > 0)) { Maxwell += I3 * L2Pc/lambda5 * exp(-lambda02c/lambda2); } p *= Maxwell; } /* Perform the correct treatment - no small angle approx. here! */ tan_h = tan(theta); tan_v = tan(phi); vz = v / sqrt(1 + tan_v*tan_v + tan_h*tan_h); vy = tan_v * vz; vx = tan_h * vz; SCATTER; } %} FINALLY %{ Table_Free(&pTable); %} MCDISPLAY %{ double xmin; double xmax; double ymin; double ymax; if ((h == 0) || (w == 0)) { magnify("xy"); circle("xy",0,0,0,radius); if (gaussian) circle("xy",0,0,0,radius/2); } else { xmin = -w/2; xmax = w/2; ymin = -h/2; ymax = h/2; magnify("xy"); multiline(5, (double)xmin, (double)ymin, 0.0, (double)xmax, (double)ymin, 0.0, (double)xmax, (double)ymax, 0.0, (double)xmin, (double)ymax, 0.0, (double)xmin, (double)ymin, 0.0); if (gaussian) circle("xy",0,0,0,sqrt(w*w+h*h)/4); } %} END