Project
Loading...
Searching...
No Matches
TRKDigitizerSpec.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
12#include "TRKDigitizerSpec.h"
18#include "Framework/Lifetime.h"
19#include "Framework/Task.h"
33
34#include <TChain.h>
35#include <TStopwatch.h>
36
37#include <string>
38
39using namespace o2::framework;
41
42namespace
43{
44std::vector<OutputSpec> makeOutChannels(o2::header::DataOrigin detOrig, bool mctruth)
45{
46 std::vector<OutputSpec> outputs;
47 outputs.emplace_back(detOrig, "DIGITS", 0, Lifetime::Timeframe);
48 outputs.emplace_back(detOrig, "DIGITSROF", 0, Lifetime::Timeframe);
49 if (mctruth) {
50 outputs.emplace_back(detOrig, "DIGITSMC2ROF", 0, Lifetime::Timeframe);
51 outputs.emplace_back(detOrig, "DIGITSMCTR", 0, Lifetime::Timeframe);
52 }
53 outputs.emplace_back(detOrig, "ROMode", 0, Lifetime::Timeframe);
54 return outputs;
55}
56} // namespace
57
58namespace o2::trk
59{
60using namespace o2::base;
62{
63 public:
65
66 TRKDPLDigitizerTask(bool mctruth = true) : BaseDPLDigitizer(InitServices::FIELD | InitServices::GEOM), mWithMCTruth(mctruth) {}
67
69 {
70 mDisableQED = ic.options().get<bool>("disable-qed");
71 }
72
74 {
75 if (mFinished) {
76 return;
77 }
79
80 // read collision context from input
81 auto context = pc.inputs().get<o2::steer::DigitizationContext*>("collisioncontext");
82 context->initSimChains(mID, mSimChains);
83 const bool withQED = context->isQEDProvided() && !mDisableQED;
84 auto& timesview = context->getEventRecords(withQED);
85 LOG(info) << "GOT " << timesview.size() << " COLLISION TIMES";
86 LOG(info) << "SIMCHAINS " << mSimChains.size();
87
88 // if there is nothing to do ... return
89 if (timesview.empty()) {
90 return;
91 }
92 TStopwatch timer;
93 timer.Start();
94 LOG(info) << " CALLING TRK DIGITIZATION ";
95
96 // mDigitizer.setDigits(&mDigits);
97 mDigitizer.setROFRecords(&mROFRecords);
98 mDigitizer.setMCLabels(&mLabels);
99
100 // digits are directly put into DPL owned resource
101 auto& digitsAccum = pc.outputs().make<std::vector<itsmft::Digit>>(Output{mOrigin, "DIGITS", 0});
102
103 auto accumulate = [this, &digitsAccum]() {
104 // accumulate result of single event processing, called after processing every event supplied
105 // AND after the final flushing via digitizer::fillOutputContainer
106 if (mDigits.empty()) {
107 return; // no digits were flushed, nothing to accumulate
108 }
109 auto ndigAcc = digitsAccum.size();
110 std::copy(mDigits.begin(), mDigits.end(), std::back_inserter(digitsAccum));
111
112 // fix ROFrecords references on ROF entries
113 auto nROFRecsOld = mROFRecordsAccum.size();
114
115 for (int i = 0; i < mROFRecords.size(); i++) {
116 auto& rof = mROFRecords[i];
117 rof.setFirstEntry(ndigAcc + rof.getFirstEntry());
118 rof.print();
119
120 if (mFixMC2ROF < mMC2ROFRecordsAccum.size()) { // fix ROFRecord entry in MC2ROF records
121 for (int m2rid = mFixMC2ROF; m2rid < mMC2ROFRecordsAccum.size(); m2rid++) {
122 // need to register the ROFRecors entry for MC event starting from this entry
123 auto& mc2rof = mMC2ROFRecordsAccum[m2rid];
124 if (rof.getROFrame() == mc2rof.minROF) {
125 mFixMC2ROF++;
126 mc2rof.rofRecordID = nROFRecsOld + i;
127 mc2rof.print();
128 }
129 }
130 }
131 }
132
133 std::copy(mROFRecords.begin(), mROFRecords.end(), std::back_inserter(mROFRecordsAccum));
134 if (mWithMCTruth) {
135 mLabelsAccum.mergeAtBack(mLabels);
136 }
137 LOG(info) << "Added " << mDigits.size() << " digits ";
138 // clean containers from already accumulated stuff
139 mLabels.clear();
140 mDigits.clear();
141 mROFRecords.clear();
142 }; // and accumulate lambda
143
144 auto& eventParts = context->getEventParts(withQED);
145 // loop over all composite collisions given from context (aka loop over all the interaction records)
146 const int bcShift = mDigitizer.getParams().getROFrameBiasInBC();
147 // loop over all composite collisions given from context (aka loop over all the interaction records)
148 for (size_t collID = 0; collID < timesview.size(); ++collID) {
149 auto irt = timesview[collID];
150 if (irt.toLong() < bcShift) { // due to the ROF misalignment the collision would go to negative ROF ID, discard
151 continue;
152 }
153 irt -= bcShift; // account for the ROF start shift
154
155 mDigitizer.setEventTime(irt);
156 mDigitizer.resetEventROFrames(); // to estimate min/max ROF for this collID
157 // for each collision, loop over the constituents event and source IDs
158 // (background signal merging is basically taking place here)
159 for (auto& part : eventParts[collID]) {
160
161 // get the hits for this event and this source
162 mHits.clear();
163 context->retrieveHits(mSimChains, o2::detectors::SimTraits::DETECTORBRANCHNAMES[mID][0].c_str(), part.sourceID, part.entryID, &mHits);
164
165 if (!mHits.empty()) {
166 LOG(debug) << "For collision " << collID << " eventID " << part.entryID
167 << " found " << mHits.size() << " hits ";
168 mDigitizer.process(&mHits, part.entryID, part.sourceID); // call actual digitization procedure
169 }
170 }
171 mMC2ROFRecordsAccum.emplace_back(collID, -1, mDigitizer.getEventROFrameMin(), mDigitizer.getEventROFrameMax());
172 accumulate();
173 }
174 mDigitizer.fillOutputContainer();
175 accumulate();
176
177 // here we have all digits and labels and we can send them to consumer (aka snapshot it onto output)
178
179 pc.outputs().snapshot(Output{mOrigin, "DIGITSROF", 0}, mROFRecordsAccum);
180 if (mWithMCTruth) {
181 pc.outputs().snapshot(Output{mOrigin, "DIGITSMC2ROF", 0}, mMC2ROFRecordsAccum);
182 auto& sharedlabels = pc.outputs().make<o2::dataformats::ConstMCTruthContainer<o2::MCCompLabel>>(Output{mOrigin, "DIGITSMCTR", 0});
183 mLabelsAccum.flatten_to(sharedlabels);
184 // free space of existing label containers
185 mLabels.clear_andfreememory();
186 mLabelsAccum.clear_andfreememory();
187 }
188 LOG(info) << mID.getName() << ": Sending ROMode= " << mROMode << " to GRPUpdater";
189 pc.outputs().snapshot(Output{mOrigin, "ROMode", 0}, mROMode);
190
191 timer.Stop();
192 LOG(info) << "Digitization took " << timer.CpuTime() << "s";
193
194 // we should be only called once; tell DPL that this process is ready to exit
195 pc.services().get<ControlService>().readyToQuit(QuitRequest::Me);
196
197 mFinished = true;
198 }
199
201 {
202 static bool initOnce{false};
203 if (!initOnce) {
204 initOnce = true;
205 auto& digipar = mDigitizer.getParams();
206
207 // configure digitizer
209 geom->fillMatrixCache(o2::math_utils::bit2Mask(o2::math_utils::TransformType::L2G)); // make sure L2G matrices are loaded
210 mDigitizer.setGeometry(geom);
211
215 digipar.setContinuous(dopt.continuous);
216 digipar.setROFrameBiasInBC(aopt.roFrameBiasInBC);
217 if (dopt.continuous) {
219 digipar.setROFrameLengthInBC(aopt.roFrameLengthInBC);
220 digipar.setROFrameLength(frameNS); // RO frame in ns
221 digipar.setStrobeDelay(aopt.strobeDelay); // Strobe delay wrt beginning of the RO frame, in ns
222 digipar.setStrobeLength(aopt.strobeLengthCont > 0 ? aopt.strobeLengthCont : frameNS - aopt.strobeDelay); // Strobe length in ns
223 } else {
224 digipar.setROFrameLength(aopt.roFrameLengthTrig); // RO frame in ns
225 digipar.setStrobeDelay(aopt.strobeDelay); // Strobe delay wrt beginning of the RO frame, in ns
226 digipar.setStrobeLength(aopt.strobeLengthTrig); // Strobe length in ns
227 }
228 // parameters of signal time response: flat-top duration, max rise time and q @ which rise time is 0
229 digipar.getSignalShape().setParameters(dopt.strobeFlatTop, dopt.strobeMaxRiseTime, dopt.strobeQRiseTime0);
230 digipar.setChargeThreshold(dopt.chargeThreshold); // charge threshold in electrons
231 digipar.setNoisePerPixel(dopt.noisePerPixel); // noise level
232 digipar.setTimeOffset(dopt.timeOffset);
233 digipar.setNSimSteps(dopt.nSimSteps);
234
236 LOG(info) << mID.getName() << " simulated in "
237 << ((mROMode == o2::parameters::GRPObject::CONTINUOUS) ? "CONTINUOUS" : "TRIGGERED")
238 << " RO mode";
239
240 // if (oTRKParams::Instance().useDeadChannelMap) {
241 // pc.inputs().get<o2::itsmft::NoiseMap*>("TRK_dead"); // trigger final ccdb update
242 // }
243
244 // init digitizer
245 mDigitizer.init();
246 }
247 // Other time-dependent parameters can be added below
248 }
249
250 void finaliseCCDB(ConcreteDataMatcher& matcher, void* obj)
251 {
252 if (matcher == ConcreteDataMatcher(detectors::DetID::ITS, "ALPIDEPARAM", 0)) {
253 LOG(info) << mID.getName() << " Alpide param updated";
255 par.printKeyValues();
256 return;
257 }
258 // if (matcher == ConcreteDataMatcher(mOrigin, "DEADMAP", 0)) {
259 // LOG(info) << mID.getName() << " static dead map updated";
260 // mDigitizer.setDeadChannelsMap((o2::itsmft::NoiseMap*)obj);
261 // return;
262 // }
263 }
264
265 private:
266 bool mWithMCTruth{true};
267 bool mFinished{false};
268 bool mDisableQED{false};
269 const o2::detectors::DetID mID{o2::detectors::DetID::TRK};
271 o2::trk::Digitizer mDigitizer{};
272 std::vector<o2::itsmft::Digit> mDigits{};
273 std::vector<o2::itsmft::ROFRecord> mROFRecords{};
274 std::vector<o2::itsmft::ROFRecord> mROFRecordsAccum{};
275 std::vector<o2::itsmft::Hit> mHits{};
276 std::vector<o2::itsmft::Hit>* mHitsP{&mHits};
279 std::vector<o2::itsmft::MC2ROFRecord> mMC2ROFRecordsAccum{};
280 std::vector<TChain*> mSimChains{};
281
282 int mFixMC2ROF = 0; // 1st entry in mc2rofRecordsAccum to be fixed for ROFRecordID
284};
285
286DataProcessorSpec getTRKDigitizerSpec(int channel, bool mctruth)
287{
288 std::string detStr = o2::detectors::DetID::getName(o2::detectors::DetID::TRK);
289 auto detOrig = o2::header::gDataOriginTRK;
290 std::vector<InputSpec> inputs;
291 inputs.emplace_back("collisioncontext", "SIM", "COLLISIONCONTEXT", static_cast<SubSpecificationType>(channel), Lifetime::Timeframe);
292 inputs.emplace_back("ITS_alppar", "ITS", "ALPIDEPARAM", 0, Lifetime::Condition, ccdbParamSpec("ITS/Config/AlpideParam"));
293 // if (oTRKParams::Instance().useDeadChannelMap) {
294 // inputs.emplace_back("TRK_dead", "TRK", "DEADMAP", 0, Lifetime::Condition, ccdbParamSpec("TRK/Calib/DeadMap"));
295 // }
296
297 return DataProcessorSpec{detStr + "Digitizer",
298 inputs, makeOutChannels(detOrig, mctruth),
299 AlgorithmSpec{adaptFromTask<TRKDPLDigitizerTask>(mctruth)},
300 Options{{"disable-qed", o2::framework::VariantType::Bool, false, {"disable QED handling"}}}};
301}
302
303} // namespace o2::trk
Definition of the base digitizer task class.
A const (ready only) version of MCTruthContainer.
Definition of the ITSMFT digit.
o2::framework::DataAllocator::SubSpecificationType SubSpecificationType
int32_t i
Header of the General Run Parameters object.
Definition of the ITSMFT ROFrame (trigger) record.
Definition of the TRK digitizer.
std::ostringstream debug
virtual void init(o2::framework::InitContext &) final
A read-only version of MCTruthContainer allowing for storage optimisation.
A container to hold and manage MC truth information/labels.
void mergeAtBack(MCTruthContainer< TruthElement > const &other)
size_t flatten_to(ContainerType &container) const
Static class with identifiers, bitmasks and names for ALICE detectors.
Definition DetID.h:58
static constexpr const char * getName(ID id)
names of defined detectors
Definition DetID.h:145
static constexpr ID ITS
Definition DetID.h:63
static const std::array< std::vector< std::string >, DetID::nDetectors > DETECTORBRANCHNAMES
Definition SimTraits.h:39
void snapshot(const Output &spec, T const &object)
o2::header::DataHeader::SubSpecificationType SubSpecificationType
decltype(auto) make(const Output &spec, Args... args)
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.
ServiceRegistryRef services()
The services registry associated with this processing context.
int getROFrameBiasInBC() const
Definition DigiParams.h:71
bool initSimChains(o2::detectors::DetID detid, std::vector< TChain * > &simchains) const
void setEventTime(const o2::InteractionTimeRecord &irt)
uint32_t getEventROFrameMin() const
Definition Digitizer.h:73
void process(const std::vector< itsmft::Hit > *hits, int evID, int srcID)
Steer conversion of hits to digits.
Definition Digitizer.cxx:89
o2::itsmft::DigiParams & getParams()
Definition Digitizer.h:49
void setROFRecords(std::vector< o2::itsmft::ROFRecord > *rec)
Definition Digitizer.h:47
void fillOutputContainer(uint32_t maxFrame=0xffffffff)
void setMCLabels(o2::dataformats::MCTruthContainer< o2::MCCompLabel > *mclb)
Definition Digitizer.h:46
void setGeometry(const o2::trk::GeometryTGeo *gm)
Definition Digitizer.h:71
uint32_t getEventROFrameMax() const
Definition Digitizer.h:74
void resetEventROFrames()
Definition Digitizer.h:75
static GeometryTGeo * Instance()
void fillMatrixCache(int mask)
TRKDPLDigitizerTask(bool mctruth=true)
void run(framework::ProcessingContext &pc)
void updateTimeDependentParams(ProcessingContext &pc)
void finaliseCCDB(ConcreteDataMatcher &matcher, void *obj)
void initDigitizerTask(framework::InitContext &ic) override
constexpr o2::header::DataOrigin gDataOriginTRK
Definition DataHeader.h:584
constexpr double LHCBunchSpacingNS
Defining PrimaryVertex explicitly as messageable.
Definition TFIDInfo.h:20
std::vector< ConfigParamSpec > ccdbParamSpec(std::string const &path, int runDependent, std::vector< CCDBMetadata > metadata={}, int qrate=0)
std::vector< ConfigParamSpec > Options
header::DataHeader::SubSpecificationType SubSpecificationType
DataProcessorSpec getTRKDigitizerSpec(int channel, bool mctruth)
class listing possible services
int roFrameBiasInBC
bias of the start of ROF wrt orbit start: t_irof = (irof*roFrameLengthInBC + roFrameBiasInBC)*BClengt...
float strobeDelay
strobe start (in ns) wrt ROF start
int roFrameLengthInBC
ROF length in BC for continuos mode.
float strobeLengthCont
if < 0, full ROF length - delay
float strobeLengthTrig
length of the strobe in ns (sig. over threshold checked in this window only)
float roFrameLengthTrig
length of RO frame in ns for triggered mode
static constexpr int L2G
Definition Cartesian.h:54
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"