Project
Loading...
Searching...
No Matches
HardwareClusterDecoder.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
14
20#include "TPCBase/Mapper.h"
21#include <algorithm>
22#include <vector>
23#include <numeric> // std::iota
24#include <fairlogger/Logger.h>
25
31
32using namespace o2::tpc;
33using namespace o2;
34using namespace o2::dataformats;
35
36int HardwareClusterDecoder::decodeClusters(std::vector<std::pair<const ClusterHardwareContainer*, std::size_t>>& inputClusters,
40{
41 if (mIntegrator == nullptr) {
42 mIntegrator.reset(new DigitalCurrentClusterIntegrator);
43 }
44 // MCLabelContainer does only allow appending new labels, so we need to write to separate
45 // containers per {sector,padrow} and merge at the end;
46 std::vector<o2::dataformats::MCTruthContainer<o2::MCCompLabel>> outMCLabelContainers;
47 if (!inMCLabels) {
48 outMCLabels = nullptr;
49 }
50 ClusterNative* outputClusterBuffer = nullptr;
51 // the number of clusters in a {sector,row}
52 int nRowClusters[constants::MAXSECTOR][constants::MAXGLOBALPADROW] = {0};
53 // offset of first cluster of {sector,row} in the output buffer
54 size_t clusterOffsets[constants::MAXSECTOR][constants::MAXGLOBALPADROW] = {0};
55 int containerRowCluster[constants::MAXSECTOR][constants::MAXGLOBALPADROW] = {0};
56 Mapper& mapper = Mapper::instance();
57 int numberOfOutputContainers = 0;
58 for (int loop = 0; loop < 2; loop++) {
59 int nTotalClusters = 0;
60 for (int i = 0; i < inputClusters.size(); i++) {
61 if (outMCLabels && inputClusters[i].second > 1) {
62 LOG(error) << "Decoding of ClusterHardware to ClusterNative with MC labels is yet only support for single 8kb pages of ClusterHardwareContainer\n";
63 return (1);
64 }
65 for (int j = 0; j < inputClusters[i].second; j++) {
66 const char* tmpPtr = reinterpret_cast<const char*>(inputClusters[i].first);
67 tmpPtr += j * 8192; //TODO: FIXME: Compute correct offset based on the size of the actual packet in the RDH
68 const ClusterHardwareContainer& cont = *(reinterpret_cast<const ClusterHardwareContainer*>(tmpPtr));
69 const CRU cru(cont.CRU);
70 const Sector sector = cru.sector();
71 const PadRegionInfo& region = mapper.getPadRegionInfo(cru.region());
72 const int rowOffset = region.getGlobalRowOffset();
73
74 // TODO: make sure that input clusters are sorted in ascending row, so we
75 // can write the MCLabels directly in consecutive order following the cluster sequence
76 // Note: also the sorting below would need to be adjusted.
77 for (int k = 0; k < cont.numberOfClusters; k++) {
78 const int padRowGlobal = rowOffset + cont.clusters[k].getRow();
79 int& nCls = nRowClusters[sector][padRowGlobal];
80 if (loop == 1) {
81 //Fill cluster in the respective output buffer
82 const ClusterHardware& cIn = cont.clusters[k];
83 ClusterNative& cOut = outputClusterBuffer[clusterOffsets[sector][padRowGlobal] + nCls];
84 float pad = cIn.getPad();
85 cOut.setPad(pad);
86 cOut.setTimeFlags(cIn.getTimeLocal() + cont.timeBinOffset, cIn.getFlags());
87 cOut.setSigmaPad(std::sqrt(cIn.getSigmaPad2()));
88 cOut.setSigmaTime(std::sqrt(cIn.getSigmaTime2()));
89 cOut.qMax = cIn.getQMax();
90 cOut.qTot = cIn.getQTot();
91 mIntegrator->integrateCluster(sector, padRowGlobal, pad, cIn.getQTot());
92 if (outMCLabels) {
93 auto& mcOut = outMCLabelContainers[containerRowCluster[sector][padRowGlobal]];
94 for (const auto& element : (*inMCLabels)[i].getLabels(k)) {
95 mcOut.addElement(nCls, element);
96 }
97 }
98 } else {
99 //Count how many output buffers we need (and how large they are below)
100 if (nCls == 0) {
101 numberOfOutputContainers++;
102 }
103 }
104 nCls++;
105 nTotalClusters++;
106 }
107 }
108 }
109 if (loop == 1) {
110 //We are done with filling the buffers, sort all output buffers
111 for (int i = 0; i < constants::MAXSECTOR; i++) {
112 for (int j = 0; j < constants::MAXGLOBALPADROW; j++) {
113 if (nRowClusters[i][j] == 0) {
114 continue;
115 }
116 if (outMCLabels) {
117 sortClustersAndMC(outputClusterBuffer + clusterOffsets[i][j], nRowClusters[i][j], outMCLabelContainers[containerRowCluster[i][j]]);
118 } else {
119 auto* cl = outputClusterBuffer + clusterOffsets[i][j];
120 std::sort(cl, cl + nRowClusters[i][j]);
121 }
122 }
123 }
124 } else {
125 //Now we know the size of all output buffers, allocate them
126 if (outMCLabels) {
127 outMCLabelContainers.resize(numberOfOutputContainers);
128 }
129 size_t rawOutputBufferSize = sizeof(ClusterCountIndex) + nTotalClusters * sizeof(ClusterNative);
130 char* rawOutputBuffer = outputAllocator(rawOutputBufferSize);
131 auto& clusterCounts = *(reinterpret_cast<ClusterCountIndex*>(rawOutputBuffer));
132 outputClusterBuffer = reinterpret_cast<ClusterNative*>(rawOutputBuffer + sizeof(ClusterCountIndex));
133 nTotalClusters = 0;
134 numberOfOutputContainers = 0;
135 for (int i = 0; i < constants::MAXSECTOR; i++) {
136 for (int j = 0; j < constants::MAXGLOBALPADROW; j++) {
137 clusterCounts.nClusters[i][j] = nRowClusters[i][j];
138 if (nRowClusters[i][j] == 0) {
139 continue;
140 }
141 containerRowCluster[i][j] = numberOfOutputContainers++;
142 clusterOffsets[i][j] = nTotalClusters;
143 nTotalClusters += nRowClusters[i][j];
144 mIntegrator->initRow(i, j);
145 }
146 }
147 memset(nRowClusters, 0, sizeof(nRowClusters));
148 }
149 }
150 // Finally merge MC label containers into one container following the cluster sequence in the
151 // output buffer
152 if (outMCLabels) {
153 auto& labels = *outMCLabels;
154 int nCls = 0;
155 for (int i = 0; i < constants::MAXSECTOR; i++) {
156 for (int j = 0; j < constants::MAXGLOBALPADROW; j++) {
157 if (nRowClusters[i][j] == 0) {
158 continue;
159 }
160 for (int k = 0, end = outMCLabelContainers[containerRowCluster[i][j]].getIndexedSize(); k < end; k++, nCls++) {
161 assert(end == nRowClusters[i][j]);
162 assert(clusterOffsets[i][j] + k == nCls);
163 for (const auto& element : outMCLabelContainers[containerRowCluster[i][j]].getLabels(k)) {
164 labels.addElement(nCls, element);
165 }
166 }
167 }
168 }
169 }
170 return (0);
171}
172
175{
176 std::vector<unsigned int> indizes(nClusters);
177 std::iota(indizes.begin(), indizes.end(), 0);
178 std::sort(indizes.begin(), indizes.end(), [&clusters](const auto a, const auto b) {
179 return clusters[a] < clusters[b];
180 });
181 std::vector<ClusterNative> buffer(clusters, clusters + nClusters);
182 ClusterNativeHelper::MCLabelContainer tmpMC = std::move(mcTruth);
183 assert(mcTruth.getIndexedSize() == 0);
184 for (int i = 0; i < nClusters; i++) {
185 clusters[i] = buffer[indizes[i]];
186 for (auto const& label : tmpMC.getLabels(indizes[i])) {
187 mcTruth.addElement(i, label);
188 }
189 }
190}
Class of a TPC cluster as produced by the hardware cluster finder (needs a postprocessing step to con...
Helper class to read the binary format of TPC ClusterNative.
Class of a TPC cluster in TPC-native coordinates (row, time)
int32_t i
Decoder to convert TPC ClusterHardware to ClusterNative.
Definition of a container to keep Monte Carlo truth external to simulation objects.
Definition of the parameter class for the detector.
Definition of the parameter class for the detector electronics.
Definition of the parameter class for the detector gas.
uint32_t j
Definition RawData.h:0
int nClusters
A container to hold and manage MC truth information/labels.
gsl::span< TruthElement > getLabels(uint32_t dataindex)
void addElement(uint32_t dataindex, TruthElement const &element, bool noElement=false)
unsigned char region() const
Definition CRU.h:64
const Sector sector() const
Definition CRU.h:65
static void sortClustersAndMC(ClusterNative *clusters, size_t nClusters, o2::dataformats::MCTruthContainer< o2::MCCompLabel > &mcTruth)
Sort clusters and MC labels in place ClusterNative defines the smaller-than relation used in the sort...
std::function< char *(size_t)> OutputAllocator
Allocator function object to provide the output buffer.
int decodeClusters(std::vector< std::pair< const o2::tpc::ClusterHardwareContainer *, std::size_t > > &inputClusters, OutputAllocator outputAllocator, const std::vector< o2::dataformats::ConstMCTruthContainerView< o2::MCCompLabel > > *inMCLabels=nullptr, o2::dataformats::MCTruthContainer< o2::MCCompLabel > *outMCLabels=nullptr)
Decode clusters provided in raw pages The function uses an allocator object to request a raw char buf...
static Mapper & instance(const std::string mappingDir="")
Definition Mapper.h:44
const PadRegionInfo & getPadRegionInfo(const unsigned char region) const
Definition Mapper.h:385
unsigned char getGlobalRowOffset() const
GLuint buffer
Definition glcorearb.h:655
GLuint GLuint end
Definition glcorearb.h:469
GLboolean GLboolean GLboolean b
Definition glcorearb.h:1233
GLuint GLsizei const GLchar * label
Definition glcorearb.h:2519
GLboolean GLboolean GLboolean GLboolean a
Definition glcorearb.h:1233
Definition of a container to keep/associate and arbitrary number of labels associated to an index wit...
constexpr int MAXSECTOR
Definition Constants.h:28
constexpr int MAXGLOBALPADROW
Definition Constants.h:34
Global TPC definitions and constants.
Definition SimTraits.h:167
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"
std::vector< Cluster > clusters