Back to home page

EIC code displayed by LXR

 
 

    


File indexing completed on 2025-04-04 08:04:42

0001 //
0002 // ********************************************************************
0003 // * License and Disclaimer                                           *
0004 // *                                                                  *
0005 // * The  Geant4 software  is  copyright of the Copyright Holders  of *
0006 // * the Geant4 Collaboration.  It is provided  under  the terms  and *
0007 // * conditions of the Geant4 Software License,  included in the file *
0008 // * LICENSE and available at  http://cern.ch/geant4/license .  These *
0009 // * include a list of copyright holders.                             *
0010 // *                                                                  *
0011 // * Neither the authors of this software system, nor their employing *
0012 // * institutes,nor the agencies providing financial support for this *
0013 // * work  make  any representation or  warranty, express or implied, *
0014 // * regarding  this  software system or assume any liability for its *
0015 // * use.  Please see the license in the file  LICENSE  and URL above *
0016 // * for the full disclaimer and the limitation of liability.         *
0017 // *                                                                  *
0018 // * This  code  implementation is the result of  the  scientific and *
0019 // * technical work of the GEANT4 collaboration.                      *
0020 // * By using,  copying,  modifying or  distributing the software (or *
0021 // * any work based  on the software)  you  agree  to acknowledge its *
0022 // * use  in  resulting  scientific  publications,  and indicate your *
0023 // * acceptance of all terms of the Geant4 Software license.          *
0024 // ********************************************************************
0025 //
0026 /// \file electromagnetic/TestEm8/src/DetectorConstruction.cc
0027 /// \brief Implementation of the DetectorConstruction class
0028 //
0029 //
0030 /////////////////////////////////////////////////////////////////////////
0031 //
0032 // TestEm8: Gaseous detector
0033 //
0034 // Created: 31.08.2010 V.Ivanchenko ob base of V.Grichine code
0035 //
0036 // Modified:
0037 //
0038 ////////////////////////////////////////////////////////////////////////
0039 //
0040 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0041 
0042 #include "DetectorConstruction.hh"
0043 
0044 #include "DetectorMessenger.hh"
0045 #include "TargetSD.hh"
0046 #include "TestParameters.hh"
0047 
0048 #include "G4Colour.hh"
0049 #include "G4GeometryManager.hh"
0050 #include "G4LogicalVolume.hh"
0051 #include "G4LogicalVolumeStore.hh"
0052 #include "G4Material.hh"
0053 #include "G4NistManager.hh"
0054 #include "G4PVPlacement.hh"
0055 #include "G4PhysicalConstants.hh"
0056 #include "G4PhysicalVolumeStore.hh"
0057 #include "G4PionPlus.hh"
0058 #include "G4ProductionCuts.hh"
0059 #include "G4Region.hh"
0060 #include "G4RegionStore.hh"
0061 #include "G4RunManager.hh"
0062 #include "G4SDManager.hh"
0063 #include "G4SolidStore.hh"
0064 #include "G4SystemOfUnits.hh"
0065 #include "G4Tubs.hh"
0066 #include "G4UnitsTable.hh"
0067 #include "G4VisAttributes.hh"
0068 #include "G4ios.hh"
0069 
0070 #include <vector>
0071 
0072 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0073 
0074 DetectorConstruction::DetectorConstruction()
0075 {
0076   fGasThickness = 23.0 * mm;
0077   fGasRadius = 10. * cm;
0078   fMaxStep = DBL_MAX;
0079 
0080   fWindowThick = 51.0 * micrometer;
0081 
0082   DefineMaterials();
0083 
0084   fDetectorMessenger = new DetectorMessenger(this);
0085 
0086   G4double cut = 0.7 * mm;
0087   fGasDetectorCuts = new G4ProductionCuts();
0088   fGasDetectorCuts->SetProductionCut(cut, "gamma");
0089   fGasDetectorCuts->SetProductionCut(cut, "e-");
0090   fGasDetectorCuts->SetProductionCut(cut, "e+");
0091   fGasDetectorCuts->SetProductionCut(cut, "proton");
0092 }
0093 
0094 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0095 
0096 DetectorConstruction::~DetectorConstruction()
0097 {
0098   delete fDetectorMessenger;
0099 }
0100 
0101 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0102 
0103 void DetectorConstruction::DefineMaterials()
0104 {
0105   // This function illustrates the possible ways to define materials
0106   G4String name, symbol;
0107   G4double density;
0108   G4int nel;
0109   G4int ncomponents;
0110   G4double fractionmass;
0111 
0112   G4NistManager* manager = G4NistManager::Instance();
0113   //
0114   // define Elements
0115   //
0116   G4Element* elH = manager->FindOrBuildElement(1);
0117   G4Element* elC = manager->FindOrBuildElement(6);
0118   G4Element* elO = manager->FindOrBuildElement(8);
0119   G4Element* elF = manager->FindOrBuildElement(9);
0120   G4Element* elNe = manager->FindOrBuildElement(10);
0121   G4Element* elXe = manager->FindOrBuildElement(54);
0122   //
0123   // simple gases at STP conditions
0124   //
0125   G4Material* Argon = manager->FindOrBuildMaterial("G4_Ar");
0126   G4Material* Kr = manager->FindOrBuildMaterial("G4_Kr");
0127   G4Material* Xe = manager->FindOrBuildMaterial("G4_Xe");
0128   //
0129   // gases at STP conditions
0130   //
0131   G4Material* CarbonDioxide = manager->FindOrBuildMaterial("G4_CARBON_DIOXIDE");
0132   G4Material* Mylar = manager->FindOrBuildMaterial("G4_MYLAR");
0133   G4Material* Methane = manager->FindOrBuildMaterial("G4_METHANE");
0134   G4Material* Propane = manager->FindOrBuildMaterial("G4_PROPANE");
0135 
0136   // propane at 10 atmospheres
0137   manager->ConstructNewGasMaterial("Propane10", "G4_PROPANE", NTP_Temperature, 10. * atmosphere);
0138 
0139   G4Material* empty = manager->FindOrBuildMaterial("G4_Galactic");
0140 
0141   // 93% Kr + 7% CH4, STP
0142   density = 3.491 * mg / cm3;
0143   G4Material* Kr7CH4 = new G4Material(name = "Kr7CH4", density, ncomponents = 2);
0144   Kr7CH4->AddMaterial(Kr, fractionmass = 0.986);
0145   Kr7CH4->AddMaterial(Methane, fractionmass = 0.014);
0146 
0147   G4double TRT_Xe_density = 5.485 * mg / cm3;
0148   G4Material* TRT_Xe = new G4Material(name = "TRT_Xe", TRT_Xe_density, nel = 1, kStateGas,
0149                                       293.15 * kelvin, 1. * atmosphere);
0150   TRT_Xe->AddElement(elXe, 1);
0151 
0152   G4double TRT_CO2_density = 1.842 * mg / cm3;
0153   G4Material* TRT_CO2 = new G4Material(name = "TRT_CO2", TRT_CO2_density, nel = 2, kStateGas,
0154                                        293.15 * kelvin, 1. * atmosphere);
0155   TRT_CO2->AddElement(elC, 1);
0156   TRT_CO2->AddElement(elO, 2);
0157 
0158   // check alternative constructor
0159   std::vector<G4String> trtatom = {"C", "O"};
0160   std::vector<G4int> trtnum = {1, 2};
0161   manager->ConstructNewMaterial("TRT_CO2p", trtatom, trtnum, TRT_CO2_density, true, kStateGas,
0162                                 NTP_Temperature, atmosphere);
0163 
0164   G4double TRT_CF4_density = 3.9 * mg / cm3;
0165   G4Material* TRT_CF4 = new G4Material(name = "TRT_CF4", TRT_CF4_density, nel = 2, kStateGas,
0166                                        293.15 * kelvin, 1. * atmosphere);
0167   TRT_CF4->AddElement(elC, 1);
0168   TRT_CF4->AddElement(elF, 4);
0169 
0170   // ATLAS TRT straw tube gas mixture (20 C, 1 atm)
0171   G4double XeCO2CF4_density = 4.76 * mg / cm3;
0172   G4Material* XeCO2CF4 = new G4Material(name = "XeCO2CF4", XeCO2CF4_density, ncomponents = 3,
0173                                         kStateGas, 293.15 * kelvin, 1. * atmosphere);
0174   XeCO2CF4->AddMaterial(TRT_Xe, 0.807);
0175   XeCO2CF4->AddMaterial(TRT_CO2, 0.039);
0176   XeCO2CF4->AddMaterial(TRT_CF4, 0.154);
0177 
0178   // C3H8,20 C, 2 atm
0179   density = 3.758 * mg / cm3;
0180   G4Material* C3H8 =
0181     new G4Material(name = "C3H8", density, nel = 2, kStateGas, 293.15 * kelvin, 2. * atmosphere);
0182   C3H8->AddElement(elC, 3);
0183   C3H8->AddElement(elH, 8);
0184 
0185   // The same material via different constructor
0186   std::vector<G4String> elmname = {"C", "H"};
0187   std::vector<G4int> atomnum = {3, 8};
0188   manager->ConstructNewIdealGasMaterial("C3H8p", elmname, atomnum, true, 293.15 * kelvin,
0189                                         2. * atmosphere);
0190 
0191   // 87.5% Xe + 7.5% CH4 + 5% C3H8, 20 C, 1. atm
0192   density = 4.9196 * mg / cm3;
0193   G4Material* XeCH4C3H8 = new G4Material(name = "XeCH4C3H8", density, ncomponents = 3, kStateGas,
0194                                          NTP_Temperature, 1. * atmosphere);
0195   XeCH4C3H8->AddMaterial(Xe, fractionmass = 0.971);
0196   XeCH4C3H8->AddMaterial(Methane, fractionmass = 0.010);
0197   XeCH4C3H8->AddMaterial(Propane, fractionmass = 0.019);
0198 
0199   // 93% Ar + 7% CH4, STP
0200   density = 1.709 * mg / cm3;
0201   G4Material* Ar7CH4 = new G4Material(name = "Ar7CH4", density, ncomponents = 2, kStateGas,
0202                                       STP_Temperature, STP_Pressure);
0203   Ar7CH4->AddMaterial(Argon, fractionmass = 0.971);
0204   Ar7CH4->AddMaterial(Methane, fractionmass = 0.029);
0205 
0206   // 80% Ar + 20% CO2, STP
0207   density = 1.8223 * mg / cm3;
0208   G4Material* Ar_80CO2_20 = new G4Material(name = "ArCO2", density, ncomponents = 2, kStateGas,
0209                                            STP_Temperature, STP_Pressure);
0210   Ar_80CO2_20->AddMaterial(Argon, fractionmass = 0.783);
0211   Ar_80CO2_20->AddMaterial(CarbonDioxide, fractionmass = 0.217);
0212 
0213   // 80% Xe + 20% CO2, STP
0214   density = 5.0818 * mg / cm3;
0215   G4Material* Xe20CO2 = new G4Material(name = "Xe20CO2", density, ncomponents = 2, kStateGas,
0216                                        STP_Temperature, STP_Pressure);
0217   Xe20CO2->AddMaterial(Xe, fractionmass = 0.922);
0218   Xe20CO2->AddMaterial(CarbonDioxide, fractionmass = 0.078);
0219 
0220   // 80% Kr + 20% CO2, STP
0221   density = 3.601 * mg / cm3;
0222   G4Material* Kr20CO2 = new G4Material(name = "Kr20CO2", density, ncomponents = 2, kStateGas,
0223                                        STP_Temperature, STP_Pressure);
0224   Kr20CO2->AddMaterial(Kr, fractionmass = 0.89);
0225   Kr20CO2->AddMaterial(CarbonDioxide, fractionmass = 0.11);
0226 
0227   // ALICE mixture TPC_Ne-CO2-2
0228   density = 0.939 * mg / cm3;
0229   G4Material* NeCO2 = new G4Material(name = "TPC_Ne-CO2-2", density, ncomponents = 3, kStateGas,
0230                                      NTP_Temperature, 1. * atmosphere);
0231   NeCO2->AddElement(elNe, fractionmass = 0.8039);
0232   NeCO2->AddElement(elO, fractionmass = 0.1426);
0233   NeCO2->AddElement(elC, fractionmass = 0.0535);
0234 
0235   // check alternative constructor
0236   std::vector<G4String> neatom = {"Ne", "O", "C"};
0237   std::vector<G4double> nefr = {0.8039, 0.1426, 0.0536};
0238   manager->ConstructNewMaterial("TPC_Ne-CO2-2p", neatom, nefr, density, true, kStateGas,
0239                                 NTP_Temperature, atmosphere);
0240 
0241   // ALICE TRD mixure 85% Xe + 15% CO2 NTP
0242   density = 4.9389 * mg / cm3;
0243   G4Material* Xe15CO2 = new G4Material(name = "Xe15CO2", density, ncomponents = 2, kStateGas,
0244                                        NTP_Temperature, atmosphere);
0245   Xe15CO2->AddMaterial(Xe, fractionmass = 0.944);
0246   Xe15CO2->AddMaterial(CarbonDioxide, fractionmass = 0.056);
0247 
0248   fGasMat = XeCH4C3H8;
0249   fWindowMat = Mylar;
0250   fWorldMaterial = empty;
0251 
0252   G4cout << *(G4Material::GetMaterialTable()) << G4endl;
0253 }
0254 
0255 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0256 
0257 G4VPhysicalVolume* DetectorConstruction::Construct()
0258 {
0259   if (nullptr != fPhysWorld) {
0260     return fPhysWorld;
0261   }
0262 
0263   G4double contThick = fWindowThick * 2 + fGasThickness;
0264   G4double contR = fWindowThick * 2 + fGasRadius;
0265 
0266   G4double worldSizeZ = contThick * 1.2;
0267   G4double worldSizeR = contR * 1.2;
0268 
0269   TestParameters::GetPointer()->SetPositionZ(-0.55 * contThick);
0270 
0271   // Printout parameters
0272   G4cout << "\n The  WORLD   is made of " << worldSizeZ / mm << "mm of "
0273          << fWorldMaterial->GetName();
0274   G4cout << ", the transverse size (R) of the world is " << worldSizeR / mm << " mm. " << G4endl;
0275   G4cout << " The CONTAINER is made of " << fWindowThick / mm << "mm of " << fWindowMat->GetName()
0276          << G4endl;
0277   G4cout << " The TARGET is made of " << fGasThickness / mm << "mm of " << fGasMat->GetName();
0278   G4cout << ", the transverse size (R) is " << fGasRadius / mm << " mm. " << G4endl;
0279   G4cout << G4endl;
0280 
0281   // World
0282   fSolidWorld = new G4Tubs("World", 0., worldSizeR, worldSizeZ / 2., 0., CLHEP::twopi);
0283 
0284   fLogicWorld = new G4LogicalVolume(fSolidWorld, fWorldMaterial, "World");
0285 
0286   fPhysWorld = new G4PVPlacement(0, G4ThreeVector(0., 0., 0.), "World", fLogicWorld, 0, false, 0);
0287 
0288   // Window
0289   fSolidContainer = new G4Tubs("Absorber", 0., contR, contThick / 2., 0., CLHEP::twopi);
0290 
0291   fLogicContainer = new G4LogicalVolume(fSolidContainer, fWindowMat, "Window");
0292 
0293   G4PVPlacement* PhysWind = new G4PVPlacement(0, G4ThreeVector(0., 0., 0.), "Window",
0294                                               fLogicContainer, fPhysWorld, false, 0);
0295 
0296   // Detector volume
0297   fSolidDetector = new G4Tubs("Gas", 0., fGasRadius, fGasThickness / 2., 0., CLHEP::twopi);
0298 
0299   fLogicDetector = new G4LogicalVolume(fSolidDetector, fGasMat, "Gas");
0300 
0301   new G4PVPlacement(0, G4ThreeVector(0., 0., 0.), "Gas", fLogicDetector, PhysWind, false, 0);
0302 
0303   // defined gas detector region
0304   fRegGasDet = new G4Region("GasDetector");
0305   fRegGasDet->SetProductionCuts(fGasDetectorCuts);
0306   fRegGasDet->AddRootLogicalVolume(fLogicDetector);
0307 
0308   // visualisation
0309   fLogicWorld->SetVisAttributes(G4VisAttributes::GetInvisible());
0310   G4VisAttributes* color1 = new G4VisAttributes(G4Colour(0.3, 0.3, 0.3));
0311   fLogicContainer->SetVisAttributes(color1);
0312   G4VisAttributes* color2 = new G4VisAttributes(G4Colour(0.0, 0.3, 0.7));
0313   fLogicDetector->SetVisAttributes(color2);
0314 
0315   if (0.0 == fGasMat->GetIonisation()->GetMeanEnergyPerIonPair()) {
0316     SetPairEnergy(20 * eV);
0317   }
0318   return fPhysWorld;
0319 }
0320 
0321 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0322 
0323 void DetectorConstruction::ConstructSDandField()
0324 {
0325   auto sd = new TargetSD("GasSD");
0326   G4SDManager::GetSDMpointer()->AddNewDetector(sd);
0327   SetSensitiveDetector(fLogicDetector, sd);
0328 }
0329 
0330 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0331 
0332 void DetectorConstruction::SetGasMaterial(const G4String& name)
0333 {
0334   // get the pointer to the existing material
0335   G4Material* mat = G4Material::GetMaterial(name, false);
0336 
0337   // create the material by its name
0338   if (!mat) {
0339     mat = G4NistManager::Instance()->FindOrBuildMaterial(name);
0340   }
0341 
0342   if (mat && mat != fGasMat) {
0343     G4cout << "### New target material: " << mat->GetName() << G4endl;
0344     fGasMat = mat;
0345     if (fLogicDetector) {
0346       fLogicDetector->SetMaterial(mat);
0347       G4RunManager::GetRunManager()->PhysicsHasBeenModified();
0348     }
0349   }
0350 }
0351 
0352 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0353 
0354 void DetectorConstruction::SetContainerMaterial(const G4String& name)
0355 {
0356   // get the pointer to the existing material
0357   G4Material* mat = G4Material::GetMaterial(name, false);
0358 
0359   // create the material by its name
0360   if (!mat) {
0361     mat = G4NistManager::Instance()->FindOrBuildMaterial(name);
0362   }
0363 
0364   if (mat && mat != fWindowMat) {
0365     G4cout << "### New material for container: " << mat->GetName() << G4endl;
0366     fWindowMat = mat;
0367     if (fLogicContainer) {
0368       fLogicContainer->SetMaterial(mat);
0369       G4RunManager::GetRunManager()->PhysicsHasBeenModified();
0370     }
0371   }
0372 }
0373 
0374 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0375 
0376 void DetectorConstruction::SetWorldMaterial(const G4String& name)
0377 {
0378   // get the pointer to the existing material
0379   G4Material* mat = G4Material::GetMaterial(name, false);
0380 
0381   // create the material by its name
0382   if (!mat) {
0383     mat = G4NistManager::Instance()->FindOrBuildMaterial(name);
0384   }
0385 
0386   if (mat && mat != fWorldMaterial) {
0387     G4cout << "### New World material: " << mat->GetName() << G4endl;
0388     fWorldMaterial = mat;
0389     if (fLogicWorld) {
0390       fLogicWorld->SetMaterial(mat);
0391       G4RunManager::GetRunManager()->PhysicsHasBeenModified();
0392     }
0393   }
0394 }
0395 
0396 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0397 
0398 void DetectorConstruction::SetGasThickness(G4double val)
0399 {
0400   if (val <= 0.0) {
0401     return;
0402   }
0403   fGasThickness = val;
0404   if (nullptr != fPhysWorld) {
0405     ChangeGeometry();
0406   }
0407 }
0408 
0409 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0410 
0411 void DetectorConstruction::SetGasRadius(G4double val)
0412 {
0413   if (val <= 0.0) {
0414     return;
0415   }
0416   fGasRadius = val;
0417   if (nullptr != fPhysWorld) {
0418     ChangeGeometry();
0419   }
0420 }
0421 
0422 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0423 
0424 void DetectorConstruction::SetContainerThickness(G4double val)
0425 {
0426   if (val <= 0.0) {
0427     return;
0428   }
0429   fWindowThick = val;
0430   if (nullptr != fPhysWorld) {
0431     ChangeGeometry();
0432   }
0433 }
0434 
0435 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0436 
0437 void DetectorConstruction::SetPairEnergy(G4double val)
0438 {
0439   if (val > 0.0) {
0440     fGasMat->GetIonisation()->SetMeanEnergyPerIonPair(val);
0441   }
0442 }
0443 
0444 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......
0445 
0446 void DetectorConstruction::ChangeGeometry()
0447 {
0448   G4double contThick = fWindowThick * 2 + fGasThickness;
0449   G4double contR = fWindowThick * 2 + fGasRadius;
0450 
0451   G4double worldSizeZ = contThick * 1.2;
0452   G4double worldSizeR = contR * 1.2;
0453 
0454   TestParameters::GetPointer()->SetPositionZ(-0.55 * contThick);
0455 
0456   fSolidWorld->SetOuterRadius(worldSizeR);
0457   fSolidWorld->SetZHalfLength(worldSizeZ * 0.5);
0458 
0459   fSolidContainer->SetOuterRadius(contR);
0460   fSolidContainer->SetZHalfLength(contThick * 0.5);
0461 
0462   fSolidDetector->SetOuterRadius(fGasRadius);
0463   fSolidDetector->SetZHalfLength(fGasThickness * 0.5);
0464 }
0465 
0466 //....oooOO0OOooo........oooOO0OOooo........oooOO0OOooo........oooOO0OOooo......