Project
Loading...
Searching...
No Matches
ClusterFinderOriginalSpec.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
16
18
19#include <iostream>
20#include <fstream>
21#include <chrono>
22#include <vector>
23#include <stdexcept>
24#include <string>
25
26#include <gsl/span>
27
32#include "Framework/Lifetime.h"
33#include "Framework/Output.h"
34#include "Framework/Task.h"
35#include "Framework/Logger.h"
36
40#include "MCHBase/Error.h"
41#include "MCHBase/ErrorMap.h"
42#include "MCHBase/PreCluster.h"
45
46namespace o2
47{
48namespace mch
49{
50
51using namespace std;
52using namespace o2::framework;
53
55{
56 public:
57 //_________________________________________________________________________________________________
59 {
61 LOG(info) << "initializing cluster finder";
62
63 auto config = ic.options().get<std::string>("mch-config");
64 if (!config.empty()) {
65 o2::conf::ConfigurableParam::updateFromFile(config, "MCHClustering", true);
66 }
67 bool run2Config = ic.options().get<bool>("run2-config");
68 mClusterFinder.init(run2Config);
69
70 mAttachInitalPrecluster = ic.options().get<bool>("attach-initial-precluster");
71
73 ic.services().get<CallbackService>().set<CallbackService::Id::Stop>([this]() {
74 LOG(info) << "cluster finder duration = " << mTimeClusterFinder.count() << " s";
75 mErrorMap.forEach([](Error error) {
76 LOGP(warning, fmt::runtime(error.asString()));
77 });
78 this->mClusterFinder.deinit();
79 });
80 }
81
82 //_________________________________________________________________________________________________
84 {
86
87 // get the input preclusters and associated digits
88 auto preClusterROFs = pc.inputs().get<gsl::span<ROFRecord>>("preclusterrofs");
89 auto preClusters = pc.inputs().get<gsl::span<PreCluster>>("preclusters");
90 auto digits = pc.inputs().get<gsl::span<Digit>>("digits");
91
92 // create the output messages for clusters and attached digits
93 auto& clusterROFs = pc.outputs().make<std::vector<ROFRecord>>(OutputRef{"clusterrofs"});
94 auto& clusters = pc.outputs().make<std::vector<Cluster>>(OutputRef{"clusters"});
95 auto& usedDigits = pc.outputs().make<std::vector<Digit>>(OutputRef{"clusterdigits"});
96
97 clusterROFs.reserve(preClusterROFs.size());
98 auto& errorMap = mClusterFinder.getErrorMap();
99 errorMap.clear();
100 for (const auto& preClusterROF : preClusterROFs) {
101
102 // prepare to clusterize the current ROF
103 auto clusterOffset = clusters.size();
104 mClusterFinder.reset();
105
106 for (const auto& preCluster : preClusters.subspan(preClusterROF.getFirstIdx(), preClusterROF.getNEntries())) {
107
108 auto preclusterDigits = digits.subspan(preCluster.firstDigit, preCluster.nDigits);
109 auto firstClusterIdx = mClusterFinder.getClusters().size();
110
111 // clusterize the current precluster
112 auto tStart = std::chrono::high_resolution_clock::now();
113 mClusterFinder.findClusters(preclusterDigits);
114 auto tEnd = std::chrono::high_resolution_clock::now();
115 mTimeClusterFinder += tEnd - tStart;
116
117 if (mAttachInitalPrecluster) {
118 // store the new clusters and associate them to all the digits of the precluster
119 writeClusters(preclusterDigits, firstClusterIdx, clusters, usedDigits);
120 }
121 }
122
123 if (!mAttachInitalPrecluster) {
124 // store all the clusters of the current ROF and the associated digits actually used in the clustering
125 writeClusters(clusters, usedDigits);
126 }
127
128 // create the cluster ROF
129 clusterROFs.emplace_back(preClusterROF.getBCData(), clusterOffset, clusters.size() - clusterOffset,
130 preClusterROF.getBCWidth());
131 }
132
133 // create the output message for clustering errors
134 auto& clusterErrors = pc.outputs().make<std::vector<Error>>(OutputRef{"clustererrors"});
135 errorMap.forEach([&clusterErrors](Error error) {
136 clusterErrors.emplace_back(error);
137 });
138 mErrorMap.add(errorMap);
139
140 LOGP(info, "Found {:4d} clusters from {:4d} preclusters in {:2d} ROFs",
141 clusters.size(), preClusters.size(), preClusterROFs.size());
142 }
143
144 private:
145 //_________________________________________________________________________________________________
146 void writeClusters(const gsl::span<const Digit>& preclusterDigits, size_t firstClusterIdx,
147 std::vector<Cluster, o2::pmr::polymorphic_allocator<Cluster>>& clusters,
148 std::vector<Digit, o2::pmr::polymorphic_allocator<Digit>>& usedDigits) const
149 {
152
153 if (firstClusterIdx == mClusterFinder.getClusters().size()) {
154 return;
155 }
156
157 auto clusterOffset = clusters.size();
158 clusters.insert(clusters.end(), mClusterFinder.getClusters().begin() + firstClusterIdx, mClusterFinder.getClusters().end());
159
160 auto digitOffset = usedDigits.size();
161 usedDigits.insert(usedDigits.end(), preclusterDigits.begin(), preclusterDigits.end());
162
163 for (auto itCluster = clusters.begin() + clusterOffset; itCluster < clusters.end(); ++itCluster) {
164 itCluster->firstDigit = digitOffset;
165 itCluster->nDigits = preclusterDigits.size();
166 }
167 }
168
169 //_________________________________________________________________________________________________
170 void writeClusters(std::vector<Cluster, o2::pmr::polymorphic_allocator<Cluster>>& clusters,
171 std::vector<Digit, o2::pmr::polymorphic_allocator<Digit>>& usedDigits) const
172 {
175
176 auto clusterOffset = clusters.size();
177 clusters.insert(clusters.end(), mClusterFinder.getClusters().begin(), mClusterFinder.getClusters().end());
178
179 auto digitOffset = usedDigits.size();
180 usedDigits.insert(usedDigits.end(), mClusterFinder.getUsedDigits().begin(), mClusterFinder.getUsedDigits().end());
181
182 for (auto itCluster = clusters.begin() + clusterOffset; itCluster < clusters.end(); ++itCluster) {
183 itCluster->firstDigit += digitOffset;
184 }
185 }
186
187 bool mAttachInitalPrecluster = false;
188 ClusterFinderOriginal mClusterFinder{};
189 ErrorMap mErrorMap{};
190 std::chrono::duration<double> mTimeClusterFinder{};
191};
192
193//_________________________________________________________________________________________________
195{
196 return DataProcessorSpec{
197 specName,
198 Inputs{InputSpec{"preclusterrofs", "MCH", "PRECLUSTERROFS", 0, Lifetime::Timeframe},
199 InputSpec{"preclusters", "MCH", "PRECLUSTERS", 0, Lifetime::Timeframe},
200 InputSpec{"digits", "MCH", "PRECLUSTERDIGITS", 0, Lifetime::Timeframe}},
201 Outputs{OutputSpec{{"clusterrofs"}, "MCH", "CLUSTERROFS", 0, Lifetime::Timeframe},
202 OutputSpec{{"clusters"}, "MCH", "CLUSTERS", 0, Lifetime::Timeframe},
203 OutputSpec{{"clusterdigits"}, "MCH", "CLUSTERDIGITS", 0, Lifetime::Timeframe},
204 OutputSpec{{"clustererrors"}, "MCH", "CLUSTERERRORS", 0, Lifetime::Timeframe}},
205 AlgorithmSpec{adaptFromTask<ClusterFinderOriginalTask>()},
206 Options{{"mch-config", VariantType::String, "", {"JSON or INI file with clustering parameters"}},
207 {"run2-config", VariantType::Bool, false, {"setup for run2 data"}},
208 {"attach-initial-precluster", VariantType::Bool, false, {"attach all digits of initial precluster to cluster"}}}};
209}
210
211} // end namespace mch
212} // end namespace o2
Definition of a data processor to run the original MLEM cluster finder.
Definition of a class to reconstruct clusters with the original MLEM algorithm.
Definition of the MCH cluster minimal structure.
definition of the MCH processing errors
Definition of the MCH precluster minimal structure.
Definition of the MCH ROFrame record.
const char * specName
static void updateFromFile(std::string const &, std::string const &paramsList="", bool unchangedOnly=false)
decltype(auto) make(const Output &spec, Args... args)
ServiceRegistryRef services()
Definition InitContext.h:34
ConfigParamRegistry const & options()
Definition InitContext.h:33
decltype(auto) get(R binding, int part=0) const
DataAllocator & outputs()
The data allocator is used to allocate memory for the output data.
InputRecord & inputs()
The inputs associated with this processing context.
HMPID cluster implementation.
Definition Cluster.h:27
void init(framework::InitContext &ic)
void run(framework::ProcessingContext &pc)
const std::vector< Digit > & getUsedDigits() const
return the list of digits used in reconstructed clusters
ErrorMap & getErrorMap()
return the counting of encountered errors
void findClusters(gsl::span< const Digit > digits)
const std::vector< Cluster > & getClusters() const
return the list of reconstructed clusters
MCH digit implementation.
Definition Digit.h:31
void forEach(ErrorFunction f) const
Definition ErrorMap.cxx:90
void add(ErrorType errorType, uint32_t id0, uint32_t id1, uint64_t n=1)
Definition ErrorMap.cxx:33
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
std::vector< ConfigParamSpec > Options
std::vector< InputSpec > Inputs
std::vector< OutputSpec > Outputs
o2::framework::DataProcessorSpec getClusterFinderOriginalSpec(const char *specName="mch-cluster-finder-original")
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
Defining DataPointCompositeObject explicitly as copiable.
cluster minimal structure
Definition Cluster.h:31
std::string asString() const
Definition Error.cxx:86
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::vector< Cluster > clusters
std::vector< Digit > digits