Project
Loading...
Searching...
No Matches
Digitizer.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
15// #include "ITSMFTBase/SegmentationAlpide.h"
18// #include "MathUtils/Cartesian.h"
19// #include "SimulationDataFormat/MCTruthContainer.h"
20// #include "DetectorsRaw/HBFUtils.h"
21
22// #include <TRandom.h>
23// #include <climits>
24// #include <vector>
25// #include <numeric>
26#include <fairlogger/Logger.h> // for LOG
27
29using o2::itsmft::Hit;
30// using Segmentation = o2::itsmft::SegmentationAlpide;
31
32using namespace o2::trk;
33// using namespace o2::base;
34
35//_______________________________________________________________________
37{
38 // mNumberOfChips = mGeometry->getNumberOfChips();
39 // mChips.resize(mNumberOfChips);
40 // for (int i = mNumberOfChips; i--;) {
41 // mChips[i].setChipIndex(i);
42 // if (mNoiseMap) {
43 // mChips[i].setNoiseMap(mNoiseMap);
44 // }
45 // if (mDeadChanMap) {
46 // mChips[i].disable(mDeadChanMap->isFullChipMasked(i));
47 // mChips[i].setDeadChanMap(mDeadChanMap);
48 // }
49 // }
50 // initializing for both collection tables
51 /*for (int i = 0; i < 2; i++) {
52 mAlpSimResp[i].initData(i);
53 }*/
54
55 // importing the charge collection tables
56 // (initialized while building O2)
57 // auto file = TFile::Open(mResponseFile.data());
58 // if (!file) {
59 // LOG(fatal) << "Cannot open response file " << mResponseFile;
60 // }
61 /*std::string response = "response";
62 for (int i=0; i<2; i++) {
63 response.append(std::to_string(i));
64 mAlpSimResp[i] = *(o2::itsmft::AlpideSimResponse*)file->Get(response.data());
65 }*/
66 // mAlpSimResp[0] = *(o2::itsmft::AlpideSimResponse*)file->Get("response0");
67 // mAlpSimResp[1] = *(o2::itsmft::AlpideSimResponse*)file->Get("response1");
68
69 // importing the parameters from DPLDigitizerParam.h
71
72 LOGP(info, "TRK Digitizer is initalised.");
73}
74
75// auto Digitizer::getChipResponse(int chipID)
76// {
77// if (mNumberOfChips < 10000) { // in MFT
78// return mAlpSimRespMFT;
79// }
80
81// if (chipID < 432) { // in ITS Inner Barrel
82// return mAlpSimRespIB;
83// } else { // in ITS Outter Barrel
84// return mAlpSimRespOB;
85// }
86// }
87
88//_______________________________________________________________________
89void Digitizer::process(const std::vector<Hit>* hits, int evID, int srcID)
90{
91 // digitize single event, the time must have been set beforehand
92
93 // LOG(info) << "Digitizing " << mGeometry->getName() << " hits of entry " << evID << " from source "
94 // << srcID << " at time " << mEventTime << " ROFrame= " << mNewROFrame << ")"
95 // << " cont.mode: " << isContinuous()
96 // << " Min/Max ROFrames " << mROFrameMin << "/" << mROFrameMax;
97
98 // // is there something to flush ?
99 // if (mNewROFrame > mROFrameMin) {
100 // fillOutputContainer(mNewROFrame - 1); // flush out all frame preceding the new one
101 // }
102
103 // int nHits = hits->size();
104 // std::vector<int> hitIdx(nHits);
105 // std::iota(std::begin(hitIdx), std::end(hitIdx), 0);
106 // // sort hits to improve memory access
107 // std::sort(hitIdx.begin(), hitIdx.end(),
108 // [hits](auto lhs, auto rhs) {
109 // return (*hits)[lhs].GetDetectorID() < (*hits)[rhs].GetDetectorID();
110 // });
111 // for (int i : hitIdx) {
112 // processHit((*hits)[i], mROFrameMax, evID, srcID);
113 // }
114 // // in the triggered mode store digits after every MC event
115 // // TODO: in the real triggered mode this will not be needed, this is actually for the
116 // // single event processing only
117 // if (!mParams.isContinuous()) {
118 // fillOutputContainer(mROFrameMax);
119 // }
120}
121
122//_______________________________________________________________________
124{
125 // // assign event time in ns
126 // mEventTime = irt;
127 // if (!mParams.isContinuous()) {
128 // mROFrameMin = 0; // in triggered mode reset the frame counters
129 // mROFrameMax = 0;
130 // }
131 // // RO frame corresponding to provided time
132 // mCollisionTimeWrtROF = mEventTime.timeInBCNS; // in triggered mode the ROF starts at BC (is there a delay?)
133 // if (mParams.isContinuous()) {
134 // auto nbc = mEventTime.differenceInBC(mIRFirstSampledTF);
135 // if (mCollisionTimeWrtROF < 0 && nbc > 0) {
136 // nbc--;
137 // }
138
139 // // we might get interactions to digitize from before
140 // // the first sampled IR
141 // if (nbc < 0) {
142 // mNewROFrame = 0;
143 // // this event is before the first RO
144 // mIsBeforeFirstRO = true;
145 // } else {
146 // mNewROFrame = nbc / mParams.getROFrameLengthInBC();
147 // mIsBeforeFirstRO = false;
148 // }
149 // LOG(info) << " NewROFrame " << mNewROFrame << " nbc " << nbc;
150
151 // // in continuous mode depends on starts of periodic readout frame
152 // mCollisionTimeWrtROF += (nbc % mParams.getROFrameLengthInBC()) * o2::constants::lhc::LHCBunchSpacingNS;
153 // } else {
154 // mNewROFrame = 0;
155 // }
156
157 // if (mNewROFrame < mROFrameMin) {
158 // LOG(error) << "New ROFrame " << mNewROFrame << " (" << irt << ") precedes currently cashed " << mROFrameMin;
159 // throw std::runtime_error("deduced ROFrame precedes already processed one");
160 // }
161
162 // if (mParams.isContinuous() && mROFrameMax < mNewROFrame) {
163 // mROFrameMax = mNewROFrame - 1; // all frames up to this are finished
164 // }
165}
166
167//_______________________________________________________________________
168void Digitizer::fillOutputContainer(uint32_t frameLast)
169{
170 // // fill output with digits from min.cached up to requested frame, generating the noise beforehand
171 // if (frameLast > mROFrameMax) {
172 // frameLast = mROFrameMax;
173 // }
174 // // make sure all buffers for extra digits are created up to the maxFrame
175 // getExtraDigBuffer(mROFrameMax);
176
177 // LOG(info) << "Filling " << mGeometry->getName() << " digits output for RO frames " << mROFrameMin << ":"
178 // << frameLast;
179
180 // o2::itsmft::ROFRecord rcROF;
181
182 // // we have to write chips in RO increasing order, therefore have to loop over the frames here
183 // for (; mROFrameMin <= frameLast; mROFrameMin++) {
184 // rcROF.setROFrame(mROFrameMin);
185 // rcROF.setFirstEntry(mDigits->size()); // start of current ROF in digits
186
187 // auto& extra = *(mExtraBuff.front().get());
188 // for (auto& chip : mChips) {
189 // if (chip.isDisabled()) {
190 // continue;
191 // }
192 // chip.addNoise(mROFrameMin, mROFrameMin, &mParams);
193 // auto& buffer = chip.getPreDigits();
194 // if (buffer.empty()) {
195 // continue;
196 // }
197 // auto itBeg = buffer.begin();
198 // auto iter = itBeg;
199 // ULong64_t maxKey = chip.getOrderingKey(mROFrameMin + 1, 0, 0) - 1; // fetch digits with key below that
200 // for (; iter != buffer.end(); ++iter) {
201 // if (iter->first > maxKey) {
202 // break; // is the digit ROFrame from the key > the max requested frame
203 // }
204 // auto& preDig = iter->second; // preDigit
205 // if (preDig.charge >= mParams.getChargeThreshold()) {
206 // int digID = mDigits->size();
207 // mDigits->emplace_back(chip.getChipIndex(), preDig.row, preDig.col, preDig.charge);
208 // mMCLabels->addElement(digID, preDig.labelRef.label);
209 // auto& nextRef = preDig.labelRef; // extra contributors are in extra array
210 // while (nextRef.next >= 0) {
211 // nextRef = extra[nextRef.next];
212 // mMCLabels->addElement(digID, nextRef.label);
213 // }
214 // }
215 // }
216 // buffer.erase(itBeg, iter);
217 // }
218 // // finalize ROF record
219 // rcROF.setNEntries(mDigits->size() - rcROF.getFirstEntry()); // number of digits
220 // if (isContinuous()) {
221 // rcROF.getBCData().setFromLong(mIRFirstSampledTF.toLong() + mROFrameMin * mParams.getROFrameLengthInBC());
222 // } else {
223 // rcROF.getBCData() = mEventTime; // RSTODO do we need to add trigger delay?
224 // }
225 // if (mROFRecords) {
226 // mROFRecords->push_back(rcROF);
227 // }
228 // extra.clear(); // clear container for extra digits of the mROFrameMin ROFrame
229 // // and move it as a new slot in the end
230 // mExtraBuff.emplace_back(mExtraBuff.front().release());
231 // mExtraBuff.pop_front();
232 // }
233}
234
235//_______________________________________________________________________
236void Digitizer::processHit(const o2::itsmft::Hit& hit, uint32_t& maxFr, int evID, int srcID)
237{
238 // // convert single hit to digits
239 // int chipID = hit.GetDetectorID();
240 // auto& chip = mChips[chipID];
241 // if (chip.isDisabled()) {
242 // LOG(debug) << "skip disabled chip " << chipID;
243 // return;
244 // }
245 // float timeInROF = hit.GetTime() * sec2ns;
246 // if (timeInROF > 20e3) {
247 // const int maxWarn = 10;
248 // static int warnNo = 0;
249 // if (warnNo < maxWarn) {
250 // LOG(warning) << "Ignoring hit with time_in_event = " << timeInROF << " ns"
251 // << ((++warnNo < maxWarn) ? "" : " (suppressing further warnings)");
252 // }
253 // return;
254 // }
255 // if (isContinuous()) {
256 // timeInROF += mCollisionTimeWrtROF;
257 // }
258 // if (mIsBeforeFirstRO && timeInROF < 0) {
259 // // disregard this hit because it comes from an event before readout starts and it does not effect this RO
260 // return;
261 // }
262
263 // // calculate RO Frame for this hit
264 // if (timeInROF < 0) {
265 // timeInROF = 0.;
266 // }
267 // float tTot = mParams.getSignalShape().getMaxDuration();
268 // // frame of the hit signal start wrt event ROFrame
269 // int roFrameRel = int(timeInROF * mParams.getROFrameLengthInv());
270 // // frame of the hit signal end wrt event ROFrame: in the triggered mode we read just 1 frame
271 // uint32_t roFrameRelMax = mParams.isContinuous() ? (timeInROF + tTot) * mParams.getROFrameLengthInv() : roFrameRel;
272 // int nFrames = roFrameRelMax + 1 - roFrameRel;
273 // uint32_t roFrameMax = mNewROFrame + roFrameRelMax;
274 // if (roFrameMax > maxFr) {
275 // maxFr = roFrameMax; // if signal extends beyond current maxFrame, increase the latter
276 // }
277
278 // // here we start stepping in the depth of the sensor to generate charge diffusion
279 // float nStepsInv = mParams.getNSimStepsInv();
280 // int nSteps = mParams.getNSimSteps();
281 // const auto& matrix = mGeometry->getMatrixL2G(hit.GetDetectorID());
282 // math_utils::Vector3D<float> xyzLocS(matrix ^ (hit.GetPosStart())); // start position in sensor frame
283 // math_utils::Vector3D<float> xyzLocE(matrix ^ (hit.GetPos())); // end position in sensor frame
284
285 // math_utils::Vector3D<float> step(xyzLocE);
286 // step -= xyzLocS;
287 // step *= nStepsInv; // position increment at each step
288 // // the electrons will injected in the middle of each step
289 // math_utils::Vector3D<float> stepH(step * 0.5);
290 // xyzLocS += stepH;
291 // xyzLocE -= stepH;
292
293 // int rowS = -1, colS = -1, rowE = -1, colE = -1, nSkip = 0;
294 // // get entrance pixel row and col
295 // while (!Segmentation::localToDetector(xyzLocS.X(), xyzLocS.Z(), rowS, colS)) { // guard-ring ?
296 // if (++nSkip >= nSteps) {
297 // return; // did not enter to sensitive matrix
298 // }
299 // xyzLocS += step;
300 // }
301 // // get exit pixel row and col
302 // while (!Segmentation::localToDetector(xyzLocE.X(), xyzLocE.Z(), rowE, colE)) { // guard-ring ?
303 // if (++nSkip >= nSteps) {
304 // return; // did not enter to sensitive matrix
305 // }
306 // xyzLocE -= step;
307 // }
308 // // estimate the limiting min/max row and col where the non-0 response is possible
309 // if (rowS > rowE) {
310 // std::swap(rowS, rowE);
311 // }
312 // if (colS > colE) {
313 // std::swap(colS, colE);
314 // }
315 // rowS -= AlpideRespSimMat::NPix / 2;
316 // rowE += AlpideRespSimMat::NPix / 2;
317 // if (rowS < 0) {
318 // rowS = 0;
319 // }
320 // if (rowE >= Segmentation::NRows) {
321 // rowE = Segmentation::NRows - 1;
322 // }
323 // colS -= AlpideRespSimMat::NPix / 2;
324 // colE += AlpideRespSimMat::NPix / 2;
325 // if (colS < 0) {
326 // colS = 0;
327 // }
328 // if (colE >= Segmentation::NCols) {
329 // colE = Segmentation::NCols - 1;
330 // }
331 // int rowSpan = rowE - rowS + 1, colSpan = colE - colS + 1; // size of plaquet where some response is expected
332
333 // float respMatrix[rowSpan][colSpan]; // response accumulated here
334 // std::fill(&respMatrix[0][0], &respMatrix[0][0] + rowSpan * colSpan, 0.f);
335
336 // float nElectrons = hit.GetEnergyLoss() * mParams.getEnergyToNElectrons(); // total number of deposited electrons
337 // nElectrons *= nStepsInv; // N electrons injected per step
338 // if (nSkip) {
339 // nSteps -= nSkip;
340 // }
341 // //
342 // int rowPrev = -1, colPrev = -1, row, col;
343 // float cRowPix = 0.f, cColPix = 0.f; // local coordinated of the current pixel center
344
345 // const o2::itsmft::AlpideSimResponse* resp = getChipResponse(chipID);
346
347 // // take into account that the AlpideSimResponse depth defintion has different min/max boundaries
348 // // although the max should coincide with the surface of the epitaxial layer, which in the chip
349 // // local coordinates has Y = +SensorLayerThickness/2
350
351 // xyzLocS.SetY(xyzLocS.Y() + resp->getDepthMax() - Segmentation::SensorLayerThickness / 2.);
352
353 // // collect charge in every pixel which might be affected by the hit
354 // for (int iStep = nSteps; iStep--;) {
355 // // Get the pixel ID
356 // Segmentation::localToDetector(xyzLocS.X(), xyzLocS.Z(), row, col);
357 // if (row != rowPrev || col != colPrev) { // update pixel and coordinates of its center
358 // if (!Segmentation::detectorToLocal(row, col, cRowPix, cColPix)) {
359 // continue; // should not happen
360 // }
361 // rowPrev = row;
362 // colPrev = col;
363 // }
364 // bool flipCol, flipRow;
365 // // note that response needs coordinates along column row (locX) (locZ) then depth (locY)
366 // auto rspmat = resp->getResponse(xyzLocS.X() - cRowPix, xyzLocS.Z() - cColPix, xyzLocS.Y(), flipRow, flipCol);
367
368 // xyzLocS += step;
369 // if (!rspmat) {
370 // continue;
371 // }
372
373 // for (int irow = AlpideRespSimMat::NPix; irow--;) {
374 // int rowDest = row + irow - AlpideRespSimMat::NPix / 2 - rowS; // destination row in the respMatrix
375 // if (rowDest < 0 || rowDest >= rowSpan) {
376 // continue;
377 // }
378 // for (int icol = AlpideRespSimMat::NPix; icol--;) {
379 // int colDest = col + icol - AlpideRespSimMat::NPix / 2 - colS; // destination column in the respMatrix
380 // if (colDest < 0 || colDest >= colSpan) {
381 // continue;
382 // }
383 // respMatrix[rowDest][colDest] += rspmat->getValue(irow, icol, flipRow, flipCol);
384 // }
385 // }
386 // }
387
388 // // fire the pixels assuming Poisson(n_response_electrons)
389 // o2::MCCompLabel lbl(hit.GetTrackID(), evID, srcID, false);
390 // auto roFrameAbs = mNewROFrame + roFrameRel;
391 // for (int irow = rowSpan; irow--;) {
392 // uint16_t rowIS = irow + rowS;
393 // for (int icol = colSpan; icol--;) {
394 // float nEleResp = respMatrix[irow][icol];
395 // if (!nEleResp) {
396 // continue;
397 // }
398 // int nEle = gRandom->Poisson(nElectrons * nEleResp); // total charge in given pixel
399 // // ignore charge which have no chance to fire the pixel
400 // if (nEle < mParams.getMinChargeToAccount()) {
401 // continue;
402 // }
403 // uint16_t colIS = icol + colS;
404 // if (mNoiseMap && mNoiseMap->isNoisy(chipID, rowIS, colIS)) {
405 // continue;
406 // }
407 // if (mDeadChanMap && mDeadChanMap->isNoisy(chipID, rowIS, colIS)) {
408 // continue;
409 // }
410 // //
411 // registerDigits(chip, roFrameAbs, timeInROF, nFrames, rowIS, colIS, nEle, lbl);
412 // }
413 // }
414}
415
416//________________________________________________________________________________
417void Digitizer::registerDigits(o2::itsmft::ChipDigitsContainer& chip, uint32_t roFrame, float tInROF, int nROF,
418 uint16_t row, uint16_t col, int nEle, o2::MCCompLabel& lbl)
419{
420 // Register digits for given pixel, accounting for the possible signal contribution to
421 // multiple ROFrame. The signal starts at time tInROF wrt the start of provided roFrame
422 // In every ROFrame we check the collected signal during strobe
423
424 // float tStrobe = mParams.getStrobeDelay() - tInROF; // strobe start wrt signal start
425 // for (int i = 0; i < nROF; i++) {
426 // uint32_t roFr = roFrame + i;
427 // int nEleROF = mParams.getSignalShape().getCollectedCharge(nEle, tStrobe, tStrobe + mParams.getStrobeLength());
428 // tStrobe += mParams.getROFrameLength(); // for the next ROF
429
430 // // discard too small contributions, they have no chance to produce a digit
431 // if (nEleROF < mParams.getMinChargeToAccount()) {
432 // continue;
433 // }
434 // if (roFr > mEventROFrameMax) {
435 // mEventROFrameMax = roFr;
436 // }
437 // if (roFr < mEventROFrameMin) {
438 // mEventROFrameMin = roFr;
439 // }
440 // auto key = chip.getOrderingKey(roFr, row, col);
441 // PreDigit* pd = chip.findDigit(key);
442 // if (!pd) {
443 // chip.addDigit(key, roFr, row, col, nEleROF, lbl);
444 // } else { // there is already a digit at this slot, account as PreDigitExtra contribution
445 // pd->charge += nEleROF;
446 // if (pd->labelRef.label == lbl) { // don't store the same label twice
447 // continue;
448 // }
449 // ExtraDig* extra = getExtraDigBuffer(roFr);
450 // int& nxt = pd->labelRef.next;
451 // bool skip = false;
452 // while (nxt >= 0) {
453 // if ((*extra)[nxt].label == lbl) { // don't store the same label twice
454 // skip = true;
455 // break;
456 // }
457 // nxt = (*extra)[nxt].next;
458 // }
459 // if (skip) {
460 // continue;
461 // }
462 // // new predigit will be added in the end of the chain
463 // nxt = extra->size();
464 // extra->emplace_back(lbl);
465 // }
466 // }
467}
Definition of the ITSMFT digit.
uint32_t col
Definition RawData.h:4
Definition of the TRK digitizer.
Container for similated points connected to a given chip.
Digit class for the ITS.
Definition Digit.h:30
void setEventTime(const o2::InteractionTimeRecord &irt)
void process(const std::vector< itsmft::Hit > *hits, int evID, int srcID)
Steer conversion of hits to digits.
Definition Digitizer.cxx:89
void fillOutputContainer(uint32_t maxFrame=0xffffffff)
std::vector< int > row