Project
Loading...
Searching...
No Matches
Clusterer.h
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
14#ifndef ALICEO2_ITS_CLUSTERER_H
15#define ALICEO2_ITS_CLUSTERER_H
16
17#define _PERFORM_TIMING_
18
19// uncomment this to not allow diagonal clusters, e.g. like |* |
20// | *|
21#define _ALLOW_DIAGONAL_ALPIDE_CLUSTERS_
22
23#include <utility>
24#include <vector>
25#include <cstring>
26#include <memory>
27#include <gsl/span>
28
37
38#ifdef _PERFORM_TIMING_
39#include <TStopwatch.h>
40#endif
41
42class TTree;
43
44namespace o2
45{
46class MCCompLabel;
47namespace dataformats
48{
49template <typename T>
50class ConstMCTruthContainerView;
51template <typename T>
53} // namespace dataformats
54
55namespace its3
56{
57
58using CompClusCont = std::vector<itsmft::CompClusterExt>;
59using PatternCont = std::vector<unsigned char>;
60using ROFRecCont = std::vector<itsmft::ROFRecord>;
61
62// template <class CompClusCont, class PatternCont, class ROFRecCont> // container types (PMR or std::vectors)
63
65{
71 using Label = o2::MCCompLabel;
74
75 public:
76 static constexpr int MaxLabels = 10;
77 static constexpr int MaxHugeClusWarn = 5; // max number of warnings for HugeCluster
78
79 struct BBox {
80 uint16_t chipID = 0xffff;
81 uint16_t rowMin = 0xffff;
82 uint16_t colMin = 0xffff;
83 uint16_t rowMax = 0;
84 uint16_t colMax = 0;
85 BBox(uint16_t c) : chipID(c) {}
86 bool isInside(uint16_t row, uint16_t col) const { return row >= rowMin && row <= rowMax && col >= colMin && col <= colMax; }
87 [[nodiscard]] auto rowSpan() const { return rowMax - rowMin + 1; }
88 [[nodiscard]] auto colSpan() const { return colMax - colMin + 1; }
90 void clear()
91 {
92 rowMin = colMin = 0xffff;
93 rowMax = colMax = 0;
94 }
95 void adjust(uint16_t row, uint16_t col)
96 {
97 if (row < rowMin) {
98 rowMin = row;
99 }
100 if (row > rowMax) {
101 rowMax = row;
102 }
103 if (col < colMin) {
104 colMin = col;
105 }
106 if (col > colMax) {
107 colMax = col;
108 }
109 }
110 };
111
112 //=========================================================
114 struct ThreadStat {
115 uint16_t firstChip = 0;
116 uint16_t nChips = 0;
117 uint32_t firstClus = 0;
118 uint32_t firstPatt = 0;
119 uint32_t nClus = 0;
120 uint32_t nPatt = 0;
121 };
122
124 Clusterer* parent = nullptr; // parent clusterer
125 int id = -1;
126 // buffers for entries in preClusterIndices in 2 columns, to avoid boundary checks, we reserve
127 // extra elements in the beginning and the end
128 int* column1 = nullptr;
129 int* column2 = nullptr;
130 int* curr = nullptr; // pointer on the 1st row of currently processed columnsX
131 int* prev = nullptr; // pointer on the 1st row of previously processed columnsX
133 // pixels[].first is the index of the next pixel of the same precluster in the pixels
134 // pixels[].second is the index of the referred pixel in the ChipPixelData (element of mChips)
135 std::vector<std::pair<int, uint32_t>> pixels;
136 std::vector<int> preClusterHeads; // index of precluster head in the pixels
137 std::vector<int> preClusterIndices;
138 uint16_t currCol = 0xffff;
139 bool noLeftCol = true;
140 std::array<Label, MaxLabels> labelsBuff;
141 std::vector<PixelData> pixArrBuff;
142 //
147 std::vector<ThreadStat> stats; // statistics for each thread results, used at merging
150 void resetColumn(int* buff) const { std::memset(buff, -1, sizeof(int) * (size - 2)); }
151
153 void swapColumnBuffers() { std::swap(prev, curr); }
154
156 void expandPreCluster(uint32_t ip, uint16_t row, int preClusIndex)
157 {
158 auto& firstIndex = preClusterHeads[preClusterIndices[preClusIndex]];
159 pixels.emplace_back(firstIndex, ip);
160 firstIndex = pixels.size() - 1;
161 curr[row] = preClusIndex;
162 }
163
165 void addNewPrecluster(uint32_t ip, uint16_t row)
166 {
167 preClusterHeads.push_back(pixels.size());
168 // new head does not point yet (-1) on other pixels, store just the entry of the pixel in the ChipPixelData
169 pixels.emplace_back(-1, ip);
170 int lastIndex = preClusterIndices.size();
171 preClusterIndices.push_back(lastIndex);
172 curr[row] = lastIndex; // store index of the new precluster in the current column buffer
173 }
174
175 void fetchMCLabels(int digID, const ConstMCTruth* labelsDig, int& nfilled);
176 void initChip(const ChipPixelData* curChipData, uint32_t first);
177 void updateChip(const ChipPixelData* curChipData, uint32_t ip);
178 void finishChip(ChipPixelData* curChipData, CompClusCont* compClus, PatternCont* patterns,
179 const ConstMCTruth* labelsDig, MCTruth* labelsClus);
180 void finishChipSingleHitFast(uint32_t hit, ChipPixelData* curChipData, CompClusCont* compClusPtr,
181 PatternCont* patternsPtr, const ConstMCTruth* labelsDigPtr, MCTruth* labelsClusPTr);
182 void process(uint16_t chip, uint16_t nChips, CompClusCont* compClusPtr, PatternCont* patternsPtr,
183 const ConstMCTruth* labelsDigPtr, MCTruth* labelsClPtr, const itsmft::ROFRecord& rofPtr);
184
186 {
187 delete[] column1;
188 delete[] column2;
189 }
190
191 ClustererThread(Clusterer* par = nullptr, int _id = -1) : parent(par), id(_id), curr(column2 + 1), prev(column1 + 1) {}
196 };
197 //=========================================================
198
199 Clusterer();
200 Clusterer(Clusterer&&) = delete;
202 ~Clusterer() = default;
203 Clusterer(const Clusterer&) = delete;
204 Clusterer& operator=(const Clusterer&) = delete;
205
206 void process(int nThreads, PixelReader& r, CompClusCont* compClus, PatternCont* patterns, ROFRecCont* vecROFRec, MCTruth* labelsCl = nullptr);
207
208 template <typename VCLUS, typename VPAT>
209 static void streamCluster(const std::vector<PixelData>& pixbuf, const std::array<Label, MaxLabels>* lblBuff, const BBox& bbox, const its3::LookUp& pattIdConverter,
210 VCLUS* compClusPtr, VPAT* patternsPtr, MCTruth* labelsClusPtr, int nlab, bool isHuge = false);
211
212 bool isContinuousReadOut() const { return mContinuousReadout; }
213 void setContinuousReadOut(bool v) { mContinuousReadout = v; }
214
215 int getMaxBCSeparationToMask() const { return mMaxBCSeparationToMask; }
216 void setMaxBCSeparationToMask(int n) { mMaxBCSeparationToMask = n; }
217
218 int getMaxRowColDiffToMask() const { return mMaxRowColDiffToMask; }
219 void setMaxRowColDiffToMask(int v) { mMaxRowColDiffToMask = v; }
220
221 int getMaxROFDepthToSquash() const { return mSquashingDepth; }
222 void setMaxROFDepthToSquash(int v) { mSquashingDepth = v; }
223
224 int getMaxBCSeparationToSquash() const { return mMaxBCSeparationToSquash; }
225 void setMaxBCSeparationToSquash(int n) { mMaxBCSeparationToSquash = n; }
226
227 void print() const;
228 void clear();
229
232 {
233 LOGP(info, "Setting TopologyDictionary with size={}", dict->getSize());
234 mPattIdConverter.setDictionary(dict);
235 // dict->print();
236 }
237
238 TStopwatch& getTimer() { return mTimer; } // cannot be const
239 TStopwatch& getTimerMerge() { return mTimerMerge; } // cannot be const
240
241 void setNChips(int n)
242 {
243 mChips.resize(n);
244 mChipsOld.resize(n);
245 }
246
247 private:
248 void flushClusters(CompClusCont* compClus, MCTruth* labels);
249
250 // clusterization options
251 bool mContinuousReadout = true;
252
254 int mMaxBCSeparationToMask = static_cast<int>(6000. / o2::constants::lhc::LHCBunchSpacingNS + 10);
255 int mMaxRowColDiffToMask = 0;
256 int mNHugeClus = 0;
257
259 int mSquashingDepth = 0;
260 int mMaxBCSeparationToSquash = 6000. / o2::constants::lhc::LHCBunchSpacingNS + 10;
261
262 std::vector<std::unique_ptr<ClustererThread>> mThreads; // buffers for threads
263 std::vector<ChipPixelData> mChips; // currently processed ROF's chips data
264 std::vector<ChipPixelData> mChipsOld; // previously processed ROF's chips data (for masking)
265 std::vector<ChipPixelData*> mFiredChipsPtr; // pointers on the fired chips data in the decoder cache
266
267 its3::LookUp mPattIdConverter;
268 //
269#ifdef _PERFORM_TIMING_
270 TStopwatch mTimer;
271 TStopwatch mTimerMerge;
272#endif
273};
274
275template <typename VCLUS, typename VPAT>
276void Clusterer::streamCluster(const std::vector<PixelData>& pixbuf, const std::array<Label, MaxLabels>* lblBuff, const Clusterer::BBox& bbox, const its3::LookUp& pattIdConverter,
277 VCLUS* compClusPtr, VPAT* patternsPtr, MCTruth* labelsClusPtr, int nlab, bool isHuge)
278{
279 if (labelsClusPtr && lblBuff) { // MC labels were requested
280 auto cnt = compClusPtr->size();
281 for (int i = nlab; i--;) {
282 labelsClusPtr->addElement(cnt, (*lblBuff)[i]);
283 }
284 }
285 auto colSpanW = bbox.colSpan();
286 auto rowSpanW = bbox.rowSpan();
287 // add to compact clusters, which must be always filled
288 std::array<unsigned char, itsmft::ClusterPattern::MaxPatternBytes> patt{};
289 for (const auto& pix : pixbuf) {
290 uint32_t ir = pix.getRowDirect() - bbox.rowMin, ic = pix.getCol() - bbox.colMin;
291 int nbits = ir * colSpanW + ic;
292 patt[nbits >> 3] |= (0x1 << (7 - (nbits % 8)));
293 }
294 uint16_t pattID = (isHuge || pattIdConverter.size() == 0) ? CompCluster::InvalidPatternID : pattIdConverter.findGroupID(rowSpanW, colSpanW, patt.data());
295 uint16_t row = bbox.rowMin, col = bbox.colMin;
296 LOGP(debug, "PattID: findGroupID({},{},{})={}", row, col, patt[0], pattID);
297 if (pattID == CompCluster::InvalidPatternID || pattIdConverter.isGroup(pattID)) {
298 if (pattID != CompCluster::InvalidPatternID) {
299 // For groupped topologies, the reference pixel is the COG pixel
300 float xCOG = 0., zCOG = 0.;
301 itsmft::ClusterPattern::getCOG(rowSpanW, colSpanW, patt.data(), xCOG, zCOG);
302 row += round(xCOG);
303 col += round(zCOG);
304 }
305 if (patternsPtr) {
306 patternsPtr->emplace_back((unsigned char)rowSpanW);
307 patternsPtr->emplace_back((unsigned char)colSpanW);
308 int nBytes = rowSpanW * colSpanW / 8;
309 if (((rowSpanW * colSpanW) % 8) != 0) {
310 nBytes++;
311 }
312 patternsPtr->insert(patternsPtr->end(), std::begin(patt), std::begin(patt) + nBytes);
313 }
314 }
315 compClusPtr->emplace_back(row, col, pattID, bbox.chipID);
316}
317
318} // namespace its3
319} // namespace o2
320#endif /* ALICEO2_ITS_CLUSTERER_H */
Definition of the ITSMFT compact cluster.
int32_t i
Definition of the ITSMFT ROFrame (trigger) record.
Header to collect LHC related constants.
Transient data classes for single pixel and set of pixels from current chip.
Abstract class for Alpide data reader class.
uint32_t col
Definition RawData.h:4
uint32_t c
Definition RawData.h:2
Definition of the SegmentationAlpide class.
Definition of the LookUp class for its3.
std::ostringstream debug
void addElement(uint32_t dataindex, TruthElement const &element, bool noElement=false)
void setDictionary(const its3::TopologyDictionary *dict)
Definition Clusterer.h:231
void setContinuousReadOut(bool v)
Definition Clusterer.h:213
static constexpr int MaxLabels
Definition Clusterer.h:76
int getMaxBCSeparationToMask() const
Definition Clusterer.h:215
int getMaxRowColDiffToMask() const
Definition Clusterer.h:218
TStopwatch & getTimerMerge()
Definition Clusterer.h:239
int getMaxROFDepthToSquash() const
Definition Clusterer.h:221
Clusterer & operator=(Clusterer &&)=delete
void setMaxBCSeparationToMask(int n)
Definition Clusterer.h:216
void setNChips(int n)
Definition Clusterer.h:241
bool isContinuousReadOut() const
Definition Clusterer.h:212
Clusterer(Clusterer &&)=delete
void clear()
load the dictionary of cluster topologies
Clusterer & operator=(const Clusterer &)=delete
int getMaxBCSeparationToSquash() const
Definition Clusterer.h:224
static void streamCluster(const std::vector< PixelData > &pixbuf, const std::array< Label, MaxLabels > *lblBuff, const BBox &bbox, const its3::LookUp &pattIdConverter, VCLUS *compClusPtr, VPAT *patternsPtr, MCTruth *labelsClusPtr, int nlab, bool isHuge=false)
Definition Clusterer.h:276
void setMaxRowColDiffToMask(int v)
Definition Clusterer.h:219
void print() const
TStopwatch & getTimer()
Definition Clusterer.h:238
Clusterer(const Clusterer &)=delete
void setMaxBCSeparationToSquash(int n)
Definition Clusterer.h:225
static constexpr int MaxHugeClusWarn
Definition Clusterer.h:77
void setMaxROFDepthToSquash(int v)
Definition Clusterer.h:222
int size() const
Definition LookUp.h:40
void setDictionary(const TopologyDictionary *dict)
Definition LookUp.cxx:37
int getSize() const
Returns the number of elements in the dicionary;.
static int getCOG(int rowSpan, int colSpan, const unsigned char patt[MaxPatternBytes], float &xCOG, float &zCOG)
Static: Compute pattern's COG position. Returns the number of fired pixels.
static constexpr uint8_t MaxRowSpan
static constexpr uint8_t MaxColSpan
static constexpr unsigned short InvalidPatternID
Definition CompCluster.h:46
< single pixel datum, with possibility to set a flag of pixel being masked out
Definition PixelData.h:33
PixelReader class for the ITSMFT.
Definition PixelReader.h:34
GLdouble n
Definition glcorearb.h:1982
GLsizeiptr size
Definition glcorearb.h:659
const GLdouble * v
Definition glcorearb.h:832
GLint GLint GLsizei GLint GLenum GLenum const void * pixels
Definition glcorearb.h:275
GLboolean r
Definition glcorearb.h:1233
GLuint id
Definition glcorearb.h:650
constexpr double LHCBunchSpacingNS
std::vector< unsigned char > PatternCont
Definition Clusterer.h:59
std::vector< itsmft::CompClusterExt > CompClusCont
Definition Clusterer.h:58
std::vector< itsmft::ROFRecord > ROFRecCont
Definition Clusterer.h:60
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
bool isInside(uint16_t row, uint16_t col) const
Definition Clusterer.h:86
bool isAcceptableSize() const
Definition Clusterer.h:89
void adjust(uint16_t row, uint16_t col)
Definition Clusterer.h:95
ClustererThread & operator=(ClustererThread &&)=delete
std::vector< PixelData > pixArrBuff
temporary buffer for building cluster labels
Definition Clusterer.h:141
void addNewPrecluster(uint32_t ip, uint16_t row)
Definition Clusterer.h:165
std::vector< ThreadStat > stats
Definition Clusterer.h:147
void finishChipSingleHitFast(uint32_t hit, ChipPixelData *curChipData, CompClusCont *compClusPtr, PatternCont *patternsPtr, const ConstMCTruth *labelsDigPtr, MCTruth *labelsClusPTr)
std::array< Label, MaxLabels > labelsBuff
Definition Clusterer.h:140
std::vector< int > preClusterIndices
Definition Clusterer.h:137
void updateChip(const ChipPixelData *curChipData, uint32_t ip)
void expandPreCluster(uint32_t ip, uint16_t row, int preClusIndex)
add new precluster at given row of current column for the fired pixel with index ip in the ChipPixelD...
Definition Clusterer.h:156
ClustererThread(const ClustererThread &)=delete
void fetchMCLabels(int digID, const ConstMCTruth *labelsDig, int &nfilled)
std::vector< std::pair< int, uint32_t > > pixels
Definition Clusterer.h:135
ClustererThread(ClustererThread &&)=delete
bool noLeftCol
flag that there is no column on the left to check
Definition Clusterer.h:139
ClustererThread & operator=(const ClustererThread &)=delete
ClustererThread(Clusterer *par=nullptr, int _id=-1)
Definition Clusterer.h:191
CompClusCont compClusters
temporary buffer for pattern calc.
Definition Clusterer.h:144
void initChip(const ChipPixelData *curChipData, uint32_t first)
void resetColumn(int *buff) const
reset column buffer, for the performance reasons we use memset
Definition Clusterer.h:150
void finishChip(ChipPixelData *curChipData, CompClusCont *compClus, PatternCont *patterns, const ConstMCTruth *labelsDig, MCTruth *labelsClus)
void swapColumnBuffers()
add cluster at row (entry ip in the ChipPixeData) to the precluster with given index
Definition Clusterer.h:153
uint16_t currCol
Column being processed.
Definition Clusterer.h:138
methods and transient data used within a thread
Definition Clusterer.h:114
o2::InteractionRecord ir(0, 0)
std::vector< int > row
LookUp pattIdConverter