Class Reference for E1039 Core & Analysis Software
PHG4TargetCoilV2Detector.cc
Go to the documentation of this file.
2 
3 #include <phparameter/PHParameters.h>
4 
5 #include <g4main/PHG4Utils.h>
6 
8 #include <phool/PHIODataNode.h>
9 #include <phool/getClass.h>
10 
11 #include <Geant4/G4Colour.hh>
12 #include <Geant4/G4LogicalVolume.hh>
13 #include <Geant4/G4Material.hh>
14 #include <Geant4/G4PVPlacement.hh>
15 #include <Geant4/G4PhysicalConstants.hh>
16 #include <Geant4/G4SystemOfUnits.hh>
17 #include <Geant4/G4Polycone.hh>
18 #include <Geant4/G4Tubs.hh>
19 #include <Geant4/G4SubtractionSolid.hh>
20 #include <Geant4/G4UserLimits.hh>
21 #include <Geant4/G4VisAttributes.hh>
22 
23 #include <cmath>
24 #include <sstream>
25 #include <algorithm>
26 
27 using namespace std;
28 
29 //_______________________________________________________________
30 PHG4TargetCoilV2Detector::PHG4TargetCoilV2Detector(PHCompositeNode *Node, PHParameters *parameters, const std::string &dnam, const int lyr)
31  : PHG4Detector(Node, dnam)
32  , params(parameters)
33  , cylinder_physi(nullptr)
34  , layer(lyr)
35 {
36 }
37 
38 //_______________________________________________________________
39 bool PHG4TargetCoilV2Detector::IsInCylinder(const G4VPhysicalVolume *volume) const
40 {
41  if (volume == cylinder_physi || volume->GetMotherLogical() == cylinder_physi->GetLogicalVolume())
42  {
43  return true;
44  }
45  return false;
46 }
47 
48 namespace {
49 
50  bool PlaceHollowTube(
51  G4LogicalVolume *mother,
52  double place,
53  G4UserLimits *g4userlimits,
54  bool overlapcheck,
55  const std::string& name,
56  G4Material *Mouter,
57  double in,
58  double out,
59  double half,
60  double t,
61  double phi0 = 0,
62  double phi = twopi
63  )
64  {
65  G4VSolid *all_solid = new G4Tubs((name+"_all").c_str(),
66  in,
67  out,
68  half,
69  phi0,
70  phi);
71 
72  G4VSolid *inner_solid = new G4Tubs((name+"_inner").c_str(),
73  in+t,
74  out-t,
75  half-t,
76  phi0,
77  phi);
78 
79  G4VSolid * outer_solid = new G4SubtractionSolid((name+"_outer").c_str(),
80  all_solid,
81  inner_solid,
82  0,
83  G4ThreeVector(0,0,0)
84  );
85 
86 
87  G4VisAttributes *vis_outer = new G4VisAttributes();
88  PHG4Utils::SetColour(vis_outer, Mouter->GetName());
89  vis_outer->SetVisibility(true);
90  vis_outer->SetForceSolid(true);
91 
92  G4LogicalVolume *outer_logic = new G4LogicalVolume(outer_solid,
93  Mouter,
94  (name+"_outer").c_str(),
95  nullptr, nullptr, g4userlimits);
96 
97  outer_logic->SetVisAttributes(vis_outer);
98 
99  new G4PVPlacement(
100  0,
101  G4ThreeVector(0, 0, place),
102  outer_logic,
103  (name+"_outer").c_str(),
104  mother, 0, false, overlapcheck);
105 
106  return true;
107  }
108 
109  bool PlaceLayeredTube(
110  G4LogicalVolume *mother,
111  double place,
112  G4UserLimits *g4userlimits,
113  bool overlapcheck,
114  const std::string& name,
115  G4Material *Minner,
116  G4Material *Mouter,
117  double in,
118  double out,
119  double half,
120  double t,
121  double phi0 = 0,
122  double phi = twopi
123  )
124  {
125  G4VSolid *all_solid = new G4Tubs((name+"_all").c_str(),
126  in,
127  out,
128  half,
129  phi0,
130  phi);
131 
132  G4VSolid *inner_solid = new G4Tubs((name+"_inner").c_str(),
133  in+t,
134  out-t,
135  half-t,
136  phi0,
137  phi);
138 
139  G4VSolid * outer_solid = new G4SubtractionSolid((name+"_outer").c_str(),
140  all_solid,
141  inner_solid,
142  0,
143  G4ThreeVector(0,0,0)
144  );
145 
146 
147  G4VisAttributes *vis_inner = new G4VisAttributes();
148  PHG4Utils::SetColour(vis_inner, Minner->GetName());
149  vis_inner->SetVisibility(true);
150  vis_inner->SetForceSolid(true);
151 
152  G4LogicalVolume *inner_logic = new G4LogicalVolume(inner_solid,
153  Minner,
154  (name+"_inner").c_str(),
155  nullptr, nullptr, g4userlimits);
156 
157  inner_logic->SetVisAttributes(vis_inner);
158 
159  new G4PVPlacement(
160  0,
161  G4ThreeVector(0, 0, place),
162  inner_logic,
163  (name+"_inner").c_str(),
164  mother, 0, false, overlapcheck);
165 
166 
167  G4VisAttributes *vis_outer = new G4VisAttributes();
168  PHG4Utils::SetColour(vis_outer, Mouter->GetName());
169  vis_outer->SetVisibility(true);
170  vis_outer->SetForceSolid(true);
171 
172  G4LogicalVolume *outer_logic = new G4LogicalVolume(outer_solid,
173  Mouter,
174  (name+"_outer").c_str(),
175  nullptr, nullptr, g4userlimits);
176 
177  outer_logic->SetVisAttributes(vis_outer);
178 
179  new G4PVPlacement(
180  0,
181  G4ThreeVector(0, 0, place),
182  outer_logic,
183  (name+"_outer").c_str(),
184  mother, 0, false, overlapcheck);
185 
186  return true;
187  }
188 
189  bool PlaceSSMainMag(
190  G4LogicalVolume *mother,
191  G4UserLimits *g4userlimits,
192  bool overlapcheck,
193  const std::string& name,
194  G4Material *mat,
195  double z_offset,
196  double in = 6.0*cm,
197  double out = 22.68*cm,
198  double z1 = 2.15*cm,
199  double z2 = 9.73*cm
200  ) {
201 
202  int n_base = 2;
203  double z_base[] = {z1, z2};
204  double ri_base[] = {in, in};
205  double ro_base[] = {out, out};
206 
207  G4VSolid *base_solid = new G4Polycone((name+"_base").c_str(),
208  0,
209  twopi,
210  n_base,
211  z_base,
212  ri_base,
213  ro_base);
214 
215  int n_sub1 = 6;
216  double z_sub1[] = {3.20*cm, 4.30*cm, 4.30*cm, 7.61*cm, 7.61*cm, z2+0.01*cm};
217  double ri_sub1[] = {9.70*cm, 9.70*cm, 6.97*cm, 6.97*cm, in, in};
218  double ro_sub1[] = {11.16*cm,11.16*cm,11.16*cm,11.16*cm,11.16*cm,11.16*cm};
219 
220  G4VSolid *sub1 = new G4Polycone((name+"_sub1").c_str(),
221  0,
222  twopi,
223  n_sub1,
224  z_sub1,
225  ri_sub1,
226  ro_sub1);
227 
228  int n_sub2 = 2;
229  double z_sub2[] = {2.76*cm, 8.57*cm};
230  double ri_sub2[] = {12.47*cm, 12.47*cm};
231  double ro_sub2[] = {17.88*cm, 17.88*cm};
232 
233  G4VSolid *sub2 = new G4Polycone((name+"_sub2").c_str(),
234  0,
235  twopi,
236  n_sub2,
237  z_sub2,
238  ri_sub2,
239  ro_sub2);
240 
241  G4VSolid * base_sub1 = new G4SubtractionSolid((name+"base_sub1").c_str(),
242  base_solid,
243  sub1,
244  0,
245  G4ThreeVector(0,0,0)
246  );
247 
248  G4VSolid * ss_solid = new G4SubtractionSolid((name+"base_sub1_sub2").c_str(),
249  base_sub1,
250  sub2,
251  0,
252  G4ThreeVector(0,0,0)
253  );
254 
255  G4VisAttributes *ss_vis = new G4VisAttributes();
256  PHG4Utils::SetColour(ss_vis, mat->GetName());
257  ss_vis->SetVisibility(true);
258  ss_vis->SetForceSolid(true);
259 
260  G4LogicalVolume *ss_logic = new G4LogicalVolume(ss_solid,
261  mat,
262  (name+"_logic").c_str(),
263  nullptr, nullptr, g4userlimits);
264 
265  ss_logic->SetVisAttributes(ss_vis);
266 
267  new G4PVPlacement(
268  0,
269  G4ThreeVector(0, 0, z_offset),
270  ss_logic,
271  (name+"_phys").c_str(),
272  mother, 0, false, overlapcheck);
273  }
274 
275 }
276 
277 //_______________________________________________________________
278 void PHG4TargetCoilV2Detector::Construct(G4LogicalVolume *logicWorld)
279 {
280  G4double z;
281  G4double a;
282  G4String symbol;
283  G4String name;
284  G4double density;
285  G4int ncomponents;
286  G4int natoms;
287 
288  G4Element *elHe = new G4Element(name="Helium", symbol="He" , z=2., a = 4.003*g/mole);
289  G4Material* lHe = new G4Material(name = "G4_lHe", density = 0.145 * g/cm3, ncomponents = 1);
290  lHe->AddElement(elHe, natoms = 1);
291 
292  // 1/(0.45/8.57+0.45/4.506+0.1/8.96) = 6.11 g/cm3
293  G4Material* Coil = new G4Material(name = "Coil", density = 6.11*g/cm3, ncomponents = 3);
294  Coil->AddMaterial(G4Material::GetMaterial("G4_Nb"), 45 * perCent);
295  Coil->AddMaterial(G4Material::GetMaterial("G4_Ti"), 45 * perCent);
296  Coil->AddMaterial(G4Material::GetMaterial("G4_Cu"), 10 * perCent);
297  std::cout<< "PHG4TargetCoilV2Detector::Construct: " << Coil << std::endl;
298 
299  // 1/(0.6/7.87+0.2/7.18+0.15/8.902+0.05/10.22) = 7.95 g/cm3
300  G4Material* SS316L = new G4Material(name = "SS316L", density = 7.95*g/cm3, ncomponents = 4);
301  SS316L->AddMaterial(G4Material::GetMaterial("G4_Fe"), 60 * perCent);
302  SS316L->AddMaterial(G4Material::GetMaterial("G4_Cr"), 20 * perCent);
303  SS316L->AddMaterial(G4Material::GetMaterial("G4_Ni"), 15 * perCent);
304  SS316L->AddMaterial(G4Material::GetMaterial("G4_Mo"), 5 * perCent);
305  std::cout<< "PHG4TargetCoilV2Detector::Construct: " << SS316L << std::endl;
306 
307  G4VisAttributes *cylinder_vis = new G4VisAttributes();
308  PHG4Utils::SetColour(cylinder_vis, "G4_He");
309  cylinder_vis->SetVisibility(true);
310  cylinder_vis->SetForceSolid(true);
311 
312  double l = 22.7 * cm; // length of cylinder
313  double gap = 4 *cm; // gap between cylinders
314  double ri = 6.0 * cm; // inner r of cylinder
315  double ro = 22.225 * cm; // inner r of cylinder
316  double ts = 0.3 * cm; // shell thickness
317 
318  //double cl = 1e-2 * cm;
319 
320  G4VSolid *cylinder_solid = new G4Tubs(G4String(GetName().c_str()),
321  ri,
322  ro,
323  l/2,
324  0*deg,
325  twopi);
326 
327  double steplimits = params->get_double_param("steplimits") * cm;
328  G4UserLimits *g4userlimits = nullptr;
329  if (isfinite(steplimits))
330  {
331  g4userlimits = new G4UserLimits(steplimits);
332  }
333 
334  G4LogicalVolume *cylinder_logic = new G4LogicalVolume(cylinder_solid,
335  lHe,
336  G4String(GetName().c_str()),
337  nullptr, nullptr, g4userlimits);
338 
339  cylinder_logic->SetVisAttributes(cylinder_vis);
340 
341  G4RotationMatrix *rotm = new G4RotationMatrix();
342  rotm->rotateX(params->get_double_param("rot_x")*deg);
343  rotm->rotateY(params->get_double_param("rot_y")*deg);
344  rotm->rotateZ(params->get_double_param("rot_z")*deg);
345  params->Print();
346  rotm->print(std::cout);
347  cylinder_physi = new G4PVPlacement(
348  rotm,
349  G4ThreeVector(
350  params->get_double_param("place_x") * cm,
351  params->get_double_param("place_y") * cm,
352  params->get_double_param("place_z") * cm),
353  cylinder_logic,
354  G4String(GetName().c_str()),
355  logicWorld, 0, false, overlapcheck);
356 
357  PlaceHollowTube(
358  cylinder_logic,
359  0,
360  g4userlimits,
361  overlapcheck,
362  "Shell",
363  SS316L, //G4Material::GetMaterial("G4_Cu"),//SS316L,
364  ri,
365  ro,
366  l/2,
367  ts,
368  0,
369  twopi
370  );
371 
372  // SS structure
373  PlaceSSMainMag(
374  cylinder_logic,
375  0,
376  overlapcheck,
377  "SSMainMag",
378  SS316L, //G4Material::GetMaterial("G4_Si"),//SS316L,
379  -(l+gap)/2,
380  ri + 1.01*ts,
381  ro - 1.01*ts,
382  2.15*cm + 1.01*ts,
383  9.73*cm - 1.01*ts
384  );
385 
386  double c1_l = 4.5 * cm;
387  double c1_ri = 12.5 *cm;
388  double c1_ro = 17.2 *cm;
389  double c1_t = 0.01*cm; //0.5 *cm;
390 
391  PlaceLayeredTube(
392  cylinder_logic,
393  -(l+gap)/2 + 5.08*cm,//-7.5 *cm,
394  g4userlimits,
395  overlapcheck,
396  "C1",
397  Coil, //G4Material::GetMaterial("G4_Fe"),//Coil,
398  SS316L, //G4Material::GetMaterial("G4_Cu"),//SS316L,
399  c1_ri-c1_t,
400  c1_ro+c1_t,
401  c1_l/2+c1_t,
402  c1_t,
403  0,
404  twopi
405  );
406 
407 
408  double c2_l = 5.7 * cm;
409  double c2_ri = 7.6 *cm;
410  double c2_ro = 9.4 *cm;
411  double c2_t = 0.5 *cm;
412 
413  PlaceLayeredTube(
414  cylinder_logic,
415  -2.8 *cm,
416  g4userlimits,
417  overlapcheck,
418  "C2",
419  Coil,
420  SS316L,
421  c2_ri-c2_t,
422  c2_ro+c2_t,
423  c2_l/2+c2_t,
424  c2_t,
425  0,
426  twopi
427  );
428 
429 
430  double c3_l = 1 * cm;
431  double c3_ri = 12.7 *cm;
432  double c3_ro = 13.7 *cm;
433  double c3_t = 0.5 *cm;
434 
435  PlaceLayeredTube(
436  cylinder_logic,
437  0.9 *cm,
438  g4userlimits,
439  overlapcheck,
440  "C3",
441  Coil,
442  SS316L,
443  c3_ri-c3_t,
444  c3_ro+c3_t,
445  c3_l/2+c3_t,
446  c3_t,
447  0,
448  twopi
449  );
450 }
451 
452 int PHG4TargetCoilV2Detector::get_elem_id(const std::string& name) const {
453  int ret = 0;
454  if(name.find("C1_inner") != std::string::npos) {
455  ret = 10;
456  } else if (name.find("C1_outer") != std::string::npos) {
457  ret = 11;
458  } else if (name.find("C2_inner") != std::string::npos) {
459  ret = 20;
460  } else if (name.find("C2_outer") != std::string::npos) {
461  ret = 21;
462  } else if (name.find("C3_inner") != std::string::npos) {
463  ret = 30;
464  } else if (name.find("C3_outer") != std::string::npos) {
465  ret = 31;
466  } else if (name.find("Shell") != std::string::npos) {
467  ret = 99;
468  }
469  return ret;
470 }
471 
472 
473 
474 
475 
476 
477 
478 
479 
480 
481 
base class for phenix detector creation
Definition: PHG4Detector.h:14
virtual std::string GetName() const
Definition: PHG4Detector.h:51
bool overlapcheck
Definition: PHG4Detector.h:62
std::string name
Definition: PHG4Detector.h:61
PHG4TargetCoilV2Detector(PHCompositeNode *Node, PHParameters *parameters, const std::string &dnam, const int layer=0)
constructor
int get_elem_id(const std::string &name) const
bool IsInCylinder(const G4VPhysicalVolume *) const
void Construct(G4LogicalVolume *world)
construct
static void SetColour(G4VisAttributes *att, const std::string &mat)
Definition: PHG4Utils.cc:75
double get_double_param(const std::string &name) const
void Print() const
Definition: half.h:103