29 #include <TGeoShape.h>
32 #include <TGeoBoolNode.h>
33 #include <TGeoCompositeShape.h>
37 #include <TGeoMatrix.h>
38 #include <TGeoParaboloid.h>
42 #include <TGeoShapeAssembly.h>
43 #include <TGeoSphere.h>
44 #include <TGeoTorus.h>
48 #include <TGeoScaledShape.h>
65 XYZRotation getXYZangles(
const Double_t* r) {
66 Double_t cosb = std::sqrt(r[0]*r[0] + r[1]*r[1]);
68 return XYZRotation(atan2(r[5], r[8]), atan2(-r[2], cosb), atan2(r[1], r[0]));
70 return XYZRotation(atan2(-r[7], r[4]),atan2(-r[2], cosb),0);
74 XYZRotation getXYZangles(
const Double_t* rotationMatrix) {
77 const Double_t *r = rotationMatrix;
78 Double_t cosb = TMath::Sqrt(r[0] * r[0] + r[1] * r[1]);
80 a = TMath::ATan2(r[5], r[8]) * rad;
81 b = TMath::ATan2(-r[2], cosb) * rad;
82 c = TMath::ATan2(r[1], r[0]) * rad;
85 a = TMath::ATan2(-r[7], r[4]) * rad;
86 b = TMath::ATan2(-r[2], cosb) * rad;
89 XYZRotation rr(a, b, c);
90 std::cout <<
" X:" << a <<
" " << rr.X() <<
" Y:" << b <<
" " << rr.Y() <<
" Z:" << c <<
" " << rr.Z()
91 <<
" lx:" << r[0] <<
" ly:" << r[4] <<
" lz:" << r[8] << std::endl;
92 return XYZRotation(a, b, c);
96 bool is_volume(
const TGeoVolume* volume) {
101 return node.
data() != 0;
104 std::string genName(
const std::string& n) {
return n; }
105 std::string genName(
const std::string& n,
const void* ptr) {
106 std::string nn = genName(n);
108 ::snprintf(text,
sizeof(text),
"%p",ptr);
116 std::map<std::string, const TNamed*>::const_iterator i = _m.find(name);
118 const char* isa = _n ? _n->IsA()->GetName() : (*i).second ? (*i).second->IsA()->GetName() :
"Unknown";
119 std::cout << isa <<
"(position): duplicate entry with name:" << name <<
" " << (
void*) _n <<
" " << (
void*) (*i).second << std::endl;
121 _m.insert(make_pair(name, _n));
140 int Z = element->Z();
141 double A = element->A();
148 e.
setAttr(
_U(formula), element->GetName());
152 atom.
setAttr(
_U(value), A>0.99 ? A : 1.00794 );
164 TGeoMaterial* geo_mat = medium->GetMaterial();
165 double d = geo_mat->GetDensity();
166 if (d < 1e-10) d = 1e-10;
168 mat.
setAttr(
_U(name), medium->GetName());
176 if (geo_mat->IsMixture()) {
177 TGeoMixture *
mix = (TGeoMixture*)geo_mat;
178 const double *wmix =
mix->GetWmixt();
179 const int *nmix =
mix->GetNmixt();
181 for (
int i = 0, n =
mix->GetNelements(); i < n; i++) {
182 TGeoElement *elt =
mix->GetElement(i);
186 for (
int i = 0, n =
mix->GetNelements(); i < n; i++) {
187 TGeoElement *elt =
mix->GetElement(i);
200 else if ( name !=
"dummy" ) {
203 TGeoElement *elt = geo_mat->GetElement(0);
204 printout(INFO,
"++ Converting non mixing material: %s",name.c_str());
211 atom.
setAttr(
_U(value), geo_mat->GetA() );
222 SolidMap::iterator sit = geo.
xmlSolids.find(shape);
230 return (*sit).second;
232 else if (shape->IsA() == TGeoShapeAssembly::Class()) {
239 TClass* isa = shape->IsA();
240 std::string shape_name = shape->GetName();
242 if (isa == TGeoBBox::Class()) {
243 const TGeoBBox* sh = (
const TGeoBBox*) shape;
251 else if (isa == TGeoTube::Class()) {
252 const TGeoTube* sh = (
const TGeoTube*) shape;
263 else if (isa == TGeoTubeSeg::Class()) {
264 const TGeoTubeSeg* sh = (
const TGeoTubeSeg*) shape;
270 solid.
setAttr(
_U(startphi), sh->GetPhi1());
271 solid.
setAttr(
_U(deltaphi), sh->GetPhi2());
275 else if (isa == TGeoEltu::Class()) {
276 const TGeoEltu* sh = (
const TGeoEltu*) shape;
284 else if (isa == TGeoTrd1::Class()) {
285 const TGeoTrd1* sh = (
const TGeoTrd1*) shape;
295 else if (isa == TGeoTrd2::Class()) {
296 const TGeoTrd2* sh = (
const TGeoTrd2*) shape;
306 else if (isa == TGeoHype::Class()) {
307 const TGeoHype* sh = (
const TGeoHype*) shape;
313 solid.
setAttr(
_U(outst), sh->GetStOut());
318 else if (isa == TGeoPgon::Class()) {
319 const TGeoPgon* sh = (
const TGeoPgon*) shape;
322 solid.
setAttr(
_U(startphi), sh->GetPhi1());
323 solid.
setAttr(
_U(deltaphi), sh->GetDphi());
324 solid.
setAttr(
_U(numsides), sh->GetNedges());
327 for (Int_t i = 0; i < sh->GetNz(); ++i) {
330 zplane.
setAttr(
_U(rmin), sh->GetRmin(i));
331 zplane.
setAttr(
_U(rmax), sh->GetRmax(i));
335 else if (isa == TGeoPcon::Class()) {
336 const TGeoPcon* sh = (
const TGeoPcon*) shape;
339 solid.
setAttr(
_U(startphi), sh->GetPhi1());
340 solid.
setAttr(
_U(deltaphi), sh->GetDphi());
343 for (Int_t i = 0; i < sh->GetNz(); ++i) {
346 zplane.
setAttr(
_U(rmin), sh->GetRmin(i));
347 zplane.
setAttr(
_U(rmax), sh->GetRmax(i));
352 else if (isa == TGeoCone::Class()) {
353 const TGeoCone* sh = (
const TGeoCone*) shape;
357 solid.
setAttr(
_U(rmin1), sh->GetRmin1());
358 solid.
setAttr(
_U(rmax1), sh->GetRmax1());
359 solid.
setAttr(
_U(rmin2), sh->GetRmin2());
360 solid.
setAttr(
_U(rmax2), sh->GetRmax2());
366 else if (isa == TGeoConeSeg::Class()) {
371 solid.
setAttr(
_U(rmin1), sh->GetRmin1());
372 solid.
setAttr(
_U(rmin2), sh->GetRmin2());
373 solid.
setAttr(
_U(rmax1), sh->GetRmax1());
374 solid.
setAttr(
_U(rmax2), sh->GetRmax2());
375 solid.
setAttr(
_U(startphi), sh->GetPhi1());
376 solid.
setAttr(
_U(deltaphi), sh->GetPhi2() - sh->GetPhi1());
380 else if (isa == TGeoParaboloid::Class()) {
381 const TGeoParaboloid* sh = (
const TGeoParaboloid*) shape;
390 else if (isa == TGeoEllipsoid::Class()) {
391 const TGeoEllipsoid* sh = (
const TGeoEllipsoid*) shape;
396 else if (isa == TGeoSphere::Class()) {
397 const TGeoSphere* sh = (
const TGeoSphere*) shape;
402 solid.
setAttr(
_U(startphi), sh->GetPhi1());
403 solid.
setAttr(
_U(deltaphi), (sh->GetPhi2() - sh->GetPhi1()));
404 solid.
setAttr(
_U(starttheta), sh->GetTheta1());
405 solid.
setAttr(
_U(deltatheta), (sh->GetTheta2() - sh->GetTheta1()));
409 else if (isa == TGeoTorus::Class()) {
410 const TGeoTorus* sh = (
const TGeoTorus*) shape;
416 solid.
setAttr(
_U(startphi), sh->GetPhi1());
417 solid.
setAttr(
_U(deltaphi), sh->GetDphi());
421 else if (isa == TGeoTrap::Class()) {
422 const TGeoTrap* sh = (
const TGeoTrap*) shape;
432 solid.
setAttr(
_U(alpha1), sh->GetAlpha1());
433 solid.
setAttr(
_U(alpha2), sh->GetAlpha2());
434 solid.
setAttr(
_U(theta), sh->GetTheta());
439 else if (isa == TGeoPara::Class()) {
440 const TGeoPara* sh = (
const TGeoPara*) shape;
446 solid.
setAttr(
_U(alpha), sh->GetAlpha());
447 solid.
setAttr(
_U(theta), sh->GetTheta());
452 else if (isa == TGeoArb8::Class()) {
453 TGeoArb8* sh = (TGeoArb8*) shape;
454 const double* vtx = sh->GetVertices();
476 else if (isa == TGeoScaledShape::Class()) {
477 TGeoScaledShape* sh = (TGeoScaledShape*) shape;
478 const double* vals = sh->GetScale()->GetScale();
479 Solid s_sh(sh->GetShape());
490 else if (isa == TGeoCompositeShape::Class() ||
491 isa == TGeoUnion::Class() ||
492 isa == TGeoIntersection::Class() ||
493 isa == TGeoSubtraction::Class() ) {
494 const TGeoCompositeShape* sh = (
const TGeoCompositeShape*) shape;
495 const TGeoBoolNode*
boolean = sh->GetBoolNode();
496 TGeoBoolNode::EGeoBoolType oper =
boolean->GetBooleanOperator();
497 TGeoMatrix* rm =
boolean->GetRightMatrix();
498 TGeoMatrix* lm =
boolean->GetLeftMatrix();
499 TGeoShape* ls =
boolean->GetLeftShape();
500 TGeoShape* rs =
boolean->GetRightShape();
503 xml_h first_solid(0), second_solid(0);
505 throw std::runtime_error(
"G4Converter: No left Detector Solid present for composite shape:" + name);
508 throw std::runtime_error(
"G4Converter: No right Detector Solid present for composite shape:" + name);
515 if (strcmp(ls->ClassName(),
"TGeoScaledShape") == 0 &&
516 strcmp(rs->ClassName(),
"TGeoBBox") == 0) {
517 if (strcmp(((TGeoScaledShape *)ls)->GetShape()->ClassName(),
"TGeoSphere") == 0) {
518 if (oper == TGeoBoolNode::kGeoIntersection) {
519 TGeoScaledShape* lls = (TGeoScaledShape *)ls;
520 TGeoBBox* rrs = (TGeoBBox*)rs;
523 double sx = lls->GetScale()->GetScale()[0];
524 double sy = lls->GetScale()->GetScale()[1];
525 double radius = ((TGeoSphere *)lls->GetShape())->GetRmax();
526 double dz = rrs->GetDZ();
527 double zorig = rrs->GetOrigin()[2];
528 double zcut2 = dz + zorig;
529 double zcut1 = 2 * zorig - zcut2;
541 if ( oper == TGeoBoolNode::kGeoSubtraction )
543 else if ( oper == TGeoBoolNode::kGeoUnion )
545 else if ( oper == TGeoBoolNode::kGeoIntersection )
549 std::string lnam = left.
attr<std::string>(
_U(name));
550 std::string rnam = right.
attr<std::string>(
_U(name));
555 first_solid.setAttr(
_U(ref), lnam);
556 const double *tr = lm->GetTranslation();
558 if ((tr[0] != 0.0) || (tr[1] != 0.0) || (tr[2] != 0.0)) {
560 obj.
setAttr(
_U(name), name+
"_"+lnam+
"_pos");
566 if (lm->IsRotation()) {
567 TGeoMatrix
const & linv = lm->Inverse();
568 XYZRotation rot = getXYZangles(linv.GetRotationMatrix());
569 if ((rot.X() != 0.0) || (rot.Y() != 0.0) || (rot.Z() != 0.0)) {
571 obj.
setAttr(
_U(name), name+
"_"+lnam+
"_rot");
578 tr = rm->GetTranslation();
581 if ((tr[0] != 0.0) || (tr[1] != 0.0) || (tr[2] != 0.0)) {
585 if (rm->IsRotation()) {
586 TGeoMatrix
const & rinv = rm->Inverse();
587 XYZRotation rot = getXYZangles(rinv.GetRotationMatrix());
588 if ((rot.X() != 0.0) || (rot.Y() != 0.0) || (rot.Z() != 0.0)) {
595 std::string err =
"Failed to handle unknown solid shape:" + name +
" of type " + std::string(shape->IsA()->GetName());
596 throw std::runtime_error(err);
607 const double* tr = trafo->GetTranslation();
608 if (tr[0] != 0.0 || tr[1] != 0.0 || tr[2] != 0.0) {
609 std::string gen_name = genName(name,trafo);
641 XYZRotation r = getXYZangles(trafo->GetRotationMatrix());
642 if (!(r.X() == 0.0 && r.Y() == 0.0 && r.Z() == 0.0)) {
643 std::string gen_name = genName(name,trafo);
675 const TGeoVolume*
v = volume;
677 std::string n = genName(
v->GetName(),
v);
678 TGeoMedium* medium =
v->GetMedium();
679 TGeoShape* sh =
v->GetShape();
683 if (
v->IsAssembly()) {
689 throw std::runtime_error(
"G4Converter: No Detector Solid present for volume:" + n);
691 throw std::runtime_error(
"G4Converter: No Detector material present for volume:" + n);
696 std::string mat_name = medium->GetName();
704 const TObjArray* dau =
const_cast<TGeoVolume*
>(
v)->GetNodes();
705 if (dau && dau->GetEntries() > 0) {
706 for (Int_t i = 0, n_dau = dau->GetEntries(); i < n_dau; ++i) {
707 TGeoNode* node =
reinterpret_cast<TGeoNode*
>(dau->At(i));
742 const TGeoVolume*
v = volume;
744 if (is_volume(volume)) {
761 if ( is_volume(volume) ) {
774 printout(WARNING,
"LCDDConverter",
"++ CollectVolume: Skip volume: %s",volume->GetName());
782 std::stringstream str;
783 str <<
"++ CheckVolumes: Volume " << n <<
" ";
784 if (is_volume(
v.ptr())) {
788 str <<
"of " << sd.
name() <<
" ";
791 str <<
"with VisAttrs " << vis.
name() <<
" ";
794 str <<
"has duplicate entries." << std::endl;
795 printout(ERROR,
"LCDDConverter",str.str().c_str());
806 TGeoMatrix* matrix = node->GetMatrix();
807 TGeoVolume* volume = node->GetVolume();
819 if ( matrix->IsRotation() ) {
825 if (is_placement(node)) {
827 for (PlacedVolume::VolIDs::const_iterator i = ids.begin(); i != ids.end(); ++i) {
829 pvid.
setAttr(
_U(field_name), (*i).first);
838 std::cout <<
"Attempt to DOUBLE-place physical volume:" << name <<
" No:" << node->GetNumber() << std::endl;
866 const std::set<Limit>& obj = lim.
limits();
867 for (std::set<Limit>::const_iterator i = obj.begin(); i != obj.end(); ++i) {
886 std::string typ = seg.
type();
889 for (_P::const_iterator i = p.begin(); i != p.end(); ++i) {
890 const _P::value_type&
v = *i;
891 if (
v->
name() ==
"lunit") {
894 v->value() ==
_toDouble(
"nanometer") ?
"namometer" :
"??";
900 double value =
_toDouble(
v->value()) / dd4hep::mm;
944 int length = 0, start = 0;
947 id.setAttr(
_U(name), name);
949 for (
const auto& i : fm ) {
954 length = f.second<0 ? -f.second : f.second;
955 idfield.
setAttr(
_U(
signed),f.second<0 ?
true :
false);
968 id.setAttr(
_U(length), length + start);
979 float red = 0, green = 0, blue = 0;
994 vis.
setAttr(
_U(drawing_style),
"wireframe");
997 attr.
rgb(red, green, blue);
1014 std::string type = f->GetTitle();
1017 fld = PluginService::Create<NamedObject*>(type +
"_Convert2Detector", &
m_detDesc, &field, &fld);
1018 printout(ALWAYS,
"LCDDConverter",
"++ %s electromagnetic field:%s of type %s",
1019 (fld.
isValid() ?
"Converted" :
"FAILED to convert "), f->GetName(), type.c_str());
1022 PluginService::Create<NamedObject*>(type +
"_Convert2Detector", &
m_detDesc, &field, &fld);
1023 except(
"LCDDConverter",
"Failed to locate plugin to convert electromagnetic field:"
1024 + std::string(f->GetName()) +
" of type " + type +
". "
1034 std::map<std::string, std::string> processors;
1035 static int s_idd = 9999999;
1037 for (Detector::Properties::const_iterator i = prp.begin(); i != prp.end(); ++i) {
1038 const std::string& nam = (*i).first;
1040 if (nam.substr(0, 6) ==
"geant4") {
1041 Detector::PropertyValues::const_iterator id_it = vals.find(
"id");
1042 if (id_it != vals.end()) {
1043 id = (*id_it).second;
1047 ::snprintf(text,
sizeof(text),
"%d", ++s_idd);
1050 processors.insert(make_pair(
id, nam));
1053 for (std::map<std::string, std::string>::const_iterator i = processors.begin(); i != processors.end(); ++i) {
1055 std::string nam = (*i).second;
1057 auto iter = vals.find(
"type");
1058 if ( iter != vals.end() ) {
1059 std::string type = iter->second;
1060 std::string tag = type +
"_Geant4_action";
1061 long result = PluginService::Create<long>(tag, &
m_detDesc, ptr, &vals);
1064 result = PluginService::Create<long>(tag, &
m_detDesc, ptr, &vals);
1066 except(
"LCDDConverter",
"Failed to locate plugin to interprete files of type"
1067 " \"" + tag +
"\" - no factory:" + type +
". " +
1071 result = *(
long*) result;
1073 except(
"LCDDConverter",
"Failed to invoke the plugin " + tag +
" of type " + type);
1075 printout(INFO,
"",
"+++ Executed Successfully Detector setup module %s.", type.c_str());
1078 printout(INFO,
"",
"+++ FAILED to execute Detector setup module %s.", nam.c_str());
1101 printout(WARNING,
"LCDDConverter",
"+++ No Detector header information availible from the geometry description.");
1104 template <
typename O,
typename C,
typename F>
void handle(
const O* o,
const C& c, F pmf) {
1105 for (
typename C::const_iterator i = c.begin(); i != c.end(); ++i) {
1106 std::string n = (*i)->GetName();
1111 template <
typename O,
typename C,
typename F>
void handleMap(
const O* o,
const C& c, F pmf) {
1112 for (
typename C::const_iterator i = c.begin(); i != c.end(); ++i)
1113 (o->*pmf)((*i).first, (*i).second);
1116 template <
typename O,
typename C,
typename F>
void handleRMap(
const O* o,
const C& c, F pmf) {
1117 for (
typename C::const_reverse_iterator i = c.rbegin(); i != c.rend(); ++i)
1118 handle(o, (*i).second, pmf);
1125 throw std::runtime_error(
"Attempt to call createGDML with an invalid geometry!");
1131 printout(ALWAYS,
"LCDDConverter",
"++ ==> Converting in memory detector description to GDML format...");
1137 "http://service-spi.web.cern.ch/service-spi/app/releases/GDML/schema/gdml.xsd");
1155 for(Detector::HandleMap::const_iterator i=mat.begin(); i!=mat.end(); ++i)
1156 geo.
materials.insert(
dynamic_cast<TGeoMedium*
>((*i).second.ptr()));
1161 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld materials.",geo.
materials.size());
1164 printout(ALWAYS,
"LCDDConverter",
"++ Collected %ld volumes.",geo.
volumes.size());
1167 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld solids.",geo.
solids.size());
1170 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld volumes.",geo.
volumes.size());
1180 throw std::runtime_error(
"Attempt to call createDetector with an invalid geometry!");
1186 printout(ALWAYS,
"LCDDConverter",
"++ ==> Dump visualisation attributes "
1187 "from in memory detector description...");
1197 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld volumes.",geo.
volumes.size());
1205 throw std::runtime_error(
"Attempt to call createDetector with an invalid geometry!");
1219 "http://www.lcsim.org/schemas/description/1.0/description.xsd");
1241 for (Detector::HandleMap::const_iterator i = fld.begin(); i != fld.end(); ++i)
1242 geo.
fields.insert((*i).second);
1244 printout(ALWAYS,
"LCDDConverter",
"++ ==> Converting in memory detector description to Detector format...");
1248 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld materials.",geo.
materials.size());
1251 printout(ALWAYS,
"LCDDConverter",
"++ Collected %ld volumes.",geo.
volumes.size());
1254 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld solids.",geo.
solids.size());
1257 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld visualization attributes.",geo.
vis.size());
1260 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld sensitive detectors.",geo.
sensitives.size());
1263 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld limit sets.",geo.
limits.size());
1266 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld regions.",geo.
regions.size());
1269 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld volumes.",geo.
volumes.size());
1272 printout(ALWAYS,
"LCDDConverter",
"++ Handled %ld fields.",geo.
fields.size());
1285 : doc(0), doc_root(0), doc_header(0), doc_idDict(0), doc_detectors(0), doc_limits(0),
1286 doc_regions(0), doc_display(0), doc_gdml(0), doc_fields(0), doc_define(0),
1287 doc_materials(0), doc_solids(0), doc_structure(0), doc_setup(0)
1291 static long dump_output(
xml_doc_t doc,
int argc,
char** argv) {
1293 return docH.
output(doc, argc > 0 ? argv[0] :
"");
1301 static long create_description(
Detector& description,
int argc,
char** argv) {
1303 return dump_output(wr.createDetector(description.
world()), argc, argv);
1306 static long create_vis(
Detector& description,
int argc,
char** argv) {
1308 return dump_output(wr.createVis(description.
world()), argc, argv);
1311 static long create_visASCII(
Detector& description,
int ,
char** argv) {
1313 wr.createVis(description.
world());
1315 std::map<std::string, xml_comp_t> vis_map;
1319 const char* sep =
";";
1320 std::ofstream os(argv[0]);
1324 auto iter = vis_map.find(ref.refStr());
1325 if ( iter != vis_map.end() ) {
1328 os <<
"vol:" << vol.nameStr() << sep <<
"vis:" << vis.nameStr() << sep
1329 <<
"visible:" << vis.visible() << sep <<
"r:"
1330 << col.R() << sep <<
"g:" << col.G() << sep <<
"b:" << col.B() << sep
1331 <<
"alpha:" << col.alpha() << sep <<
"line_style:"
1332 << vis.attr < std::string > (
_U(line_style)) << sep
1333 <<
"drawing_style:" << vis.attr < std::string> (
_U(drawing_style)) << sep
1334 <<
"show_daughters:" << vis.show_daughters() << sep << std::endl;