10 #include <top/modules/OpticalGun/OpticalGunModule.h>
13 #include <top/geometry/TOPGeometryPar.h>
16 #include <framework/gearbox/Unit.h>
17 #include <framework/logging/Logger.h>
21 #include <Math/EulerAngles.h>
24 #include <top/reconstruction_cpp/func.h>
27 using namespace ROOT::Math;
47 OpticalGunModule::OpticalGunModule() :
Module()
60 addParam(
"na",
m_na,
"source numerical aperture. It is used only by the Gaussian distribution", 0.50);
62 "source angular distribution: uniform, Lambertian, an arbitrary TFormula, or Gaussian "
63 "(numerical aperture instead of minAlpha and maxAlpha). If you are writing a TFormula, "
64 "assume the angles are measured in degrees. The conversion to radians is done internally.",
67 addParam(
"phi",
m_phi,
"first rotation angle (around z) [deg]", 0.0);
68 addParam(
"theta",
m_theta,
"second rotation angle (around x) [deg]", 0.0);
69 addParam(
"psi",
m_psi,
"third rotation angle (around z) [deg]", 0.0);
71 "start time [ns]. If TOPCalPulseGenerator is in path this is relative to the first cal pulse", 0.0);
74 "average number of photons per pulse, if positive, otherwise exactly one", 0.0);
76 "TOP slot ID (1-16): if valid, source position and rotation angles assumed to be given in a local bar frame, "
77 "otherwise Belle II frame is assumed", 0);
78 addParam(
"slitDX",
m_slitDX,
"slit size in x [cm], if positive, otherwise full open", 0.0);
79 addParam(
"slitDY",
m_slitDY,
"slit size in y [cm], if positive, otherwise full open", 0.0);
80 addParam(
"slitX0",
m_slitX0,
"slit x-offset in respect to source [cm] ", 0.0);
81 addParam(
"slitY0",
m_slitY0,
"slit y-offset in respect to source [cm] ", 0.0);
82 addParam(
"slitZ",
m_slitZ,
"slit distance to source [cm], if > 0.01, otherwise slit full open", 0.0);
97 if (m_wavelength < 150 or m_wavelength > 1000)
98 B2FATAL(
"Wavelength does not correspond to optical photons.");
99 if (m_na < 0 or m_na > 1)
100 B2FATAL(
"Numerical aperture must be between 0 and 1.");
102 B2FATAL(
"Minimum emission angle must be positive");
104 B2FATAL(
"Maximum emission angle must be positive");
106 B2FATAL(
"Minimum emission angle must me smaller than the maximum emission angle");
114 <<
"Checking if it's a valid, positively-defined TFormula.");
116 int result = testFormula.Compile();
122 double value = testFormula.Eval(testPoint *
Unit::deg);
125 << testPoint <<
" deg (value = " << value <<
")");
142 if (not geo->isModuleIDValid(
m_slotID)) B2FATAL(
"Slot ID is not valid");
143 const auto& T = geo->getModule(
m_slotID).getTransformation();
163 for (
int i = 0; i < numPhotons; i++) {
171 XYZPoint point(x, y, 0);
184 XYZVector momentum =
m_energy * direction;
190 double alphaPol = 2.0 * M_PI * gRandom->Rndm();
191 XYZVector polarization(cos(alphaPol), sin(alphaPol), 0);
192 func::rotateUz(polarization, direction);
195 double emissionTime = gRandom->Gaus(startTime,
m_pulseWidth);
208 part->setProductionVertex(
static_cast<XYZVector
>(point));
209 part->setProductionTime(emissionTime);
210 part->setMomentum(momentum);
212 part->setDecayVertex(polarization);
219 const XYZVector& direction)
const
221 if (
m_slitZ < 0.01)
return true;
222 if (direction.Z() < 1.0e-6)
return false;
224 double pathLength = (
m_slitZ - point.Z()) / direction.Z();
226 double x = point.X() + pathLength * direction.X();
230 double y = point.Y() + pathLength * direction.Y();
245 x = gRandom->Gaus(0., asin(
m_na) / 2.45);
246 y = gRandom->Gaus(0., asin(
m_na) / 2.45);
247 z = 1. - x * x - y * y;
249 return XYZVector(x, y,
sqrt(z));
255 double sinTheta =
sqrt(1.0 - cosTheta * cosTheta);
256 double phi = 2.0 * M_PI * gRandom->Rndm();
257 return XYZVector(cos(phi) * sinTheta, sin(phi) * sinTheta, cosTheta);
264 double sinTheta =
sqrt(1.0 - cosTheta * cosTheta);
265 double phi = 2.0 * M_PI * gRandom->Rndm();
266 return XYZVector(cos(phi) * sinTheta, sin(phi) * sinTheta, cosTheta);
272 double phi = 2.0 * M_PI * gRandom->Rndm();
273 return XYZVector(cos(phi) * sin(alpha), sin(phi) * sin(alpha), cos(alpha));
@ c_PrimaryParticle
bit 0: Particle is primary particle.
@ c_StableInGenerator
bit 1: Particle is stable, i.e., not decaying in the generator.
void setDescription(const std::string &description)
Sets the description of the module.
void setPropertyFlags(unsigned int propertyFlags)
Sets the flags for the module properties.
@ c_ParallelProcessingCertified
This module can be run in parallel processing mode safely (All I/O must be done through the data stor...
std::string m_angularDistribution
source angular distribution
double m_minAlpha
minimum emission angle
TF1 * m_customDistribution
Custom angular distribution, that uses m_angularDistribution as formula.
double m_slitZ
slit distance from source
double m_cosMinAlpha
cos of m_minAlpha
StoreArray< TOPSimCalPulse > m_simCalPulses
simulated cal pulse collection
double m_slitDX
slit size in x
double m_wavelength
source wavelenght [nm]
double m_cosMaxAlpha
cos of m_maxAlpha
double m_slitDY
slit size in y
double m_slitX0
slit x-offset in respect to source
double m_na
source numerical aperture.
double m_slitY0
slit y-offset in respect to source
double m_phi
first rotation angle (around z) [deg]
double m_x
source position in x
double m_theta
second rotation angle (around x) [deg]
double m_startTime
start time
double m_maxAlpha
maximum emission angle
double m_energy
photon energy (from wavelength)
double m_diameter
source diameter
double m_psi
third rotation angle (around z) [deg]
double m_z
source position in z
double m_y
source position in y
double m_pulseWidth
pulse duration (Gaussian sigma)
double m_numPhotons
average number of photons in a pulse
StoreArray< MCParticle > m_MCParticles
MC particles collection.
ROOT::Math::Transform3D m_transform
transformation to BelleII frame
bool registerInDataStore(DataStore::EStoreFlags storeFlags=DataStore::c_WriteOut)
Register the object/array in the DataStore.
T * appendNew()
Construct a new T object at the end of the array.
const TOPGeometry * getGeometry() const
Returns pointer to geometry object using basf2 units.
static TOPGeometryPar * Instance()
Static method to obtain the pointer to its instance.
static const double c_hc
Planck constant times speed of light in [eV*nm].
static const double deg
degree to radians
static const double eV
[electronvolt]
void addParam(const std::string &name, T ¶mVariable, const std::string &description, const T &defaultValue)
Adds a new parameter to the module.
#define REG_MODULE(moduleName)
Register the given module (without 'Module' suffix) with the framework.
double sqrt(double a)
sqrt for double
ROOT::Math::XYZVector getDirectionGaussian() const
Return photon direction according to a projected 2D gaussian distribution based on numerical aperture...
virtual ~OpticalGunModule()
Destructor.
ROOT::Math::XYZVector getDirectionUniform() const
Return photon direction according to a projected uniform distribution with opening angle alpha.
ROOT::Math::XYZVector getDirectionCustom() const
Return photon direction according to a custom angular distribution given by TFormula.
virtual void initialize() override
Initialize the Module.
virtual void event() override
Event processor.
bool isInsideSlit(const ROOT::Math::XYZPoint &point, const ROOT::Math::XYZVector &direction) const
Checks if photon passes the slit.
ROOT::Math::XYZVector getDirectionLambertian() const
Return photon direction according to a lambertian distribution with opening angle alpha.
Abstract base class for different kinds of events.