95 const G4double realTemperture = (273.15 + 23.) * CLHEP::kelvin;
105 G4double h2odensity = 1.000 * CLHEP::g / CLHEP::cm3;
106 G4double a = 1.01 * CLHEP::g / CLHEP::mole;
107 G4Element* elH =
new G4Element(
"Hydrogen",
"H", 1., a);
108 a = 16.00 * CLHEP::g / CLHEP::mole;
109 G4Element* elO =
new G4Element(
"Oxygen",
"O", 8., a);
110 G4Material* medH2O =
new G4Material(
"Water", h2odensity, 2);
111 medH2O->AddElement(elH, 2);
112 medH2O->AddElement(elO, 1);
124 const double num_senseWire =
static_cast<double>(geo.
getNSenseWires());
125 const double num_fieldWire =
static_cast<double>(geo.
getNFieldWires());
126 double totalCS = M_PI * (rmin_outerWall * rmin_outerWall - rmax_innerWall * rmax_innerWall);
129 double senseCS = M_PI * (diameter_senseWire / 2) * (diameter_senseWire / 2) * num_senseWire;
132 double fieldCS = M_PI * (diameter_fieldWire / 2) * (diameter_fieldWire / 2) * num_fieldWire;
135 const double denHelium = medHelium->GetDensity() / 2.0;
136 const double denEthane = medEthane->GetDensity() / 2.0;
137 const double denAluminum = medAluminum->GetDensity() * (fieldCS / totalCS);
138 const double denTungsten = medTungsten->GetDensity() * (senseCS / totalCS);
139 const double density = denHelium + denEthane + denAluminum + denTungsten;
140 G4Material* cdcMed =
new G4Material(
"CDCGasWire", density, 4, kStateGas, realTemperture);
141 cdcMed->AddMaterial(medHelium, denHelium / density);
142 cdcMed->AddMaterial(medEthane, denEthane / density);
143 cdcMed->AddMaterial(medTungsten, denTungsten / density);
144 cdcMed->AddMaterial(medAluminum, denAluminum / density);
146 G4Material* cdcMedGas = cdcMed;
154 const double density2 = denHelium + denEthane;
155 cdcMedGas =
new G4Material(
"CDCRealGas", density2, 2, kStateGas, realTemperture);
156 cdcMedGas->AddMaterial(medHelium, denHelium / density2);
157 cdcMedGas->AddMaterial(medEthane, denEthane / density2);
161 G4cout << *(G4Material::GetMaterialTable());
165 const auto& motherRmin = mother.
getRmin();
166 const auto& motherRmax = mother.getRmax();
167 const auto& motherZ = mother.getZ();
168 G4Polycone* solid_cdc =
169 new G4Polycone(
"solidCDC", 0 * CLHEP::deg, 360.* CLHEP::deg,
170 mother.getNNodes(), motherZ.data(),
171 motherRmin.data(), motherRmax.data());
172 m_logicalCDC =
new G4LogicalVolume(solid_cdc, medAir,
"logicalCDC", 0, 0, 0);
176 "physicalCDC", &topVolume,
false, 0);
179 G4Region* aRegion =
new G4Region(
"CDCEnvelope");
183 m_VisAttributes.push_back(
new G4VisAttributes(
true, G4Colour(0., 1., 0.)));
185 const int iOuterWall = wall.getId();
186 const string wallName = wall.getName();
187 const double wallRmin = wall.getRmin();
188 const double wallRmax = wall.getRmax();
189 const double wallZfwd = wall.getZfwd();
190 const double wallZbwd = wall.getZbwd();
191 const double length = (wallZfwd - wallZbwd) / 2.0;
195 if (strstr((wallName).c_str(),
"MiddleWall") !=
nullptr) {
198 medWall = medAluminum;
200 G4Tubs* outerWallTubeShape =
new G4Tubs(
"solid" + wallName, wallRmin * CLHEP::cm,
201 wallRmax * CLHEP::cm, length * CLHEP::cm, 0 * CLHEP::deg, 360.*CLHEP::deg);
203 G4LogicalVolume* outerWallTube =
new G4LogicalVolume(outerWallTubeShape, medWall,
"solid" + wallName, 0, 0, 0);
205 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (length + wallZbwd)*CLHEP::cm), outerWallTube,
"logical" + wallName,
210 m_VisAttributes.push_back(
new G4VisAttributes(
true, G4Colour(0., 1., 0.)));
212 const string wallName = wall.getName();
213 const double wallRmin = wall.getRmin();
214 const double wallRmax = wall.getRmax();
215 const double wallZfwd = wall.getZfwd();
216 const double wallZbwd = wall.getZbwd();
217 const double length = (wallZfwd - wallZbwd) / 2.0;
218 const int iInnerWall = wall.getId();
221 if (strstr(wallName.c_str(),
"MiddleWall") !=
nullptr) {
223 }
else if (strstr(wallName.c_str(),
"MiddleGlue") !=
nullptr) {
226 medWall = medAluminum;
229 G4Tubs* innerWallTubeShape =
new G4Tubs(
"solid" + wallName, wallRmin * CLHEP::cm,
230 wallRmax * CLHEP::cm, length * CLHEP::cm, 0 * CLHEP::deg, 360.*CLHEP::deg);
231 G4LogicalVolume* innerWallTube =
new G4LogicalVolume(innerWallTubeShape, medWall,
"logical" + wallName, 0, 0, 0);
233 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (length + wallZbwd)*CLHEP::cm), innerWallTube,
"physical" + wallName,
246 for (uint iSLayer = 0; iSLayer < nSLayer; ++iSLayer) {
248 double rmin_sensitive_left, rmax_sensitive_left;
249 double rmin_sensitive_middle, rmax_sensitive_middle;
250 double rmin_sensitive_right, rmax_sensitive_right;
251 double zback_sensitive_left, zfor_sensitive_left;
252 double zback_sensitive_middle, zfor_sensitive_middle;
253 double zback_sensitive_right, zfor_sensitive_right;
256 rmin_sensitive_left, rmax_sensitive_left, zback_sensitive_left, zfor_sensitive_left,
257 rmin_sensitive_middle, rmax_sensitive_middle, zback_sensitive_middle, zfor_sensitive_middle,
258 rmin_sensitive_right, rmax_sensitive_right, zback_sensitive_right, zfor_sensitive_right)) {
263 if ((zfor_sensitive_left - zback_sensitive_left) > length_feedthrough) {
280 G4Tubs* leftTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_leftTube") % iSLayer).str().c_str(),
281 rmin_sensitive_left * CLHEP::cm,
282 rmax_sensitive_left * CLHEP::cm, length_feedthrough * CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
283 G4LogicalVolume* leftTube =
new G4LogicalVolume(leftTubeShape, cdcMed,
284 (boost::format(
"logicalCDCLayer_%1%_leftTube") % iSLayer).str().c_str(), 0, 0, 0);
285 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zback_sensitive_left + length_feedthrough / 2.0)*CLHEP::cm), leftTube,
286 (boost::format(
"physicalCDCLayer_%1%_leftTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
288 G4Tubs* leftSensitiveTubeShape =
new G4Tubs((boost::format(
"solidSD_CDCLayer_%1%_left") % iSLayer).str().c_str(),
289 rmin_sensitive_left * CLHEP::cm, rmax_sensitive_left * CLHEP::cm,
290 (zfor_sensitive_left - zback_sensitive_left - length_feedthrough)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
291 G4LogicalVolume* leftSensitiveTube =
new G4LogicalVolume(leftSensitiveTubeShape, cdcMed,
292 (boost::format(
"logicalSD_CDCLayer_%1%_left") % iSLayer).str().c_str(), 0, 0, 0);
293 leftSensitiveTube->SetSensitiveDetector(
m_sensitive);
294 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_left + zback_sensitive_left + length_feedthrough)*CLHEP::cm / 2.0),
295 leftSensitiveTube, (boost::format(
"physicalSD_CDCLayer_%1%_left") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
313 G4Tubs* leftTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_leftTube") % iSLayer).str().c_str(),
314 rmin_sensitive_left * CLHEP::cm,
315 rmax_sensitive_left * CLHEP::cm, (zfor_sensitive_left - zback_sensitive_left)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
316 G4LogicalVolume* leftTube =
new G4LogicalVolume(leftTubeShape, cdcMed,
317 (boost::format(
"logicalCDCLayer_%1%_leftTube") % iSLayer).str().c_str(), 0, 0, 0);
318 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_left + zback_sensitive_left)*CLHEP::cm / 2.0), leftTube,
319 (boost::format(
"physicalCDCLayer_%1%_leftTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
323 G4Tubs* leftMidTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_leftMidTube") % iSLayer).str().c_str(),
324 rmin_sensitive_middle * CLHEP::cm, rmax_sensitive_middle * CLHEP::cm,
325 (length_feedthrough - zfor_sensitive_left + zback_sensitive_left)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
326 G4LogicalVolume* leftMidTube =
new G4LogicalVolume(leftMidTubeShape, cdcMed,
327 (boost::format(
"logicalCDCLayer_%1%_leftMidTube") % iSLayer).str().c_str(), 0, 0, 0);
329 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (length_feedthrough + zfor_sensitive_left + zback_sensitive_left)*CLHEP::cm / 2.0),
330 leftMidTube, (boost::format(
"physicalCDCLayer_%1%_leftMidTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
333 zback_sensitive_middle = length_feedthrough + zback_sensitive_left;
337 if ((zfor_sensitive_right - zback_sensitive_right) > length_feedthrough) {
354 G4Tubs* rightTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_rightTube") % iSLayer).str().c_str(),
355 rmin_sensitive_right * CLHEP::cm, rmax_sensitive_right * CLHEP::cm, length_feedthrough * CLHEP::cm / 2.0, 0 * CLHEP::deg,
357 G4LogicalVolume* rightTube =
new G4LogicalVolume(rightTubeShape, cdcMed,
358 (boost::format(
"logicalCDCLayer_%1%_rightTube") % iSLayer).str().c_str(), 0, 0, 0);
360 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_right - length_feedthrough / 2.0)*CLHEP::cm), rightTube,
361 (boost::format(
"physicalCDCLayer_%1%_rightTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
365 G4Tubs* rightSensitiveTubeShape =
new G4Tubs((boost::format(
"solidSD_CDCLayer_%1%_right") % iSLayer).str().c_str(),
366 rmin_sensitive_right * CLHEP::cm, rmax_sensitive_right * CLHEP::cm,
367 (zfor_sensitive_right - zback_sensitive_right - length_feedthrough)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
368 G4LogicalVolume* rightSensitiveTube =
new G4LogicalVolume(rightSensitiveTubeShape, cdcMed,
369 (boost::format(
"logicalSD_CDCLayer_%1%_right") % iSLayer).str().c_str(), 0, 0, 0);
370 rightSensitiveTube->SetSensitiveDetector(
m_sensitive);
372 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_right + zback_sensitive_right - length_feedthrough)*CLHEP::cm / 2.0),
373 rightSensitiveTube, (boost::format(
"physicalSD_CDCLayer_%1%_right") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
392 G4Tubs* rightTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_rightTube") % iSLayer).str().c_str(),
393 rmin_sensitive_right * CLHEP::cm, rmax_sensitive_right * CLHEP::cm, (zfor_sensitive_right - zback_sensitive_right)*CLHEP::cm / 2.0,
394 0 * CLHEP::deg, 360.*CLHEP::deg);
395 G4LogicalVolume* rightTube =
new G4LogicalVolume(rightTubeShape, cdcMed,
396 (boost::format(
"logicalCDCLayer_%1%_rightTube") % iSLayer).str().c_str(), 0, 0, 0);
398 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_right + zback_sensitive_right)*CLHEP::cm / 2.0), rightTube,
399 (boost::format(
"physicalCDCLayer_%1%_rightTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
403 G4Tubs* rightMidTubeShape =
new G4Tubs((boost::format(
"solidCDCLayer_%1%_rightMidTube") % iSLayer).str().c_str(),
404 rmin_sensitive_middle * CLHEP::cm, rmax_sensitive_middle * CLHEP::cm,
405 (length_feedthrough - zfor_sensitive_right + zback_sensitive_right)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
406 G4LogicalVolume* rightMidTube =
new G4LogicalVolume(rightMidTubeShape, cdcMed,
407 (boost::format(
"logicalCDCLayer_%1%_rightMidTube") % iSLayer).str().c_str(), 0, 0, 0);
408 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zback_sensitive_right - length_feedthrough + zfor_sensitive_right)*CLHEP::cm / 2.0),
409 rightMidTube, (boost::format(
"physicalCDCLayer_%1%_rightMidTube") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
412 zfor_sensitive_middle = zfor_sensitive_right - length_feedthrough;
417 G4Tubs* middleSensitiveTubeShape =
new G4Tubs((boost::format(
"solidSD_CDCLayer_%1%_middle") % iSLayer).str().c_str(),
418 rmin_sensitive_middle * CLHEP::cm, rmax_sensitive_middle * CLHEP::cm,
419 (zfor_sensitive_middle - zback_sensitive_middle)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
420 G4LogicalVolume* middleSensitiveTube =
new G4LogicalVolume(middleSensitiveTubeShape, cdcMedGas,
421 (boost::format(
"logicalSD_CDCLayer_%1%_middle") % iSLayer).str().c_str(), 0, 0, 0);
424 G4UserLimits* uLimits =
new G4UserLimits(8.5 * CLHEP::cm);
426 middleSensitiveTube->SetUserLimits(uLimits);
427 middleSensitiveTube->SetSensitiveDetector(
m_sensitive);
429 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfor_sensitive_middle + zback_sensitive_middle)*CLHEP::cm / 2.0), middleSensitiveTube,
430 (boost::format(
"physicalSD_CDCLayer_%1%_middle") % iSLayer).str().c_str(),
m_logicalCDC,
false, iSLayer);
434 G4String sName =
"sWire";
439 G4double tAtZ0 = -wb0.
Z() / (wf0.
Z() - wb0.
Z());
442 const G4double epsl = 25.e-4;
443 G4double reductionBwd = (zback_sensitive_middle + epsl) / wb0.
Z();
445 wb0 = reductionBwd * (wb0 - wAtZ0) + wAtZ0;
447 G4double reductionFwd = (zfor_sensitive_middle - epsl) / wf0.
Z();
448 wf0 = reductionFwd * (wf0 - wAtZ0) + wAtZ0;
450 const G4double wireHalfLength = 0.5 * (wf0 - wb0).Mag() * CLHEP::cm;
453 G4Tubs* middleSensitiveSwireShape =
new G4Tubs(sName, 0., sWireRadius, wireHalfLength, 0., 360. * CLHEP::deg);
454 G4LogicalVolume* middleSensitiveSwire =
new G4LogicalVolume(middleSensitiveSwireShape, medTungsten, sName);
458 G4String fName =
"fWire";
460 G4Tubs* middleSensitiveFwireShape =
new G4Tubs(fName, 0., fWireRadius, wireHalfLength, 0., 360. * CLHEP::deg);
461 G4LogicalVolume* middleSensitiveFwire =
new G4LogicalVolume(middleSensitiveFwireShape, medAluminum, fName);
468 const G4double dphi = M_PI / nCells;
471 for (
int ic = 0; ic < nCells; ++ic) {
475 G4double tAtZ02 = -wb.
Z() / (wf.
Z() - wb.
Z());
477 G4double reductionBwd2 = (zback_sensitive_middle + epsl) / wb.
Z();
478 wb = reductionBwd2 * (wb - wAtZ02) + wAtZ02;
479 G4double reductionFwd2 = (zfor_sensitive_middle - epsl) / wf.
Z();
480 wf = reductionFwd2 * (wf - wAtZ02) + wAtZ02;
482 G4double thetaYZ = -asin((wf - wb).Y() / (wf - wb).Mag());
484 B2Vector3D fMinusBInZX((wf - wb).X(), 0., (wf - wb).Z());
485 G4double thetaZX = asin((unitZ.
Cross(fMinusBInZX)).Y() / fMinusBInZX.
Mag());
486 G4RotationMatrix rotM;
488 rotM.rotateX(thetaYZ * CLHEP::rad);
489 rotM.rotateY(thetaZX * CLHEP::rad);
491 G4ThreeVector xyz(0.5 * (wb.
X() + wf.
X()) * CLHEP::cm,
492 0.5 * (wb.
Y() + wf.
Y()) * CLHEP::cm, 0.);
496 new G4PVPlacement(G4Transform3D(rotM, xyz), middleSensitiveSwire, sName, middleSensitiveTube,
false, ic);
500 G4double rF = rmax_sensitive_middle - 0.5 * diameter;
502 G4double phi = atan2(wbF.
Y(), wbF.
X());
503 wbF.
SetX(rF * cos(phi));
504 wbF.
SetY(rF * sin(phi));
507 rF = rmax_sensitive_middle - 0.5 * diameter;
508 phi = atan2(wfF.
Y(), wfF.
X());
509 wfF.
SetX(rF * cos(phi));
510 wfF.
SetY(rF * sin(phi));
512 thetaYZ = -asin((wfF - wbF).Y() / (wfF - wbF).Mag());
514 fMinusBInZX = wfF - wbF;
515 fMinusBInZX.
SetY(0.);
516 thetaZX = asin((unitZ.
Cross(fMinusBInZX)).Y() / fMinusBInZX.
Mag());
518 G4RotationMatrix rotM1;
519 rotM1.rotateX(thetaYZ * CLHEP::rad);
520 rotM1.rotateY(thetaZX * CLHEP::rad);
522 xyz.setX(0.5 * (wbF.
X() + wfF.
X()) * CLHEP::cm);
523 xyz.setY(0.5 * (wbF.
Y() + wfF.
Y()) * CLHEP::cm);
525 if (iSLayer != nSLayer - 1) {
527 new G4PVPlacement(G4Transform3D(rotM1, xyz), middleSensitiveFwire, fName, middleSensitiveTube,
false, ic);
533 phi = atan2(wbF.
Y(), wbF.
X());
534 wbF.
SetX(rF * cos(phi + dphi));
535 wbF.
SetY(rF * sin(phi + dphi));
539 phi = atan2(wfF.
Y(), wfF.
X());
540 wfF.
SetX(rF * cos(phi + dphi));
541 wfF.
SetY(rF * sin(phi + dphi));
543 thetaYZ = -asin((wfF - wbF).Y() / (wfF - wbF).Mag());
545 fMinusBInZX = wfF - wbF;
546 fMinusBInZX.
SetY(0.);
547 thetaZX = asin((unitZ.
Cross(fMinusBInZX)).Y() / fMinusBInZX.
Mag());
549 G4RotationMatrix rotM2;
550 rotM2.rotateX(thetaYZ * CLHEP::rad);
551 rotM2.rotateY(thetaZX * CLHEP::rad);
553 xyz.setX(0.5 * (wbF.
X() + wfF.
X()) * CLHEP::cm);
554 xyz.setY(0.5 * (wbF.
Y() + wfF.
Y()) * CLHEP::cm);
557 new G4PVPlacement(G4Transform3D(rotM2, xyz), middleSensitiveFwire, fName, middleSensitiveTube,
false, ic + nCells);
561 rF = rmax_sensitive_middle - 0.5 * diameter;
562 phi = atan2(wbF.
Y(), wbF.
X());
563 wbF.
SetX(rF * cos(phi + dphi));
564 wbF.
SetY(rF * sin(phi + dphi));
567 rF = rmax_sensitive_middle - 0.5 * diameter;
568 phi = atan2(wfF.
Y(), wfF.
X());
569 wfF.
SetX(rF * cos(phi + dphi));
570 wfF.
SetY(rF * sin(phi + dphi));
572 thetaYZ = -asin((wfF - wbF).Y() / (wfF - wbF).Mag());
574 fMinusBInZX = wfF - wbF;
575 fMinusBInZX.
SetY(0.);
576 thetaZX = asin((unitZ.
Cross(fMinusBInZX)).Y() / fMinusBInZX.
Mag());
578 G4RotationMatrix rotM3;
579 rotM3.rotateX(thetaYZ * CLHEP::rad);
580 rotM3.rotateY(thetaZX * CLHEP::rad);
582 xyz.setX(0.5 * (wbF.
X() + wfF.
X()) * CLHEP::cm);
583 xyz.setY(0.5 * (wbF.
Y() + wfF.
Y()) * CLHEP::cm);
585 if (iSLayer != nSLayer - 1) {
586 new G4PVPlacement(G4Transform3D(rotM3, xyz), middleSensitiveFwire, fName, middleSensitiveTube,
false, ic + 2 * nCells);
596 m_VisAttributes.push_back(
new G4VisAttributes(
true, G4Colour(1., 1., 0.)));
598 for (
const auto& epLayer : endplate.getEndPlateLayers()) {
599 const int iEPLayer = epLayer.getILayer();
600 const string name = epLayer.getName();
601 const double rmin = epLayer.getRmin();
602 const double rmax = epLayer.getRmax();
603 const double zbwd = epLayer.getZbwd();
604 const double zfwd = epLayer.getZfwd();
605 const double length = (zfwd - zbwd) / 2.0;
607 G4Tubs* tube =
new G4Tubs(
"solidCDCEndplate" + name, rmin * CLHEP::cm,
608 rmax * CLHEP::cm, length * CLHEP::cm, 0 * CLHEP::deg, 360.*CLHEP::deg);
609 G4LogicalVolume* logical =
new G4LogicalVolume(tube,
Materials::get(
"G4_Al"),
610 "logicalCDCEndplate" + name, 0, 0);
612 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (zfwd + zbwd)*CLHEP::cm / 2.0), logical,
613 "physicalCDCEndplate" + name,
m_logicalCDC,
false, iEPLayer);
622 const int iEB = frontend.getId();
623 const double ebInnerR = frontend.getRmin();
624 const double ebOuterR = frontend.getRmax();
625 const double ebBZ = frontend.getZbwd();
626 const double ebFZ = frontend.getZfwd();
628 G4Tubs* ebTubeShape =
new G4Tubs((boost::format(
"solidSD_ElectronicsBoard_Layer%1%") % iEB).str().c_str(), ebInnerR * CLHEP::cm,
629 ebOuterR * CLHEP::cm, (ebFZ - ebBZ)*CLHEP::cm / 2.0, 0 * CLHEP::deg, 360.*CLHEP::deg);
631 G4LogicalVolume* ebTube =
new G4LogicalVolume(ebTubeShape, medNEMA_G10_Plate,
632 (boost::format(
"logicalSD_ElectronicsBoard_Layer%1%") % iEB).str().c_str(), 0, 0, 0);
636 new G4PVPlacement(0, G4ThreeVector(0.0, 0.0, (ebFZ + ebBZ)*CLHEP::cm / 2.0), ebTube,
637 (boost::format(
"physicalSD_ElectronicsBoard_Layer%1%") % iEB).str().c_str(),
m_logicalCDC,
false, iEB);
658 for (
const auto& rib : geo.
getRibs()) {
660 const int id = rib.getId();
661 const double length = rib.getLength();
662 const double width = rib.getWidth();
663 const double thick = rib.getThick();
664 const double rotx = rib.getRotX();
665 const double roty = rib.getRotY();
666 const double rotz = rib.getRotZ();
667 const double x = rib.getX();
668 const double y = rib.getY();
669 const double z = rib.getZ();
670 const int offset = rib.getOffset();
671 const int ndiv = rib.getNDiv();
673 const string solidName =
"solidRib" + to_string(
id);
674 const string logicalName =
"logicalRib" + to_string(
id);
675 G4Box* boxShape =
new G4Box(solidName, 0.5 * length * CLHEP::cm,
676 0.5 * width * CLHEP::cm,
677 0.5 * thick * CLHEP::cm);
679 const double rmax = 0.5 * length;
680 const double rmin = max((rmax - thick), 0.);
681 G4Tubs* tubeShape =
new G4Tubs(solidName,
684 0.5 * width * CLHEP::cm,
691 G4LogicalVolume* logicalV =
new G4LogicalVolume(boxShape, medAluminum, logicalName, 0, 0, 0);
692 if (
id > 39 &&
id < 78)
693 logicalV =
new G4LogicalVolume(boxShape, medCopper, logicalName, 0, 0, 0);
694 if ((
id > 77 &&
id < 94) || (
id > 131 &&
id < 146))
695 logicalV =
new G4LogicalVolume(boxShape, medNEMA_G10_Plate, logicalName, 0, 0, 0);
696 if (
id > 93 &&
id < 110)
697 logicalV =
new G4LogicalVolume(tubeShape, medCopper, logicalName, 0, 0, 0);
698 if (
id > 109 &&
id < 126)
699 logicalV =
new G4LogicalVolume(tubeShape, medH2O, logicalName, 0, 0, 0);
700 if (
id > 127 &&
id < 132)
701 logicalV =
new G4LogicalVolume(boxShape, medHV, logicalName, 0, 0, 0);
711 const double phi = 360.0 / ndiv;
713 G4RotationMatrix rot = G4RotationMatrix();
715 if (
id > 93 &&
id < 126) dz = 0;
717 G4ThreeVector arm(x * CLHEP::cm, y * CLHEP::cm, z * CLHEP::cm - dz * CLHEP::cm / 2.0);
722 rot.rotateZ(0.5 * phi * CLHEP::deg);
723 arm.rotateZ(0.5 * phi * CLHEP::deg);
725 for (
int i = 0; i < ndiv; ++i) {
726 const string physicalName =
"physicalRib_" + to_string(
id) +
" " + to_string(i);
727 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
729 rot.rotateZ(phi * CLHEP::deg);
730 arm.rotateZ(phi * CLHEP::deg);
738 for (
const auto& rib2 : geo.
getRib2s()) {
740 const int id = rib2.getId();
741 const double length = rib2.getLength();
742 const double width = rib2.getWidth();
743 const double thick = rib2.getThick();
744 const double width2 = rib2.getWidth2();
745 const double thick2 = rib2.getThick2();
746 const double rotx = rib2.getRotX();
747 const double roty = rib2.getRotY();
748 const double rotz = rib2.getRotZ();
749 const double x = rib2.getX();
750 const double y = rib2.getY();
751 const double z = rib2.getZ();
752 const int ndiv = rib2.getNDiv();
754 const string solidName =
"solidRib2" + to_string(
id);
755 const string logicalName =
"logicalRib2" + to_string(
id);
756 G4Trd* trdShape =
new G4Trd(solidName,
757 0.5 * thick * CLHEP::cm,
758 0.5 * thick2 * CLHEP::cm,
759 0.5 * width * CLHEP::cm,
760 0.5 * width2 * CLHEP::cm,
761 0.5 * length * CLHEP::cm);
763 G4LogicalVolume* logicalV =
new G4LogicalVolume(trdShape, medAluminum, logicalName, 0, 0, 0);
766 logicalV =
new G4LogicalVolume(trdShape, medCopper, logicalName, 0, 0, 0);
770 const double phi = 360.0 / ndiv;
772 G4RotationMatrix rot = G4RotationMatrix();
773 G4ThreeVector arm(x * CLHEP::cm, y * CLHEP::cm, z * CLHEP::cm - thick * CLHEP::cm / 2.0);
778 for (
int i = 0; i < ndiv; ++i) {
779 const string physicalName =
"physicalRib2_" + to_string(
id) +
" " + to_string(i);
780 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
782 rot.rotateZ(phi * CLHEP::deg);
783 arm.rotateZ(phi * CLHEP::deg);
791 for (
const auto& rib3 : geo.
getRib3s()) {
793 const int id = rib3.getId();
794 const double length = rib3.getLength();
795 const double width = rib3.getWidth();
796 const double thick = rib3.getThick();
797 const double r = rib3.getR();
798 const double x = rib3.getX();
799 const double y = rib3.getY();
800 const double z = rib3.getZ();
801 const double rx = rib3.getRx();
802 const double ry = rib3.getRy();
803 const double rz = rib3.getRz();
804 const int offset = rib3.getOffset();
805 const int ndiv = rib3.getNDiv();
807 const string logicalName =
"logicalRib3" + to_string(
id);
808 G4VSolid* boxShape =
new G4Box(
"Block",
809 0.5 * length * CLHEP::cm,
810 0.5 * width * CLHEP::cm,
811 0.5 * thick * CLHEP::cm);
812 G4VSolid* tubeShape =
new G4Tubs(
"Hole",
819 G4RotationMatrix rotsub = G4RotationMatrix();
820 rotsub.rotateX(90. * CLHEP::deg);
821 G4ThreeVector trnsub(rx * CLHEP::cm - x * CLHEP::cm, ry * CLHEP::cm - y * CLHEP::cm,
822 rz * CLHEP::cm - z * CLHEP::cm + 0.5 * thick * CLHEP::cm);
823 G4VSolid* coolingBlock =
new G4SubtractionSolid(
"Block-Hole",
826 G4Transform3D(rotsub,
829 G4LogicalVolume* logicalV =
new G4LogicalVolume(coolingBlock, medCopper, logicalName, 0, 0, 0);
833 const double phi = 360.0 / ndiv;
835 G4RotationMatrix rot = G4RotationMatrix();
836 G4ThreeVector arm(x * CLHEP::cm, y * CLHEP::cm, z * CLHEP::cm - thick * CLHEP::cm / 2.0);
839 rot.rotateZ(0.5 * phi * CLHEP::deg);
840 arm.rotateZ(0.5 * phi * CLHEP::deg);
842 for (
int i = 0; i < ndiv; ++i) {
843 const string physicalName =
"physicalRib3_" + to_string(
id) +
" " + to_string(i);
844 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
846 rot.rotateZ(phi * CLHEP::deg);
847 arm.rotateZ(phi * CLHEP::deg);
855 for (
const auto& rib4 : geo.
getRib4s()) {
857 const int id = rib4.getId();
858 const double length = rib4.getLength();
859 const double width = rib4.getWidth();
860 const double thick = rib4.getThick();
861 const double length2 = rib4.getLength2();
862 const double width2 = rib4.getWidth2();
863 const double thick2 = rib4.getThick2();
864 const double x = rib4.getX();
865 const double y = rib4.getY();
866 const double z = rib4.getZ();
867 const double x2 = rib4.getX2();
868 const double y2 = rib4.getY2();
869 const double z2 = rib4.getZ2();
870 const int offset = rib4.getOffset();
871 const int ndiv = rib4.getNDiv();
873 const string logicalName =
"logicalRib4" + to_string(
id);
874 G4VSolid* baseShape =
new G4Box(
"Base",
875 0.5 * length * CLHEP::cm,
876 0.5 * width * CLHEP::cm,
877 0.5 * thick * CLHEP::cm);
878 G4VSolid* sqShape =
new G4Box(
"Sq",
879 0.5 * length2 * CLHEP::cm,
880 0.5 * width2 * CLHEP::cm,
881 0.5 * thick2 * CLHEP::cm);
883 G4RotationMatrix rotsub = G4RotationMatrix();
884 double dzc = (z2 - thick2 / 2.) - (z - thick / 2.);
885 G4ThreeVector trnsub(x2 * CLHEP::cm - x * CLHEP::cm,
886 y2 * CLHEP::cm - y * CLHEP::cm,
888 G4VSolid* sqHoleBase =
new G4SubtractionSolid(
"Box-Sq",
891 G4Transform3D(rotsub,
895 G4LogicalVolume* logicalV =
new G4LogicalVolume(sqHoleBase, medCopper, logicalName, 0, 0, 0);
897 logicalV =
new G4LogicalVolume(sqHoleBase, medNEMA_G10_Plate, logicalName, 0, 0, 0);
900 logicalV->SetSensitiveDetector(sensitiveDetector);
906 const double phi = 360.0 / ndiv;
908 G4RotationMatrix rot = G4RotationMatrix();
909 G4ThreeVector arm(x * CLHEP::cm, y * CLHEP::cm, z * CLHEP::cm - thick * CLHEP::cm / 2.0);
912 rot.rotateZ(0.5 * phi * CLHEP::deg);
913 arm.rotateZ(0.5 * phi * CLHEP::deg);
915 for (
int i = 0; i < ndiv; ++i) {
916 const string physicalName =
"physicalRib4_" + to_string(
id) +
" " + to_string(i);
917 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
919 rot.rotateZ(phi * CLHEP::deg);
920 arm.rotateZ(phi * CLHEP::deg);
927 for (
const auto& rib5 : geo.
getRib5s()) {
929 const int id = rib5.getId();
930 const double dr = rib5.getDr();
931 const double dz = rib5.getDz();
932 const double width = rib5.getWidth();
933 const double thick = rib5.getThick();
934 const double rin = rib5.getRin();
935 const double x = rib5.getX();
936 const double y = rib5.getY();
937 const double z = rib5.getZ();
938 const double rotx = rib5.getRotx();
939 const double roty = rib5.getRoty();
940 const double rotz = rib5.getRotz();
941 const int offset = rib5.getOffset();
942 const int ndiv = rib5.getNDiv();
944 const string solidName =
"solidRib5" + to_string(
id);
945 const string logicalName =
"logicalRib5" + to_string(
id);
947 const double rmax = rin + thick;
948 const double rmin = rin;
949 const double dphi = 2. * atan2(dz, dr);
950 const double ddphi = thick *
tan(dphi) / rin;
951 const double ddphi2 = width / 2. * width / 2. / (x + dr) / rin;
952 const double cphi = dphi - ddphi - ddphi2;
953 G4Tubs* tubeShape =
new G4Tubs(solidName,
956 0.5 * width * CLHEP::cm,
960 G4LogicalVolume* logicalV =
new G4LogicalVolume(tubeShape, medAluminum, logicalName, 0, 0, 0);
964 const double phi = 360.0 / ndiv;
966 G4RotationMatrix rot = G4RotationMatrix();
969 G4ThreeVector arm(x * CLHEP::cm, y * CLHEP::cm, z * CLHEP::cm - rin * CLHEP::cm - thick * CLHEP::cm);
974 rot.rotateZ(0.5 * phi * CLHEP::deg);
975 arm.rotateZ(0.5 * phi * CLHEP::deg);
977 for (
int i = 0; i < ndiv; ++i) {
978 const string physicalName =
"physicalRib5_" + to_string(
id) +
" " + to_string(i);
979 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
981 rot.rotateZ(phi * CLHEP::deg);
982 arm.rotateZ(phi * CLHEP::deg);
1076 string Aluminum = content.getString(
"Aluminum");
1079 G4double density = 1.000 * CLHEP::g / CLHEP::cm3;
1080 G4double a = 1.01 * CLHEP::g / CLHEP::mole;
1081 G4Element* elH =
new G4Element(
"Hydrogen",
"H", 1., a);
1082 a = 16.00 * CLHEP::g / CLHEP::mole;
1083 G4Element* elO =
new G4Element(
"Oxygen",
"O", 8., a);
1084 G4Material* medH2O =
new G4Material(
"Water", density, 2);
1085 medH2O->AddElement(elH, 2);
1086 medH2O->AddElement(elO, 1);
1094 m_VisAttributes.push_back(
new G4VisAttributes(
true, G4Colour(0., 1., 0.)));
1095 const int nCover = content.getNumberNodes(
"Covers/Cover");
1096 for (
int iCover = 0; iCover < nCover; ++iCover) {
1097 GearDir coverContent(content);
1098 coverContent.
append((boost::format(
"/Covers/Cover[%1%]/") % (iCover + 1)).str());
1099 const string scoverID = coverContent.
getString(
"@id");
1100 const int coverID = atoi(scoverID.c_str());
1101 const string coverName = coverContent.
getString(
"Name");
1102 const double coverInnerR1 = coverContent.
getLength(
"InnerR1");
1103 const double coverInnerR2 = coverContent.
getLength(
"InnerR2");
1104 const double coverOuterR1 = coverContent.
getLength(
"OuterR1");
1105 const double coverOuterR2 = coverContent.
getLength(
"OuterR2");
1106 const double coverThick = coverContent.
getLength(
"Thickness");
1107 const double coverPosZ = coverContent.
getLength(
"PosZ");
1109 const double rmin1 = coverInnerR1;
1110 const double rmax1 = coverOuterR1;
1111 const double rmin2 = coverInnerR2;
1112 const double rmax2 = coverOuterR2;
1123 if (coverID == 7 || coverID == 10) {
1124 createCone(rmin1, rmax1, rmin2, rmax2, coverThick, coverPosZ, coverID, medAluminum, coverName);
1126 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medAluminum, coverName);
1129 if (coverID > 22 && coverID < 29)
1130 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medCopper, coverName);
1131 if (coverID > 28 && coverID < 35)
1132 createTorus(rmin1, rmax1, coverThick, coverPosZ, coverID, medCopper, coverName);
1133 if (coverID > 34 && coverID < 41)
1134 createTorus(rmin1, rmax1, coverThick, coverPosZ, coverID, medH2O, coverName);
1135 if (coverID == 45 || coverID == 46)
1136 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medLV, coverName);
1137 if (coverID == 47 || coverID == 48)
1138 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medFiber, coverName);
1139 if (coverID == 49 || coverID == 50)
1140 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medCAT7, coverName);
1141 if (coverID == 51 || coverID == 52)
1142 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medTRG, coverName);
1144 createTube(rmin1, rmax1, coverThick, coverPosZ, coverID, medHV, coverName);
1147 const int nCover2 = content.getNumberNodes(
"Covers/Cover2");
1148 for (
int iCover2 = 0; iCover2 < nCover2; ++iCover2) {
1149 GearDir cover2Content(content);
1150 cover2Content.
append((boost::format(
"/Cover2s/Cover2[%1%]/") % (iCover2 + 1)).str());
1151 const string scover2ID = cover2Content.
getString(
"@id");
1152 const int cover2ID = atoi(scover2ID.c_str());
1153 const string cover2Name = cover2Content.
getString(
"Name");
1154 const double cover2InnerR = cover2Content.
getLength(
"InnerR");
1155 const double cover2OuterR = cover2Content.
getLength(
"OuterR");
1156 const double cover2StartPhi = cover2Content.
getLength(
"StartPhi");
1157 const double cover2DeltaPhi = cover2Content.
getLength(
"DeltaPhi");
1158 const double cover2Thick = cover2Content.
getLength(
"Thickness");
1159 const double cover2PosZ = cover2Content.
getLength(
"PosZ");
1162 createTube2(cover2InnerR, cover2OuterR, cover2StartPhi, cover2DeltaPhi, cover2Thick, cover2PosZ, cover2ID, medHV, cover2Name);
1163 if (cover2ID > 10 && cover2ID < 14)
1164 createTube2(cover2InnerR, cover2OuterR, cover2StartPhi, cover2DeltaPhi, cover2Thick, cover2PosZ, cover2ID, medFiber, cover2Name);
1165 if (cover2ID > 13 && cover2ID < 23)
1166 createTube2(cover2InnerR, cover2OuterR, cover2StartPhi, cover2DeltaPhi, cover2Thick, cover2PosZ, cover2ID, medCAT7, cover2Name);
1167 if (cover2ID > 22 && cover2ID < 29)
1168 createTube2(cover2InnerR, cover2OuterR, cover2StartPhi, cover2DeltaPhi, cover2Thick, cover2PosZ, cover2ID, medTRG, cover2Name);
1171 const int nRibs = content.getNumberNodes(
"Covers/Rib");
1172 for (
int iRib = 0; iRib < nRibs; ++iRib) {
1174 ribContent.
append((boost::format(
"/Covers/Rib[%1%]/") % (iRib + 1)).str());
1175 const string sribID = ribContent.
getString(
"@id");
1176 const int ribID = atoi(sribID.c_str());
1178 const double length = ribContent.
getLength(
"Length");
1179 const double width = ribContent.
getLength(
"Width");
1180 const double thick = ribContent.
getLength(
"Thickness");
1181 const double rotX = ribContent.
getLength(
"RotX");
1182 const double rotY = ribContent.
getLength(
"RotY");
1183 const double rotZ = ribContent.
getLength(
"RotZ");
1184 const double cX = ribContent.
getLength(
"PosX");
1185 const double cY = ribContent.
getLength(
"PosY");
1186 const double cZ = ribContent.
getLength(
"PosZ");
1187 const int offset = atoi((ribContent.
getString(
"Offset")).c_str());
1188 const int number = atoi((ribContent.
getString(
"NDiv")).c_str());
1190 const string solidName =
"solidRib" + to_string(ribID);
1191 const string logicalName =
"logicalRib" + to_string(ribID);
1192 G4Box* boxShape =
new G4Box(solidName, 0.5 * length * CLHEP::cm,
1193 0.5 * width * CLHEP::cm,
1194 0.5 * thick * CLHEP::cm);
1195 const double rmax = 0.5 * length;
1196 const double rmin = max((rmax - thick), 0.);
1197 G4Tubs* tubeShape =
new G4Tubs(solidName,
1200 0.5 * width * CLHEP::cm,
1207 G4LogicalVolume* logicalV =
new G4LogicalVolume(boxShape, medAluminum, logicalName, 0, 0, 0);
1208 if (ribID > 39 && ribID < 78)
1209 logicalV =
new G4LogicalVolume(boxShape, medCopper, logicalName, 0, 0, 0);
1210 if ((ribID > 77 && ribID < 94) || (ribID > 131 && ribID < 146))
1211 logicalV =
new G4LogicalVolume(boxShape, medNEMA_G10_Plate, logicalName, 0, 0, 0);
1212 if (ribID > 93 && ribID < 110)
1213 logicalV =
new G4LogicalVolume(tubeShape, medCopper, logicalName, 0, 0, 0);
1214 if (ribID > 109 && ribID < 126)
1215 logicalV =
new G4LogicalVolume(tubeShape, medH2O, logicalName, 0, 0, 0);
1217 if (ribID > 127 && ribID < 132)
1218 logicalV =
new G4LogicalVolume(boxShape, medHV, logicalName, 0, 0, 0);
1228 const double phi = 360.0 / number;
1230 G4RotationMatrix rot = G4RotationMatrix();
1233 if (ribID > 93 && ribID < 126) dz = 0;
1234 G4ThreeVector arm(cX * CLHEP::cm, cY * CLHEP::cm, cZ * CLHEP::cm - dz * CLHEP::cm / 2.0);
1240 rot.rotateZ(0.5 * phi * CLHEP::deg);
1241 arm.rotateZ(0.5 * phi * CLHEP::deg);
1243 for (
int i = 0; i < number; ++i) {
1244 const string physicalName =
"physicalRib_" + to_string(ribID) +
" " + to_string(i);
1245 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
1247 rot.rotateZ(phi * CLHEP::deg);
1248 arm.rotateZ(phi * CLHEP::deg);
1253 const int nRib2s = content.getNumberNodes(
"Covers/Rib2");
1254 for (
int iRib2 = 0; iRib2 < nRib2s; ++iRib2) {
1256 rib2Content.
append((boost::format(
"/Covers/Rib2[%1%]/") % (iRib2 + 1)).str());
1257 const string srib2ID = rib2Content.
getString(
"@id");
1258 const int rib2ID = atoi(srib2ID.c_str());
1260 const double length = rib2Content.
getLength(
"Length");
1261 const double width = rib2Content.
getLength(
"Width");
1262 const double thick = rib2Content.
getLength(
"Thickness");
1263 const double width2 = rib2Content.
getLength(
"Width2");
1264 const double thick2 = rib2Content.
getLength(
"Thickness2");
1265 const double rotX = rib2Content.
getLength(
"RotX");
1266 const double rotY = rib2Content.
getLength(
"RotY");
1267 const double rotZ = rib2Content.
getLength(
"RotZ");
1268 const double cX = rib2Content.
getLength(
"PosX");
1269 const double cY = rib2Content.
getLength(
"PosY");
1270 const double cZ = rib2Content.
getLength(
"PosZ");
1271 const int number = atoi((rib2Content.
getString(
"NDiv")).c_str());
1273 const string solidName =
"solidRib2" + to_string(rib2ID);
1274 const string logicalName =
"logicalRib2" + to_string(rib2ID);
1275 G4Trd* trdShape =
new G4Trd(solidName,
1276 0.5 * thick * CLHEP::cm,
1277 0.5 * thick2 * CLHEP::cm,
1278 0.5 * width * CLHEP::cm,
1279 0.5 * width2 * CLHEP::cm,
1280 0.5 * length * CLHEP::cm);
1282 G4LogicalVolume* logicalV =
new G4LogicalVolume(trdShape, medAluminum, logicalName, 0, 0, 0);
1284 logicalV =
new G4LogicalVolume(trdShape, medCopper, logicalName, 0, 0, 0);
1289 const double phi = 360.0 / number;
1291 G4RotationMatrix rot = G4RotationMatrix();
1292 G4ThreeVector arm(cX * CLHEP::cm, cY * CLHEP::cm, cZ * CLHEP::cm - thick * CLHEP::cm / 2.0);
1297 for (
int i = 0; i < number; ++i) {
1298 const string physicalName =
"physicalRib2_" + to_string(rib2ID) +
" " + to_string(i);
1299 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
1301 rot.rotateZ(phi * CLHEP::deg);
1302 arm.rotateZ(phi * CLHEP::deg);
1307 const int nRib3s = content.getNumberNodes(
"Covers/Rib3");
1308 for (
int iRib3 = 0; iRib3 < nRib3s; ++iRib3) {
1310 rib3Content.
append((boost::format(
"/Covers/Rib3[%1%]/") % (iRib3 + 1)).str());
1311 const string srib3ID = rib3Content.
getString(
"@id");
1312 const int rib3ID = atoi(srib3ID.c_str());
1314 const double length = rib3Content.
getLength(
"Length");
1315 const double width = rib3Content.
getLength(
"Width");
1316 const double thick = rib3Content.
getLength(
"Thickness");
1317 const double r = rib3Content.
getLength(
"HoleR");
1318 const double cX = rib3Content.
getLength(
"PosX");
1319 const double cY = rib3Content.
getLength(
"PosY");
1320 const double cZ = rib3Content.
getLength(
"PosZ");
1321 const double hX = rib3Content.
getLength(
"HoleX");
1322 const double hY = rib3Content.
getLength(
"HoleY");
1323 const double hZ = rib3Content.
getLength(
"HoleZ");
1324 const int offset = atoi((rib3Content.
getString(
"Offset")).c_str());
1325 const int number = atoi((rib3Content.
getString(
"NDiv")).c_str());
1327 const string logicalName =
"logicalRib3" + to_string(rib3ID);
1328 G4VSolid* boxShape =
new G4Box(
"Block",
1329 0.5 * length * CLHEP::cm,
1330 0.5 * width * CLHEP::cm,
1331 0.5 * thick * CLHEP::cm);
1332 G4VSolid* tubeShape =
new G4Tubs(
"Hole",
1338 G4RotationMatrix rotsub = G4RotationMatrix();
1339 G4ThreeVector trnsub(cX * CLHEP::cm - hX * CLHEP::cm, cY * CLHEP::cm - hY * CLHEP::cm,
1340 cZ * CLHEP::cm - hZ * CLHEP::cm + 0.5 * thick * CLHEP::cm);
1341 G4VSolid* coolingBlock =
new G4SubtractionSolid(
"Block-Hole",
1344 G4Transform3D(rotsub,
1347 G4LogicalVolume* logicalV =
new G4LogicalVolume(coolingBlock, medCopper, logicalName, 0, 0, 0);
1351 const double phi = 360.0 / number;
1353 G4RotationMatrix rot = G4RotationMatrix();
1354 G4ThreeVector arm(cX * CLHEP::cm, cY * CLHEP::cm, cZ * CLHEP::cm - thick * CLHEP::cm / 2.0);
1357 rot.rotateZ(0.5 * phi * CLHEP::deg);
1358 arm.rotateZ(0.5 * phi * CLHEP::deg);
1360 for (
int i = 0; i < number; ++i) {
1361 const string physicalName =
"physicalRib3_" + to_string(rib3ID) +
" " + to_string(i);
1362 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
1364 rot.rotateZ(phi * CLHEP::deg);
1365 arm.rotateZ(phi * CLHEP::deg);
1370 const int nRib4s = content.getNumberNodes(
"Covers/Rib4");
1371 for (
int iRib4 = 0; iRib4 < nRib4s; ++iRib4) {
1373 rib4Content.
append((boost::format(
"/Covers/Rib4[%1%]/") % (iRib4 + 1)).str());
1374 const string srib4ID = rib4Content.
getString(
"@id");
1375 const int rib4ID = atoi(srib4ID.c_str());
1377 const double length = rib4Content.
getLength(
"Length");
1378 const double width = rib4Content.
getLength(
"Width");
1379 const double thick = rib4Content.
getLength(
"Thickness");
1380 const double length2 = rib4Content.
getLength(
"Length2");
1381 const double width2 = rib4Content.
getLength(
"Width2");
1382 const double thick2 = rib4Content.
getLength(
"Thickness2");
1383 const double cX = rib4Content.
getLength(
"PosX");
1384 const double cY = rib4Content.
getLength(
"PosY");
1385 const double cZ = rib4Content.
getLength(
"PosZ");
1386 const double hX = rib4Content.
getLength(
"HoleX");
1387 const double hY = rib4Content.
getLength(
"HoleY");
1388 const double hZ = rib4Content.
getLength(
"HoleZ");
1389 const int offset = atoi((rib4Content.
getString(
"Offset")).c_str());
1390 const int number = atoi((rib4Content.
getString(
"NDiv")).c_str());
1392 const string logicalName =
"logicalRib4" + to_string(rib4ID);
1393 G4VSolid* baseShape =
new G4Box(
"Base",
1394 0.5 * length * CLHEP::cm,
1395 0.5 * width * CLHEP::cm,
1396 0.5 * thick * CLHEP::cm);
1397 G4VSolid* sqShape =
new G4Box(
"Sq",
1398 0.5 * length2 * CLHEP::cm,
1399 0.5 * width2 * CLHEP::cm,
1400 0.5 * thick2 * CLHEP::cm);
1401 G4RotationMatrix rotsub = G4RotationMatrix();
1402 double dzc = (hZ - thick2 / 2.) - (cZ - thick / 2.);
1403 G4ThreeVector trnsub(hX * CLHEP::cm - cX * CLHEP::cm,
1404 hY * CLHEP::cm - cY * CLHEP::cm,
1406 G4VSolid* sqHoleBase =
new G4SubtractionSolid(
"Base-Sq",
1409 G4Transform3D(rotsub,
1413 G4LogicalVolume* logicalV =
new G4LogicalVolume(sqHoleBase, medCopper, logicalName, 0, 0, 0);
1415 logicalV =
new G4LogicalVolume(sqHoleBase, medNEMA_G10_Plate, logicalName, 0, 0, 0);
1419 const double phi = 360.0 / number;
1421 G4RotationMatrix rot = G4RotationMatrix();
1422 G4ThreeVector arm(cX * CLHEP::cm, cY * CLHEP::cm, cZ * CLHEP::cm - thick * CLHEP::cm / 2.0);
1425 rot.rotateZ(0.5 * phi * CLHEP::deg);
1426 arm.rotateZ(0.5 * phi * CLHEP::deg);
1428 for (
int i = 0; i < number; ++i) {
1429 const string physicalName =
"physicalRib4_" + to_string(rib4ID) +
" " + to_string(i);
1430 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
1432 rot.rotateZ(phi * CLHEP::deg);
1433 arm.rotateZ(phi * CLHEP::deg);
1438 const int nRib5s = content.getNumberNodes(
"Covers/Rib5");
1439 for (
int iRib5 = 0; iRib5 < nRib5s; ++iRib5) {
1441 rib5Content.
append((boost::format(
"/Covers/Rib5[%1%]/") % (iRib5 + 1)).str());
1442 const string srib5ID = rib5Content.
getString(
"@id");
1443 const int rib5ID = atoi(srib5ID.c_str());
1445 const double dr = rib5Content.
getLength(
"DeltaR");
1446 const double dz = rib5Content.
getLength(
"DeltaZ");
1447 const double width = rib5Content.
getLength(
"Width");
1448 const double thick = rib5Content.
getLength(
"Thickness");
1449 const double rin = rib5Content.
getLength(
"Rin");
1450 const double rotX = rib5Content.
getLength(
"RotX");
1451 const double rotY = rib5Content.
getLength(
"RotY");
1452 const double rotZ = rib5Content.
getLength(
"RotZ");
1453 const double cX = rib5Content.
getLength(
"PosX");
1454 const double cY = rib5Content.
getLength(
"PosY");
1455 const double cZ = rib5Content.
getLength(
"PosZ");
1456 const int offset = atoi((rib5Content.
getString(
"Offset")).c_str());
1457 const int number = atoi((rib5Content.
getString(
"NDiv")).c_str());
1459 const string solidName =
"solidRib5" + to_string(rib5ID);
1460 const string logicalName =
"logicalRib5" + to_string(rib5ID);
1461 const double rmax = rin + thick;
1462 const double rmin = rin;
1463 const double dphi = 2. * atan2(dz, dr);
1464 const double ddphi = thick *
tan(dphi) / rin;
1465 const double ddphi2 = width / 2. * width / 2. / (cX + dr) / rin;
1466 const double cphi = dphi - ddphi - ddphi2;
1467 G4Tubs* tubeShape =
new G4Tubs(solidName,
1470 0.5 * width * CLHEP::cm,
1474 G4LogicalVolume* logicalV =
new G4LogicalVolume(tubeShape, medAluminum, logicalName, 0, 0, 0);
1478 const double phi = 360.0 / number;
1480 G4RotationMatrix rot = G4RotationMatrix();
1483 G4ThreeVector arm(cX * CLHEP::cm, cY * CLHEP::cm, cZ * CLHEP::cm - rin * CLHEP::cm - thick * CLHEP::cm);
1489 rot.rotateZ(0.5 * phi * CLHEP::deg);
1490 arm.rotateZ(0.5 * phi * CLHEP::deg);
1492 for (
int i = 0; i < number; ++i) {
1493 const string physicalName =
"physicalRib5_" + to_string(rib5ID) +
" " + to_string(i);
1494 new G4PVPlacement(G4Transform3D(rot, arm), logicalV,
1496 rot.rotateZ(phi * CLHEP::deg);
1497 arm.rotateZ(phi * CLHEP::deg);