Newer
Older
//==========================================================================
Markus Frank
committed
//--------------------------------------------------------------------------
// Copyright (C) Organisation europeenne pour la Recherche nucleaire (CERN)
// All rights reserved.
//
// For the licensing terms see $DD4hepINSTALL/LICENSE.
// For the list of contributors see $DD4hepINSTALL/doc/CREDITS.
//
// \author Markus Frank
// \date 2015-11-09
//
//==========================================================================
// Framework include files
#include <DDG4/Geant4DetectorConstruction.h>
Markus Frank
committed
/// Namespace for the AIDA detector description toolkit
Markus Frank
committed
/// Namespace for the Geant4 based simulation part of the AIDA detector description toolkit
Markus Frank
committed
/// Class to create Geant4 detector geometry from TGeo representation in memory
/**
* On demand (ie. when calling "Construct") the dd4hep geometry is converted
Markus Frank
committed
* to Geant4 with all volumes, assemblies, shapes, materials etc.
* The actuak work is performed by the Geant4Converter class called by this method.
*
* \author M.Frank
* \version 1.0
* \ingroup DD4HEP_SIMULATION
*/
class Geant4DetectorGeometryConstruction : public Geant4DetectorConstruction {
/// Property: Dump geometry hierarchy
Markus Frank
committed
bool m_dumpHierarchy = false;
/// Property: Flag to debug materials during conversion mechanism
Markus Frank
committed
bool m_debugMaterials = false;
/// Property: Flag to debug elements during conversion mechanism
Markus Frank
committed
bool m_debugElements = false;
/// Property: Flag to debug shapes during conversion mechanism
bool m_debugShapes = false;
/// Property: Flag to debug volumes during conversion mechanism
Markus Frank
committed
bool m_debugVolumes = false;
/// Property: Flag to debug placements during conversion mechanism
Markus Frank
committed
bool m_debugPlacements = false;
/// Property: Flag to debug regions during conversion mechanism
Markus Frank
committed
bool m_debugRegions = false;
Markus Frank
committed
/// Property: Flag to debug regions during conversion mechanism
bool m_debugSurfaces = false;
Markus Frank
committed
/// Property: Flag to dump all placements after the conversion procedure
bool m_printPlacements = false;
/// Property: Flag to dump all sensitives after the conversion procedure
bool m_printSensitives = false;
/// Property: Printout level of info object
int m_geoInfoPrintLevel;
Markus Frank
committed
/// Property: G4 GDML dump file name (default: empty. If non empty, dump)
std::string m_dumpGDML;
/// Write GDML file
int writeGDML(const char* gdml_output);
/// Print geant4 volume
int printVolumeObj(const char* vol_path, PlacedVolume pv);
/// Print geant4 volume tree
int printVolumeTree(const char* vol_path);
/// Print geant4 volume
int printVolume(const char* vol_path);
/// Check geant4 volume
int checkVolume(const char* vol_path);
/// Print geant4 material
int printMaterial(const char* mat_name);
Markus Frank
committed
public:
/// Initializing constructor for DDG4
Geant4DetectorGeometryConstruction(Geant4Context* ctxt, const std::string& nam);
/// Default destructor
virtual ~Geant4DetectorGeometryConstruction();
/// Geometry construction callback. Called at "Construct()"
void constructGeo(Geant4DetectorConstructionContext* ctxt) override;
/// Install command control messenger to write GDML file from command prompt.
virtual void installCommandMessenger() override;
Markus Frank
committed
};
} // End namespace sim
} // End namespace dd4hep
Markus Frank
committed
// Framework include files
#include <DD4hep/InstanceCount.h>
#include <DD4hep/DetectorTools.h>
#include <DD4hep/DD4hepUnits.h>
#include <DD4hep/Printout.h>
#include <DD4hep/Detector.h>
Markus Frank
committed
#include <DDG4/Geant4HierarchyDump.h>
#include <DDG4/Geant4UIMessenger.h>
#include <DDG4/Geant4Converter.h>
#include <DDG4/Geant4Kernel.h>
#include <DDG4/Factories.h>
Markus Frank
committed
#include <TGeoScaledShape.h>
Markus Frank
committed
// Geant4 include files
#include <G4LogicalVolume.hh>
#include <G4PVPlacement.hh>
#include <G4Material.hh>
#include <G4Version.hh>
#include <G4VSolid.hh>
#include "CLHEP/Units/SystemOfUnits.h"
#include <G4GDMLParser.hh>
Markus Frank
committed
#include <cmath>
Markus Frank
committed
using namespace std;
using namespace dd4hep;
using namespace dd4hep::sim;
Markus Frank
committed
DECLARE_GEANT4ACTION(Geant4DetectorGeometryConstruction)
/// Initializing constructor for other clients
Geant4DetectorGeometryConstruction::Geant4DetectorGeometryConstruction(Geant4Context* ctxt, const string& nam)
: Geant4DetectorConstruction(ctxt,nam)
Markus Frank
committed
{
declareProperty("DebugMaterials", m_debugMaterials);
declareProperty("DebugElements", m_debugElements);
declareProperty("DebugShapes", m_debugShapes);
declareProperty("DebugVolumes", m_debugVolumes);
declareProperty("DebugPlacements", m_debugPlacements);
declareProperty("DebugRegions", m_debugRegions);
Markus Frank
committed
declareProperty("DebugSurfaces", m_debugSurfaces);
declareProperty("PrintPlacements", m_printPlacements);
declareProperty("PrintSensitives", m_printSensitives);
declareProperty("GeoInfoPrintLevel", m_geoInfoPrintLevel = DEBUG);
declareProperty("DumpHierarchy", m_dumpHierarchy);
declareProperty("DumpGDML", m_dumpGDML="");
Markus Frank
committed
InstanceCount::increment(this);
}
/// Default destructor
Geant4DetectorGeometryConstruction::~Geant4DetectorGeometryConstruction() {
InstanceCount::decrement(this);
}
/// Geometry construction callback. Called at "Construct()"
void Geant4DetectorGeometryConstruction::constructGeo(Geant4DetectorConstructionContext* ctxt) {
Geant4Mapping& g4map = Geant4Mapping::instance();
DetElement world = ctxt->description.world();
Geant4Converter conv(ctxt->description, outputLevel());
conv.debugMaterials = m_debugMaterials;
conv.debugElements = m_debugElements;
Markus Frank
committed
conv.debugShapes = m_debugShapes;
conv.debugVolumes = m_debugVolumes;
conv.debugPlacements = m_debugPlacements;
conv.debugRegions = m_debugRegions;
Markus Frank
committed
conv.debugSurfaces = m_debugSurfaces;
ctxt->geometry = conv.create(world).detach();
ctxt->geometry->printLevel = outputLevel();
Markus Frank
committed
g4map.attach(ctxt->geometry);
G4VPhysicalVolume* w = ctxt->geometry->world();
Markus Frank
committed
// Save away the reference to the world volume
context()->kernel().setWorld(w);
// Create Geant4 volume manager only if not yet available
Markus Frank
committed
g4map.volumeManager();
if ( m_dumpHierarchy ) {
Markus Frank
committed
dmp.dump("",w);
}
ctxt->world = w;
if ( !m_dumpGDML.empty() ) writeGDML(m_dumpGDML.c_str());
else if ( ::getenv("DUMP_GDML") ) writeGDML(::getenv("DUMP_GDML"));
enableUI();
}
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
/// Print geant4 material
int Geant4DetectorGeometryConstruction::printMaterial(const char* mat_name) {
if ( mat_name ) {
auto& g4map = Geant4Mapping::instance().data();
for ( auto it = g4map.g4Materials.begin(); it != g4map.g4Materials.end(); ++it ) {
const auto* mat = (*it).second;
if ( mat->GetName() == mat_name ) {
const auto* ion = mat->GetIonisation();
printP2("+++ Dump of GEANT4 material: %s", mat_name);
cout << mat;
if ( ion ) {
cout << " MEE: ";
cout << setprecision(12);
cout << ion->GetMeanExcitationEnergy()/CLHEP::eV;
cout << " [eV]";
}
else
cout << " MEE: UNKNOWN";
cout << endl << endl;
return 1;
}
}
warning("+++ printMaterial: FAILED to find the material %s", mat_name);
}
warning("+++ printMaterial: Property materialName not set!");
return 0;
}
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
/// Print geant4 volume
int Geant4DetectorGeometryConstruction::printVolumeObj(const char* vol_path, PlacedVolume pv) {
if ( pv.isValid() ) {
const G4LogicalVolume* vol = 0;
auto& g4map = Geant4Mapping::instance().data();
auto pit = g4map.g4Placements.find(pv.ptr());
auto vit = g4map.g4Volumes.find(pv.volume());
if ( vit != g4map.g4Volumes.end() ) {
vol = (*vit).second;
auto* sol = vol->GetSolid();
const auto* mat = vol->GetMaterial();
const auto* ion = mat->GetIonisation();
Solid sh = pv.volume().solid();
printP2("++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++++");
printP2("+++ Dump of GEANT4 solid: %s", vol_path);
cout << mat;
if ( ion ) {
cout << " MEE: ";
cout << setprecision(12);
cout << ion->GetMeanExcitationEnergy()/CLHEP::eV;
cout << " [eV]";
}
else
cout << " MEE: UNKNOWN";
cout << endl << *sol;
printP2("+++ Dump of ROOT solid: %s", vol_path);
sh->InspectShape();
if ( sh->IsA() == TGeoScaledShape::Class() ) {
TGeoScaledShape* scaled = (TGeoScaledShape*)sh.ptr();
const Double_t* scale = scaled->GetScale()->GetScale();
double dot = scale[0]*scale[1]*scale[2];
printP2("+++ TGeoScaledShape: %8.3g %8.3g %8.3g [%s]", scale[0], scale[1], scale[2],
dot > 0e0 ? "RIGHT handed" : "LEFT handed");
}
else if ( pit != g4map.g4Placements.end() ) {
const G4VPhysicalVolume* pl = (*pit).second;
const G4RotationMatrix* rot = pl->GetRotation();
const G4ThreeVector& tr = pl->GetTranslation();
G4Transform3D transform(rot ? *rot : G4RotationMatrix(), tr);
HepGeom::Scale3D sc;
HepGeom::Rotate3D rr;
G4Translate3D tt;
transform.getDecomposition(sc,rr,tt);
double dot = sc(0,0)*sc(1,1)*sc(2,2);
printP2("+++ TGeoShape: %8.3g %8.3g %8.3g [%s]", sc(0,0), sc(1,1), sc(2,2),
dot > 0e0 ? "RIGHT handed" : "LEFT handed");
}
const TGeoMatrix* m = pv->GetMatrix();
printP2("+++ TGeoMatrix: %s",
m->TestBit(TGeoMatrix::kGeoReflection) ? "LEFT handed" : "RIGHT handed");
printP2("+++ Shape: %s cubic volume: %8.3g mm^3 area: %8.3g mm^2",
sol->GetName().c_str(), sol->GetCubicVolume(), sol->GetSurfaceArea());
return 1;
}
else {
auto ai = g4map.g4AssemblyVolumes.find(pv.ptr());
if ( ai != g4map.g4AssemblyVolumes.end() ) {
Volume v = pv.volume();
warning("+++ printVolume: volume %s is an assembly...need to resolve imprint",vol_path);
for(Int_t i=0; i < v->GetNdaughters(); ++i) {
TGeoNode* dau_nod = v->GetNode(i);
string p = vol_path + string("/") + dau_nod->GetName();
printVolumeObj(p.c_str(), dau_nod);
}
return 0;
}
}
warning("+++ printVolume: FAILED to find the volume %s in geant4 mapping...",vol_path);
return 0;
}
warning("+++ printVolume: FAILED to dump invalid volume",vol_path);
return 0;
}
/// Print geant4 volume
int Geant4DetectorGeometryConstruction::printVolume(const char* vol_path) {
if ( vol_path ) {
Detector& det = context()->kernel().detectorDescription();
PlacedVolume top = det.world().placement();
PlacedVolume pv = detail::tools::findNode(top, vol_path);
return printVolumeObj(vol_path, pv);
}
warning("+++ printVolume: Property VolumePath not set. [Ignored]");
return 0;
}
/// Print geant4 volume
int Geant4DetectorGeometryConstruction::printVolumeTree(const char* vol_path) {
if ( vol_path ) {
string p = vol_path;
Detector& det = context()->kernel().detectorDescription();
PlacedVolume top = det.world().placement();
PlacedVolume pv = detail::tools::findNode(top, vol_path);
if ( printVolumeObj(p.c_str(), pv) ) {
TGeoVolume* vol = pv->GetVolume();
for(Int_t i=0; i < vol->GetNdaughters(); ++i) {
PlacedVolume dau_pv(vol->GetNode(i));
//TGeoMatrix* tr = d->GetMatrix();
string path = (p + "/") + dau_pv.name();
warning("+++ printVolume: %s -> %s", dau_pv.name(), p.c_str());
if ( printVolumeTree(path.c_str()) ) {
}
}
}
}
warning("+++ printVolume: Property VolumePath not set. [Ignored]");
return 0;
}
/// Check geant4 volume
int Geant4DetectorGeometryConstruction::checkVolume(const char* vol_path) {
if ( vol_path ) {
Detector& det = context()->kernel().detectorDescription();
PlacedVolume top = det.world().placement();
PlacedVolume pv = detail::tools::findNode(top, vol_path);
if ( pv.isValid() ) {
auto& g4map = Geant4Mapping::instance().data();
auto it = g4map.g4Volumes.find(pv.volume());
if ( it != g4map.g4Volumes.end() ) {
const G4LogicalVolume* vol = (*it).second;
auto* g4_sol = vol->GetSolid();
Box rt_sol = pv.volume().solid();
printP2("Geant4 Shape: %s cubic volume: %8.3g mm^3 area: %8.3g mm^2",
g4_sol->GetName().c_str(), g4_sol->GetCubicVolume(), g4_sol->GetSurfaceArea());
#if G4VERSION_NUMBER>=1040
G4ThreeVector pMin, pMax;
double conv = (dd4hep::centimeter/CLHEP::centimeter)/2.0;
g4_sol->BoundingLimits(pMin,pMax);
printP2("Geant4 Bounding box extends: %8.3g %8.3g %8.3g",
(pMax.x()-pMin.x())*conv, (pMax.y()-pMin.y())*conv, (pMax.z()-pMin.z())*conv);
printP2("ROOT Bounding box dimensions: %8.3g %8.3g %8.3g",
rt_sol->GetDX(), rt_sol->GetDY(), rt_sol->GetDZ());
return 1;
}
}
warning("+++ checkVolume: FAILED to find the volume %s from the top volume",vol_path);
}
warning("+++ checkVolume: Property VolumePath not set. [Ignored]");
return 0;
}
/// Write GDML file
int Geant4DetectorGeometryConstruction::writeGDML(const char* output) {
G4VPhysicalVolume* w = context()->world();
if ( output && ::strlen(output) > 0 && output != m_dumpGDML.c_str() )
m_dumpGDML = output;
//#ifdef GEANT4_HAS_GDML
if ( !m_dumpGDML.empty() ) {
Markus Frank
committed
G4GDMLParser parser;
parser.Write(m_dumpGDML.c_str(), w);
info("+++ writeGDML: Wrote GDML file: %s", m_dumpGDML.c_str());
return 1;
}
else {
const char* gdml_dmp = ::getenv("DUMP_GDML");
if ( gdml_dmp ) {
G4GDMLParser parser;
Markus Frank
committed
parser.Write(gdml_dmp, w);
info("+++ writeGDML: Wrote GDML file: %s", gdml_dmp);
return 1;
Markus Frank
committed
}
warning("+++ writeGDML: Neither property DumpGDML nor environment DUMP_GDML set. No file written!");
return 0;
}
/// Install command control messenger to write GDML file from command prompt.
void Geant4DetectorGeometryConstruction::installCommandMessenger() {
this->Geant4DetectorConstruction::installCommandMessenger();
m_control->addCall("writeGDML", "GDML: write geometry to file: '"+m_dumpGDML+
"' [uses argument - or - property DumpGDML]",
Callback(this).make(&Geant4DetectorGeometryConstruction::writeGDML),1);
m_control->addCall("printVolume", "Print Geant4 volume properties [uses argument]",
Callback(this).make(&Geant4DetectorGeometryConstruction::printVolume),1);
m_control->addCall("printVolumeTree", "Print Geant4 volume tree with properties [uses argument]",
Callback(this).make(&Geant4DetectorGeometryConstruction::printVolumeTree),1);
m_control->addCall("checkVolume", "Check Geant4 volume properties [uses argument]",
Callback(this).make(&Geant4DetectorGeometryConstruction::checkVolume),1);
m_control->addCall("printMaterial", "Print Geant4 material properties [uses argument]",
Callback(this).make(&Geant4DetectorGeometryConstruction::printMaterial),1);
Markus Frank
committed
}