3 #include <phparameter/PHParameters.h>
11 #include <TSQLStatement.h>
13 #include <Geant4/G4Box.hh>
14 #include <Geant4/G4Tubs.hh>
15 #include <Geant4/G4SubtractionSolid.hh>
16 #include <Geant4/G4Colour.hh>
17 #include <Geant4/G4LogicalVolume.hh>
18 #include <Geant4/G4Material.hh>
19 #include <Geant4/G4PVPlacement.hh>
20 #include <Geant4/G4SystemOfUnits.hh>
21 #include <Geant4/G4UserLimits.hh>
22 #include <Geant4/G4VisAttributes.hh>
23 #include <Geant4/G4NistManager.hh>
40 if(volume == block_physi)
51 std::cout <<
"SQDipoleMagnet - begin construction of " << vName <<
" from " << dbfile << std::endl;
54 std::unique_ptr<TSQLStatement> stmt;
62 std::map<int, G4VSolid*> solids;
65 stmt.reset(db_svc->
Process(
"SELECT sID, sName, xLength, yLength, zLength FROM SolidBoxes WHERE sName like '" + vName +
"%'"));
66 while(stmt->NextResultRow() && (!stmt->IsNull(0)))
68 int sID = stmt->GetInt(0);
69 if(solids.find(sID) != solids.end())
71 std::cout <<
"ERROR - SQDipoleMagnet Construction: duplicate solid ID " << sID << std::endl;
75 G4String sName = stmt->GetString(1);
76 double xLength = stmt->GetDouble(2)*cm;
77 double yLength = stmt->GetDouble(3)*cm;
78 double zLength = stmt->GetDouble(4)*cm;
81 std::cout <<
"SQDipoleMagnet Construct: create solid box " << sID <<
" " << sName <<
" " << xLength <<
" " << yLength <<
" " << zLength << std::endl;
83 solids[sID] =
new G4Box(sName, xLength/2., yLength/2., zLength/2.);
87 stmt.reset(db_svc->
Process(
"SELECT sID, sName, length, radiusMin, radiusMax FROM SolidTubes WHERE sName like '" + vName +
"%'"));
88 while(stmt->NextResultRow() && (!stmt->IsNull(0)))
90 int sID = stmt->GetInt(0);
91 if(solids.find(sID) != solids.end())
93 std::cout <<
"ERROR - SQDipoleMagnet Construction: duplicate solid ID " << sID << std::endl;
97 G4String sName = stmt->GetString(1);
98 double length = stmt->GetDouble(2)*cm;
99 double radiusMin = stmt->GetDouble(3)*cm;
100 double radiusMax = stmt->GetDouble(4)*cm;
103 std::cout <<
"SQDipoleMagnet Construct: create solid tube " << sID <<
" " << sName <<
" " << length <<
" " << radiusMax <<
" " << radiusMin << std::endl;
105 solids[sID] =
new G4Tubs(sName, radiusMin, radiusMax, length/2., 0., 360.*deg);
109 stmt.reset(db_svc->
Process(
"SELECT sID, sName, shellID, holeID, rotX, rotY, rotZ, posX, posY, posZ FROM SubtractionSolids WHERE sName like '" + vName +
"%'"));
110 while(stmt->NextResultRow() && (!stmt->IsNull(0)))
112 int sID = stmt->GetInt(0);
113 if(solids.find(sID) != solids.end())
115 std::cout <<
"ERROR - SQDipoleMagnet Construction: duplicate solid ID " << sID << std::endl;
119 G4String sName = stmt->GetString(1);
120 int shellID = stmt->GetInt(2);
121 int holeID = stmt->GetInt(3);
124 std::cout <<
"SQDipoleMagnet Construct: create solid subtraction " << sID <<
" " << sName <<
" " << shellID <<
" " << holeID << std::endl;
126 if(solids.find(holeID) == solids.end() || solids.find(shellID) == solids.end())
128 std::cout <<
"ERROR - SQDipoleMagnet Construction: cannot find solid component for " << sName << std::endl;
132 G4RotationMatrix rot;
133 rot.rotateX(stmt->GetDouble(4)*rad);
134 rot.rotateY(stmt->GetDouble(5)*rad);
135 rot.rotateZ(stmt->GetDouble(6)*rad);
138 pos.setX(stmt->GetDouble(7)*cm);
139 pos.setY(stmt->GetDouble(8)*cm);
140 pos.setZ(stmt->GetDouble(9)*cm);
142 G4Transform3D trans(rot, pos);
143 solids[sID] =
new G4SubtractionSolid(sName, solids[shellID], solids[holeID], trans);
147 std::map<int, G4LogicalVolume*> logicals;
148 stmt.reset(db_svc->
Process(
"SELECT lvID, lvName, sID, mName FROM LogicalVolumes WHERE lvName like '" + vName +
"%'"));
149 while(stmt->NextResultRow() && (!stmt->IsNull(0)))
151 int lvID = stmt->GetInt(0);
152 if(logicals.find(lvID) != logicals.end())
154 std::cout <<
"ERROR - SQDipoleMagnet Construction: duplicate logical volume ID " << lvID << std::endl;
158 G4String lvName = stmt->GetString(1);
159 int sID = stmt->GetInt(2);
160 G4String mName = stmt->GetString(3);
163 std::cout <<
"SQDipoleMagnet Construct: create logical volume " << lvID <<
" " << lvName <<
" " << sID <<
" " << mName << std::endl;
165 logicals[lvID] =
new G4LogicalVolume(solids[sID], G4Material::GetMaterial(mName), lvName);
169 std::map<int, int> lvRefCount;
170 for(
auto it = logicals.begin(); it != logicals.end(); ++it)
172 lvRefCount[it->first] = 0;
179 std::string topPVName;
180 stmt.reset(db_svc->
Process(
"SELECT pvID, pvName, lvID FROM PhysicalVolumes WHERE motherID=0 AND depth=1 AND pvName like '" + vName +
"%'"));
181 while(stmt->NextResultRow())
185 topPVID = stmt->GetInt(0);
186 topPVName = stmt->GetString(1);
187 topLVID = stmt->GetInt(2);
192 std::cout <<
"ERROR - SQDipoleMagnet Construction: top volume is ambiguous " << nTopPV << std::endl;
196 if(logicals.find(topLVID) == logicals.end())
198 std::cout <<
"ERROR - SQDipoleMagnet Construction: top logical volume is missing " << std::endl;
204 std::cout <<
"SQDipoleMagnet Construct: create top physical volume " << topPVID <<
" " << topLVID <<
" " << topPVName << std::endl;
208 G4RotationMatrix topRot;
213 G4ThreeVector topLoc;
218 block_physi =
new G4PVPlacement(G4Transform3D(topRot, topLoc), logicals[topLVID], topPVName.c_str(), logicWorld,
false, lvRefCount[topLVID],
overlapcheck);
219 ++lvRefCount[topLVID];
222 stmt.reset(db_svc->
Process(
"SELECT pvName, lvID, motherID, xRel, yRel, zRel, rotX, rotY, rotZ FROM PhysicalVolumes WHERE motherID>0 AND pvName like '" + vName +
"%'"));
223 while(stmt->NextResultRow() && (!stmt->IsNull(0)))
226 G4String pvName = stmt->GetString(0);
227 int lvID = stmt->GetInt(1);
228 int motherID = stmt->GetInt(2);
231 pos.setX(stmt->GetDouble(3)*cm);
232 pos.setY(stmt->GetDouble(4)*cm);
233 pos.setZ(stmt->GetDouble(5)*cm);
235 G4RotationMatrix rot;
236 rot.rotateX(stmt->GetDouble(6)*rad);
237 rot.rotateY(stmt->GetDouble(7)*rad);
238 rot.rotateZ(stmt->GetDouble(8)*rad);
242 std::cout <<
"SQDipoleMagnet Construct: create physical volume " << pvName <<
" " << lvID <<
" " << motherID << std::endl;
245 new G4PVPlacement(G4Transform3D(rot, pos), logicals[lvID], pvName.c_str(), logicals[motherID],
false, lvRefCount[lvID],
overlapcheck);
Standard interface with SQL database.
TSQLStatement * Process(const char *query)
base class for phenix detector creation
virtual int Verbosity() const
double get_double_param(const std::string &name) const
std::string get_string_param(const std::string &name) const
bool IsInBlock(G4VPhysicalVolume *) const
virtual void Construct(G4LogicalVolume *world)
construct
virtual ~SQG4DipoleMagnetDetector(void)
destructor
SQG4DipoleMagnetDetector(PHCompositeNode *node, PHParameters *parameters, const std::string &dnam="BLOCK", const int lyr=0)
constructor