Class Reference for E1039 Core & Analysis Software
PHG4BNLTargetCoilDetector.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 PHG4BNLTargetCoilDetector::PHG4BNLTargetCoilDetector(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 PHG4BNLTargetCoilDetector::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 bool PlaceHollowPolycone(
50  G4LogicalVolume *mother,
51  double place,
52  G4UserLimits *g4userlimits,
53  bool overlapcheck,
54  const std::string& name,
55  G4Material *Mouter,
56  int o_n_z,
57  std::vector<double> o_z_plane,
58  std::vector<double> o_r_inner,
59  std::vector<double> o_r_outer,
60  int i_n_z,
61  std::vector<double> i_z_plane,
62  std::vector<double> i_r_inner,
63  std::vector<double> i_r_outer,
64  double phi0 = 0,
65  double phi = twopi
66  )
67 {
68  G4VSolid *all_solid = new G4Polycone((name+"_all").c_str(),
69  phi0,
70  phi,
71  o_n_z,
72  o_z_plane.data(),
73  o_r_inner.data(),
74  o_r_outer.data());
75 
76  G4VSolid *inner_solid = new G4Polycone((name+"_inner").c_str(),
77  phi0,
78  phi,
79  i_n_z,
80  i_z_plane.data(),
81  i_r_inner.data(),
82  i_r_outer.data());
83 
84  G4VSolid * outer_solid = new G4SubtractionSolid((name+"_outer").c_str(),
85  all_solid,
86  inner_solid,
87  0,
88  G4ThreeVector(0,0,0)
89  );
90 
91 
92  G4VisAttributes *vis_outer = new G4VisAttributes();
93  PHG4Utils::SetColour(vis_outer, Mouter->GetName());
94  vis_outer->SetVisibility(true);
95  vis_outer->SetForceSolid(true);
96 
97  G4LogicalVolume *outer_logic = new G4LogicalVolume(outer_solid,
98  Mouter,
99  (name+"_outer").c_str(),
100  nullptr, nullptr, g4userlimits);
101 
102  outer_logic->SetVisAttributes(vis_outer);
103 
104  new G4PVPlacement(
105  0,
106  G4ThreeVector(0, 0, place),
107  outer_logic,
108  (name+"_outer").c_str(),
109  mother, 0, false, overlapcheck);
110 
111  return true;
112 }
113 
114 //bool PlaceHollowTube(
115 // G4LogicalVolume *mother,
116 // double place,
117 // G4UserLimits *g4userlimits,
118 // bool overlapcheck,
119 // const std::string& name,
120 // G4Material *Mouter,
121 // double in,
122 // double out,
123 // double half,
124 // double t,
125 // double phi0 = 0,
126 // double phi = twopi
127 // )
128 //{
129 // G4VSolid *all_solid = new G4Tubs((name+"_all").c_str(),
130 // in,
131 // out,
132 // half,
133 // phi0,
134 // phi);
135 //
136 // G4VSolid *inner_solid = new G4Tubs((name+"_inner").c_str(),
137 // in+t,
138 // out-t,
139 // half-t,
140 // phi0,
141 // phi);
142 //
143 // G4VSolid * outer_solid = new G4SubtractionSolid((name+"_outer").c_str(),
144 // all_solid,
145 // inner_solid,
146 // 0,
147 // G4ThreeVector(0,0,0)
148 // );
149 //
150 //
151 // G4VisAttributes *vis_outer = new G4VisAttributes();
152 // PHG4Utils::SetColour(vis_outer, Mouter->GetName());
153 // vis_outer->SetVisibility(true);
154 // vis_outer->SetForceSolid(true);
155 //
156 // G4LogicalVolume *outer_logic = new G4LogicalVolume(outer_solid,
157 // Mouter,
158 // (name+"_outer").c_str(),
159 // nullptr, nullptr, g4userlimits);
160 //
161 // outer_logic->SetVisAttributes(vis_outer);
162 //
163 // new G4PVPlacement(
164 // 0,
165 // G4ThreeVector(0, 0, place),
166 // outer_logic,
167 // (name+"_outer").c_str(),
168 // mother, 0, false, overlapcheck);
169 //
170 // return true;
171 //}
172 
173 bool PlaceLayeredTube(
174  G4LogicalVolume *mother,
175  double place,
176  G4UserLimits *g4userlimits,
177  bool overlapcheck,
178  const std::string& name,
179  G4Material *Minner,
180  G4Material *Mouter,
181  double in,
182  double out,
183  double half,
184  double t,
185  double phi0 = 0,
186  double phi = twopi
187  )
188 {
189  G4VSolid *all_solid = new G4Tubs((name+"_all").c_str(),
190  in,
191  out,
192  half,
193  phi0,
194  phi);
195 
196  G4VSolid *inner_solid = new G4Tubs((name+"_inner").c_str(),
197  in+t,
198  out-t,
199  half-t,
200  phi0,
201  phi);
202 
203  G4VSolid * outer_solid = new G4SubtractionSolid((name+"_outer").c_str(),
204  all_solid,
205  inner_solid,
206  0,
207  G4ThreeVector(0,0,0)
208  );
209 
210 
211  G4VisAttributes *vis_inner = new G4VisAttributes();
212  PHG4Utils::SetColour(vis_inner, Minner->GetName());
213  vis_inner->SetVisibility(true);
214  vis_inner->SetForceSolid(true);
215 
216  G4LogicalVolume *inner_logic = new G4LogicalVolume(inner_solid,
217  Minner,
218  (name+"_inner").c_str(),
219  nullptr, nullptr, g4userlimits);
220 
221  inner_logic->SetVisAttributes(vis_inner);
222 
223  new G4PVPlacement(
224  0,
225  G4ThreeVector(0, 0, place),
226  inner_logic,
227  (name+"_inner").c_str(),
228  mother, 0, false, overlapcheck);
229 
230 
231  G4VisAttributes *vis_outer = new G4VisAttributes();
232  PHG4Utils::SetColour(vis_outer, Mouter->GetName());
233  vis_outer->SetVisibility(true);
234  vis_outer->SetForceSolid(true);
235 
236  G4LogicalVolume *outer_logic = new G4LogicalVolume(outer_solid,
237  Mouter,
238  (name+"_outer").c_str(),
239  nullptr, nullptr, g4userlimits);
240 
241  outer_logic->SetVisAttributes(vis_outer);
242 
243  new G4PVPlacement(
244  0,
245  G4ThreeVector(0, 0, place),
246  outer_logic,
247  (name+"_outer").c_str(),
248  mother, 0, false, overlapcheck);
249 
250  return true;
251 }
252 }
253 //_______________________________________________________________
254 void PHG4BNLTargetCoilDetector::Construct(G4LogicalVolume *logicWorld)
255 {
256  G4double z;
257  G4double a;
258  G4String symbol;
259  G4String name;
260  G4double density;
261  G4int ncomponents;
262  G4int natoms;
263 
264  G4Element *elHe = new G4Element(name="Helium", symbol="He" , z=2., a = 4.003*g/mole);
265  G4Material* lHe = new G4Material(name = "G4_lHe", density = 0.145 * g/cm3, ncomponents = 1);
266  lHe->AddElement(elHe, natoms = 1);
267 
268  // 1/(0.45/8.57+0.45/4.506+0.1/8.96) = 6.11 g/cm3
269  G4Material* Coil = new G4Material(name = "Coil", density = 6.11*g/cm3, ncomponents = 3);
270  Coil->AddMaterial(G4Material::GetMaterial("G4_Nb"), 45 * perCent);
271  Coil->AddMaterial(G4Material::GetMaterial("G4_Ti"), 45 * perCent);
272  Coil->AddMaterial(G4Material::GetMaterial("G4_Cu"), 10 * perCent);
273  std::cout<< "PHG4BNLTargetCoilDetector::Construct: " << Coil << std::endl;
274 
275  // 1/(0.6/7.87+0.2/7.18+0.15/8.902+0.05/10.22) = 7.95 g/cm3
276  G4Material* SS316L = new G4Material(name = "SS316L", density = 7.95*g/cm3, ncomponents = 4);
277  SS316L->AddMaterial(G4Material::GetMaterial("G4_Fe"), 60 * perCent);
278  SS316L->AddMaterial(G4Material::GetMaterial("G4_Cr"), 20 * perCent);
279  SS316L->AddMaterial(G4Material::GetMaterial("G4_Ni"), 15 * perCent);
280  SS316L->AddMaterial(G4Material::GetMaterial("G4_Mo"), 5 * perCent);
281  std::cout<< "PHG4BNLTargetCoilDetector::Construct: " << SS316L << std::endl;
282 
283 
284  G4VisAttributes *cylinder_vis = new G4VisAttributes();
285  PHG4Utils::SetColour(cylinder_vis, "G4_He");
286  cylinder_vis->SetVisibility(true);
287  cylinder_vis->SetForceSolid(true);
288 
289  // determine length of cylinder using PHENIX's rapidity coverage if flag is true
290 
291  double g1 = 2.3*cm; //gap inner
292  double g2 = 3.1*cm; //gap outer
293  double zend = 13.8*cm; //z end
294  double ri = 4.0*cm;
295  double rm = 7.9*cm;
296  double ro = 16.5*cm;
297  double ts = 0.3 * cm; // shell thickness
298 
299  int o_n_z = 3;
300  std::vector<double> o_z_plane = {g1, g2, zend};
301  std::vector<double> o_r_inner = {ri, ri, ri};
302  std::vector<double> o_r_outer = {rm, ro, ro};
303 
304  int i_n_z = 3;
305  std::vector<double> i_z_plane = {g1+ts, g2, zend-ts};
306  std::vector<double> i_r_inner = {ri+ts, ri+ts, ri+ts};
307  std::vector<double> i_r_outer = {rm, ro-ts, ro-ts};
308 
309 
310  G4VSolid *cylinder_solid = new G4Polycone(G4String(GetName().c_str()),
311  0*degree,
312  twopi,
313  o_n_z,
314  o_z_plane.data(),
315  o_r_inner.data(),
316  o_r_outer.data());
317 
318  double steplimits = params->get_double_param("steplimits") * cm;
319  G4UserLimits *g4userlimits = nullptr;
320  if (isfinite(steplimits))
321  {
322  g4userlimits = new G4UserLimits(steplimits);
323  }
324 
325  G4LogicalVolume *cylinder_logic = new G4LogicalVolume(cylinder_solid,
326  lHe,
327  G4String(GetName().c_str()),
328  nullptr, nullptr, g4userlimits);
329 
330  cylinder_logic->SetVisAttributes(cylinder_vis);
331 
332  G4RotationMatrix *rotm = new G4RotationMatrix();
333  rotm->rotateX(params->get_double_param("rot_x")*deg);
334  rotm->rotateY(params->get_double_param("rot_y")*deg);
335  rotm->rotateZ(params->get_double_param("rot_z")*deg);
336  params->Print();
337  rotm->print(std::cout);
338  cylinder_physi = new G4PVPlacement(
339  rotm,
340  G4ThreeVector(
341  params->get_double_param("place_x") * cm,
342  params->get_double_param("place_y") * cm,
343  params->get_double_param("place_z") * cm),
344  cylinder_logic,
345  G4String(GetName().c_str()),
346  logicWorld, 0, false, overlapcheck);
347 
348  PlaceHollowPolycone(
349  cylinder_logic,
350  0,
351  g4userlimits,
352  overlapcheck,
353  "Shell",
354  SS316L,
355  o_n_z,
356  o_z_plane,
357  o_r_inner,
358  o_r_outer,
359  i_n_z,
360  i_z_plane,
361  i_r_inner,
362  i_r_outer
363  );
364 
365  double center = 0*cm;
366  double coil_t = 0.5*cm;
367 
368  double coil_z1 = 4.0*cm;
369  double coil_z2 = 8.3*cm;
370  double coil_ri = 11.4*cm;
371  double coil_ro = 14.8*cm;
372 
373  PlaceLayeredTube(
374  cylinder_logic,
375  ((coil_z1+coil_z2)/2 - center),
376  g4userlimits,
377  overlapcheck,
378  "C1",
379  Coil,
380  SS316L,
381  coil_ri,
382  coil_ro,
383  (coil_z2-coil_z1)/2,
384  coil_t,
385  0,
386  twopi
387  );
388 
389  coil_z1 = 3.3*cm;
390  coil_z2 = 8.1*cm;
391  coil_ri = 6.9*cm;
392  coil_ro = 10.1*cm;
393 
394  PlaceLayeredTube(
395  cylinder_logic,
396  ((coil_z1+coil_z2)/2 - center),
397  g4userlimits,
398  overlapcheck,
399  "C2",
400  Coil,
401  SS316L,
402  coil_ri,
403  coil_ro,
404  (coil_z2-coil_z1)/2,
405  coil_t,
406  0,
407  twopi
408  );
409 
410  coil_z1 = 2.9*cm;
411  coil_z2 = 4.8*cm;
412  coil_ri = 4.5*cm;
413  coil_ro = 5.7*cm;
414 
415  PlaceLayeredTube(
416  cylinder_logic,
417  ((coil_z1+coil_z2)/2 - center),
418  g4userlimits,
419  overlapcheck,
420  "C3",
421  Coil,
422  SS316L,
423  coil_ri,
424  coil_ro,
425  (coil_z2-coil_z1)/2,
426  coil_t,
427  0,
428  twopi
429  );
430 }
431 
432 int PHG4BNLTargetCoilDetector::get_elem_id(const std::string& name) const {
433  int ret = 0;
434  if(name.find("C1_inner") != std::string::npos) {
435  ret = 10;
436  } else if (name.find("C1_outer") != std::string::npos) {
437  ret = 11;
438  } else if (name.find("C2_inner") != std::string::npos) {
439  ret = 20;
440  } else if (name.find("C2_outer") != std::string::npos) {
441  ret = 21;
442  } else if (name.find("C3_inner") != std::string::npos) {
443  ret = 30;
444  } else if (name.find("C3_outer") != std::string::npos) {
445  ret = 31;
446  } else if (name.find("Shell") != std::string::npos) {
447  ret = 99;
448  }
449  return ret;
450 }
451 
452 
453 
454 
455 
456 
457 
458 
void Construct(G4LogicalVolume *world)
construct
bool IsInCylinder(const G4VPhysicalVolume *) const
int get_elem_id(const std::string &name) const
PHG4BNLTargetCoilDetector(PHCompositeNode *Node, PHParameters *parameters, const std::string &dnam, const int layer=0)
constructor
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
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