Skip to content

Commit

Permalink
Merge branch 'HEP-FCC:main' into main
Browse files Browse the repository at this point in the history
  • Loading branch information
zwu0922 authored Sep 4, 2024
2 parents 1df17ba + 389cfd9 commit de26696
Show file tree
Hide file tree
Showing 8 changed files with 1,385 additions and 21 deletions.
21 changes: 4 additions & 17 deletions RecFCCeeCalorimeter/CMakeLists.txt
Original file line number Diff line number Diff line change
Expand Up @@ -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
Expand Down
4 changes: 2 additions & 2 deletions RecFCCeeCalorimeter/src/components/AugmentClustersFCCee.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -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));
Expand Down Expand Up @@ -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.
Expand Down
12 changes: 11 additions & 1 deletion RecFCCeeCalorimeter/src/components/CalibrateCaloClusters.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -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
{
Expand Down Expand Up @@ -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) << " : ";
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,224 @@
#include "CellPositionsHCalPhiThetaSegTool.h"

#include "edm4hep/CalorimeterHitCollection.h"
#include <DDRec/DetectorData.h>

using dd4hep::DetElement;

DECLARE_COMPONENT(CellPositionsHCalPhiThetaSegTool)

CellPositionsHCalPhiThetaSegTool::CellPositionsHCalPhiThetaSegTool(
const std::string &type,
const std::string &name,
const IInterface *parent)
: AlgTool(type, name, parent)
{
declareInterface<ICellPositionsTool>(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<dd4hep::DDSegmentation::FCCSWGridPhiTheta_k4geo *>(
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<std::string> 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<dd4hep::rec::LayeredCalorimeterData>();
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; i<m_numLayersHCalThreeParts.size(); i++)
{
numLayersProvided += m_numLayersHCalThreeParts[i];
}

if (m_layersRetrieved->size() != 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<double> CellPositionsHCalPhiThetaSegTool::calculateLayerRadii(unsigned int startIndex, unsigned int endIndex) {
std::vector<double> 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<double> 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<double> CellPositionsHCalPhiThetaSegTool::calculateLayerRadiiEndcap() {
std::vector<double> 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(); }
Original file line number Diff line number Diff line change
@@ -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 <DDRec/DetectorData.h>

// 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<double> calculateLayerRadii(unsigned int startIndex, unsigned int endIndex);

virtual std::vector<double> calculateLayerRadiiBarrel();

virtual std::vector<double> calculateLayerRadiiEndcap();

private:
/// Pointer to the geometry service
SmartIF<IGeoSvc> m_geoSvc;
/// Name of the hadronic calorimeter readout
Gaudi::Property<std::string> m_readoutName{this, "readoutName", "HCalBarrelReadout"};
/// Name of the hadronic calorimeter
Gaudi::Property<std::string> 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<double> m_radii;
/// layers retrieved from the geometry
const std::vector<dd4hep::rec::LayeredCalorimeterStruct::Layer>* m_layersRetrieved = nullptr;
/// for the HCal Endcap, one needs to provide the number of layers in each cylinder
Gaudi::Property<std::vector<int>> m_numLayersHCalThreeParts{this, "numLayersHCalThreeParts", {6,9,22}};
};
#endif /* RECCALORIMETER_CellPositionsHCalPhiThetaSegTool_H */
Loading

0 comments on commit de26696

Please sign in to comment.