Project
Loading...
Searching...
No Matches
MaterialManager.cxx
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
14
17#include "TVirtualMC.h"
18#include "TString.h" // for TString
19#include <TGeoMedium.h>
20#include <TGeoManager.h>
21#include <TList.h>
22#include <iostream>
23#include <utility>
24#include <fairlogger/Logger.h>
25#ifdef NDEBUG
26#undef NDEBUG
27#endif
28#include <cassert>
29#include <set>
30#include "rapidjson/document.h"
31#include "rapidjson/istreamwrapper.h"
32#include "rapidjson/ostreamwrapper.h"
33#include "rapidjson/prettywriter.h"
34#include <algorithm>
35#include <SimConfig/SimParams.h>
36
37using namespace o2::base;
38namespace rj = rapidjson;
39
40namespace
41{
43template <typename K, typename V>
44void writeSingleJSONParamBatch(std::unordered_map<K, const char*> const& idToName, std::map<K, V> const& valMap, V defaultValue, rapidjson::Value& parent, rapidjson::Document::AllocatorType& a)
45{
46 for (auto& itName : idToName) {
47 auto itVal = valMap.find(itName.first);
48 if (itVal != valMap.end()) {
49 parent.AddMember(rj::Value(itName.second, std::strlen(itName.second), a), rj::Value(itVal->second), a);
50 continue;
51 }
52 parent.AddMember(rj::Value(itName.second, std::strlen(itName.second), a), rj::Value(defaultValue), a);
53 }
54}
55
57static constexpr const char* jsonKeyID = "local_id";
58static constexpr const char* jsonKeyIDGlobal = "global_id";
59static constexpr const char* jsonKeyDefault = "default";
60static constexpr const char* jsonKeyCuts = "cuts";
61static constexpr const char* jsonKeyProcesses = "processes";
62static constexpr const char* jsonKeyEnableSpecialCuts = "enableSpecialCuts";
63static constexpr const char* jsonKeyEnableSpecialProcesses = "enableSpecialProcesses";
64} // namespace
65
66const std::unordered_map<EProc, const char*> MaterialManager::mProcessIDToName = {
67 {EProc::kPAIR, "PAIR"},
68 {EProc::kCOMP, "COMP"},
69 {EProc::kPHOT, "PHOT"},
70 {EProc::kPFIS, "PFIS"},
71 {EProc::kDRAY, "DRAY"},
72 {EProc::kANNI, "ANNI"},
73 {EProc::kBREM, "BREM"},
74 {EProc::kHADR, "HADR"},
75 {EProc::kMUNU, "MUNU"},
76 {EProc::kDCAY, "DCAY"},
77 {EProc::kLOSS, "LOSS"},
78 {EProc::kMULS, "MULS"},
79 {EProc::kCKOV, "CKOV"},
80 {EProc::kRAYL, "RAYL"},
81 {EProc::kLABS, "LABS"}};
82
83const std::unordered_map<ECut, const char*> MaterialManager::mCutIDToName = {
84 {ECut::kCUTGAM, "CUTGAM"},
85 {ECut::kCUTELE, "CUTELE"},
86 {ECut::kCUTNEU, "CUTNEU"},
87 {ECut::kCUTHAD, "CUTHAD"},
88 {ECut::kCUTMUO, "CUTMUO"},
89 {ECut::kBCUTE, "BCUTE"},
90 {ECut::kBCUTM, "BCUTM"},
91 {ECut::kDCUTE, "DCUTE"},
92 {ECut::kDCUTM, "DCUTM"},
93 {ECut::kPPCUTM, "PPCUTM"},
94 {ECut::kTOFMAX, "TOFMAX"}};
95
96// Constructing a map between module names and local material density values
97void MaterialManager::initDensityMap()
98{
100 if (globalDensityFactor < 0) {
101 LOG(fatal) << "Negative value "
102 << globalDensityFactor
103 << " found for global material density!\n";
104 }
105 std::string token;
106 std::istringstream input(
107 o2::conf::SimMaterialParams::Instance().localDensityFactor);
108 std::vector<std::string> inputModuleNames;
109 std::vector<std::string> inputDensityValues;
110 while (std::getline(input, token, ',')) {
111 std::size_t pos = token.find(':');
112 inputModuleNames.push_back(token.substr(0, pos));
113 inputDensityValues.push_back(token.substr(pos + 1));
114 }
115 for (std::size_t i = 0; i < inputModuleNames.size(); i++) {
116 if (std::stof(inputDensityValues[i]) < 0) {
117 LOG(fatal) << "Negative value " << std::stof(inputDensityValues[i])
118 << " found for material density in module "
119 << inputModuleNames[i] << "!\n";
120 }
121 mDensityMap[inputModuleNames[i]] = std::stof(inputDensityValues[i]);
122 }
123 mDensityMapInitialized = true;
124}
125
126float MaterialManager::getDensity(std::string const& modname)
127{
128 if (!mDensityMapInitialized) {
129 initDensityMap();
130 }
131 if (mDensityMap.find(modname) != mDensityMap.end()) {
132 return mDensityMap[modname];
133 }
135}
136
137void MaterialManager::Material(const char* modname, Int_t imat, const char* name, Float_t a, Float_t z, Float_t dens,
138 Float_t radl, Float_t absl, Float_t* buf, Int_t nwbuf)
139{
140 TString uniquename = modname;
141 auto densityFactor = getDensity(modname);
142 uniquename.Append("_");
143 uniquename.Append(name);
144 if (TVirtualMC::GetMC()) {
145 // Check this!!!
146 int kmat = -1;
147 TVirtualMC::GetMC()->Material(kmat, uniquename.Data(), a, z,
148 dens * densityFactor, radl, absl, buf, nwbuf);
149 mMaterialMap[modname][imat] = kmat;
150 insertMaterialName(uniquename.Data(), kmat);
151 } else {
152 auto uid = gGeoManager->GetListOfMaterials()->GetSize();
153 auto mat = gGeoManager->Material(uniquename.Data(), a, z,
154 dens * densityFactor, uid, radl, absl);
155 mMaterialMap[modname][imat] = uid;
156 insertMaterialName(uniquename.Data(), uid);
157 }
158}
159
172void MaterialManager::Mixture(const char* modname, Int_t imat, const char* name, Float_t* a, Float_t* z, Float_t dens,
173 Int_t nlmat, Float_t* wmat)
174{
175 TString uniquename = modname;
176 auto densityFactor = getDensity(modname);
177 uniquename.Append("_");
178 uniquename.Append(name);
179
180 if (TVirtualMC::GetMC()) {
181 // Check this!!!
182 int kmat = -1;
183 TVirtualMC::GetMC()->Mixture(kmat, uniquename.Data(), a, z,
184 dens * densityFactor, nlmat, wmat);
185 mMaterialMap[modname][imat] = kmat;
186 insertMaterialName(uniquename.Data(), kmat);
187
188 } else {
189 auto uid = gGeoManager->GetListOfMaterials()->GetSize();
190 if (nlmat < 0) {
191 nlmat = -nlmat;
192 Double_t amol = 0;
193 Int_t i;
194 for (i = 0; i < nlmat; i++) {
195 amol += a[i] * wmat[i];
196 }
197 for (i = 0; i < nlmat; i++) {
198 wmat[i] *= a[i] / amol;
199 }
200 }
201 auto mix = gGeoManager->Mixture(uniquename.Data(), a, z,
202 dens * densityFactor, nlmat, wmat, uid);
203 mMaterialMap[modname][imat] = uid;
204 insertMaterialName(uniquename.Data(), uid);
205 }
206}
207
208void MaterialManager::Medium(const char* modname, Int_t numed, const char* name, Int_t nmat, Int_t isvol, Int_t ifield,
209 Float_t fieldm, Float_t tmaxfd, Float_t stemax, Float_t deemax, Float_t epsil,
210 Float_t stmin, Float_t* ubuf, Int_t nbuf)
211{
212 TString uniquename = modname;
213 uniquename.Append("_");
214 uniquename.Append(name);
215
216 if (TVirtualMC::GetMC()) {
217 // Check this!!!
218 int kmed = -1;
219 const int kmat = getMaterialID(modname, nmat);
220 TVirtualMC::GetMC()->Medium(kmed, uniquename.Data(), kmat, isvol, ifield, fieldm, tmaxfd, stemax, deemax, epsil,
221 stmin, ubuf, nbuf);
222 mMediumMap[modname][numed] = kmed;
223 insertMediumName(uniquename.Data(), kmed);
224 insertTGeoMedium(modname, numed);
225 } else {
226 auto uid = gGeoManager->GetListOfMedia()->GetSize();
227 auto med = gGeoManager->Medium(uniquename.Data(), uid, getMaterialID(modname, nmat), isvol, ifield, fieldm, tmaxfd,
228 stemax, deemax, epsil, stmin);
229 mMediumMap[modname][numed] = uid;
230 insertMediumName(uniquename.Data(), uid);
231 insertTGeoMedium(modname, numed);
232 }
233}
234
235void MaterialManager::Processes(ESpecial special, int globalindex,
236 const std::initializer_list<std::pair<EProc, int>>& parIDValMap)
237{
238 for (auto& m : parIDValMap) {
239 Process(special, globalindex, m.first, m.second);
240 }
241}
242
243void MaterialManager::Cuts(ESpecial special, int globalindex,
244 const std::initializer_list<std::pair<ECut, Float_t>>& parIDValMap)
245{
246 for (auto& m : parIDValMap) {
247 Cut(special, globalindex, m.first, m.second);
248 }
249}
250
251void MaterialManager::Cut(ESpecial special, int globalindex, ECut cut, Float_t val)
252{
253 // this check is needed, in principal only for G3, otherwise SegFault
254 if (val < 0.) {
255 return;
256 }
257 // if low energy neutron transport is requested setting kCUTNEU will set to 0.005eV
258 if (mLowNeut && cut == ECut::kCUTNEU) {
259 LOG(info) << "Due to low energy neutrons, neutron cut value " << val << " discarded and reset to 5e-12";
260 val = 5.e-12;
261 }
262
263 auto it = mCutIDToName.find(cut);
264 if (it == mCutIDToName.end()) {
265 return;
266 }
267 if (special == ESpecial::kFALSE) {
268 auto ins = mDefaultCutMap.insert({cut, val});
269 if (ins.second) {
270 TVirtualMC::GetMC()->SetCut(it->second, val);
271 }
273 } else if (mApplySpecialCuts) {
274 auto ins = mMediumCutMap[globalindex].insert({cut, val});
275 if (ins.second) {
276 TVirtualMC::GetMC()->Gstpar(globalindex, it->second, val);
277 }
278 }
279}
280
281void MaterialManager::Process(ESpecial special, int globalindex, EProc process, int val)
282{
283 // this check is needed, in principal only for G3, otherwise SegFault
284 if (val < 0) {
285 return;
286 }
287 auto it = mProcessIDToName.find(process);
288 if (it == mProcessIDToName.end()) {
289 return;
290 }
291 if (special == ESpecial::kFALSE) {
292 auto ins = mDefaultProcessMap.insert({process, val});
293 if (ins.second) {
294 TVirtualMC::GetMC()->SetProcess(it->second, val);
295 }
297 } else if (mApplySpecialProcesses) {
298 auto ins = mMediumProcessMap[globalindex].insert({process, val});
299 if (ins.second) {
300 TVirtualMC::GetMC()->Gstpar(globalindex, it->second, val);
301 }
302 }
303}
304
306{
307 for (auto& p : mMaterialMap) {
308 auto name = p.first;
309 std::cout << "Materials for key " << name << "\n";
310 for (auto& e : p.second) {
311 std::cout << "internal id " << e.first << " to " << e.second << "\n";
312 }
313 }
314}
315
317{
318 for (auto& p : mMediumMap) {
319 auto name = p.first;
320 std::cout << "Tracking media for key " << name << "\n";
321 for (auto& e : p.second) {
322 auto medname = getMediumNameFromMediumID(e.second);
323 std::cout << medname << " ";
324 std::cout << "internal id " << e.first << " to " << e.second << "\n";
325 }
326 }
327}
328
330{
331 stream << "Summary of process settings per media.\n";
332 stream << "-- Default process settings:\n";
333 for (auto& p : mDefaultProcessMap) {
334 auto it = mProcessIDToName.find(p.first);
335 if (it != mProcessIDToName.end()) {
336 stream << "\t" << it->second << " = " << p.second << "\n";
337 }
338 }
339 if (mApplySpecialProcesses && mMediumProcessMap.size() > 0) {
340 stream << "-- Custom process settings for single media:\n";
341 for (auto& m : mMediumProcessMap) {
342 stream << "Global medium ID " << m.first << " (module = " << getModuleFromMediumID(m.first)
343 << ", medium name = " << getMediumNameFromMediumID(m.first) << "):\n";
344 for (auto& p : m.second) {
345 auto it = mProcessIDToName.find(p.first);
346 if (it != mProcessIDToName.end()) {
347 stream << "\t" << it->second << " = " << p.second << "\n";
348 }
349 }
350 }
351 }
352}
353
354void MaterialManager::printCuts(std::ostream& stream) const
355{
356 stream << "Summary of cut settings per media.\n";
357 stream << "-- Default cut settings:\n";
358 for (auto& c : mDefaultCutMap) {
359 auto it = mCutIDToName.find(c.first);
360 if (it != mCutIDToName.end()) {
361 stream << "\t" << it->second << " = " << c.second << "\n";
362 }
363 }
364 if (mApplySpecialCuts && mMediumCutMap.size() > 0) {
365 stream << "-- Custom cut settings for single media:\n";
366 for (auto& m : mMediumCutMap) {
367 stream << "Global medium ID " << m.first << " (module = " << getModuleFromMediumID(m.first)
368 << ", medium name = " << getMediumNameFromMediumID(m.first) << "):\n";
369 for (auto& c : m.second) {
370 auto it = mCutIDToName.find(c.first);
371 if (it != mCutIDToName.end()) {
372 stream << "\t" << it->second << " = " << c.second << "\n";
373 }
374 }
375 }
376 }
377}
378
379// insert material name
380void MaterialManager::insertMaterialName(const char* uniquename, int index)
381{
382 assert(mMaterialNameToGlobalIndexMap.find(uniquename) == mMaterialNameToGlobalIndexMap.end());
383 mMaterialNameToGlobalIndexMap[uniquename] = index;
384}
385
386// inserts the last
387void MaterialManager::insertTGeoMedium(std::string modname, int localindex)
388{
389 auto p = std::make_pair(modname, localindex);
390 assert(mTGeoMediumMap.find(p) == mTGeoMediumMap.end());
391 auto list = gGeoManager->GetListOfMedia();
392 mTGeoMediumMap[p] = (TGeoMedium*)list->At(list->GetEntries() - 1);
393
394 LOG(debug) << "mapping " << modname << " " << localindex << " to " << mTGeoMediumMap[p]->GetName();
395}
396
397void MaterialManager::insertMediumName(const char* uniquename, int index)
398{
399 assert(mMediumNameToGlobalIndexMap.find(uniquename) == mMediumNameToGlobalIndexMap.end());
400 mMediumNameToGlobalIndexMap[uniquename] = index;
401}
402
403// find TGeoMedium instance given a detector prefix and a local (vmc) medium index
404TGeoMedium* MaterialManager::getTGeoMedium(std::string const& modname, int localindex)
405{
406 auto p = std::make_pair(modname, localindex);
407 auto iter = mTGeoMediumMap.find(p);
408 if (iter == mTGeoMediumMap.end()) {
409 LOG(warning) << "No medium registered for " << modname << " index " << localindex << "\n";
410 return nullptr;
411 }
412 return iter->second;
413}
414
415// find TGeoMedium instance given the full medium name
416// mainly forwards directly to TGeoManager and raises a warning if medium is nullptr
417TGeoMedium* MaterialManager::getTGeoMedium(const char* mediumname)
418{
419 auto med = gGeoManager->GetMedium(mediumname);
420 assert(med != nullptr);
421 return med;
422}
423
425{
426 const std::string filenameIn = filename.empty() ? o2::MaterialManagerParam::Instance().inputFile : filename;
427 if (filenameIn.empty()) {
428 return;
429 }
430 std::ifstream is(filenameIn);
431 if (!is.is_open()) {
432 LOG(fatal) << "Cannot open MC cuts/processes file " << filenameIn;
433 return;
434 }
435 auto digestCutsFromJSON = [this](int globalindex, rj::Value& cuts) {
436 auto special = globalindex < 0 ? ESpecial::kFALSE : ESpecial::kTRUE;
437 for (auto& cut : cuts.GetObject()) {
438 auto name = cut.name.GetString();
439 bool found = false;
440 for (auto& cn : mCutIDToName) {
441 if (std::strcmp(name, cn.second) == 0) {
442 Cut(special, globalindex, cn.first, cut.value.GetFloat());
443 found = true;
444 }
445 }
446 if (!found) {
447 LOG(warn) << "Unknown cut parameter " << name;
448 }
449 }
450 };
451 auto digestProcessesFromJSON = [this](int globalindex, rj::Value& processes) {
452 auto special = globalindex < 0 ? ESpecial::kFALSE : ESpecial::kTRUE;
453 for (auto& proc : processes.GetObject()) {
454 auto name = proc.name.GetString();
455 for (auto& pn : mProcessIDToName) {
456 if (std::strcmp(name, pn.second) == 0) {
457 Process(special, globalindex, pn.first, proc.value.GetInt());
458 }
459 }
460 }
461 };
462
463 rj::IStreamWrapper isw(is);
464 rj::Document d;
465 d.ParseStream(isw);
466
467 if (special == ESpecial::kFALSE && d.HasMember(jsonKeyDefault)) {
468 // defaults
469 auto& defaultParams = d[jsonKeyDefault];
470 if (defaultParams.HasMember(jsonKeyCuts)) {
471 digestCutsFromJSON(-1, defaultParams[jsonKeyCuts]);
472 }
473 if (defaultParams.HasMember(jsonKeyProcesses)) {
474 digestProcessesFromJSON(-1, defaultParams[jsonKeyProcesses]);
475 }
476 } else if (special == ESpecial::kTRUE) {
477 // read whether to apply special cuts and processes at all
478 if (d.HasMember(jsonKeyEnableSpecialCuts)) {
479 enableSpecialCuts(d[jsonKeyEnableSpecialCuts].GetBool());
480 }
481 if (d.HasMember(jsonKeyEnableSpecialProcesses)) {
482 enableSpecialProcesses(d[jsonKeyEnableSpecialProcesses].GetBool());
483 }
484 // special
485 for (auto& m : d.GetObject()) {
486 if (m.name.GetString()[0] == '\0' || !m.value.IsArray()) {
487 // do not parse anything with empty key, these at the most meant to be comments
488 continue;
489 }
490 for (auto& batch : m.value.GetArray()) {
491 if (std::strcmp(m.name.GetString(), jsonKeyDefault) == 0) {
492 // don't do defaults here
493 continue;
494 }
495 // set via their global indices
496 auto index = getMediumID(m.name.GetString(), batch[jsonKeyID].GetInt());
497 if (index < 0) {
498 continue;
499 }
500 if (batch.HasMember(jsonKeyCuts)) {
501 digestCutsFromJSON(index, batch[jsonKeyCuts]);
502 }
503 if (batch.HasMember(jsonKeyProcesses)) {
504 digestProcessesFromJSON(index, batch[jsonKeyProcesses]);
505 }
506 }
507 }
508 }
509}
510
512{
513 const std::string filenameOut = filename.empty() ? o2::MaterialManagerParam::Instance().outputFile : filename;
514 if (filenameOut.empty()) {
515 return;
516 }
517
518 // write parameters as global AND module specific
519 std::ofstream os(filenameOut);
520 if (!os.is_open()) {
521 LOG(error) << "Cannot create file " << filenameOut;
522 return;
523 }
524
525 rj::Document d;
526 rj::Document::AllocatorType& a = d.GetAllocator();
527 d.SetObject();
528
529 // add each local medium with params per module
530 for (auto& itMed : mMediumMap) {
531 // prepare array for module
532 rj::Value toAdd(rj::kArrayType);
533 // extract each medium's local and global index
534 for (auto& locToGlob : itMed.second) {
535 auto globalindex = locToGlob.second;
536 auto itCut = mMediumCutMap.find(globalindex);
537 auto itProc = mMediumProcessMap.find(globalindex);
538 // prepare a batch summarising localID, globaldID, cuts and processes
539 rj::Value oLoc(rj::kObjectType);
540 // IDs
541 oLoc.AddMember(rj::Value(jsonKeyID, std::strlen(jsonKeyID), a), rj::Value(locToGlob.first), a);
542 oLoc.AddMember(rj::Value(jsonKeyIDGlobal, std::strlen(jsonKeyIDGlobal)), rj::Value(locToGlob.second), a);
543 // add medium and material name
544 auto mediumIt = mTGeoMediumMap.find({itMed.first, locToGlob.first});
545 const char* medName = mediumIt->second->GetName();
546 const char* matName = mediumIt->second->GetMaterial()->GetName();
547 // not using variables for key names cause they are only written for info but not read
548 oLoc.AddMember(rj::Value("medium_name", 11, a), rj::Value(medName, std::strlen(medName), a), a);
549 oLoc.AddMember(rj::Value("material_name", 13, a), rj::Value(matName, std::strlen(matName), a), a);
550 // prepare for cuts
551 if (itCut != mMediumCutMap.end()) {
552 rj::Value cutMap(rj::kObjectType);
553 writeSingleJSONParamBatch(mCutIDToName, itCut->second, -1.f, cutMap, a);
554 oLoc.AddMember(rj::Value(jsonKeyCuts, std::strlen(jsonKeyCuts), a), cutMap, a);
555 }
556 // prepare for processes
557 if (itProc != mMediumProcessMap.end()) {
558 rj::Value procMap(rj::kObjectType);
559 writeSingleJSONParamBatch(mProcessIDToName, itProc->second, -1, procMap, a);
560 oLoc.AddMember(rj::Value(jsonKeyProcesses, std::strlen(jsonKeyProcesses), a), procMap, a);
561 }
562 // append this medium to module array
563 toAdd.PushBack(oLoc, a);
564 }
565 // append the entire module array
566 d.AddMember(rj::Value(itMed.first.c_str(), itMed.first.size(), a), toAdd, a);
567 }
568 // also add default parameters
569 rj::Value cutMapDef(rj::kObjectType);
570 rj::Value procMapDef(rj::kObjectType);
571 writeSingleJSONParamBatch(mCutIDToName, mDefaultCutMap, -1.f, cutMapDef, a);
572 writeSingleJSONParamBatch(mProcessIDToName, mDefaultProcessMap, -1, procMapDef, a);
573 rj::Value defaultParams(rj::kObjectType);
574 defaultParams.AddMember(rj::Value(jsonKeyCuts, std::strlen(jsonKeyCuts), a), cutMapDef, a);
575 defaultParams.AddMember(rj::Value(jsonKeyProcesses, std::strlen(jsonKeyProcesses), a), procMapDef, a);
576 d.AddMember(rj::Value(jsonKeyDefault, std::strlen(jsonKeyDefault), a), defaultParams, a);
577
578 d.AddMember(rj::Value(jsonKeyEnableSpecialCuts, std::strlen(jsonKeyEnableSpecialCuts), a), rj::Value(mApplySpecialCuts), a);
579 d.AddMember(rj::Value(jsonKeyEnableSpecialProcesses, std::strlen(jsonKeyEnableSpecialProcesses), a), rj::Value(mApplySpecialProcesses), a);
580 // now write to file
581 rj::OStreamWrapper osw(os);
582 rj::PrettyWriter<rj::OStreamWrapper> writer(osw);
583 writer.SetIndent(' ', 2);
584 d.Accept(writer);
585}
586
587void MaterialManager::loadCutsAndProcessesFromFile(const char* modname, const char* filename)
588{
589 // Implementation of a method to set cuts and processes as done in AliRoot.
590 // The file is expected to contain columns of the form
591 // MODNAME LOCALMEDIUMID CUT0 ... CUT9 FLAG0 ... FLAG11
592 // where cuts and flags correspond to keys denoted in cutnames and procnames below.
593
594 const int NCUTS = 10; // number of cut columns expected in file
595 const int NFLAGS = 12; // number of process flag columns expected in file
596
598 using namespace o2::base;
599 ECut cutnames[NCUTS] = {ECut::kCUTGAM,
609
611 // missing STRA for the moment
612 EProc procnames[NFLAGS - 1] = {EProc::kANNI,
623
624 std::ifstream cutfile(filename);
625
626 if (!cutfile.is_open()) {
627 LOG(warn) << "File " << filename << " does not exist; Cannot apply cuts";
628 return;
629 }
630
631 // reading from file
632 float cut[NCUTS]; // to store cut values
633 int flag[NFLAGS]; // to store flags
634 int itmed, iret;
635 char line[256];
636 char detName[7];
637
638 while (cutfile.getline(line, 256)) {
639 // Initialise cuts and flags for this line
640 for (int i = 0; i < NCUTS; i++) {
641 cut[i] = -99;
642 }
643 for (int i = 0; i < NFLAGS; i++) {
644 flag[i] = -99;
645 }
646 if (strlen(line) == 0) {
647 continue;
648 }
649 // ignore comments marked by *
650 if (line[0] == '*') {
651 continue;
652 }
653 // Read the numbers
654 iret = sscanf(line, "%6s %d %f %f %f %f %f %f %f %f %f %f %d %d %d %d %d %d %d %d %d %d %d %d",
655 detName, &itmed, &cut[0], &cut[1], &cut[2], &cut[3], &cut[4], &cut[5], &cut[6], &cut[7], &cut[8],
656 &cut[9], &flag[0], &flag[1], &flag[2], &flag[3], &flag[4], &flag[5], &flag[6], &flag[7],
657 &flag[8], &flag[9], &flag[10], &flag[11]);
658 if (!iret) {
659 // nothing read
660 continue;
661 }
662
663 // apply cuts via material manager interface
664 for (int i = 0; i < NCUTS; ++i) {
665 if (cut[i] >= 0.) {
666 SpecialCut(detName, itmed, cutnames[i], cut[i]);
667 }
668 }
669
670 // apply process flags
671 for (int i = 0; i < NFLAGS - 1; ++i) {
672 if (flag[i] >= 0) {
673 SpecialProcess(detName, itmed, procnames[i], flag[i]);
674 }
675 }
676 } // end loop over lines
677}
678
682void MaterialManager::SpecialCuts(const char* modname, int localindex,
683 const std::initializer_list<std::pair<ECut, Float_t>>& parIDValMap)
684{
685 int globalindex = getMediumID(modname, localindex);
686 if (globalindex != -1) {
687 Cuts(ESpecial::kTRUE, globalindex, parIDValMap);
688 }
689}
690
691void MaterialManager::SpecialCut(const char* modname, int localindex, ECut parID, Float_t val)
692{
693 int globalindex = getMediumID(modname, localindex);
694 if (globalindex != -1) {
695 Cut(ESpecial::kTRUE, globalindex, parID, val);
696 } else {
697 LOG(warn) << "SpecialCut: NO GLOBALINDEX FOUND FOR " << modname << " " << localindex;
698 }
699}
700
702void MaterialManager::SpecialProcess(const char* modname, int localindex, EProc parID, int val)
703{
704 int globalindex = getMediumID(modname, localindex);
705 if (globalindex != -1) {
706 Process(ESpecial::kTRUE, globalindex, parID, val);
707 } else {
708 LOG(warn) << "SpecialProcess: NO GLOBALINDEX FOUND FOR " << modname << " " << localindex;
709 }
710}
711
712int MaterialManager::getMaterialID(const char* modname, int imat) const
713{
714 auto lookupiter = mMaterialMap.find(modname);
715 if (lookupiter == mMaterialMap.end()) {
716 return -1;
717 }
718 auto lookup = lookupiter->second;
719
720 auto iter = lookup.find(imat);
721 if (iter != lookup.end()) {
722 return iter->second;
723 }
724 return -1;
725}
726
727int MaterialManager::getMediumID(const char* modname, int imed) const
728{
729 auto lookupiter = mMediumMap.find(modname);
730 if (lookupiter == mMediumMap.end()) {
731 return -1;
732 }
733 auto lookup = lookupiter->second;
734
735 auto iter = lookup.find(imed);
736 if (iter != lookup.end()) {
737 return iter->second;
738 }
739 return -1;
740}
741
742// fill the medium index mapping into a standard vector
743// the vector gets sized properly and will be overridden
744void MaterialManager::getMediumIDMappingAsVector(const char* modname, std::vector<int>& mapping) const
745{
746 mapping.clear();
747
748 auto lookupiter = mMediumMap.find(modname);
749 if (lookupiter == mMediumMap.end()) {
750 return;
751 }
752 auto lookup = lookupiter->second;
753
754 // get the biggest mapped value (maps are sorted in keys)
755 auto maxkey = lookup.rbegin()->first;
756 // resize mapping and initialize with -1 by default
757 mapping.resize(maxkey + 1, -1);
758 // fill vector with entries from map
759 for (auto& p : lookup) {
760 mapping[p.first] = p.second;
761 }
762}
763
764void MaterialManager::getMediaWithSpecialProcess(EProc process, std::vector<int>& mediumProcessVector) const
765{
766 // clear
767 mediumProcessVector.clear();
768 // resize to maximum number of global IDs for which special processes are set. In case process is not
769 // implemented for a certain medium, value is -1
770 mediumProcessVector.resize(mMediumProcessMap.size(), -1);
771 // find media
772 for (auto& m : mMediumProcessMap) {
773 // loop over processes in medium
774 for (auto& p : m.second) {
775 // push medium ID if process is there
776 if (p.first == process) {
777 mediumProcessVector[m.first] = p.second;
778 break;
779 }
780 }
781 }
782}
783
784void MaterialManager::getMediaWithSpecialCut(ECut cut, std::vector<Float_t>& mediumCutVector) const
785{
786 // clear
787 mediumCutVector.clear();
788 // resize to maximum number of global IDs for which special cuts are set. In case cut is not implemented
789 // for a certain medium, value is -1.
790 mediumCutVector.resize(mMediumCutMap.size(), -1.);
791 // find media
792 for (auto& m : mMediumCutMap) {
793 // loop over cuts in medium
794 for (auto& c : m.second) {
795 // push medium ID if cut is there
796 if (c.first == cut) {
797 mediumCutVector[m.first] = c.second;
798 break;
799 }
800 }
801 }
802}
803
805void MaterialManager::getDefaultProcesses(std::vector<std::pair<EProc, int>>& processVector)
806{
807 processVector.clear();
808 for (auto& m : mDefaultProcessMap) {
809 processVector.emplace_back(m.first, m.second);
810 }
811}
813void MaterialManager::getDefaultCuts(std::vector<std::pair<ECut, Float_t>>& cutVector)
814{
815 cutVector.clear();
816 for (auto& m : mDefaultCutMap) {
817 cutVector.emplace_back(m.first, m.second);
818 }
819}
821void MaterialManager::getSpecialProcesses(int globalindex, std::vector<std::pair<EProc, int>>& processVector)
822{
823 processVector.clear();
824 if (mMediumProcessMap.find(globalindex) != mMediumProcessMap.end()) {
825 for (auto& m : mMediumProcessMap[globalindex]) {
826 processVector.emplace_back(m.first, m.second);
827 }
828 }
829}
831void MaterialManager::getSpecialProcesses(const char* modname, int localindex, std::vector<std::pair<EProc, int>>& processVector)
832{
833 int globalindex = getMediumID(modname, localindex);
834 if (globalindex != -1) {
835 getSpecialProcesses(globalindex, processVector);
836 }
837}
839void MaterialManager::getSpecialCuts(int globalindex, std::vector<std::pair<ECut, Float_t>>& cutVector)
840{
841 cutVector.clear();
842 if (mMediumCutMap.find(globalindex) != mMediumCutMap.end()) {
843 for (auto& m : mMediumCutMap[globalindex]) {
844 cutVector.emplace_back(m.first, m.second);
845 }
846 }
847}
849void MaterialManager::getSpecialCuts(const char* modname, int localindex, std::vector<std::pair<ECut, Float_t>>& cutVector)
850{
851 int globalindex = getMediumID(modname, localindex);
852 if (globalindex != -1) {
853 getSpecialCuts(globalindex, cutVector);
854 }
855}
856
857const char* MaterialManager::getModuleFromMediumID(int globalindex) const
858{
859 // loop over module names and corresponding local<->global mapping
860 for (auto& m : mMediumMap) {
861 for (auto& i : m.second) {
862 // is the global index there?
863 if (i.second == globalindex) {
864 // \note maybe unsafe in case mMediumMap is altered in the same scope where the returned C string is used
865 // since that points to memory of string it was derived from.
866 return m.first.c_str();
867 }
868 }
869 }
870 // module is UNKNOWN if global medium ID could not be found.
871 return "UNKNOWN";
872}
873
875const char* MaterialManager::getMediumNameFromMediumID(int globalindex) const
876{
877 // Get the name of the medium.
878 // TODO: avoid linear search
879 for (auto& n : mMediumNameToGlobalIndexMap) {
880 if (n.second == globalindex) {
881 // \note maybe unsafe in case mMediumMap is altered in the same scope where the returned C string is used since
882 // that points to memory of string it was derived from.
883 return n.first.c_str();
884 }
885 }
886 return "UNKNOWN";
887}
888
891void MaterialManager::printContainingMedia(std::string const& volumename)
892{
893 auto vol = gGeoManager->FindVolumeFast(volumename.c_str());
894 if (vol == nullptr) {
895 LOG(warn) << "No volume found; Cannot query medias";
896 }
897 std::set<TGeoMedium const*> media;
898
899 // a little well encapsulated helper to code the recursive visitor
900 // pure lambda cannot be recursive
901 std::function<void(TGeoVolume const* vol, std::set<TGeoMedium const*>& mediumset)> recursivevisitor;
902 recursivevisitor = [&recursivevisitor](TGeoVolume const* vol, std::set<TGeoMedium const*>& mediumset) {
903 // exclude assemblies
904 if (!vol->IsAssembly()) {
905 mediumset.insert(vol->GetMedium());
906 }
907 const int n = vol->GetNdaughters();
908 for (int i = 0; i < n; ++i) {
909 auto daughter = vol->GetNode(i)->GetVolume();
910 recursivevisitor(daughter, mediumset);
911 }
912 };
913 recursivevisitor(vol, media);
914
915 // simply print the media
916 for (auto m : media) {
917 std::cout << m->GetName() << "\n";
918 }
919}
920
uint8_t lookup(const char input) noexcept
int32_t i
ClassImp(IdPath)
uint16_t pos
Definition RawData.h:3
uint32_t c
Definition RawData.h:2
std::ostringstream debug
void printCuts(std::ostream &stream) const
Print all cuts for all media as well as defaults.
void getSpecialCuts(int globalindex, std::vector< std::pair< ECut, Float_t > > &cutVector)
Get special cuts for global medium ID.
void SpecialCuts(const char *modname, int localindex, const std::initializer_list< std::pair< ECut, Float_t > > &parIDValMap)
void getMediumIDMappingAsVector(const char *modname, std::vector< int > &mapping) const
void getDefaultProcesses(std::vector< std::pair< EProc, int > > &processVector)
Fill vector with default processes.
void printProcesses(std::ostream &stream) const
Print all processes for all media as well as defaults.
void enableSpecialCuts(bool val=true)
void writeCutsAndProcessesToJSON(std::string const &filename="")
void loadCutsAndProcessesFromFile(const char *modname, const char *filename)
load cuts and process flags from a data file (like AliRoot did)
void loadCutsAndProcessesFromJSON(ESpecial special=ESpecial::kFALSE, std::string const &filename="")
void getMediaWithSpecialProcess(EProc process, std::vector< int > &mediumProcessVector) const
get global medium IDs where special process is set along with process value
int getMediumID(const char *modname, int imed) const
void getSpecialProcesses(int globalindex, std::vector< std::pair< EProc, int > > &processVector)
Get special processes for global medium ID.
const char * getMediumNameFromMediumID(int globalindex) const
Get medium name from global medium ID.
void getMediaWithSpecialCut(ECut cut, std::vector< Float_t > &mediumCutVector) const
get global medium IDs where special cut is set along with cut value
static void printContainingMedia(std::string const &volumename)
void Mixture(const char *modname, Int_t imat, const char *name, Float_t *a, Float_t *z, Float_t dens, Int_t nlmat, Float_t *wmat)
void SpecialProcess(const char *modname, int localindex, EProc parID, int val)
Custom setting of process or cut given parameter name and value.
void Medium(const char *modname, Int_t numed, const char *name, Int_t nmat, Int_t isvol, Int_t ifield, Float_t fieldm, Float_t tmaxfd, Float_t stemax, Float_t deemax, Float_t epsil, Float_t stmin, Float_t *ubuf=nullptr, Int_t nbuf=0)
void SpecialCut(const char *modname, int localindex, ECut parID, Float_t val)
Custom setting of process or cut given parameter name and value.
void Material(const char *modname, Int_t imat, const char *name, Float_t a, Float_t z, Float_t dens, Float_t radl, Float_t absl, Float_t *buf=nullptr, Int_t nwbuf=0)
void enableSpecialProcesses(bool val=true)
Set flags whether to use special cuts and process settings.
const char * getModuleFromMediumID(int globalindex) const
Get module name which has medium of a certain global medium ID.
ESpecial
scoped enum to decide whether settings are done globally or for a certain medium
int getMaterialID(const char *modname, int imat) const
void getDefaultCuts(std::vector< std::pair< ECut, Float_t > > &cutVector)
Fill vector with default cuts.
TGeoMedium * getTGeoMedium(const std::string &modname, int localid)
GLdouble n
Definition glcorearb.h:1982
const GLfloat * m
Definition glcorearb.h:4066
GLuint index
Definition glcorearb.h:781
GLuint const GLchar * name
Definition glcorearb.h:781
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLuint GLfloat * val
Definition glcorearb.h:1582
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
GLuint GLuint stream
Definition glcorearb.h:1806
GLenum GLuint GLenum GLsizei const GLchar * buf
Definition glcorearb.h:2514
GLdouble GLdouble GLdouble z
Definition glcorearb.h:843
ECut
cuts available
EProc
processes available
std::string filename()
Definition list.h:40
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"