Project
Loading...
Searching...
No Matches
DigitFilteringSpec.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
13
18#include "Framework/Logger.h"
20#include "Framework/Task.h"
22#include "MCHBase/SanityCheck.h"
23#include "MCHStatus/StatusMap.h"
30#include <fmt/format.h>
31#include <functional>
32#include <iostream>
33#include <string>
34#include <vector>
35
36using namespace o2::dataformats;
37using namespace o2::framework;
38
39namespace o2::mch
40{
42{
43 public:
44 DigitFilteringTask(bool useMC, bool useStatusMap) : mUseMC{useMC}, mUseStatusMap{useStatusMap} {}
45
46 void init(InitContext& ic)
47 {
48 mSanityCheck = DigitFilterParam::Instance().sanityCheck;
49 mMinADC = DigitFilterParam::Instance().minADC;
50 mRejectBackground = DigitFilterParam::Instance().rejectBackground;
51 mStatusMask = DigitFilterParam::Instance().statusMask;
52 mTimeCalib = DigitFilterParam::Instance().timeOffset;
53
54 mUpdateDigitsST1 = DigitModifierParam::Instance().updateST1;
55 mUpdateDigitsST2 = DigitModifierParam::Instance().updateST2;
56
57 auto stop = [this]() {
58 LOG(info) << "digit filtering duration = "
59 << std::chrono::duration<double, std::milli>(mElapsedTime).count() << " ms";
60 };
61 ic.services().get<CallbackService>().set<CallbackService::Id::Stop>(stop);
62 }
63
64 void shiftDigitsTime(gsl::span<ROFRecord> rofs, gsl::span<Digit> digits)
65 {
66 for (auto i = 0; i < rofs.size(); i++) {
67 ROFRecord& rof = rofs[i];
68 rof.getBCData() += mTimeCalib;
69 }
70
71 for (auto i = 0; i < digits.size(); i++) {
72 Digit& d = digits[i];
73 d.setTime(d.getTime() + mTimeCalib);
74 }
75 }
76
78 {
79 StatusMap defaultStatusMap;
80
81 // get input
82 auto iRofs = pc.inputs().get<gsl::span<ROFRecord>>("rofs");
83 auto iDigits = pc.inputs().get<gsl::span<Digit>>("digits");
84 auto iLabels = mUseMC ? pc.inputs().get<MCTruthContainer<MCCompLabel>*>("labels") : nullptr;
85 auto statusMap = mUseStatusMap && pc.inputs().isValid("statusmap") ? pc.inputs().get<StatusMap*>("statusmap") : nullptr;
86
87 bool abort{false};
88
89 auto tStart = std::chrono::high_resolution_clock::now();
90
91 const auto& tinfo = pc.services().get<o2::framework::TimingInfo>();
92 if (tinfo.runNumber != 0) {
93 mRunNumber = tinfo.runNumber;
94 }
95
96 if (mSanityCheck) {
97 LOGP(info, "performing sanity checks");
98 auto error = sanityCheck(iRofs, iDigits);
99
100 if (!isOK(error)) {
101 if (error.nofOutOfBounds > 0) {
102 LOGP(error, "{}", asString(error));
103 LOGP(error, "in a TF with {} rofs and {} digits", iRofs.size(), iDigits.size());
104 abort = true;
105 }
106 }
107 }
108
109 // create the output messages
110 auto& oRofs = pc.outputs().make<std::vector<ROFRecord>>(OutputRef{"rofs"});
111 auto& oDigits = pc.outputs().make<std::vector<Digit>>(OutputRef{"digits"});
112 auto oLabels = mUseMC ? &pc.outputs().make<MCTruthContainer<MCCompLabel>>(OutputRef{"labels"}) : nullptr;
113
114 if (!abort) {
115
116 mDigitModifier = createDigitModifier(mRunNumber,
117 mUpdateDigitsST1,
118 mUpdateDigitsST2);
119
120 bool selectSignal = false;
121 mIsGoodDigit = createDigitFilter(mMinADC,
122 mRejectBackground,
123 selectSignal,
124 statusMap ? *statusMap : defaultStatusMap,
125 mStatusMask);
126 // at digit filtering stage it is important to keep the 3rd parameter
127 // (selectSignal) to false in the call above : the idea is to not cut
128 // too much on the tails of the charge distributions otherwise
129 // the clustering resolution will suffer.
130 // That's why we only apply the "reject background" filter, which
131 // is a loose background cut that does not penalize the signal
132
133 int cursor{0};
134 for (const auto& irof : iRofs) {
135 const auto digits = iDigits.subspan(irof.getFirstIdx(), irof.getNEntries());
136
137 // filter the digits from the current ROF
138 for (auto i = 0; i < digits.size(); i++) {
139 auto digit = digits[i];
140
141 // modify the digit if needed
142 if (mDigitModifier) {
143 mDigitModifier(digit);
144 }
145
146 // check the digit quality
147 if (mIsGoodDigit(digit)) {
148 oDigits.emplace_back(digit);
149 if (iLabels) {
150 oLabels->addElements(oLabels->getIndexedSize(), iLabels->getLabels(i + irof.getFirstIdx()));
151 }
152 }
153 }
154
155 int nofGoodDigits = oDigits.size() - cursor;
156 if (nofGoodDigits > 0) {
157 // we create an ouput ROF only if at least one digit from
158 // the input ROF passed the filtering
159 oRofs.emplace_back(ROFRecord(irof.getBCData(),
160 cursor,
161 nofGoodDigits,
162 irof.getBCWidth()));
163 cursor += nofGoodDigits;
164 }
165 }
166 }
167
168 auto labelMsg = mUseMC ? fmt::format("| {} labels (out of {})", oLabels->getNElements(), iLabels->getNElements()) : "";
169
170 LOGP(info, "Kept after filtering : {} rofs (out of {}) | {} digits (out of {}) {}",
171 oRofs.size(), iRofs.size(),
172 oDigits.size(), iDigits.size(),
173 labelMsg);
174
175 if (mTimeCalib != 0) {
176 shiftDigitsTime(oRofs, oDigits);
177 }
178
179 if (abort) {
180 LOGP(error, "Sanity check failed");
181 }
182 auto tEnd = std::chrono::high_resolution_clock::now();
183 mElapsedTime += tEnd - tStart;
184 }
185
186 private:
187 int mRunNumber{0};
188 bool mRejectBackground{false};
189 bool mSanityCheck{false};
190 bool mUseMC{false};
191 bool mUseStatusMap{false};
192 int mMinADC{1};
193 int32_t mTimeCalib{0};
194 uint32_t mStatusMask{0};
195 bool mUpdateDigitsST1{false};
196 bool mUpdateDigitsST2{false};
197 DigitFilter mIsGoodDigit;
198 DigitModifier mDigitModifier;
199 std::chrono::duration<double> mElapsedTime{};
200};
201
204 std::string_view specName,
205 std::string_view inputDigitDataDescription,
206 std::string_view outputDigitDataDescription,
207 std::string_view inputDigitRofDataDescription,
208 std::string_view outputDigitRofDataDescription,
209 std::string_view inputDigitLabelDataDescription,
210 std::string_view outputDigitLabelDataDescription)
211
212{
213 std::string input =
214 fmt::format("digits:MCH/{}/0;rofs:MCH/{}/0",
215 inputDigitDataDescription,
216 inputDigitRofDataDescription);
217 if (useMC) {
218 input += fmt::format(";labels:MCH/{}/0", inputDigitLabelDataDescription);
219 }
220
221 bool useStatusMap = DigitFilterParam::Instance().statusMask != 0;
222
223 if (useStatusMap) {
224 input += ";statusmap:MCH/STATUSMAP/0";
225 }
226
227 std::string output =
228 fmt::format("digits:MCH/{}/0;rofs:MCH/{}/0",
229 outputDigitDataDescription,
230 outputDigitRofDataDescription);
231 if (useMC) {
232 output += fmt::format(";labels:MCH/{}/0", outputDigitLabelDataDescription);
233 }
234
235 std::vector<OutputSpec> outputs;
236 auto matchers = select(output.c_str());
237 for (auto& matcher : matchers) {
238 outputs.emplace_back(DataSpecUtils::asOutputSpec(matcher));
239 }
240
241 return DataProcessorSpec{
242 specName.data(),
243 Inputs{select(input.c_str())},
244 outputs,
245 AlgorithmSpec{adaptFromTask<DigitFilteringTask>(useMC, useStatusMap)},
246 Options{}};
247}
248} // namespace o2::mch
int32_t i
void output(const std::map< std::string, ChannelStat > &channels)
Definition rawdump.cxx:197
Definition of a container to keep Monte Carlo truth external to simulation objects.
Definition of the MCH ROFrame record.
const char * specName
A container to hold and manage MC truth information/labels.
decltype(auto) make(const Output &spec, Args... args)
ServiceRegistryRef services()
Definition InitContext.h:34
bool isValid(std::string const &s) const
Helper method to be used to check if a given part of the InputRecord is present.
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.
ServiceRegistryRef services()
The services registry associated with this processing context.
void run(ProcessingContext &pc)
DigitFilteringTask(bool useMC, bool useStatusMap)
void shiftDigitsTime(gsl::span< ROFRecord > rofs, gsl::span< Digit > digits)
MCH digit implementation.
Definition Digit.h:31
int32_t getTime() const
Definition Digit.h:44
void setTime(int32_t t)
Definition Digit.h:43
const BCData & getBCData() const
get the interaction record
Definition ROFRecord.h:46
Definition of a container to keep/associate and arbitrary number of labels associated to an index wit...
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
std::vector< ConfigParamSpec > Options
std::vector< InputSpec > select(char const *matcher="")
std::vector< InputSpec > Inputs
o2::framework::DataProcessorSpec getDigitFilteringSpec(bool useMC, std::string_view specName="mch-digit-filtering", std::string_view inputDigitDataDescription="DIGITS", std::string_view outputDigitDataDescription="F-DIGITS", std::string_view inputDigitRofDataDescription="DIGITROFS", std::string_view outputDigitRofDataDescription="F-DIGITROFS", std::string_view inputDigitLabelDataDescription="DIGITLABELS", std::string_view outputDigitLabelDataDescription="F-DIGITLABELS")
std::string asString(const o2::mch::TrackMCH &t)
Definition TrackMCH.cxx:104
std::function< void(Digit &)> DigitModifier
std::function< bool(const Digit &)> DigitFilter
Definition DigitFilter.h:21
DigitFilter createDigitFilter(uint32_t minADC, bool rejectBackground, bool selectSignal, const StatusMap &statusMap={}, uint32_t statusMask=0)
bool isOK(const SanityError &error)
SanityError sanityCheck(gsl::span< const ROFRecord > rofs, gsl::span< const T > items)
Definition SanityCheck.h:48
DigitModifier createDigitModifier(int runNumber, bool updateST1, bool updateST2)
static OutputSpec asOutputSpec(InputSpec const &spec)
uint32_t statusMask
mask to reject digits based on the statusmap (0 = no rejection)
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::vector< Digit > digits