File indexing completed on 2025-04-04 08:04:12
0001
0002
0003
0004
0005
0006
0007
0008
0009
0010
0011
0012
0013
0014
0015
0016
0017
0018
0019
0020
0021
0022
0023
0024
0025
0026
0027
0028
0029
0030 #include "DetectorConstruction.hh"
0031
0032 #include "DetectorMessenger.hh"
0033 #include "TrackerSD.hh"
0034
0035 #include "G4AutoDelete.hh"
0036 #include "G4Box.hh"
0037 #include "G4Colour.hh"
0038 #include "G4GeometryManager.hh"
0039 #include "G4GeometryTolerance.hh"
0040 #include "G4GlobalMagFieldMessenger.hh"
0041 #include "G4LogicalVolume.hh"
0042 #include "G4Material.hh"
0043 #include "G4NistManager.hh"
0044 #include "G4PVPlacement.hh"
0045 #include "G4SDManager.hh"
0046 #include "G4SystemOfUnits.hh"
0047 #include "G4Tubs.hh"
0048 #include "G4UserLimits.hh"
0049 #include "G4VisAttributes.hh"
0050
0051 using namespace B2;
0052
0053 namespace B2a
0054 {
0055
0056
0057
0058 G4ThreadLocal G4GlobalMagFieldMessenger* DetectorConstruction::fMagFieldMessenger = nullptr;
0059
0060 DetectorConstruction::DetectorConstruction()
0061 {
0062 fMessenger = new DetectorMessenger(this);
0063
0064 fNbOfChambers = 5;
0065 fLogicChamber = new G4LogicalVolume*[fNbOfChambers];
0066 }
0067
0068
0069
0070 DetectorConstruction::~DetectorConstruction()
0071 {
0072 delete[] fLogicChamber;
0073 delete fStepLimit;
0074 delete fMessenger;
0075 }
0076
0077
0078
0079 G4VPhysicalVolume* DetectorConstruction::Construct()
0080 {
0081
0082 DefineMaterials();
0083
0084
0085 return DefineVolumes();
0086 }
0087
0088
0089
0090 void DetectorConstruction::DefineMaterials()
0091 {
0092
0093
0094 G4NistManager* nistManager = G4NistManager::Instance();
0095
0096
0097 nistManager->FindOrBuildMaterial("G4_AIR");
0098
0099
0100 fTargetMaterial = nistManager->FindOrBuildMaterial("G4_Pb");
0101
0102
0103 fChamberMaterial = nistManager->FindOrBuildMaterial("G4_Xe");
0104
0105
0106 G4cout << *(G4Material::GetMaterialTable()) << G4endl;
0107 }
0108
0109
0110
0111 G4VPhysicalVolume* DetectorConstruction::DefineVolumes()
0112 {
0113 G4Material* air = G4Material::GetMaterial("G4_AIR");
0114
0115
0116
0117 G4double chamberSpacing = 80 * cm;
0118
0119 G4double chamberWidth = 20.0 * cm;
0120 G4double targetLength = 5.0 * cm;
0121
0122 G4double trackerLength = (fNbOfChambers + 1) * chamberSpacing;
0123
0124 G4double worldLength = 1.2 * (2 * targetLength + trackerLength);
0125
0126 G4double targetRadius = 0.5 * targetLength;
0127 targetLength = 0.5 * targetLength;
0128 G4double trackerSize = 0.5 * trackerLength;
0129
0130
0131
0132
0133
0134 G4GeometryManager::GetInstance()->SetWorldMaximumExtent(worldLength);
0135
0136 G4cout << "Computed tolerance = "
0137 << G4GeometryTolerance::GetInstance()->GetSurfaceTolerance() / mm << " mm" << G4endl;
0138
0139 auto worldS = new G4Box("world",
0140 worldLength / 2, worldLength / 2, worldLength / 2);
0141 auto worldLV = new G4LogicalVolume(worldS,
0142 air,
0143 "World");
0144
0145
0146
0147 auto worldPV = new G4PVPlacement(nullptr,
0148 G4ThreeVector(),
0149 worldLV,
0150 "World",
0151 nullptr,
0152 false,
0153 0,
0154 fCheckOverlaps);
0155
0156
0157
0158 G4ThreeVector positionTarget = G4ThreeVector(0, 0, -(targetLength + trackerSize));
0159
0160 auto targetS = new G4Tubs("target", 0., targetRadius, targetLength, 0. * deg, 360. * deg);
0161 fLogicTarget = new G4LogicalVolume(targetS, fTargetMaterial, "Target", nullptr, nullptr, nullptr);
0162 new G4PVPlacement(nullptr,
0163 positionTarget,
0164 fLogicTarget,
0165 "Target",
0166 worldLV,
0167 false,
0168 0,
0169 fCheckOverlaps);
0170
0171 G4cout << "Target is " << 2 * targetLength / cm << " cm of " << fTargetMaterial->GetName()
0172 << G4endl;
0173
0174
0175
0176 G4ThreeVector positionTracker = G4ThreeVector(0, 0, 0);
0177
0178 auto trackerS = new G4Tubs("tracker", 0, trackerSize, trackerSize, 0. * deg, 360. * deg);
0179 auto trackerLV = new G4LogicalVolume(trackerS, air, "Tracker", nullptr, nullptr, nullptr);
0180 new G4PVPlacement(nullptr,
0181 positionTracker,
0182 trackerLV,
0183 "Tracker",
0184 worldLV,
0185 false,
0186 0,
0187 fCheckOverlaps);
0188
0189
0190
0191 G4VisAttributes boxVisAtt(G4Colour::White());
0192 G4VisAttributes chamberVisAtt(G4Colour::Yellow());
0193
0194 worldLV->SetVisAttributes(boxVisAtt);
0195 fLogicTarget->SetVisAttributes(boxVisAtt);
0196 trackerLV->SetVisAttributes(boxVisAtt);
0197
0198
0199
0200 G4cout << "There are " << fNbOfChambers << " chambers in the tracker region. " << G4endl
0201 << "The chambers are " << chamberWidth / cm << " cm of " << fChamberMaterial->GetName()
0202 << G4endl << "The distance between chamber is " << chamberSpacing / cm << " cm" << G4endl;
0203
0204 G4double firstPosition = -trackerSize + chamberSpacing;
0205 G4double firstLength = trackerLength / 10;
0206 G4double lastLength = trackerLength;
0207
0208 G4double halfWidth = 0.5 * chamberWidth;
0209 G4double rmaxFirst = 0.5 * firstLength;
0210
0211 G4double rmaxIncr = 0.0;
0212 if (fNbOfChambers > 0) {
0213 rmaxIncr = 0.5 * (lastLength - firstLength) / (fNbOfChambers - 1);
0214 if (chamberSpacing < chamberWidth) {
0215 G4Exception("DetectorConstruction::DefineVolumes()", "InvalidSetup", FatalException,
0216 "Width>Spacing");
0217 }
0218 }
0219
0220 for (G4int copyNo = 0; copyNo < fNbOfChambers; copyNo++) {
0221 G4double Zposition = firstPosition + copyNo * chamberSpacing;
0222 G4double rmax = rmaxFirst + copyNo * rmaxIncr;
0223
0224 auto chamberS = new G4Tubs("Chamber_solid", 0, rmax, halfWidth, 0. * deg, 360. * deg);
0225
0226 fLogicChamber[copyNo] =
0227 new G4LogicalVolume(chamberS, fChamberMaterial, "Chamber_LV", nullptr, nullptr, nullptr);
0228
0229 fLogicChamber[copyNo]->SetVisAttributes(chamberVisAtt);
0230
0231 new G4PVPlacement(nullptr,
0232 G4ThreeVector(0, 0, Zposition),
0233 fLogicChamber[copyNo],
0234 "Chamber_PV",
0235 trackerLV,
0236 false,
0237 copyNo,
0238 fCheckOverlaps);
0239 }
0240
0241
0242
0243
0244
0245
0246
0247
0248 G4double maxStep = 0.5 * chamberWidth;
0249 fStepLimit = new G4UserLimits(maxStep);
0250 trackerLV->SetUserLimits(fStepLimit);
0251
0252
0253
0254
0255
0256
0257
0258
0259
0260
0261
0262 return worldPV;
0263 }
0264
0265
0266
0267 void DetectorConstruction::ConstructSDandField()
0268 {
0269
0270
0271 G4String trackerChamberSDname = "/TrackerChamberSD";
0272 auto trackerSD = new TrackerSD(trackerChamberSDname, "TrackerHitsCollection");
0273 G4SDManager::GetSDMpointer()->AddNewDetector(trackerSD);
0274
0275
0276 SetSensitiveDetector("Chamber_LV", trackerSD, true);
0277
0278
0279
0280
0281 G4ThreeVector fieldValue = G4ThreeVector();
0282 fMagFieldMessenger = new G4GlobalMagFieldMessenger(fieldValue);
0283 fMagFieldMessenger->SetVerboseLevel(1);
0284
0285
0286 G4AutoDelete::Register(fMagFieldMessenger);
0287 }
0288
0289
0290
0291 void DetectorConstruction::SetTargetMaterial(G4String materialName)
0292 {
0293 G4NistManager* nistManager = G4NistManager::Instance();
0294
0295 G4Material* pttoMaterial = nistManager->FindOrBuildMaterial(materialName);
0296
0297 if (fTargetMaterial != pttoMaterial) {
0298 if (pttoMaterial) {
0299 fTargetMaterial = pttoMaterial;
0300 if (fLogicTarget) fLogicTarget->SetMaterial(fTargetMaterial);
0301 G4cout << G4endl << "----> The target is made of " << materialName << G4endl;
0302 }
0303 else {
0304 G4cout << G4endl << "--> WARNING from SetTargetMaterial : " << materialName << " not found"
0305 << G4endl;
0306 }
0307 }
0308 }
0309
0310
0311
0312 void DetectorConstruction::SetChamberMaterial(G4String materialName)
0313 {
0314 G4NistManager* nistManager = G4NistManager::Instance();
0315
0316 G4Material* pttoMaterial = nistManager->FindOrBuildMaterial(materialName);
0317
0318 if (fChamberMaterial != pttoMaterial) {
0319 if (pttoMaterial) {
0320 fChamberMaterial = pttoMaterial;
0321 for (G4int copyNo = 0; copyNo < fNbOfChambers; copyNo++) {
0322 if (fLogicChamber[copyNo]) fLogicChamber[copyNo]->SetMaterial(fChamberMaterial);
0323 }
0324 G4cout << G4endl << "----> The chambers are made of " << materialName << G4endl;
0325 }
0326 else {
0327 G4cout << G4endl << "--> WARNING from SetChamberMaterial : " << materialName << " not found"
0328 << G4endl;
0329 }
0330 }
0331 }
0332
0333
0334
0335 void DetectorConstruction::SetMaxStep(G4double maxStep)
0336 {
0337 if ((fStepLimit) && (maxStep > 0.)) fStepLimit->SetMaxAllowedStep(maxStep);
0338 }
0339
0340
0341
0342 void DetectorConstruction::SetCheckOverlaps(G4bool checkOverlaps)
0343 {
0344 fCheckOverlaps = checkOverlaps;
0345 }
0346
0347
0348
0349 }