From 24fc6e0ca4dfc3907af7df89b0e9dd1865ac1f1e Mon Sep 17 00:00:00 2001 From: mmlynari <104267016+mmlynari@users.noreply.github.com> Date: Wed, 28 Aug 2024 22:53:42 +0200 Subject: [PATCH 1/3] update CellPositionsTool for HCal Allegro (#91) * update CellPositionsTool for HCal Allegro * revert changes in CellPositionsHCalBarrelPhiThetaSegTool and add CellPositionsHCalPhiThetaSegTool * remove a typo * include comments and improve documentation * do not use GaudiAlg * replace GaudiAlg by GaudiKernel --- .../CellPositionsHCalPhiThetaSegTool.cpp | 224 ++++++++++++++++++ .../CellPositionsHCalPhiThetaSegTool.h | 83 +++++++ 2 files changed, 307 insertions(+) create mode 100644 RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.cpp create mode 100644 RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.h diff --git a/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.cpp b/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.cpp new file mode 100644 index 00000000..754db0f5 --- /dev/null +++ b/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.cpp @@ -0,0 +1,224 @@ +#include "CellPositionsHCalPhiThetaSegTool.h" + +#include "edm4hep/CalorimeterHitCollection.h" +#include + +using dd4hep::DetElement; + +DECLARE_COMPONENT(CellPositionsHCalPhiThetaSegTool) + +CellPositionsHCalPhiThetaSegTool::CellPositionsHCalPhiThetaSegTool( + const std::string &type, + const std::string &name, + const IInterface *parent) + : AlgTool(type, name, parent) +{ + declareInterface(this); +} + +StatusCode CellPositionsHCalPhiThetaSegTool::initialize() +{ + StatusCode sc = AlgTool::initialize(); + if (sc.isFailure()) + return sc; + m_geoSvc = service("GeoSvc"); + if (!m_geoSvc) + { + error() << "Unable to locate Geometry service." << endmsg; + return StatusCode::FAILURE; + } + // get PhiTheta segmentation + m_segmentation = dynamic_cast( + m_geoSvc->getDetector()->readout(m_readoutName).segmentation().segmentation()); + if (m_segmentation == nullptr) + { + error() << "There is no phi-theta segmentation!!!!" << endmsg; + return StatusCode::FAILURE; + } + // Take readout bitfield decoder from GeoSvc + m_decoder = m_geoSvc->getDetector()->readout(m_readoutName).idSpec().decoder(); + + // check if decoder contains "layer" + std::vector fields; + for (uint itField = 0; itField < m_decoder->size(); itField++) + { + fields.push_back((*m_decoder)[itField].name()); + } + auto iter = std::find(fields.begin(), fields.end(), "layer"); + if (iter == fields.end()) + { + error() << "Readout does not contain field: 'layer'" << endmsg; + return StatusCode::FAILURE; + } + + // retrieve radii from the LayeredCalorimeterData extension + dd4hep::Detector* detector = m_geoSvc->getDetector(); + if (!detector) + { + error() << "Unable to retrieve the detector." << endmsg; + return StatusCode::FAILURE; + } + + DetElement caloDetElem = detector->detector(m_detectorName); + if (!caloDetElem.isValid()) + { + error() << "Unable to retrieve the detector element: " << m_detectorName << endmsg; + return StatusCode::FAILURE; + } + + dd4hep::rec::LayeredCalorimeterData* theExtension = caloDetElem.extension(); + if (!theExtension) + { + error() << "The detector element does not have the required LayeredCalorimeterData extension." << endmsg; + return StatusCode::FAILURE; + } + + // Debug information to check the number of layers retrieved from the LayeredCalorimeterData extension + m_layersRetrieved = &(theExtension->layers); + debug() << "Number of layers retrieved: " << m_layersRetrieved->size() << endmsg; + + if (m_detectorName=="HCalBarrel"){ + m_radii = CellPositionsHCalPhiThetaSegTool::calculateLayerRadiiBarrel(); + } + else if (m_detectorName=="HCalThreePartsEndcap"){ + m_radii = CellPositionsHCalPhiThetaSegTool::calculateLayerRadiiEndcap(); + } + else{ + error() << "Provided detector name in m_detectorName " << m_detectorName << " is not matching, expected inputs are HCalBarrel or HCalThreePartsEndcap" << endmsg; + return StatusCode::FAILURE; + } + + unsigned int numLayersProvided = 0; + + if (m_detectorName=="HCalThreePartsEndcap") + { + // Check that the vector containing number of layers in each cylinder is provided + if (m_numLayersHCalThreeParts.empty()) + { + error() << "The vector m_numLayersHCalThreeParts is empty." << endmsg; + return StatusCode::FAILURE; + } + // Check that the total number of layers provided in the steering file + // matches the total number of layers in the geometry xml file + for (unsigned int i=0; isize() != numLayersProvided) + { + error() << "Total number of radial layers provided in m_numLayersHCalThreeParts " << numLayersProvided << " does not match the numbers retrieved from the xml file " << m_layersRetrieved->size() << endmsg; + return StatusCode::FAILURE; + } + } + return sc; +} + +void CellPositionsHCalPhiThetaSegTool::getPositions(const edm4hep::CalorimeterHitCollection &aCells, + edm4hep::CalorimeterHitCollection &outputColl) +{ + debug() << "Input collection size : " << aCells.size() << endmsg; + // Loop through cell collection + for (const auto &cell : aCells) + { + auto outSeg = CellPositionsHCalPhiThetaSegTool::xyzPosition(cell.getCellID()); + + auto edmPos = edm4hep::Vector3f(); + edmPos.x = outSeg.x() / dd4hep::mm; + edmPos.y = outSeg.y() / dd4hep::mm; + edmPos.z = outSeg.z() / dd4hep::mm; + + auto positionedHit = cell.clone(); + positionedHit.setPosition(edmPos); + outputColl.push_back(positionedHit); + + // Debug information about cell position + debug() << "Cell energy (GeV) : " << positionedHit.getEnergy() << "\tcellID " << positionedHit.getCellID() << endmsg; + debug() << "Position of cell (mm) : \t" << outSeg.x() / dd4hep::mm << "\t" << outSeg.y() / dd4hep::mm << "\t" + << outSeg.z() / dd4hep::mm << endmsg; + } + debug() << "Output positions collection size: " << outputColl.size() << endmsg; +} + + +dd4hep::Position CellPositionsHCalPhiThetaSegTool::xyzPosition(const uint64_t &aCellId) const +{ + dd4hep::DDSegmentation::CellID volumeId = aCellId; + m_decoder->set(volumeId, "phi", 0); + m_decoder->set(volumeId, "theta", 0); + + int layer = m_decoder->get(volumeId, "layer"); + // get radius in cm + double radius = m_radii[layer]; + // get local position (for r=1) + auto inSeg = m_segmentation->position(aCellId); + // scale by radius to get global position + dd4hep::Position outSeg(inSeg.x() * radius, inSeg.y() * radius, inSeg.z() * radius); + + // MM: TBD the z-coordinate still needs to be carefully validated + // at the first glance it seems to be off in some cases in the Endcap + debug() << "Layer : " << layer << "\tradius : " << radius << " cm" << endmsg; + debug() << "Local position : x = " << inSeg.x() << " y = " << inSeg.y() << " z = " << inSeg.z() << endmsg; + debug() << "Global position : x = " << outSeg.x() << " y = " << outSeg.y() << " z = " << outSeg.z() << endmsg; + + return outSeg; +} + +int CellPositionsHCalPhiThetaSegTool::layerId(const uint64_t &aCellId) +{ + int layer; + layer = m_decoder->get(aCellId, "layer"); + return layer; +} + +// calculate layer radii from LayeredCalorimeterData extension +// which is included in the geometry description +std::vector CellPositionsHCalPhiThetaSegTool::calculateLayerRadii(unsigned int startIndex, unsigned int endIndex) { + std::vector radii; + + for (unsigned int idxLayer = startIndex; idxLayer < endIndex; ++idxLayer) { + const dd4hep::rec::LayeredCalorimeterStruct::Layer & theLayer = m_layersRetrieved->at(idxLayer); + // inner radius of a given layer + double layerInnerRadius = theLayer.distance; + // radial dimension of a given layer + double layerThickness = theLayer.sensitive_thickness; + + // Calculate mid-radius for the current layer (layerInnerRadius+layerOuterRadius)/2 + double layerMidRadius = layerInnerRadius + layerThickness / 2; + + radii.push_back(layerMidRadius); + } + return radii; +} + +// calculateLayerRadii should be used for HCalTileBarrel which is formed +// by a single cylinder with a constant number of radial layers +std::vector CellPositionsHCalPhiThetaSegTool::calculateLayerRadiiBarrel() { + return calculateLayerRadii(0, m_layersRetrieved->size()); +} + +// calculateLayerRadiiEndcap should be used for HCalThreePartsEndcap which is formed +// by three cylinders along the z-coordinate and each cylinder has a different number of radial layers +std::vector CellPositionsHCalPhiThetaSegTool::calculateLayerRadiiEndcap() { + std::vector radii; + + // Calculate radii for each part (cylinder) and merge the results + unsigned int upperIndexLayerRadiiPartOne = m_numLayersHCalThreeParts[0]; + unsigned int upperIndexLayerRadiiPartTwo = upperIndexLayerRadiiPartOne + m_numLayersHCalThreeParts[1]; + + // Part 1 + auto partOneRadii = calculateLayerRadii(0, upperIndexLayerRadiiPartOne); + radii.insert(radii.end(), partOneRadii.begin(), partOneRadii.end()); + + // Part 2 + auto partTwoRadii = calculateLayerRadii(upperIndexLayerRadiiPartOne, upperIndexLayerRadiiPartTwo); + radii.insert(radii.end(), partTwoRadii.begin(), partTwoRadii.end()); + + // Part 3 + auto partThreeRadii = calculateLayerRadii(upperIndexLayerRadiiPartTwo, m_layersRetrieved->size()); + radii.insert(radii.end(), partThreeRadii.begin(), partThreeRadii.end()); + + return radii; +} + +StatusCode CellPositionsHCalPhiThetaSegTool::finalize() { return AlgTool::finalize(); } diff --git a/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.h b/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.h new file mode 100644 index 00000000..bc51d110 --- /dev/null +++ b/RecFCCeeCalorimeter/src/components/CellPositionsHCalPhiThetaSegTool.h @@ -0,0 +1,83 @@ +#ifndef RECCALORIMETER_CellPositionsHCalPhiThetaSegTool_H +#define RECCALORIMETER_CellPositionsHCalPhiThetaSegTool_H + +// Gaudi +#include "GaudiKernel/AlgTool.h" +#include "GaudiKernel/ServiceHandle.h" + +// k4geo +#include "detectorCommon/DetUtils_k4geo.h" +#include "detectorSegmentations/FCCSWGridPhiTheta_k4geo.h" + +// k4FWCore +#include "k4FWCore/DataHandle.h" +#include "k4Interface/IGeoSvc.h" +#include "k4Interface/ICellPositionsTool.h" + +// DD4hep +#include "DD4hep/Readout.h" +#include "DD4hep/Volumes.h" +#include "DDSegmentation/Segmentation.h" +#include + +// ROOT +#include "TGeoManager.h" + +class IGeoSvc; +namespace DD4hep { +namespace DDSegmentation { +class Segmentation; +} +} + +/** @class CellPositionsHCalPhiThetaSegTool + * + * Tool to determine each Calorimeter cell position. + * + * For the FCCee HCal Barrel and EndCap with phi-theta segmentation, + * determined from the segmentation and the LayeredCalorimeterData extension. + * The LayeredCalorimeterData extension is part of the geometry description. + * + * @author Michaela Mlynarikova + */ + +class CellPositionsHCalPhiThetaSegTool : public AlgTool, virtual public ICellPositionsTool { +public: + CellPositionsHCalPhiThetaSegTool(const std::string& type, const std::string& name, const IInterface* parent); + ~CellPositionsHCalPhiThetaSegTool() = default; + + virtual StatusCode initialize() final; + + virtual StatusCode finalize() final; + + virtual void getPositions(const edm4hep::CalorimeterHitCollection& aCells, edm4hep::CalorimeterHitCollection& outputColl) final; + + virtual dd4hep::Position xyzPosition(const uint64_t& aCellId) const final; + + virtual int layerId(const uint64_t& aCellId) final; + + virtual std::vector calculateLayerRadii(unsigned int startIndex, unsigned int endIndex); + + virtual std::vector calculateLayerRadiiBarrel(); + + virtual std::vector calculateLayerRadiiEndcap(); + +private: + /// Pointer to the geometry service + SmartIF m_geoSvc; + /// Name of the hadronic calorimeter readout + Gaudi::Property m_readoutName{this, "readoutName", "HCalBarrelReadout"}; + /// Name of the hadronic calorimeter + Gaudi::Property m_detectorName{this, "detectorName", "HCalBarrel"}; + /// Theta-phi segmentation + dd4hep::DDSegmentation::FCCSWGridPhiTheta_k4geo* m_segmentation = nullptr; + /// Cellid decoder + dd4hep::DDSegmentation::BitFieldCoder* m_decoder = nullptr; + /// vector to store calculated layer radii + std::vector m_radii; + /// layers retrieved from the geometry + const std::vector* m_layersRetrieved = nullptr; + /// for the HCal Endcap, one needs to provide the number of layers in each cylinder + Gaudi::Property> m_numLayersHCalThreeParts{this, "numLayersHCalThreeParts", {6,9,22}}; +}; +#endif /* RECCALORIMETER_CellPositionsHCalPhiThetaSegTool_H */ \ No newline at end of file From 952997a4d3ea23f2ce66ccbe67f23afd24a4a9e6 Mon Sep 17 00:00:00 2001 From: Giovanni Marchiori <39376142+giovannimarchiori@users.noreply.github.com> Date: Thu, 29 Aug 2024 10:10:42 +0200 Subject: [PATCH 2/3] fix crash due to wrong treatment of pointers in ONNX-related code, and remove one photon-ID variable from decorations when not needed (#108) * fix crash due to wrong treatment of pointers (see https://github.com/HEP-FCC/k4RecCalorimeter/pull/104#discussion_r1735675543) * simplify code --- .../src/components/AugmentClustersFCCee.cpp | 4 ++-- .../src/components/CalibrateCaloClusters.cpp | 12 +++++++++++- RecFCCeeCalorimeter/src/components/PhotonIDTool.cpp | 11 ++++++++++- 3 files changed, 23 insertions(+), 4 deletions(-) diff --git a/RecFCCeeCalorimeter/src/components/AugmentClustersFCCee.cpp b/RecFCCeeCalorimeter/src/components/AugmentClustersFCCee.cpp index 8cf0900c..b64fc869 100644 --- a/RecFCCeeCalorimeter/src/components/AugmentClustersFCCee.cpp +++ b/RecFCCeeCalorimeter/src/components/AugmentClustersFCCee.cpp @@ -129,9 +129,9 @@ StatusCode AugmentClustersFCCee::initialize() showerShapeDecorations.push_back(Form("energy_fraction_%s_layer_%d", detector, layer)); showerShapeDecorations.push_back(Form("theta_%s_layer_%d", detector, layer)); showerShapeDecorations.push_back(Form("phi_%s_layer_%d", detector, layer)); - showerShapeDecorations.push_back(Form("maxcell_E_%s_layer_%d", detector, layer)); // pi0/photon shape var only calculated in EMB if (m_do_photon_shapeVar && m_systemIDs[k] == systemID_EMB) { + showerShapeDecorations.push_back(Form("maxcell_E_%s_layer_%d", detector, layer)); showerShapeDecorations.push_back(Form("width_theta_%s_layer_%d", detector, layer)); showerShapeDecorations.push_back(Form("width_module_%s_layer_%d", detector, layer)); showerShapeDecorations.push_back(Form("Ratio_E_max_2ndmax_%s_layer_%d", detector, layer)); @@ -651,7 +651,7 @@ StatusCode AugmentClustersFCCee::execute([[maybe_unused]] const EventContext &ev return StatusCode::FAILURE; } width_theta[layer+startPositionToFill] = (sumEnLayer[layer+startPositionToFill] > 0.) ? std::sqrt(std::fabs(w_theta2)) : 0. ; - } + } double w_module2 = module2_E_layer[layer+startPositionToFill] / sumEnLayer[layer+startPositionToFill] - std::pow(module_E_layer[layer+startPositionToFill] / sumEnLayer[layer+startPositionToFill], 2); // Very small but negative values caused by computational precision are allowed, // otherwise crash. diff --git a/RecFCCeeCalorimeter/src/components/CalibrateCaloClusters.cpp b/RecFCCeeCalorimeter/src/components/CalibrateCaloClusters.cpp index 4a7cefba..d48ba837 100644 --- a/RecFCCeeCalorimeter/src/components/CalibrateCaloClusters.cpp +++ b/RecFCCeeCalorimeter/src/components/CalibrateCaloClusters.cpp @@ -169,12 +169,22 @@ StatusCode CalibrateCaloClusters::finalize() { if (m_ortSession) delete m_ortSession; + if (m_ortEnv) delete m_ortEnv; + for (auto& name : m_input_names) { + delete name; + } + + for (auto& name : m_output_names) { + delete name; + } + return Gaudi::Algorithm::finalize(); } + edm4hep::ClusterCollection *CalibrateCaloClusters::initializeOutputClusters( const edm4hep::ClusterCollection *inClusters) const { @@ -271,7 +281,7 @@ StatusCode CalibrateCaloClusters::readCalibrationFile(const std::string &calibra #if ORT_API_VERSION < 13 m_output_names.emplace_back(AllocatedStringPtr(m_ortSession->GetOutputName(i, allocator), allocDeleter).release()); #else - m_output_names.emplace_back(m_ortSession->GetOutputNameAllocated(i, allocator).get()); + m_output_names.emplace_back(m_ortSession->GetOutputNameAllocated(i, allocator).release()); #endif m_output_shapes = m_ortSession->GetOutputTypeInfo(i).GetTensorTypeAndShapeInfo().GetShape(); debug() << "\t" << m_output_names.at(i) << " : "; diff --git a/RecFCCeeCalorimeter/src/components/PhotonIDTool.cpp b/RecFCCeeCalorimeter/src/components/PhotonIDTool.cpp index 85f191c7..8a713ccc 100644 --- a/RecFCCeeCalorimeter/src/components/PhotonIDTool.cpp +++ b/RecFCCeeCalorimeter/src/components/PhotonIDTool.cpp @@ -134,9 +134,18 @@ StatusCode PhotonIDTool::finalize() { if (m_ortSession) delete m_ortSession; + if (m_ortEnv) delete m_ortEnv; + for (auto& name : m_input_names) { + delete name; + } + + for (auto& name : m_output_names) { + delete name; + } + return Gaudi::Algorithm::finalize(); } @@ -341,7 +350,7 @@ StatusCode PhotonIDTool::readMVAFiles(const std::string& mvaInputsFileName, #if ORT_API_VERSION < 13 m_output_names.emplace_back(AllocatedStringPtr(m_ortSession->GetOutputName(i, allocator), allocDeleter).release()); #else - m_output_names.emplace_back(m_ortSession->GetOutputNameAllocated(i, allocator).get()); + m_output_names.emplace_back(m_ortSession->GetOutputNameAllocated(i, allocator).release()); #endif m_output_shapes = m_ortSession->GetOutputTypeInfo(i).GetTensorTypeAndShapeInfo().GetShape(); From 389cfd960e7a79708672db6a66829df42c54d9ff Mon Sep 17 00:00:00 2001 From: Giovanni Marchiori <39376142+giovannimarchiori@users.noreply.github.com> Date: Wed, 4 Sep 2024 15:11:08 +0200 Subject: [PATCH 3/3] move main ALLEGRO tests to ddsim+k4run (#109) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * move main ALLEGRO tests to ddsim+k4run * move module imports closer to when they are used (BrieucĂ…'s style) * use Zqq events for ALLEGRO test * trying to fix error in automatic tests * add crosstalk to test * implement comments by Juraj * Try one more suggestion by Juraj * check also that the download of the other files is successful --- RecFCCeeCalorimeter/CMakeLists.txt | 27 +- .../tests/options/ALLEGRO_o1_v03.sh | 66 ++ .../tests/options/ALLEGRO_o1_v03_digi_reco.py | 985 ++++++++++++++++++ 3 files changed, 1055 insertions(+), 23 deletions(-) create mode 100755 RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03.sh create mode 100644 RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03_digi_reco.py diff --git a/RecFCCeeCalorimeter/CMakeLists.txt b/RecFCCeeCalorimeter/CMakeLists.txt index 565fdd3c..69686b19 100644 --- a/RecFCCeeCalorimeter/CMakeLists.txt +++ b/RecFCCeeCalorimeter/CMakeLists.txt @@ -27,24 +27,11 @@ install(TARGETS k4RecFCCeeCalorimeterPlugins install(DIRECTORY ${CMAKE_CURRENT_LIST_DIR}/tests DESTINATION ${CMAKE_INSTALL_DATADIR}/${CMAKE_PROJECT_NAME}/RecFCCeeCalorimeter) -add_test(NAME FCCeeLAr_simulateForReco - COMMAND k4run RecFCCeeCalorimeter/tests/options/runCaloSim.py - WORKING_DIRECTORY ${PROJECT_SOURCE_DIR} -) -set_test_env(FCCeeLAr_simulateForReco) - -add_test(NAME FCCeeLAr_simRecoAllegroV3 - COMMAND k4run RecFCCeeCalorimeter/tests/options/run_thetamodulemerged.py - WORKING_DIRECTORY ${PROJECT_SOURCE_DIR} -) -set_test_env(FCCeeLAr_simRecoAllegroV3) - -add_test(NAME FCCeeLAr_slidingWindowClustering - COMMAND k4run RecFCCeeCalorimeter/tests/options/runFullCaloSystem_ReconstructionSW_noiseFromFile.py - WORKING_DIRECTORY ${PROJECT_SOURCE_DIR} +add_test(NAME ALLEGRO_o1_v03_sim_reco + COMMAND ${PROJECT_SOURCE_DIR}/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03.sh + WORKING_DIRECTORY ${PROJECT_SOURCE_DIR}/build/Testing/Temporary ) -set_test_env(FCCeeLAr_slidingWindowClustering) -set_tests_properties(FCCeeLAr_slidingWindowClustering PROPERTIES DEPENDS "FCCeeLAr_simulateForReco") +set_test_env(ALLEGRO_o1_v03_sim_reco) add_test(NAME FCCeeLAr_benchmarkCalibration COMMAND k4run RecFCCeeCalorimeter/tests/options/fcc_ee_caloBenchmarkCalibration.py @@ -63,9 +50,3 @@ add_test(NAME FCCeeLAr_benchmarkCorrection WORKING_DIRECTORY ${PROJECT_SOURCE_DIR} ) set_test_env(FCCeeLAr_benchmarkCorrection) - -add_test(NAME FCCeeLAr_runxtalk - COMMAND k4run RecCalorimeter/tests/options/runEcalBarrel_ReconstructionTopoClusters_crosstalk.py - WORKING_DIRECTORY ${PROJECT_SOURCE_DIR} -) -set_test_env(FCCeeLAr_runxtalk) diff --git a/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03.sh b/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03.sh new file mode 100755 index 00000000..9f5f1447 --- /dev/null +++ b/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03.sh @@ -0,0 +1,66 @@ +#!/bin/bash + +# Define the function for downloading files +download_file() { + local url="$1" + + # Attempt to download the file using wget + wget "$url" + + # Check the exit status of wget + if [ $? -ne 0 ]; then + # if wget failed, exit the script with status code 1 + echo "Download failed." + exit 1 + fi +} + +# set-up the Key4hep environment if not already set +if [[ -z "${KEY4HEP_STACK}" ]]; then + echo "Error: Key4hep environment not set" + return 1 +fi + +# download the events to reconstruct +if ! -test ./pythia_ee_z_qq_10evt.hepmc; then + echo "Downloading files needed for simulation" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/gen/pythia_ee_z_qq_10evt.hepmc" +fi + +# run the SIM step (for debug do not run it if files already present. Comment the if and fi lines for production) +# if ! test -f ALLEGRO_sim_ee_z_qq.root; then +echo "Performing the Geant4 simulation with ddsim" +ddsim --inputFiles pythia_ee_z_qq_10evt.hepmc --numberOfEvents -1 --outputFile ALLEGRO_sim_ee_z_qq.root --compactFile $K4GEO/FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml +# fi +#if ! test -f ALLEGRO_sim_e_barrel.root; then +#echo "Generating events and performing the Geant4 simulation with ddsim" +#ddsim --enableGun --gun.distribution uniform --gun.energy "10*GeV" --gun.particle e- --numberOfEvents 10 --outputFile ALLEGRO_sim_e_barrel.root --random.enableEventSeed --random.seed 42 --compactFile $K4GEO/FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml +#ddsim --enableGun --gun.distribution uniform --gun.energy "10*GeV" --gun.particle e- --numberOfEvents 10 --outputFile ALLEGRO_sim_e_endcap.root --random.enableEventSeed --random.seed 42 --compactFile $K4GEO/FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml +#ddsim --enableGun --gun.distribution uniform --gun.energy "10*GeV" --gun.particle pi- --numberOfEvents 10 --outputFile ALLEGRO_sim_pi_barrel.root --random.enableEventSeed --random.seed 42 --compactFile $K4GEO/FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml +#ddsim --enableGun --gun.distribution uniform --gun.energy "10*GeV" --gun.particle pi- --numberOfEvents 10 --outputFile ALLEGRO_sim_pi_endcap.root --random.enableEventSeed --random.seed 42 --compactFile $K4GEO/FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml +#fi + +retcode=$? +if [ $retcode -ne 0 ]; then + echo "Simulation failed" + exit $retcode +fi + +# get the files needed for calibration, noise, neighbor finding, etc +if ! test -f ./neighbours_map_ecalB_thetamodulemerged_hcalB_thetaphi.root; then # assumes that if the last file exists, all the other as well + echo "Downloading files needed for reconstruction" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/capacitances_ecalBarrelFCCee_theta.root" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/cellNoise_map_electronicsNoiseLevel_ecalB_thetamodulemerged.root" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/cellNoise_map_electronicsNoiseLevel_ecalB_thetamodulemerged_hcalB_thetaphi.root" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/elecNoise_ecalBarrelFCCee_theta.root" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/lgbm_calibration-CaloClusters.onnx" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/lgbm_calibration-CaloTopoClusters.onnx" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/neighbours_map_ecalB_thetamodulemerged.root" + download_file "https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/neighbours_map_ecalB_thetamodulemerged_hcalB_thetaphi.root" + + # add here the lines to get the files for the photon ID +fi + +# run the RECO step +SCRIPT_DIR=$( cd -- "$( dirname -- "${BASH_SOURCE[0]}" )" &> /dev/null && pwd ) # workaround to have ctests working +k4run $SCRIPT_DIR/ALLEGRO_o1_v03_digi_reco.py diff --git a/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03_digi_reco.py b/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03_digi_reco.py new file mode 100644 index 00000000..766131c4 --- /dev/null +++ b/RecFCCeeCalorimeter/tests/options/ALLEGRO_o1_v03_digi_reco.py @@ -0,0 +1,985 @@ +# +# COMMON IMPORTS +# + +# Logger +from Gaudi.Configuration import INFO # DEBUG, VERBOSE +# units and physical constants +from GaudiKernel.PhysicalConstants import pi + + +# +# SETTINGS +# + +# - general settings +# +inputfile = "ALLEGRO_sim_ee_z_qq.root" # input file produced with ddsim +Nevts = -1 # -1 means all events +addNoise = True # add noise or not to the cell energy +addCrosstalk = True # switch on/off the crosstalk +dumpGDML = False # create GDML file of detector model +runHCal = True # if false, it will produce only ECAL clusters. if true, it will also produce ECAL+HCAL clusters + +# - what to save in output file +# +# always drop unpositioned / uncalibrated cells, excepts for tests and debugging +# dropUncalibratedCells = True +# dropUnpositionedCells = True +dropUncalibratedCells = False +dropUnpositionedCells = False + +# for big productions, save significant space removing hits and cells +# however, hits and cluster cells might be wanted for small productions for detailed event displays +# cluster cells are not needed for the training of the MVA energy regression nor the photon ID since needed quantities are stored in cluster shapeParameters +# saveHits = False +# saveCells = False +# saveClusterCells = False +saveHits = True +saveCells = True +saveClusterCells = True + +# ECAL barrel parameters for digitisation +ecalBarrelSamplingFraction = [0.3800493723322256] * 1 + [0.13494147915064658] * 1 + [0.142866851721152] * 1 + [0.14839315921940666] * 1 + [0.15298362570665006] * 1 + [0.15709704561942747] * 1 + [0.16063717490147533] * 1 + [0.1641723795419055] * 1 + [0.16845490287689746] * 1 + [0.17111520115997653] * 1 + [0.1730605163148862] * 1 +ecalBarrelUpstreamParameters = [[0.028158491043365624, -1.564259408365951, -76.52312805346982, 0.7442903558010191, -34.894692961350195, -74.19340877431723]] +ecalBarrelDownstreamParameters = [[0.00010587711361028165, 0.0052371999097777355, 0.69906696456064, -0.9348243433360095, -0.0364714212117143, 8.360401126995626]] + +ecalBarrelLayers = len(ecalBarrelSamplingFraction) +resegmentECalBarrel = False + +# - parameters for clustering +# +doSWClustering = True +doTopoClustering = True + +# cluster energy corrections +# simple parametrisations of up/downstream losses for ECAL-only clusters +# not to be applied for ECAL+HCAL clustering +# superseded by MVA calibration, but turned on here for the purpose of testing that the code is not broken +applyUpDownstreamCorrections = True + +# BDT regression from total cluster energy and fraction of energy in each layer (after correction for sampling fraction) +# not to be applied (yet) for ECAL+HCAL clustering (MVA trained only on ECAL so far) +applyMVAClusterEnergyCalibration = True + +# calculate cluster energy and barycenter per layer and save it as extra parameters +addShapeParameters = True +ecalBarrelThetaWeights = [-1, 3.0, 3.0, 3.0, 4.25, 4.0, 4.0, 4.0, 4.0, 4.0, 4.0] # to be recalculated for V03, separately for topo and calo clusters... + +# run photon ID algorithm +# not run by default in production, but to be turned on here for the purpose of testing that the code is not broken +# currently off till we provide the onnx files +runPhotonIDTool = False + +# +# ALGORITHMS AND SERVICES SETUP +# + +# Input: load the output of the SIM step +from Configurables import k4DataSvc, PodioInput +podioevent = k4DataSvc('EventDataSvc') +podioevent.input = inputfile +input_reader = PodioInput('InputReader') + + +# Detector geometry +# prefix all xmls with path_to_detector +# if K4GEO is empty, this should use relative path to working directory +from Configurables import GeoSvc +import os +geoservice = GeoSvc("GeoSvc") +path_to_detector = os.environ.get("K4GEO", "") +detectors_to_use = [ + 'FCCee/ALLEGRO/compact/ALLEGRO_o1_v03/ALLEGRO_o1_v03.xml' +] +geoservice.detectors = [ + os.path.join(path_to_detector, _det) for _det in detectors_to_use +] +geoservice.OutputLevel = INFO + +# GDML dump of detector model +if dumpGDML: + from Configurables import GeoToGdmlDumpSvc + gdmldumpservice = GeoToGdmlDumpSvc("GeoToGdmlDumpSvc") + +# Digitisation (merging hits into cells, EM scale calibration via sampling fractions) + +# - ECAL readouts +ecalBarrelReadoutName = "ECalBarrelModuleThetaMerged" # barrel, original segmentation (baseline) +ecalBarrelReadoutName2 = "ECalBarrelModuleThetaMerged2" # barrel, after re-segmentation (for optimisation studies) +ecalEndcapReadoutName = "ECalEndcapTurbine" # endcap, turbine-like (baseline) +# - HCAL readouts +if runHCal: + hcalBarrelReadoutName = "HCalBarrelReadout" # barrel, original segmentation (row-phi) + hcalBarrelReadoutName2 = "BarHCal_Readout_phitheta" # barrel, groups together cells of different row within same theta slice + hcalEndcapReadoutName = "HCalEndcapReadout" # endcap, original segmentation +else: + hcalBarrelReadoutName = "" + hcalBarrelReadoutName2 = "" + hcalEndcapReadoutName = "" + +# - EM scale calibration (sampling fraction) +from Configurables import CalibrateInLayersTool +# * ECAL barrel +calibEcalBarrel = CalibrateInLayersTool("CalibrateECalBarrel", + samplingFraction=ecalBarrelSamplingFraction, + readoutName=ecalBarrelReadoutName, + layerFieldName="layer") +# * ECAL endcap +calibEcalEndcap = CalibrateInLayersTool("CalibrateECalEndcap", + samplingFraction=[0.16419] * 1 + [0.192898] * 1 + [0.18783] * 1 + [0.193203] * 1 + [0.193928] * 1 + [0.192286] * 1 + [0.199959] * 1 + [0.200153] * 1 + [0.212635] * 1 + [0.180345] * 1 + [0.18488] * 1 + [0.194762] * 1 + [0.197775] * 1 + [0.200504] * 1 + [0.205555] * 1 + [0.203601] * 1 + [0.210877] * 1 + [0.208376] * 1 + [0.216345] * 1 + [0.201452] * 1 + [0.202134] * 1 + [0.207566] * 1 + [0.208152] * 1 + [0.209889] * 1 + [0.211743] * 1 + [0.213188] * 1 + [0.215864] * 1 + [0.22972] * 1 + [0.192515] * 1 + [0.0103233] * 1, + readoutName=ecalEndcapReadoutName, + layerFieldName="layer") + +if runHCal: + from Configurables import CalibrateCaloHitsTool + # HCAL barrel + calibHCalBarrel = CalibrateCaloHitsTool( + "CalibrateHCalBarrel", invSamplingFraction="29.4202") + # HCAL endcap + calibHCalEndcap = CalibrateCaloHitsTool( + "CalibrateHCalEndcap", invSamplingFraction="29.4202") # FIXME: to be updated for ddsim + +from Configurables import CreateCaloCells +from Configurables import CreateCaloCellPositionsFCCee +from Configurables import CellPositionsECalBarrelModuleThetaSegTool +from Configurables import CellPositionsECalEndcapTurbineSegTool +if addCrosstalk: + from Configurables import ReadCaloCrosstalkMap + # read the crosstalk map + readCrosstalkMap = ReadCaloCrosstalkMap("ReadCrosstalkMap", + fileName="https://fccsw.web.cern.ch/fccsw/filesForSimDigiReco/ALLEGRO/ALLEGRO_o1_v03/xtalk_neighbours_map_ecalB_thetamodulemerged.root", + OutputLevel=INFO) + +# Create cells in ECal barrel (needed if one wants to apply cell calibration, +# which is not performed by ddsim) +# - merge hits into cells according to initial segmentation +ecalBarrelCellsName = "ECalBarrelCells" +createEcalBarrelCells = CreateCaloCells("CreateECalBarrelCells", + doCellCalibration=True, + calibTool=calibEcalBarrel, + crosstalksTool=readCrosstalkMap, + addCrosstalk=addCrosstalk, + addCellNoise=False, + filterCellNoise=False, + addPosition=True, + OutputLevel=INFO, + hits=ecalBarrelReadoutName, + cells=ecalBarrelCellsName) + +# - add to Ecal barrel cells the position information +# (good for physics, all coordinates set properly) +cellPositionEcalBarrelTool = CellPositionsECalBarrelModuleThetaSegTool( + "CellPositionsECalBarrel", + readoutName=ecalBarrelReadoutName, + OutputLevel=INFO +) +ecalBarrelPositionedCellsName = ecalBarrelReadoutName + "Positioned" +createEcalBarrelPositionedCells = CreateCaloCellPositionsFCCee( + "CreateECalBarrelPositionedCells", + OutputLevel=INFO +) +createEcalBarrelPositionedCells.positionsTool = cellPositionEcalBarrelTool +createEcalBarrelPositionedCells.hits.Path = ecalBarrelCellsName +createEcalBarrelPositionedCells.positionedHits.Path = ecalBarrelPositionedCellsName + +# - now, if we want to also save cells with coarser granularity: +if resegmentECalBarrel: + # 2. step - rewrite the cellId using the merged theta-module segmentation + # (merging several modules and severla theta readout cells). + # Add noise at this step if you derived the noise already assuming merged cells + # Step 2a: compute new cellID of cells based on new readout + # (merged module-theta segmentation with variable merging vs layer) + from Configurables import RedoSegmentation + resegmentEcalBarrelTool = RedoSegmentation("ReSegmentationEcal", + # old bitfield (readout) + oldReadoutName=ecalBarrelReadoutName, + # specify which fields are going to be altered (deleted/rewritten) + oldSegmentationIds=["module", "theta"], + # new bitfield (readout), with new segmentation (merged modules and theta cells) + newReadoutName=ecalBarrelReadoutName2, + OutputLevel=INFO, + debugPrint=200, + inhits=ecalBarrelCellsName, + outhits="ECalBarrelCellsMerged") + + # Step 2b: merge new cells with same cellID together + # do not apply cell calibration again since cells were already + # calibrated in Step 1 + ecalBarrelCellsName2 = "ECalBarrelCells2" + createEcalBarrelCells2 = CreateCaloCells("CreateECalBarrelCells2", + doCellCalibration=False, + addCellNoise=False, + filterCellNoise=False, + OutputLevel=INFO, + hits="ECalBarrelCellsMerged", + cells=ecalBarrelCellsName2) + + cellPositionEcalBarrelTool2 = CellPositionsECalBarrelModuleThetaSegTool( + "CellPositionsECalBarrel2", + readoutName=ecalBarrelReadoutName2, + OutputLevel=INFO + ) + ecalBarrelPositionedCellsName2 = ecalBarrelReadoutName2 + "Positioned" + createEcalBarrelPositionedCells2 = CreateCaloCellPositionsFCCee( + "CreateECalBarrelPositionedCells2", + OutputLevel=INFO + ) + createEcalBarrelPositionedCells2.positionsTool = cellPositionEcalBarrelTool2 + createEcalBarrelPositionedCells2.hits.Path = ecalBarrelCellsName2 + createEcalBarrelPositionedCells2.positionedHits.Path = ecalBarrelPositionedCellsName2 + + +# Create cells in ECal endcap (needed if one wants to apply cell calibration, +# which is not performed by ddsim) +ecalEndcapCellsName = "ECalEndcapCells" +createEcalEndcapCells = CreateCaloCells("CreateEcalEndcapCaloCells", + doCellCalibration=True, + calibTool=calibEcalEndcap, + addCellNoise=False, + filterCellNoise=False, + OutputLevel=INFO, + hits=ecalEndcapReadoutName, + cells=ecalEndcapCellsName) + +# Add to Ecal endcap cells the position information +# (good for physics, all coordinates set properly) +cellPositionEcalEndcapTool = CellPositionsECalEndcapTurbineSegTool( + "CellPositionsECalEndcap", + readoutName=ecalEndcapReadoutName, + OutputLevel=INFO +) +ecalEndcapPositionedCellsName = ecalEndcapReadoutName + "Positioned" +createEcalEndcapPositionedCells = CreateCaloCellPositionsFCCee( + "CreateECalEndcapPositionedCells", + OutputLevel=INFO +) +createEcalEndcapPositionedCells.positionsTool = cellPositionEcalEndcapTool +createEcalEndcapPositionedCells.hits.Path = ecalEndcapCellsName +createEcalEndcapPositionedCells.positionedHits.Path = ecalEndcapPositionedCellsName + + +if addNoise: + ecalBarrelNoisePath = "elecNoise_ecalBarrelFCCee_theta.root" + ecalBarrelNoiseRMSHistName = "h_elecNoise_fcc_" + from Configurables import NoiseCaloCellsVsThetaFromFileTool + noiseBarrel = NoiseCaloCellsVsThetaFromFileTool("NoiseBarrel", + cellPositionsTool=cellPositionEcalBarrelTool, + readoutName=ecalBarrelReadoutName, + noiseFileName=ecalBarrelNoisePath, + elecNoiseRMSHistoName=ecalBarrelNoiseRMSHistName, + setNoiseOffset=False, + activeFieldName="layer", + addPileup=False, + filterNoiseThreshold=0, + numRadialLayers=11, + scaleFactor=1 / 1000., # MeV to GeV + OutputLevel=INFO) + + # needs to be migrated! + # from Configurables import TubeLayerPhiEtaCaloTool + # barrelGeometry = TubeLayerPhiEtaCaloTool("EcalBarrelGeo", + # readoutName=ecalBarrelReadoutNamePhiEta, + # activeVolumeName="LAr_sensitive", + # activeFieldName="layer", + # activeVolumesNumber=12, + # fieldNames=["system"], + # fieldValues=[4]) + + # cells with noise not filtered + # createEcalBarrelCellsNoise = CreateCaloCells("CreateECalBarrelCellsNoise", + # doCellCalibration=False, + # addCellNoise=True, + # filterCellNoise=False, + # OutputLevel=INFO, + # hits="ECalBarrelCellsStep2", + # noiseTool=noiseBarrel, + # geometryTool=barrelGeometry, + # cells=EcalBarrelCellsName) + + # cells with noise filtered + # createEcalBarrelCellsNoise = CreateCaloCells("CreateECalBarrelCellsNoise_filtered", + # doCellCalibration=False, + # addCellNoise=True, + # filterCellNoise=True, + # OutputLevel=INFO, + # hits="ECalBarrelCellsStep2", + # noiseTool=noiseBarrel, + # geometryTool=barrelGeometry, + # cells=EcalBarrelCellsName) + + +if runHCal: + # Create cells in HCal barrel + # 1 - merge hits into cells with the default readout + hcalBarrelCellsName = "HCalBarrelCells" + createHCalBarrelCells = CreateCaloCells("CreateHCalBarrelCells", + doCellCalibration=True, + calibTool=calibHCalBarrel, + addCellNoise=False, + filterCellNoise=False, + addPosition=True, + hits=hcalBarrelReadoutName, + cells=hcalBarrelCellsName, + OutputLevel=INFO) + + # 2 - attach positions to the cells (cell positions needed for RedoSegmentation!) + from Configurables import CellPositionsHCalBarrelPhiThetaSegTool + cellPositionHCalBarrelTool = CellPositionsHCalBarrelPhiThetaSegTool( + "CellPositionsHCalBarrel", + readoutName=hcalBarrelReadoutName, + OutputLevel=INFO + ) + hcalBarrelPositionedCellsName = "HCalBarrelPositionedCells" + createHCalBarrelPositionedCells = CreateCaloCellPositionsFCCee( + "CreateHCalBarrelPositionedCells", + OutputLevel=INFO + ) + createHCalBarrelPositionedCells.positionsTool = cellPositionHCalBarrelTool + createHCalBarrelPositionedCells.hits.Path = hcalBarrelCellsName + createHCalBarrelPositionedCells.positionedHits.Path = hcalBarrelPositionedCellsName + + # 3 - compute new cellID of cells based on new readout - removing row information + # We use a RedoSegmentation. Using a RewriteBitField with removeIds=["row"], + # wont work because there are tiles with same layer/theta/phi but different row + # as a consequence there will be multiple cells with same cellID in the output collection + # and this will screw up the SW clustering + hcalBarrelCellsName2 = "HCalBarrelCells2" + + # first we create new hits with the readout without the row information + # and then merge them into new cells + from Configurables import RedoSegmentation + rewriteHCalBarrel = RedoSegmentation("ReSegmentationHCal", + # old bitfield (readout) + oldReadoutName=hcalBarrelReadoutName, + # specify which fields are going to be altered (deleted/rewritten) + oldSegmentationIds=["row", "theta", "phi"], + # new bitfield (readout), with new segmentation (theta-phi grid) + newReadoutName=hcalBarrelReadoutName2, + OutputLevel=INFO, + debugPrint=200, + inhits=hcalBarrelPositionedCellsName, + outhits="HCalBarrelCellsWithoutRow") + + createHCalBarrelCells2 = CreateCaloCells("CreateHCalBarrelCells2", + doCellCalibration=False, + addCellNoise=False, + filterCellNoise=False, + OutputLevel=INFO, + hits=rewriteHCalBarrel.outhits.Path, + cells=hcalBarrelCellsName2) + + # 4 - attach positions to the new cells + from Configurables import CellPositionsHCalBarrelPhiThetaSegTool + hcalBarrelPositionedCellsName2 = hcalBarrelReadoutName2 + "Positioned" + cellPositionHCalBarrelTool2 = CellPositionsHCalBarrelPhiThetaSegTool( + "CellPositionsHCalBarrel2", + readoutName=hcalBarrelReadoutName2, + OutputLevel=INFO + ) + createHCalBarrelPositionedCells2 = CreateCaloCellPositionsFCCee( + "CreateHCalBarrelPositionedCells2", + OutputLevel=INFO + ) + createHCalBarrelPositionedCells2.positionsTool = cellPositionHCalBarrelTool2 + createHCalBarrelPositionedCells2.hits.Path = hcalBarrelCellsName2 + createHCalBarrelPositionedCells2.positionedHits.Path = hcalBarrelPositionedCellsName2 + + # Create cells in HCal endcap + # createHCalEndcapCells = CreateCaloCells("CreateHCalEndcapCaloCells", + # doCellCalibration=True, + # calibTool=calibHCalEndcap, + # addCellNoise=False, + # filterCellNoise=False, + # OutputLevel=INFO) + # createHCalEndcapCells.hits.Path="HCalEndcapHits" + # createHCalEndcapCells.cells.Path="HCalEndcapCells" + +else: + hcalBarrelCellsName = "emptyCaloCells" + hcalBarrelPositionedCellsName = "emptyCaloCells" + hcalBarrelCellsName2 = "emptyCaloCells" + hcalBarrelPositionedCellsName2 = "emptyCaloCells" + cellPositionHCalBarrelTool = None + cellPositionHCalBarrelTool2 = None + +# Empty cells for parts of calorimeter not implemented yet +from Configurables import CreateEmptyCaloCellsCollection +createemptycells = CreateEmptyCaloCellsCollection("CreateEmptyCaloCells") +createemptycells.cells.Path = "emptyCaloCells" + +if doSWClustering: + + # Produce sliding window clusters + from Configurables import CaloTowerToolFCCee + from Configurables import CreateCaloClustersSlidingWindowFCCee + + # Clustering parameters + # - phi-theta window sizes + windT = 9 + windP = 17 + posT = 5 + posP = 11 + dupT = 7 + dupP = 13 + finT = 9 + finP = 17 + # - minimal energy to create a cluster in GeV (FCC-ee detectors have to reconstruct low energy particles) + threshold = 0.040 + + # ECAL-only clusters + ecalBarrelTowers = CaloTowerToolFCCee("CreateECalBarrelTowers", + deltaThetaTower=4 * 0.009817477 / 4, deltaPhiTower=2 * 2 * pi / 1536., + ecalBarrelReadoutName=ecalBarrelReadoutName, + ecalEndcapReadoutName="", + ecalFwdReadoutName="", + hcalBarrelReadoutName="", + hcalExtBarrelReadoutName="", + hcalEndcapReadoutName="", + hcalFwdReadoutName="", + OutputLevel=INFO) + ecalBarrelTowers.ecalBarrelCells.Path = ecalBarrelPositionedCellsName + ecalBarrelTowers.ecalEndcapCells.Path = "emptyCaloCells" + ecalBarrelTowers.ecalFwdCells.Path = "emptyCaloCells" + ecalBarrelTowers.hcalBarrelCells.Path = "emptyCaloCells" + ecalBarrelTowers.hcalExtBarrelCells.Path = "emptyCaloCells" + ecalBarrelTowers.hcalEndcapCells.Path = "emptyCaloCells" + ecalBarrelTowers.hcalFwdCells.Path = "emptyCaloCells" + + createECalBarrelClusters = CreateCaloClustersSlidingWindowFCCee("CreateECalBarrelClusters", + towerTool=ecalBarrelTowers, + nThetaWindow=windT, nPhiWindow=windP, + nThetaPosition=posT, nPhiPosition=posP, + nThetaDuplicates=dupT, nPhiDuplicates=dupP, + nThetaFinal=finT, nPhiFinal=finP, + energyThreshold=threshold, + energySharingCorrection=False, + attachCells=True, + OutputLevel=INFO + ) + createECalBarrelClusters.clusters.Path = "EMBCaloClusters" + createECalBarrelClusters.clusterCells.Path = "EMBCaloClusterCells" + + ecalEndcapTowers = CaloTowerToolFCCee("CreateECalEndcapTowers", + deltaThetaTower=4 * 0.009817477 / 4, deltaPhiTower=2 * 2 * pi / 1536., + ecalBarrelReadoutName="", + ecalEndcapReadoutName=ecalEndcapReadoutName, + ecalFwdReadoutName="", + hcalBarrelReadoutName="", + hcalExtBarrelReadoutName="", + hcalEndcapReadoutName="", + hcalFwdReadoutName="", + OutputLevel=INFO) + ecalEndcapTowers.ecalBarrelCells.Path = "emptyCaloCells" + ecalEndcapTowers.ecalEndcapCells.Path = ecalEndcapPositionedCellsName + ecalEndcapTowers.ecalFwdCells.Path = "emptyCaloCells" + ecalEndcapTowers.hcalBarrelCells.Path = "emptyCaloCells" + ecalEndcapTowers.hcalExtBarrelCells.Path = "emptyCaloCells" + ecalEndcapTowers.hcalEndcapCells.Path = "emptyCaloCells" + ecalEndcapTowers.hcalFwdCells.Path = "emptyCaloCells" + + createECalEndcapClusters = CreateCaloClustersSlidingWindowFCCee("CreateECalEndcapClusters", + towerTool=ecalEndcapTowers, + nThetaWindow=windT, nPhiWindow=windP, + nThetaPosition=posT, nPhiPosition=posP, + nThetaDuplicates=dupT, nPhiDuplicates=dupP, + nThetaFinal=finT, nPhiFinal=finP, + energyThreshold=threshold, + energySharingCorrection=False, + attachCells=True, + OutputLevel=INFO + ) + createECalEndcapClusters.clusters.Path = "EMECCaloClusters" + createECalEndcapClusters.clusterCells.Path = "EMECCaloClusterCells" + + if applyUpDownstreamCorrections: + from Configurables import CorrectCaloClusters + correctECalBarrelClusters = CorrectCaloClusters("CorrectECalBarrelClusters", + inClusters=createECalBarrelClusters.clusters.Path, + outClusters="Corrected" + createECalBarrelClusters.clusters.Path, + systemIDs=[4], + numLayers=[ecalBarrelLayers], + firstLayerIDs=[0], + lastLayerIDs=[ecalBarrelLayers - 1], + readoutNames=[ecalBarrelReadoutName], + upstreamParameters=ecalBarrelUpstreamParameters, + upstreamFormulas=[ + ['[0]+[1]/(x-[2])', '[0]+[1]/(x-[2])']], + downstreamParameters=ecalBarrelDownstreamParameters, + downstreamFormulas=[ + ['[0]+[1]*x', '[0]+[1]/sqrt(x)', '[0]+[1]/x']], + OutputLevel=INFO + ) + + if addShapeParameters: + from Configurables import AugmentClustersFCCee + augmentECalBarrelClusters = AugmentClustersFCCee("AugmentECalBarrelClusters", + inClusters=createECalBarrelClusters.clusters.Path, + outClusters="Augmented" + createECalBarrelClusters.clusters.Path, + systemIDs=[4], + systemNames=["EMB"], + numLayers=[ecalBarrelLayers], + readoutNames=[ecalBarrelReadoutName], + layerFieldNames=["layer"], + thetaRecalcWeights=[ecalBarrelThetaWeights], + do_photon_shapeVar=runPhotonIDTool, + OutputLevel=INFO + ) + + if applyMVAClusterEnergyCalibration: + inClusters = "" + if addShapeParameters: + inClusters = augmentECalBarrelClusters.outClusters.Path + else: + inClusters = createECalBarrelClusters.clusters.Path + + from Configurables import CalibrateCaloClusters + calibrateECalBarrelClusters = CalibrateCaloClusters("calibrateECalBarrelClusters", + inClusters=inClusters, + outClusters="Calibrated" + createECalBarrelClusters.clusters.Path, + systemIDs=[4], + systemNames=["EMB"], + numLayers=[ecalBarrelLayers], + firstLayerIDs=[0], + readoutNames=[ + ecalBarrelReadoutName], + layerFieldNames=["layer"], + calibrationFile="lgbm_calibration-CaloClusters.onnx", + OutputLevel=INFO + ) + + if runPhotonIDTool: + if not addShapeParameters: + print("Photon ID tool cannot be run if shower shape parameters are not calculated") + runPhotonIDTool = False + else: + inClusters = "" + if applyMVAClusterEnergyCalibration: + inClusters = calibrateECalBarrelClusters.outClusters.Path + else: + inClusters = augmentECalBarrelClusters.outClusters.Path + + from Configurables import PhotonIDTool + photonIDECalBarrelClusters = PhotonIDTool("photonIDECalBarrelClusters", + inClusters=inClusters, + outClusters="PhotonID" + inClusters, + mvaModelFile="bdt-photonid-weights-CaloClusters.onnx", + mvaInputsFile="bdt-photonid-inputs-CaloClusters.json", + OutputLevel=INFO + ) + + # ECAL + HCAL clusters + if runHCal: + towers = CaloTowerToolFCCee("towers", + deltaThetaTower=4 * 0.009817477 / 4, deltaPhiTower=2 * 2 * pi / 1536., + ecalBarrelReadoutName=ecalBarrelReadoutName, + ecalEndcapReadoutName=ecalEndcapReadoutName, + ecalFwdReadoutName="", + hcalBarrelReadoutName=hcalBarrelReadoutName2, + hcalExtBarrelReadoutName="", + hcalEndcapReadoutName="", + hcalFwdReadoutName="", + OutputLevel=INFO) + towers.ecalBarrelCells.Path = ecalBarrelPositionedCellsName + towers.ecalEndcapCells.Path = ecalEndcapPositionedCellsName + towers.ecalFwdCells.Path = "emptyCaloCells" + towers.hcalBarrelCells.Path = hcalBarrelPositionedCellsName2 + towers.hcalExtBarrelCells.Path = "emptyCaloCells" + towers.hcalEndcapCells.Path = "emptyCaloCells" + towers.hcalFwdCells.Path = "emptyCaloCells" + + createClusters = CreateCaloClustersSlidingWindowFCCee("CreateCaloClusters", + towerTool=towers, + nThetaWindow=windT, nPhiWindow=windP, + nThetaPosition=posT, nPhiPosition=posP, + nThetaDuplicates=dupT, nPhiDuplicates=dupP, + nThetaFinal=finT, nPhiFinal=finP, + energyThreshold=threshold, + energySharingCorrection=False, + attachCells=True, + OutputLevel=INFO + ) + createClusters.clusters.Path = "CaloClusters" + createClusters.clusterCells.Path = "CaloClusterCells" + + # add here E+H cluster calibration tool or anything else for E+H clusters + + +if doTopoClustering: + + # Produce topoclusters + from Configurables import CaloTopoClusterInputTool + from Configurables import TopoCaloNeighbours + from Configurables import TopoCaloNoisyCells + from Configurables import CaloTopoClusterFCCee + + # ECAL only + createECalBarrelTopoInput = CaloTopoClusterInputTool("CreateECalBarrelTopoInput", + ecalBarrelReadoutName=ecalBarrelReadoutName, + ecalEndcapReadoutName="", + ecalFwdReadoutName="", + hcalBarrelReadoutName="", + hcalExtBarrelReadoutName="", + hcalEndcapReadoutName="", + hcalFwdReadoutName="", + OutputLevel=INFO) + + createECalBarrelTopoInput.ecalBarrelCells.Path = ecalBarrelPositionedCellsName + createECalBarrelTopoInput.ecalEndcapCells.Path = "emptyCaloCells" + createECalBarrelTopoInput.ecalFwdCells.Path = "emptyCaloCells" + createECalBarrelTopoInput.hcalBarrelCells.Path = "emptyCaloCells" + createECalBarrelTopoInput.hcalExtBarrelCells.Path = "emptyCaloCells" + createECalBarrelTopoInput.hcalEndcapCells.Path = "emptyCaloCells" + createECalBarrelTopoInput.hcalFwdCells.Path = "emptyCaloCells" + + ecalBarrelNeighboursMap = "neighbours_map_ecalB_thetamodulemerged.root" + ecalBarrelNoiseMap = "cellNoise_map_electronicsNoiseLevel_ecalB_thetamodulemerged.root" + + readECalBarrelNeighboursMap = TopoCaloNeighbours("ReadECalBarrelNeighboursMap", + fileName=ecalBarrelNeighboursMap, + OutputLevel=INFO) + + # Noise levels per cell + readECalBarrelNoisyCellsMap = TopoCaloNoisyCells("ReadECalBarrelNoisyCellsMap", + fileName=ecalBarrelNoiseMap, + OutputLevel=INFO) + + createECalBarrelTopoClusters = CaloTopoClusterFCCee("CreateECalBarrelTopoClusters", + TopoClusterInput=createECalBarrelTopoInput, + # expects neighbours map from cellid->vec < neighbourIds > + neigboursTool=readECalBarrelNeighboursMap, + # tool to get noise level per cellid + noiseTool=readECalBarrelNoisyCellsMap, + # cell positions tools for all sub - systems + positionsECalBarrelTool=cellPositionEcalBarrelTool, + # positionsHCalBarrelTool=cellPositionHCalBarrelTool2, + # positionsHCalBarrelNoSegTool=cellPositionHCalBarrelNoSegTool, + # positionsHCalExtBarrelTool=cellPositionHCalExtBarrelTool, + # positionsHCalExtBarrelTool = HCalExtBcells, + # positionsEMECTool = EMECcells, + # positionsHECTool = HECcells, + # positionsEMFwdTool = ECalFwdcells, + # positionsHFwdTool = HCalFwdcells, + noSegmentationHCal=False, + seedSigma=4, + neighbourSigma=2, + lastNeighbourSigma=0, + OutputLevel=INFO) + createECalBarrelTopoClusters.clusters.Path = "EMBCaloTopoClusters" + createECalBarrelTopoClusters.clusterCells.Path = "EMBCaloTopoClusterCells" + + # no topoclusters for ECAL endcap yet: no noise and neighbour maps provided + + if applyUpDownstreamCorrections: + from Configurables import CorrectCaloClusters + correctECalBarrelTopoClusters = CorrectCaloClusters( + "CorrectECalBarrelTopoClusters", + inClusters=createECalBarrelTopoClusters.clusters.Path, + outClusters="Corrected" + createECalBarrelTopoClusters.clusters.Path, + systemIDs=[4], + numLayers=[ecalBarrelLayers], + firstLayerIDs=[0], + lastLayerIDs=[ecalBarrelLayers - 1], + readoutNames=[ecalBarrelReadoutName], + # do not split the following line or it will break scripts that update the values of the corrections + upstreamParameters=ecalBarrelUpstreamParameters, + upstreamFormulas=[['[0]+[1]/(x-[2])', '[0]+[1]/(x-[2])']], + # do not split the following line or it will break scripts that update the values of the corrections + downstreamParameters=ecalBarrelDownstreamParameters, + downstreamFormulas=[['[0]+[1]*x', '[0]+[1]/sqrt(x)', '[0]+[1]/x']], + OutputLevel=INFO + ) + + if addShapeParameters: + from Configurables import AugmentClustersFCCee + augmentECalBarrelTopoClusters = AugmentClustersFCCee("augmentECalBarrelTopoClusters", + inClusters=createECalBarrelTopoClusters.clusters.Path, + outClusters="Augmented" + createECalBarrelTopoClusters.clusters.Path, + systemIDs=[4], + systemNames=["EMB"], + numLayers=[ecalBarrelLayers], + readoutNames=[ecalBarrelReadoutName], + layerFieldNames=["layer"], + thetaRecalcWeights=[ecalBarrelThetaWeights], + do_photon_shapeVar=runPhotonIDTool, + OutputLevel=INFO) + + if applyMVAClusterEnergyCalibration: + inClusters = "" + if addShapeParameters: + inClusters = "Augmented" + createECalBarrelTopoClusters.clusters.Path + else: + inClusters = createECalBarrelTopoClusters.clusters.Path + + from Configurables import CalibrateCaloClusters + calibrateECalBarrelTopoClusters = CalibrateCaloClusters("calibrateECalBarrelTopoClusters", + inClusters=inClusters, + outClusters="Calibrated" + createECalBarrelTopoClusters.clusters.Path, + systemIDs=[4], + systemNames=["EMB"], + numLayers=[ecalBarrelLayers], + firstLayerIDs=[0], + readoutNames=[ + ecalBarrelReadoutName], + layerFieldNames=["layer"], + calibrationFile="lgbm_calibration-CaloTopoClusters.onnx", + OutputLevel=INFO + ) + + if runPhotonIDTool: + if not addShapeParameters: + print("Photon ID tool cannot be run if shower shape parameters are not calculated") + runPhotonIDTool = False + else: + inClusters = "" + if applyMVAClusterEnergyCalibration: + inClusters = calibrateECalBarrelTopoClusters.outClusters.Path + else: + inClusters = augmentECalBarrelTopoClusters.outClusters.Path + + from Configurables import PhotonIDTool + photonIDECalBarrelTopoClusters = PhotonIDTool("photonIDECalBarrelTopoClusters", + inClusters=inClusters, + outClusters="PhotonID" + inClusters, + mvaModelFile="bdt-photonid-weights-CaloTopoClusters.onnx", + mvaInputsFile="bdt-photonid-inputs-CaloTopoClusters.json", + OutputLevel=INFO) + + # ECAL + HCAL + if runHCal: + createTopoInput = CaloTopoClusterInputTool("CreateTopoInput", + ecalBarrelReadoutName=ecalBarrelReadoutName, + ecalEndcapReadoutName="", + ecalFwdReadoutName="", + hcalBarrelReadoutName=hcalBarrelReadoutName2, + hcalExtBarrelReadoutName="", + hcalEndcapReadoutName="", + hcalFwdReadoutName="", + OutputLevel=INFO) + + createTopoInput.ecalBarrelCells.Path = ecalBarrelPositionedCellsName + createTopoInput.ecalEndcapCells.Path = "emptyCaloCells" + createTopoInput.ecalFwdCells.Path = "emptyCaloCells" + createTopoInput.hcalBarrelCells.Path = hcalBarrelPositionedCellsName2 + createTopoInput.hcalExtBarrelCells.Path = "emptyCaloCells" + createTopoInput.hcalEndcapCells.Path = "emptyCaloCells" + createTopoInput.hcalFwdCells.Path = "emptyCaloCells" + + cellPositionHCalBarrelNoSegTool = None + cellPositionHCalExtBarrelTool = None + + neighboursMap = "neighbours_map_ecalB_thetamodulemerged_hcalB_thetaphi.root" + noiseMap = "cellNoise_map_electronicsNoiseLevel_ecalB_thetamodulemerged_hcalB_thetaphi.root" + + readNeighboursMap = TopoCaloNeighbours("ReadNeighboursMap", + fileName=neighboursMap, + OutputLevel=INFO) + + # Noise levels per cell + readNoisyCellsMap = TopoCaloNoisyCells("ReadNoisyCellsMap", + fileName=noiseMap, + OutputLevel=INFO) + + createTopoClusters = CaloTopoClusterFCCee("CreateTopoClusters", + TopoClusterInput=createTopoInput, + # expects neighbours map from cellid->vec < neighbourIds > + neigboursTool=readNeighboursMap, + # tool to get noise level per cellid + noiseTool=readNoisyCellsMap, + # cell positions tools for all sub - systems + positionsECalBarrelTool=cellPositionEcalBarrelTool, + positionsHCalBarrelTool=cellPositionHCalBarrelTool2, + # positionsHCalBarrelNoSegTool=cellPositionHCalBarrelNoSegTool, + # positionsHCalExtBarrelTool=cellPositionHCalExtBarrelTool, + # positionsHCalExtBarrelTool = HCalExtBcells, + # positionsEMECTool = EMECcells, + # positionsHECTool = HECcells, + # positionsEMFwdTool = ECalFwdcells, + # positionsHFwdTool = HCalFwdcells, + noSegmentationHCal=False, + seedSigma=4, + neighbourSigma=2, + lastNeighbourSigma=0, + OutputLevel=INFO) + createTopoClusters.clusters.Path = "CaloTopoClusters" + createTopoClusters.clusterCells.Path = "CaloTopoClusterCells" + +# Output +from Configurables import PodioOutput +out = PodioOutput("out", + OutputLevel=INFO) +out.filename = "ALLEGRO_sim_digi_reco.root" + +out.outputCommands = ["keep *", + "drop emptyCaloCells"] + +# drop the uncalibrated cells +if dropUncalibratedCells: + out.outputCommands.append("drop %s" % ecalBarrelReadoutName) + out.outputCommands.append("drop %s" % ecalBarrelReadoutName2) + out.outputCommands.append("drop %s" % ecalEndcapReadoutName) + if runHCal: + out.outputCommands.append("drop %s" % hcalBarrelReadoutName) + # out.outputCommands.append("drop %s" % hcalEndcapReadoutName) + +# drop the unpositioned ECal and HCal barrel and endcap cells +if dropUnpositionedCells: + if runHCal: + out.outputCommands += ["drop %s" % ecalBarrelCellsName, + "drop %s" % ecalEndcapCellsName, + "drop %s" % hcalBarrelCellsName, + "drop %s" % hcalBarrelPositionedCellsName, + "drop %s" % hcalBarrelCellsName2, + "drop %s" % rewriteHCalBarrel.outhits.Path] + + else: + out.outputCommands += ["drop HCal*", + "drop %s" % ecalBarrelCellsName, + "drop %s" % ecalEndcapCellsName] + + # drop the intermediate ecal barrel cells in case of a resegmentation + if resegmentECalBarrel: + out.outputCommands.append("drop ECalBarrelCellsMerged") + out.outputCommands.append("drop %s" % ecalBarrelCellsName2) + +# drop lumi, vertex, DCH, Muons (unless want to keep for event display) +out.outputCommands.append("drop Lumi*") +# out.outputCommands.append("drop Vertex*") +# out.outputCommands.append("drop DriftChamber_simHits*") +out.outputCommands.append("drop MuonTagger*") + +# drop hits/positioned cells/cluster cells if desired +if not saveHits: + out.outputCommands.append("drop *%sContributions" % ecalBarrelReadoutName) + out.outputCommands.append("drop *%sContributions" % ecalBarrelReadoutName2) + out.outputCommands.append("drop *%sContributions" % ecalEndcapReadoutName) +if not saveCells: + out.outputCommands.append("drop %s" % ecalBarrelPositionedCellsName) + out.outputCommands.append("drop %s" % ecalEndcapPositionedCellsName) + if resegmentECalBarrel: + out.outputCommands.append("drop %s" % ecalBarrelPositionedCellsName2) + if runHCal: + out.outputCommands.append("drop %s" % hcalBarrelPositionedCellsName2) +if not saveClusterCells: + out.outputCommands.append("drop Calo*ClusterCells*") + +# if we decorate the clusters, we can drop the non-decorated ones +# commented in tests, for debugging +# if addShapeParameters: +# out.outputCommands.append("drop %s" % augmentECalBarrelClusters.inClusters) + +# CPU information +from Configurables import AuditorSvc, ChronoAuditor +chra = ChronoAuditor() +audsvc = AuditorSvc() +audsvc.Auditors = [chra] +out.AuditExecute = True + +# Configure list of external services +ExtSvc = [geoservice, podioevent, audsvc] +if dumpGDML: + ExtSvc += [gdmldumpservice] + +# Setup alg sequence +TopAlg = [ + input_reader, + createEcalBarrelCells, + createEcalBarrelPositionedCells, + createEcalEndcapCells, + createEcalEndcapPositionedCells, +] +createEcalBarrelCells.AuditExecute = True +createEcalBarrelPositionedCells.AuditExecute = True +createEcalEndcapCells.AuditExecute = True +createEcalEndcapPositionedCells.AuditExecute = True + +if resegmentECalBarrel: + TopAlg += [ + resegmentEcalBarrelTool, + createEcalBarrelCells2, + createEcalBarrelPositionedCells2, + ] + resegmentEcalBarrelTool.AuditExecute = True + createEcalBarrelCells2.AuditExecute = True + createEcalBarrelPositionedCells2.AuditExecute = True + +if runHCal: + TopAlg += [ + createHCalBarrelCells, + createHCalBarrelPositionedCells, + rewriteHCalBarrel, + createHCalBarrelCells2, + createHCalBarrelPositionedCells2, + # createHCalEndcapCells + ] + createHCalBarrelCells.AuditExecute = True + createHCalBarrelPositionedCells.AuditExecute = True + rewriteHCalBarrel.AuditExecute = True + createHCalBarrelCells2.AuditExecute = True + createHCalBarrelPositionedCells2.AuditExecute = True + +if doSWClustering or doTopoClustering: + TopAlg += [createemptycells] + createemptycells.AuditExecute = True + + if doSWClustering: + TopAlg += [createECalBarrelClusters, createECalEndcapClusters] + createECalBarrelClusters.AuditExecute = True + createECalEndcapClusters.AuditExecute = True + + if applyUpDownstreamCorrections: + TopAlg += [correctECalBarrelClusters] + correctECalBarrelClusters.AuditExecute = True + + if addShapeParameters: + TopAlg += [augmentECalBarrelClusters] + augmentECalBarrelClusters.AuditExecute = True + + if applyMVAClusterEnergyCalibration: + TopAlg += [calibrateECalBarrelClusters] + calibrateECalBarrelClusters.AuditExecute = True + + if runPhotonIDTool: + TopAlg += [photonIDECalBarrelClusters] + photonIDECalBarrelClusters.AuditExecute = True + + if runHCal: + TopAlg += [createClusters] + createClusters.AuditExecute = True + + if doTopoClustering: + TopAlg += [createECalBarrelTopoClusters] + createECalBarrelTopoClusters.AuditExecute = True + + if applyUpDownstreamCorrections: + TopAlg += [correctECalBarrelTopoClusters] + correctECalBarrelTopoClusters.AuditExecute = True + + if addShapeParameters: + TopAlg += [augmentECalBarrelTopoClusters] + augmentECalBarrelTopoClusters.AuditExecute = True + + if applyMVAClusterEnergyCalibration: + TopAlg += [calibrateECalBarrelTopoClusters] + calibrateECalBarrelTopoClusters.AuditExecute = True + + if runPhotonIDTool: + TopAlg += [photonIDECalBarrelTopoClusters] + photonIDECalBarrelTopoClusters.AuditExecute = True + + if runHCal: + TopAlg += [createTopoClusters] + createTopoClusters.AuditExecute = True + +TopAlg += [ + out +] + +from Configurables import ApplicationMgr +ApplicationMgr( + TopAlg=TopAlg, + EvtSel='NONE', + EvtMax=Nevts, + ExtSvc=ExtSvc, + StopOnSignal=True, +)