Back to home page

EIC code displayed by LXR

 
 

    


File indexing completed on 2025-06-30 07:52:11

0001 // This file is part of the ACTS project.
0002 //
0003 // Copyright (C) 2016 CERN for the benefit of the ACTS project
0004 //
0005 // This Source Code Form is subject to the terms of the Mozilla Public
0006 // License, v. 2.0. If a copy of the MPL was not distributed with this
0007 // file, You can obtain one at https://mozilla.org/MPL/2.0/.
0008 
0009 #include "Acts/Material/detail/AverageMaterials.hpp"
0010 
0011 #include "Acts/Material/Material.hpp"
0012 
0013 #include <cmath>
0014 
0015 Acts::MaterialSlab Acts::detail::combineSlabs(const MaterialSlab& slab1,
0016                                               const MaterialSlab& slab2) {
0017   const auto& mat1 = slab1.material();
0018   const auto& mat2 = slab2.material();
0019 
0020   // NOTE 2020-08-26 msmk
0021   // the following computations provide purely geometrical averages of the
0022   // material properties. what we are really interested in are the material
0023   // properties that best describe the material interaction of the averaged
0024   // material. It is unclear if the latter can be computed in a general fashion,
0025   // so we stick to the purely geometric averages for now.
0026 
0027   // NOTE 2021-03-24 corentin
0028   // In the case of the atomic number, the averaging is based on the log
0029   // to properly account for the energy loss in multiple material.
0030 
0031   // use double for (intermediate) computations to avoid precision loss
0032   const double thickness1 = static_cast<double>(slab1.thickness());
0033   const double thickness2 = static_cast<double>(slab2.thickness());
0034 
0035   if (thickness1 == 0) {
0036     return slab2;
0037   }
0038   if (thickness2 == 0) {
0039     return slab1;
0040   }
0041 
0042   // the thickness properties are purely additive
0043   const double thickness = thickness1 + thickness2;
0044 
0045   // if the two materials are the same there is no need for additional
0046   // computation
0047   if (mat1 == mat2) {
0048     return {mat1, static_cast<float>(thickness)};
0049   }
0050 
0051   // molar amount-of-substance assuming a unit area, i.e. volume = thickness*1*1
0052   const double molarDensity1 = static_cast<double>(mat1.molarDensity());
0053   const double molarDensity2 = static_cast<double>(mat2.molarDensity());
0054 
0055   const double molarAmount1 = molarDensity1 * thickness1;
0056   const double molarAmount2 = molarDensity2 * thickness2;
0057   const double molarAmount = molarAmount1 + molarAmount2;
0058 
0059   // handle vacuum specially
0060   if (!(0.0 < molarAmount)) {
0061     return {Material::Vacuum(), static_cast<float>(thickness)};
0062   }
0063 
0064   // radiation/interaction length follows from consistency argument
0065   const double thicknessX01 = static_cast<double>(slab1.thicknessInX0());
0066   const double thicknessX02 = static_cast<double>(slab2.thicknessInX0());
0067   const double thicknessL01 = static_cast<double>(slab1.thicknessInL0());
0068   const double thicknessL02 = static_cast<double>(slab2.thicknessInL0());
0069 
0070   const double thicknessInX0 = thicknessX01 + thicknessX02;
0071   const double thicknessInL0 = thicknessL01 + thicknessL02;
0072 
0073   const float x0 = static_cast<float>(thickness / thicknessInX0);
0074   const float l0 = static_cast<float>(thickness / thicknessInL0);
0075 
0076   // assume two slabs of materials with N1,N2 atoms/molecules each with atomic
0077   // masses A1,A2 and nuclear charges. We have a total of N = N1 + N2
0078   // atoms/molecules and should have average atomic masses and nuclear charges
0079   // of
0080   //
0081   //     A = (N1*A1 + N2*A2) / (N1+N2) = (N1/N)*A1 + (N2/N)*A2 = W1*A1 + W2*A2
0082   //
0083   // the number of atoms/molecules in a given volume V with molar density rho is
0084   //
0085   //     N = V * rho * Na
0086   //
0087   // where Na is the Avogadro constant. the weighting factors follow as
0088   //
0089   //     Wi = (Vi*rhoi*Na) / (V1*rho1*Na + V2*rho2*Na)
0090   //        = (Vi*rhoi) / (V1*rho1 + V2*rho2)
0091   //
0092   // which can be computed from the molar amount-of-substance above.
0093   const double ar1 = static_cast<double>(mat1.Ar());
0094   const double ar2 = static_cast<double>(mat2.Ar());
0095 
0096   const double molarWeight1 = molarAmount1 / molarAmount;
0097   const double molarWeight2 = molarAmount2 / molarAmount;
0098   const float ar = static_cast<float>(molarWeight1 * ar1 + molarWeight2 * ar2);
0099 
0100   // In the case of the atomic number, its main use is the computation
0101   // of the mean excitation energy approximated in ATL-SOFT-PUB-2008-003 as :
0102   //     I = 16 eV * Z^(0.9)
0103   // This mean excitation energy will then be used to compute energy loss
0104   // which will be proportional to :
0105   //     Eloss ~ ln(1/I)*thickness
0106   // In the case of two successive material :
0107   //     Eloss = El1 + El2
0108   //           ~ ln(Z1)*t1 + ln(Z2)*t2
0109   //           ~ ln(Z)*t
0110   // To respect this the average atomic number thus need to be defined as :
0111   //     ln(Z)*t = ln(Z1)*t1 + ln(Z2)*t2
0112   //           Z = Exp( ln(Z1)*t1/t + ln(Z2)*t2/t )
0113   const double z1 = static_cast<double>(mat1.Z());
0114   const double z2 = static_cast<double>(mat2.Z());
0115 
0116   const double thicknessWeight1 = thickness1 / thickness;
0117   const double thicknessWeight2 = thickness2 / thickness;
0118   float z = 0.f;
0119   if (z1 > 0. && z2 > 0.) {
0120     z = static_cast<float>(std::exp(thicknessWeight1 * std::log(z1) +
0121                                     thicknessWeight2 * std::log(z2)));
0122   } else {
0123     z = static_cast<float>(thicknessWeight1 * z1 + thicknessWeight2 * z2);
0124   }
0125 
0126   // compute average molar density by dividing the total amount-of-substance by
0127   // the total volume for the same unit area, i.e. volume = totalThickness*1*1
0128   const float molarDensity = static_cast<float>(molarAmount / thickness);
0129 
0130   const double molarElectronAmount1 = mat1.molarElectronDensity() * thickness1;
0131   const double molarElectronAmount2 = mat2.molarElectronDensity() * thickness2;
0132   const double molarElectronAmount =
0133       molarElectronAmount1 + molarElectronAmount2;
0134   const float molarElectronDensity =
0135       static_cast<float>(molarElectronAmount / thickness);
0136 
0137   float meanExcitationEnergy = 0.f;
0138   if (mat1.meanExcitationEnergy() > 0 && mat2.meanExcitationEnergy() > 0) {
0139     meanExcitationEnergy = static_cast<float>(
0140         std::exp(thicknessWeight1 * std::log(mat1.meanExcitationEnergy())) *
0141         std::exp(thicknessWeight2 * std::log(mat2.meanExcitationEnergy())));
0142   } else {
0143     meanExcitationEnergy =
0144         static_cast<float>(thicknessWeight1 * mat1.meanExcitationEnergy() +
0145                            thicknessWeight2 * mat2.meanExcitationEnergy());
0146   }
0147 
0148   return {
0149       Material::fromMolarDensity(x0, l0, ar, z, molarDensity,
0150                                  molarElectronDensity, meanExcitationEnergy),
0151       static_cast<float>(thickness)};
0152 }