44#include <oneapi/tbb.h>
49using namespace std::string_literals;
56 uint32_t nClSector = 0;
64 for (uint32_t k = 0; k < native->
nClusters[
i][
j]; k++) {
66 float x = 0,
y = 0,
z = 0;
67 if (continuousMaxTimeBin == 0) {
68 transform->Transform(
i,
j, clin.getPad(), clin.getTime(),
x,
y,
z);
70 transform->TransformInTimeFrame(
i,
j, clin.getPad(), clin.getTime(),
x,
y,
z, continuousMaxTimeBin);
72 auto& clout =
clusters[
i].get()[nClSector];
77 clout.amp = clin.qTot;
78 clout.flags = clin.getFlags();
90 memset((
void*)&native, 0,
sizeof(native));
92 for (uint32_t
j = 0;
j < nRawClusters[
i];
j++) {
104 for (uint32_t
j = 0;
j < nRawClusters[
i];
j++) {
123 for (uint32_t k = 0; k < native.
nClusters[
i][
j]; k++) {
137 for (uint32_t k = 0; k <
digits.nTPCDigits[
i]; k++) {
153 const char* page = (
const char*)zspages.
sector[
i].
zsPtr[
j][k];
160 if (hdr2->
flags & TPCZSHDRV2::ZSFlags::nTimeBinSpanBit8) {
177#ifdef GPUCA_TPC_GEOMETRY_O2
188 int32_t curRegion = 0, outputRegion = 0;
189 uint32_t encodeBits = 0;
190 uint32_t zsVersion = 0;
191 uint32_t iSector = 0;
197 int32_t
endpoint = 0, outputEndpoint = 0;
198 int64_t hbf = -1, nexthbf = 0;
199 zsPage* page =
nullptr;
200 uint8_t* pagePtr =
nullptr;
201 int32_t bcShiftInFirstHBF = 0;
202 int32_t firstTimebinInPage = -1;
203 float encodeBitsFactor = 0;
204 bool needAnotherPage =
false;
205 uint32_t packetCounter = 0;
206 uint32_t pageCounter = 0;
207 void ZSfillEmpty(
void*
ptr, int32_t shift, uint32_t feeId, int32_t
orbit, int32_t linkid);
208 static void ZSstreamOut(uint16_t* bufIn, uint32_t& lenIn, uint8_t* bufOut, uint32_t& lenOut, uint32_t nBits);
212inline void zsEncoder::ZSfillEmpty(
void*
ptr, int32_t shift, uint32_t feeId, int32_t
orbit, int32_t linkid)
216 o2::raw::RDHUtils::setHeartBeatBC(*rdh, shift);
226inline void zsEncoder::ZSstreamOut(uint16_t* bufIn, uint32_t& lenIn, uint8_t* bufOut, uint32_t& lenOut, uint32_t nBits)
228 uint32_t
byte = 0,
bits = 0;
229 uint32_t
mask = (1 << nBits) - 1;
230 for (uint32_t
i = 0;
i < lenIn;
i++) {
234 bufOut[lenOut++] = (
uint8_t)(
byte & 0xFF);
240 bufOut[lenOut++] = byte;
249static inline auto ZSEncoderGetDigits(
const DigitArray& in, int32_t
i) {
return in[
i].data(); }
250static inline auto ZSEncoderGetNDigits(
const DigitArray& in, int32_t
i) {
return in[
i].size(); }
255struct zsEncoderRow :
public zsEncoder {
256 std::array<uint16_t, TPCZSHDR::TPC_ZS_PAGE_SIZE> streamBuffer = {};
257 std::array<uint8_t, TPCZSHDR::TPC_ZS_PAGE_SIZE> streamBuffer8 = {};
262 int32_t endpointStart = 0;
263 int32_t nRowsInTB = 0;
264 uint32_t streamSize = 0, streamSize8 = 0;
265 constexpr static int32_t RAWLNK = rdh_utils::UserLogicLinkID;
267 bool checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
271 uint32_t encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
274 void decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* page, uint32_t
endpoint, uint32_t firstOrbit, uint32_t triggerBC = 0);
279 int32_t endpointa = GPUTPCGeometry::GetRegion(
a.getRow());
280 int32_t endpointb = GPUTPCGeometry::GetRegion(
b.getRow());
281 endpointa = 2 * endpointa + (
a.getRow() >= GPUTPCGeometry::GetRegionStart(endpointa) + GPUTPCGeometry::GetRegionRows(endpointa) / 2);
282 endpointb = 2 * endpointb + (
b.getRow() >= GPUTPCGeometry::GetRegionStart(endpointb) + GPUTPCGeometry::GetRegionRows(endpointb) / 2);
283 if (endpointa != endpointb) {
284 return endpointa <= endpointb;
286 if (
a.getTimeStamp() !=
b.getTimeStamp()) {
287 return a.getTimeStamp() <
b.getTimeStamp();
289 if (
a.getRow() !=
b.getRow()) {
290 return a.getRow() <
b.getRow();
292 return a.getPad() <
b.getPad();
295bool zsEncoderRow::checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
298 if (lastRow != tmpBuffer[k].getRow()) {
299 endpointStart = GPUTPCGeometry::GetRegionStart(curRegion);
301 if (tmpBuffer[k].getRow() >= endpointStart + GPUTPCGeometry::GetRegionRows(curRegion) / 2) {
303 endpointStart += GPUTPCGeometry::GetRegionRows(curRegion) / 2;
306 for (uint32_t l = k + 1; l < tmpBuffer.size(); l++) {
307 if (tmpBuffer[l].getRow() == tmpBuffer[k].getRow() && tmpBuffer[l].
getTimeStamp() == tmpBuffer[k].
getTimeStamp() && tmpBuffer[l].getPad() == tmpBuffer[l - 1].getPad() + 1) {
313 if (lastEndpoint >= 0 && lastTime != -1 && (int32_t)hdr->nTimeBinSpan + tmpBuffer[k].getTimeStamp() - lastTime >= 256) {
317 uint32_t sizeChk = (uint32_t)(pagePtr -
reinterpret_cast<uint8_t*
>(page));
318 sizeChk += 2 * (nRowsInTB + (tmpBuffer[k].getRow() != lastRow && tmpBuffer[k].getTimeStamp() == lastTime));
319 sizeChk += streamSize8;
320 sizeChk += (lastTime != tmpBuffer[k].getTimeStamp()) && ((sizeChk + (streamSize * encodeBits + 7) / 8) & 1);
321 sizeChk += (tmpBuffer[k].getTimeStamp() != lastTime || tmpBuffer[k].getRow() != lastRow) ? 3 : 0;
322 sizeChk += (lastTime != -1 && tmpBuffer[k].getTimeStamp() > lastTime) ? ((tmpBuffer[k].
getTimeStamp() - lastTime - 1) * 2) : 0;
324 const uint32_t streamSizeChkBits = streamSize * encodeBits + ((lastTime != tmpBuffer[k].getTimeStamp() && (streamSize * encodeBits) % 8) ? (8 - (streamSize * encodeBits) % 8) : 0);
333 return endpoint != lastEndpoint || tmpBuffer[k].getTimeStamp() != lastTime;
336bool zsEncoderRow::writeSubPage()
338 if (pagePtr !=
reinterpret_cast<uint8_t*
>(page)) {
339 pagePtr += 2 * nRowsInTB;
340 ZSstreamOut(streamBuffer.data(), streamSize, streamBuffer8.data(), streamSize8, encodeBits);
341 pagePtr = std::copy(streamBuffer8.data(), streamBuffer8.data() + streamSize8, pagePtr);
342 if (pagePtr -
reinterpret_cast<uint8_t*
>(page) > 8192) {
343 throw std::runtime_error(
"internal error during ZS encoding");
346 for (int32_t l = 1; l < nRowsInTB; l++) {
347 curTBHdr->rowAddr1()[l - 1] += 2 * nRowsInTB;
353uint32_t zsEncoderRow::encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
356 if (lastTime != -1) {
357 hdr->nTimeBinSpan += tmpBuffer[k].getTimeStamp() - lastTime - 1;
358 pagePtr += (tmpBuffer[k].getTimeStamp() - lastTime - 1) * 2;
361 if ((pagePtr -
reinterpret_cast<uint8_t*
>(page)) & 1) {
364 curTBHdr =
reinterpret_cast<TPCZSTBHDR*
>(pagePtr);
369 if (tmpBuffer[k].getRow() != lastRow) {
370 curTBHdr->rowMask |= 1 << (tmpBuffer[k].getRow() - endpointStart);
371 ZSstreamOut(streamBuffer.data(), streamSize, streamBuffer8.data(), streamSize8, encodeBits);
373 curTBHdr->rowAddr1()[nRowsInTB - 1] = (pagePtr -
reinterpret_cast<uint8_t*
>(page)) + streamSize8;
376 nSeq = streamBuffer8.data() + streamSize8++;
380 streamBuffer8[streamSize8++] = tmpBuffer[k].getPad();
381 streamBuffer8[streamSize8++] = streamSize + seqLen;
382 for (int32_t l = 0; l < seqLen; l++) {
383 streamBuffer[streamSize++] = (uint16_t)(tmpBuffer[k + l].getChargeFloat() * encodeBitsFactor + 0.5f);
388void zsEncoderRow::decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* decPage, uint32_t decEndpoint, uint32_t firstOrbit, uint32_t triggerBC)
390 const uint8_t* decPagePtr =
reinterpret_cast<const uint8_t*
>(decPage);
397 decPagePtr +=
sizeof(*decHDR);
399 throw std::runtime_error(
"invalid ZS version "s +
std::to_string(decHDR->
version) +
" (1 or 2 expected)"s);
401 const float decodeBitsFactor = 1.f / (1 << (encodeBits - 10));
402 uint32_t
mask = (1 << encodeBits) - 1;
403 int32_t cruid = decHDR->
cruID;
404 uint32_t sector = cruid / 10;
405 if (sector != iSector) {
406 throw std::runtime_error(
"invalid TPC sector");
408 int32_t region = cruid % 10;
409 if ((uint32_t)region != decEndpoint / 2) {
410 throw std::runtime_error(
"CRU ID / endpoint mismatch");
412 int32_t nRowsRegion = GPUTPCGeometry::GetRegionRows(region);
416 if ((decPagePtr -
reinterpret_cast<const uint8_t*
>(decPage)) & 1) {
420 bool upperRows = tbHdr->
rowMask & 0x8000;
421 if (tbHdr->
rowMask != 0 && ((upperRows) ^ ((decEndpoint & 1) != 0))) {
422 throw std::runtime_error(
"invalid endpoint");
424 const int32_t rowOffset = GPUTPCGeometry::GetRegionStart(region) + (upperRows ? (nRowsRegion / 2) : 0);
425 const int32_t
nRows = upperRows ? (nRowsRegion - nRowsRegion / 2) : (nRowsRegion / 2);
426 const int32_t nRowsUsed = __builtin_popcount((uint32_t)(tbHdr->
rowMask & 0x7FFF));
427 decPagePtr += nRowsUsed ? (2 * nRowsUsed) : 2;
429 for (int32_t
m = 0;
m <
nRows;
m++) {
430 if ((tbHdr->
rowMask & (1 <<
m)) == 0) {
433 const uint8_t* rowData = rowPos == 0 ? decPagePtr : (
reinterpret_cast<const uint8_t*
>(decPage) + tbHdr->rowAddr1()[rowPos - 1]);
434 const int32_t nSeqRead = *rowData;
435 const uint8_t* adcData = rowData + 2 * nSeqRead + 1;
436 int32_t nADC = (rowData[2 * nSeqRead] * encodeBits + 7) / 8;
437 decPagePtr += 1 + 2 * nSeqRead + nADC;
438 uint32_t
byte = 0,
bits = 0, posXbits = 0;
439 std::array<uint16_t, TPCZSHDR::TPC_ZS_PAGE_SIZE> decBuffer;
440 for (int32_t
n = 0;
n < nADC;
n++) {
441 byte |= *(adcData++) <<
bits;
443 while (
bits >= encodeBits) {
444 decBuffer[posXbits++] =
byte &
mask;
445 byte =
byte >> encodeBits;
450 for (int32_t
n = 0;
n < nSeqRead;
n++) {
451 const int32_t decSeqLen = rowData[(
n + 1) * 2] - (
n ? rowData[
n * 2] : 0);
452 for (int32_t
o = 0;
o < decSeqLen;
o++) {
464struct zsEncoderLinkBased :
public zsEncoder {
467 int32_t inverseChannelMapping[5][32];
468 int32_t nSamples = 0;
470 bool finishPage =
false;
471 std::vector<uint16_t> adcValues = {};
472 std::bitset<80> bitmask = {};
474 void createBitmask(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
479void zsEncoderLinkBased::init()
482 for (int32_t
i = 0;
i < 5;
i++) {
483 for (int32_t
j = 0;
j < 32;
j++) {
484 inverseChannelMapping[
i][
j] = -1;
487 for (int32_t iCRU = 0; iCRU < 2; iCRU++) {
488 for (int32_t iChannel = 0; iChannel < 80; iChannel++) {
489 int32_t sampaOnFEC = 0, channelOnSAMPA = 0;
491 if (inverseChannelMapping[sampaOnFEC][channelOnSAMPA] != -1 && inverseChannelMapping[sampaOnFEC][channelOnSAMPA] != iChannel) {
492 GPUError(
"ERROR: Channel conflict: %d %d: %d vs %d", sampaOnFEC, channelOnSAMPA, inverseChannelMapping[sampaOnFEC][channelOnSAMPA], iChannel);
493 throw std::runtime_error(
"ZS error");
495 inverseChannelMapping[sampaOnFEC][channelOnSAMPA] = iChannel;
498 for (int32_t
i = 0;
i < 5;
i++) {
499 for (int32_t
j = 0;
j < 32;
j++) {
500 if (inverseChannelMapping[
i][
j] == -1) {
501 GPUError(
"ERROR: Map missing for sampa %d channel %d",
i,
j);
502 throw std::runtime_error(
"ZS error");
508void zsEncoderLinkBased::createBitmask(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
515 for (l = k; l < tmpBuffer.size(); l++) {
516 const auto&
a = tmpBuffer[l];
517 int32_t cruinsector = GPUTPCGeometry::GetRegion(
a.getRow());
521 int32_t fecInPartition = fec.
getIndex() - mapper.getPartitionInfo(cru.
partition()).getSectorFECOffset();
522 int32_t tmpEndpoint = 2 * cruinsector + (fecInPartition >= (mapper.getPartitionInfo(cru.
partition()).getNumberOfFECs() + 1) / 2);
524 link = fecInPartition;
530 bitmask[channel] = 1;
531 adcValues.emplace_back((uint16_t)(
a.getChargeFloat() * encodeBitsFactor + 0.5f));
539 int32_t cruinsectora = GPUTPCGeometry::GetRegion(
a.getRow());
540 int32_t cruinsectorb = GPUTPCGeometry::GetRegion(
b.getRow());
541 if (cruinsectora != cruinsectorb) {
542 return cruinsectora < cruinsectorb;
550 int32_t fecInPartitiona = feca.
getIndex() - mapper.getPartitionInfo(cru.
partition()).getSectorFECOffset();
551 int32_t fecInPartitionb = fecb.
getIndex() - mapper.getPartitionInfo(cru.
partition()).getSectorFECOffset();
553 int32_t endpointa = 2 * cruinsectora + (fecInPartitiona >= (mapper.getPartitionInfo(cru.
partition()).getNumberOfFECs() + 1) / 2);
554 int32_t endpointb = 2 * cruinsectorb + (fecInPartitionb >= (mapper.getPartitionInfo(cru.
partition()).getNumberOfFECs() + 1) / 2);
555 if (endpointa != endpointb) {
556 return endpointa < endpointb;
558 if (
a.getTimeStamp() !=
b.getTimeStamp()) {
559 return a.getTimeStamp() <
b.getTimeStamp();
561 if (fecInPartitiona != fecInPartitionb) {
562 return fecInPartitiona < fecInPartitionb;
569struct zsEncoderImprovedLinkBased :
public zsEncoderLinkBased {
570 bool checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
571 uint32_t encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
572 void decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* page, uint32_t
endpoint, uint32_t firstOrbit, uint32_t triggerBC = 0);
576 constexpr static int32_t RAWLNK = rdh_utils::ILBZSLinkID;
579bool zsEncoderImprovedLinkBased::checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
581 createBitmask(tmpBuffer, k);
582 finishPage =
endpoint != lastEndpoint;
583 if (firstTimebinInPage != -1 && tmpBuffer[k].
getTimeStamp() - firstTimebinInPage >= 1 << (
sizeof(hdr->nTimeBinSpan) * 8)) {
587 uint32_t sizeChk = (uint32_t)(pagePtr -
reinterpret_cast<uint8_t*
>(page));
601uint32_t zsEncoderImprovedLinkBased::encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
604 pagePtr +=
sizeof(*tbHdr);
605 tbHdr->
bunchCrossing = (tmpBuffer[k].getTimeStamp() - firstTimebinInPage) * LHCBCPERTIMEBIN;
608 tbHdr->
bitMaskLow = (bitmask & std::bitset<80>(0xFFFFFFFFFFFFFFFFlu)).to_ulong();
611 hdr->nTimeBinSpan = tmpBuffer[k].getTimeStamp() - firstTimebinInPage;
612 hdr->nTimebinHeaders++;
616 uint32_t tmpIn = nSamples;
617 ZSstreamOut(adcValues.data(), tmpIn, pagePtr, tmp, encodeBits);
620 uint64_t* payloadPtr = (uint64_t*)pagePtr;
624 for (uint32_t
i = 0;
i < nSamples;
i++) {
632bool zsEncoderImprovedLinkBased::writeSubPage()
637void zsEncoderImprovedLinkBased::initPage()
640 hdr->nTimebinHeaders = 0;
641 hdr->firstZSDataOffset = 0;
644void zsEncoderImprovedLinkBased::decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* decPage, uint32_t decEndpoint, uint32_t firstOrbit, uint32_t triggerBC)
647 const uint8_t* decPagePtr =
reinterpret_cast<const uint8_t*
>(decPage);
654 decPagePtr +=
sizeof(*decHDR);
655 if (decHDR->
version != ZSVersion::ZSVersionLinkBasedWithMeta) {
659 throw std::runtime_error(
"Magic word missing");
661 const float decodeBitsFactor = 1.f / (1 << (encodeBits - 10));
662 uint32_t
mask = (1 << encodeBits) - 1;
663 int32_t cruid = decHDR->
cruID;
664 uint32_t sector = cruid / 10;
665 if (sector != iSector) {
666 throw std::runtime_error(
"invalid TPC sector");
668 int32_t region = cruid % 10;
674 const int32_t feeLink = tbHdr->
fecInPartition - (decEndpoint & 1) * ((mapper.getPartitionInfo(cru.
partition()).getNumberOfFECs() + 1) / 2);
675 auto fillADC = [&outputBuffer](int32_t cru, int32_t rowInSector, int32_t padInRow, int32_t timeBin,
float adcValue) {
676 outputBuffer.emplace_back(
o2::tpc::Digit{cruid, adcValue, rowInSector, padInRow, timeBin});
680 raw_processing_helpersa::processZSdata((
const char*)decPagePtr,
size, rdh_utils::getFEEID(cruid, decEndpoint & 1, feeLink), o2::raw::RDHUtils::getHeartBeatOrbit(*rdh), firstOrbit, decHDR->
timeOffset, fillADC);
683 throw std::runtime_error(
"ZS TB Hdr does not have linkZS magic word");
687 LOGP(
debug,
"zsEncoderImprovedLinkBased::decodePage skipping digits hdr->tOff {} + hdr->bc {} + (orbit {} - firstOrbit {}) * maxBunch {} - triggerBC {} = {} < 0", decHDR->
timeOffset, tbHdr->
bunchCrossing, o2::raw::RDHUtils::getHeartBeatOrbit(*rdh), firstOrbit,
o2::constants::lhc::LHCMaxBunches, triggerBC, timeBin);
690 const uint8_t* adcData = (
const uint8_t*)(decPagePtr +
sizeof(*tbHdr));
692 int32_t nADC = bitmask.count();
693 std::vector<uint16_t> decBuffer(nADC);
695 uint32_t
byte = 0,
bits = 0, posXbits = 0;
696 while (posXbits < nADC) {
697 byte |= *(adcData++) <<
bits;
699 while (
bits >= encodeBits) {
700 decBuffer[posXbits++] =
byte &
mask;
701 byte =
byte >> encodeBits;
706 const uint64_t* adcData64 = (
const uint64_t*)adcData;
707 for (int32_t
j = 0;
j < nADC;
j++) {
711 for (int32_t
j = 0, k = 0;
j < bitmask.size();
j++) {
713 int32_t sampaOnFEC = 0, channelOnSAMPA = 0;
714 mapper.getSampaAndChannelOnFEC(cruid,
j, sampaOnFEC, channelOnSAMPA);
715 const auto padSecPos = mapper.padSecPos(cruid, tbHdr->
fecInPartition, sampaOnFEC, channelOnSAMPA);
716 const auto& padPos = padSecPos.getPadPos();
727struct zsEncoderDenseLinkBased :
public zsEncoderLinkBased {
728 bool checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
729 uint32_t encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k);
730 void decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* page, uint32_t
endpoint, uint32_t firstOrbit, uint32_t triggerBC = 0);
733 void amendPageErrorMessage(std::ostringstream& oss,
const o2::header::RAWDataHeader* rdh,
const TPCZSHDRV2* decHDR,
const uint8_t* payloadEnd,
const uint8_t* decPagePtr, uint32_t nOutput);
735 uint16_t curTimeBin = 0;
736 std::vector<uint8_t> sequenceBuffer;
737 std::vector<uint16_t> sequenceBufferADC;
739 constexpr static int32_t RAWLNK = rdh_utils::DLBZSLinkID;
740 constexpr static int32_t v2nbits = 10;
743bool zsEncoderDenseLinkBased::checkInput(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
745 createBitmask(tmpBuffer, k);
746 finishPage =
endpoint != lastEndpoint;
747 uint16_t newTimeBin = tmpBuffer[k].getTimeStamp() - firstTimebinInPage;
748 bool retVall = finishPage || newTimeBin != curTimeBin;
752uint32_t zsEncoderDenseLinkBased::encodeSequence(std::vector<o2::tpc::Digit>& tmpBuffer, uint32_t k)
754 if (sequenceBuffer.size() == 0) {
756 if (zsVersion == ZSVersion::ZSVersionDenseLinkBasedV2) {
759 sequenceBuffer.emplace_back(
bc << 4);
760 sequenceBuffer.emplace_back(
bc >> 4);
761 curTimeBin = tmpBuffer[k].getTimeStamp() - firstTimebinInPage;
762 hdr->nTimeBinSpan = curTimeBin & 0xFF;
763 if (curTimeBin & 0x100) {
764 hdr->flags |= TPCZSHDRV2::ZSFlags::nTimeBinSpanBit8;
766 hdr->nTimebinHeaders++;
770 sequenceBuffer.emplace_back(link);
771 uint8_t* plink = &sequenceBuffer.back();
773 std::bitset<10> bitmaskL2;
774 for (int32_t
i = 9;
i >= 0;
i--) {
775 bitmaskL2.set(
i, ((bitmask >> (
i * 8)) & std::bitset<80>(0xFF)).any());
777 if (bitmaskL2.all()) {
778 *plink |= 0b00100000;
780 *plink |= (bitmaskL2.to_ulong() >> 2) & 0b11000000;
781 sequenceBuffer.emplace_back(bitmaskL2.to_ulong() & 0xFF);
784 for (int32_t
i = 0;
i < 10;
i++) {
785 if (bitmaskL2.test(
i)) {
786 sequenceBuffer.emplace_back(((bitmask >> (
i * 8)) & std::bitset<80>(0xFF)).to_ulong());
792 sequenceBufferADC.insert(sequenceBufferADC.end(), adcValues.begin(), adcValues.end());
798bool zsEncoderDenseLinkBased::writeSubPage()
800 uint32_t
offset = sequenceBuffer.size();
801 if (sequenceBufferADC.size()) {
802 bool need12bit = zsVersion != ZSVersion::ZSVersionDenseLinkBasedV2;
803 uint32_t needNow = 0;
804 if (zsVersion == ZSVersion::ZSVersionDenseLinkBasedV2) {
805 for (uint32_t
i = 0;
i < sequenceBufferADC.size();
i++) {
806 if (sequenceBufferADC[
i] >= (1 << v2nbits)) {
812 uint32_t encodeBitsBlock = encodeBits;
814 encodeBitsBlock = v2nbits;
815 sequenceBuffer[0] |= 0x10;
817 sequenceBuffer.resize(
offset + (sequenceBufferADC.size() * encodeBitsBlock + 7) / 8);
819 uint32_t tmpIn = sequenceBufferADC.size();
820 ZSstreamOut(sequenceBufferADC.data(), tmpIn, sequenceBuffer.data() +
offset, tmp, encodeBitsBlock);
821 sequenceBufferADC.clear();
824 if (sequenceBuffer.size()) {
826 uint32_t
size = sequenceBuffer.size();
827 uint32_t fill = std::min(sizeLeft,
size);
828 memcpy(pagePtr, sequenceBuffer.data(), fill);
832 sequenceBuffer.erase(sequenceBuffer.begin(), sequenceBuffer.begin() + fill);
834 sequenceBuffer.clear();
836 finishPage = finishPage ||
size >= sizeLeft || needAnotherPage;
842void zsEncoderDenseLinkBased::initPage()
845 hdr->nTimebinHeaders = 0;
846 memcpy(pagePtr, sequenceBuffer.data(), sequenceBuffer.size());
848 pagePtr += sequenceBuffer.size();
849 sequenceBuffer.clear();
853void zsEncoderDenseLinkBased::decodePage(std::vector<o2::tpc::Digit>& outputBuffer,
const zsPage* decPage, uint32_t decEndpoint, uint32_t firstOrbit, uint32_t triggerBC)
856 const uint8_t* decPagePtr =
reinterpret_cast<const uint8_t*
>(decPage);
863 if (decHDR->
version < ZSVersion::ZSVersionDenseLinkBased || decHDR->
version > ZSVersion::ZSVersionDenseLinkBasedV2) {
867 throw std::runtime_error(
"Magic word missing");
870 const float decodeBitsFactor = 1.f / (1 << (encodeBits - 10));
871 int32_t cruid = decHDR->
cruID;
872 uint32_t sector = cruid / 10;
873 if (sector != iSector) {
874 throw std::runtime_error(
"invalid TPC sector");
876 int32_t region = cruid % 10;
878 std::vector<uint8_t> tmpBuffer;
879 bool extendFailure =
false;
880 uint32_t nOutput = 0;
881 uint32_t minTimeBin = -1, maxTimeBin = 0;
883 int32_t sizeLeftInPage = payloadEnd - decPagePtr;
884 if (sizeLeftInPage <= 0) {
885 throw std::runtime_error(
"Decoding ran beyond end of page before processing extended timebin");
889 throw std::runtime_error(
"pageExtends signaled, but current page is not full");
895 if ((uint16_t)(o2::raw::RDHUtils::getPageCounter(*rdh) + 1) != o2::raw::RDHUtils::getPageCounter(*rdhNext)) {
896 GPUError(
"Incomplete HBF: Payload extended to next page, but next page missing in stream (packet counters %d %d)", (int32_t)o2::raw::RDHUtils::getPageCounter(*rdh), (int32_t)o2::raw::RDHUtils::getPageCounter(*rdhNext));
897 extendFailure =
true;
898 decPagePtr = payloadEnd;
904 memcpy(tmpBuffer.data(), decPagePtr, sizeLeftInPage);
906 decPagePtr = tmpBuffer.data();
907 payloadEnd = decPagePtr + tmpBuffer.size();
910 uint16_t linkBC = (*((
const uint16_t*)decPagePtr) & 0xFFF0) >> 4;
911 bool v2Flag = decHDR->
version == ZSVersion::ZSVersionDenseLinkBasedV2 && *((
const uint8_t*)decPagePtr) & 0x10;
912 if (decHDR->
version == ZSVersion::ZSVersionDenseLinkBasedV2) {
915 decPagePtr +=
sizeof(uint16_t);
916 std::vector<int32_t>
links;
917 std::vector<std::bitset<80>> bitmasks;
918 uint32_t nTotalSamples = 0;
919 for (uint32_t l = 0; l < linkCount; l++) {
922 uint8_t decLink = decLinkX & 0b00011111;
923 std::bitset<10> bitmaskL2;
924 if (decLinkX & 0b00100000) {
927 bitmaskL2 = std::bitset<10>(((((uint16_t)decLinkX) & 0b11000000) << 2) | (uint16_t)*((
const uint8_t*)decPagePtr));
931 std::bitset<80> bitmask(0);
932 for (int32_t
i = 0;
i < 10;
i++) {
933 if (bitmaskL2.test(
i)) {
934 bitmask |= std::bitset<80>(*((
const uint8_t*)decPagePtr)) <<
i * 8;
938 links.emplace_back(decLink);
939 bitmasks.emplace_back(bitmask);
940 nTotalSamples += bitmask.count();
944 int32_t encodeBitsBlock = v2Flag ? v2nbits : encodeBits;
945 decPagePtr += (nTotalSamples * encodeBitsBlock + 7) / 8;
948 int32_t timeBin = (int32_t(linkBC) + (int32_t)(o2::raw::RDHUtils::getHeartBeatOrbit(*rdh) - firstOrbit) *
o2::constants::lhc::LHCMaxBunches - int32_t(triggerBC)) / LHCBCPERTIMEBIN;
949 if (timeBin < 0 || nTotalSamples == 0) {
950 if (timeBin < 0 && nTotalSamples > 0) {
951 LOGP(
debug,
"zsEncoderDenseLinkBased::decodePage skipping digits (linkBC {} + orbit {} - firstOrbit {}) * maxBunch {} - triggerBC {} = {} < 0, nTotalSamples {}", linkBC, o2::raw::RDHUtils::getHeartBeatOrbit(*rdh), firstOrbit,
o2::constants::lhc::LHCMaxBunches, triggerBC, timeBin, nTotalSamples);
955 if (timeBin > maxTimeBin) {
956 maxTimeBin = timeBin;
958 if (timeBin < minTimeBin) {
959 minTimeBin = timeBin;
962 std::vector<uint16_t>
samples(nTotalSamples);
963 uint32_t
mask = (1 << encodeBitsBlock) - 1;
964 uint32_t
byte = 0,
bits = 0, posXbits = 0;
965 while (posXbits < nTotalSamples) {
966 byte |= *(adcData++) <<
bits;
968 while (
bits >= encodeBitsBlock && posXbits < nTotalSamples) {
970 byte =
byte >> encodeBitsBlock;
971 bits -= encodeBitsBlock;
974 uint32_t samplePos = 0;
976 for (uint32_t l = 0; l < linkCount; l++) {
978 const auto& bitmask = bitmasks[l];
979 int32_t nADC = bitmask.count();
981 for (int32_t
j = 0;
j < bitmask.size();
j++) {
983 int32_t sampaOnFEC = 0, channelOnSAMPA = 0;
984 mapper.getSampaAndChannelOnFEC(cruid,
j, sampaOnFEC, channelOnSAMPA);
985 const auto padSecPos = mapper.padSecPos(cruid, decLink, sampaOnFEC, channelOnSAMPA);
986 const auto& padPos = padSecPos.getPadPos();
995 if (triggerBC > 0 && hdrMinTimeBin < 0) {
999 int32_t hdrMaxTimeBin = hdrMinTimeBin + decHDR->
nTimeBinSpan + ((decHDR->
flags & TPCZSHDRV2::ZSFlags::nTimeBinSpanBit8) ? 256 : 0);
1001 if (!extendFailure && nOutput != decHDR->
nADCsamples) {
1002 std::ostringstream oss;
1003 oss <<
"Number of decoded digits " << nOutput <<
" does not match value from MetaInfo " << decHDR->
nADCsamples;
1004 amendPageErrorMessage(oss, rdh, decHDR,
nullptr,
nullptr, nOutput);
1005 throw std::runtime_error(oss.str());
1008 if (decHDR->
nADCsamples && (minTimeBin < hdrMinTimeBin || maxTimeBin > hdrMaxTimeBin)) {
1009 std::ostringstream oss;
1010 oss <<
"Incorrect time bin range in MetaInfo, header reports " << hdrMinTimeBin <<
" - " << hdrMaxTimeBin <<
"(timeOffset: " << decHDR->
timeOffset <<
" + (orbit: " << o2::raw::RDHUtils::getHeartBeatOrbit(*rdh) <<
" - firstOrbit " << firstOrbit <<
") * LHCMaxBunches - triggerBC: " << triggerBC <<
", decoded data is " << minTimeBin <<
" - " << maxTimeBin;
1011 amendPageErrorMessage(oss, rdh, decHDR, payloadEnd, decPagePtr, nOutput);
1012 throw std::runtime_error(oss.str());
1016 std::ostringstream oss;
1017 oss <<
"Decoding ran over end of page";
1018 amendPageErrorMessage(oss, rdh, decHDR, payloadEnd, decPagePtr, nOutput);
1019 throw std::runtime_error(oss.str());
1022 std::ostringstream oss;
1023 oss <<
"Decoding didn't reach end of page";
1024 amendPageErrorMessage(oss, rdh, decHDR, payloadEnd, decPagePtr, nOutput);
1025 throw std::runtime_error(oss.str());
1029void zsEncoderDenseLinkBased::amendPageErrorMessage(std::ostringstream& oss,
const o2::header::RAWDataHeader* rdh,
const TPCZSHDRV2* decHDR,
const uint8_t* payloadEnd,
const uint8_t* decPagePtr, uint32_t nOutput)
1031 if (payloadEnd && decPagePtr) {
1032 oss <<
" (payloadEnd " << (
void*)payloadEnd <<
" - decPagePtr " << (
void*)decPagePtr <<
" - " << (payloadEnd - decPagePtr) <<
" bytes left, " << nOutput <<
" of " << decHDR->
nADCsamples <<
" digits decoded)\n";
1036 constexpr size_t bufferSize = 3 * std::max(
sizeof(*rdh),
sizeof(*decHDR)) + 1;
1038 for (
size_t i = 0;
i <
sizeof(*rdh);
i++) {
1039 snprintf(dumpBuffer + 3 *
i, 4,
"%02X ", (int32_t)((uint8_t*)rdh)[
i]);
1041 oss <<
"RDH of page: " <<
dumpBuffer <<
"\n";
1042 for (
size_t i = 0;
i <
sizeof(*decHDR);
i++) {
1043 snprintf(dumpBuffer + 3 *
i, 4,
"%02X ", (int32_t)((uint8_t*)decHDR)[
i]);
1045 oss <<
"Meta header of page: " <<
dumpBuffer <<
"\n";
1053struct zsEncoderRun :
public T {
1054 uint32_t run(std::vector<zsPage>*
buffer, std::vector<o2::tpc::Digit>& tmpBuffer,
size_t* totalSize =
nullptr);
1055 size_t compare(std::vector<zsPage>*
buffer, std::vector<o2::tpc::Digit>& tmpBuffer);
1057 using T::bcShiftInFirstHBF;
1058 using T::checkInput;
1060 using T::decodePage;
1061 using T::encodeBits;
1062 using T::encodeBitsFactor;
1063 using T::encodeSequence;
1065 using T::firstTimebinInPage;
1073 using T::lastEndpoint;
1076 using T::needAnotherPage;
1078 using T::outputEndpoint;
1079 using T::outputRegion;
1080 using T::packetCounter;
1083 using T::pageCounter;
1088 using T::writeSubPage;
1089 using T::ZSfillEmpty;
1094inline uint32_t zsEncoderRun<T>::run(std::vector<zsPage>*
buffer, std::vector<o2::tpc::Digit>& tmpBuffer,
size_t* totalSize)
1096 uint32_t totalPages = 0;
1097 zsPage singleBuffer;
1102 bcShiftInFirstHBF =
ir ?
ir->
bc : 0;
1103 int32_t orbitShift =
ir ?
ir->
orbit : 0;
1105 int32_t rawendpoint = 0;
1106 (
void)(rawcru + rawendpoint);
1107 encodeBitsFactor = (1 << (encodeBits - 10));
1110 for (uint32_t k = 0; k <= tmpBuffer.size();) {
1111 bool mustWritePage =
false, mustWriteSubPage =
false;
1112 if (needAnotherPage) {
1113 needAnotherPage =
false;
1114 mustWritePage =
true;
1116 if (k < tmpBuffer.size()) {
1120 throw std::runtime_error(
"Received digit before the defined first orbit");
1122 if (hbf != nexthbf) {
1124 mustWritePage =
true;
1127 if (lastRow != tmpBuffer[k].getRow()) {
1128 curRegion = GPUTPCGeometry::GetRegion(tmpBuffer[k].getRow());
1130 mustWriteSubPage = checkInput(tmpBuffer, k);
1133 mustWritePage =
true;
1136 if (mustWritePage || mustWriteSubPage) {
1137 mustWritePage |= writeSubPage();
1139 if (page && mustWritePage) {
1140 if constexpr (std::is_same_v<T, struct zsEncoderDenseLinkBased>) {
1144 uint8_t* triggerWord =
nullptr;
1145 if (hbf != nexthbf ||
endpoint != lastEndpoint) {
1152 needAnotherPage =
true;
1154 if (this->sequenceBuffer.size()) {
1155 needAnotherPage =
true;
1159 triggerWord = pagePtr;
1168 throw std::runtime_error(
"TPC ZS page overflow");
1170 memcpy(pagehdr, hdr,
sizeof(*hdr));
1177 *totalSize += !std::is_same_v<T, struct zsEncoderDenseLinkBased> && (lastEndpoint == -1 || hbf == nexthbf) ?
TPCZSHDR::TPC_ZS_PAGE_SIZE : (pagePtr - (
uint8_t*)page);
1180 size = CAMath::nextMultipleOf<o2::raw::RDHUtils::GBTWord128>(
size);
1184 maxhbf = std::max<int32_t>(maxhbf, hbf);
1185 minhbf = std::min<int32_t>(minhbf, hbf);
1191 o2::raw::RDHUtils::setHeartBeatBC(*rdh, bcShiftInFirstHBF);
1201 if (k >= tmpBuffer.size() && !needAnotherPage) {
1205 if (mustWritePage) {
1206 if (!needAnotherPage) {
1207 if (hbf != nexthbf) {
1210 outputRegion = curRegion;
1217 page = &singleBuffer;
1219 if (
buffer[outputEndpoint].
size() == 0 && nexthbf > orbitShift) {
1220 buffer[outputEndpoint].emplace_back();
1221 ZSfillEmpty(&
buffer[outputEndpoint].back(), bcShiftInFirstHBF, rdh_utils::getFEEID(iSector * 10 + outputEndpoint / 2, outputEndpoint & 1, this->RAWLNK), orbitShift, this->RAWLNK);
1227 buffer[outputEndpoint].emplace_back();
1228 page = &
buffer[outputEndpoint].back();
1230 pagePtr =
reinterpret_cast<uint8_t*
>(page);
1231 std::fill(page->begin(), page->end(), 0);
1233 if constexpr (std::is_same_v<T, struct zsEncoderDenseLinkBased>) {
1234 hdr = &this->hdrBuffer;
1236 hdr =
reinterpret_cast<decltype(hdr)
>(pagePtr);
1237 pagePtr +=
sizeof(*hdr);
1239 hdr->version = zsVersion;
1240 hdr->cruID = iSector * 10 + outputRegion;
1241 hdr->nTimeBinSpan = 0;
1242 hdr->nADCsamples = 0;
1243 rawcru = iSector * 10 + outputRegion;
1244 rawendpoint = outputEndpoint & 1;
1246 firstTimebinInPage = tmpBuffer[k].getTimeStamp();
1250 if (needAnotherPage) {
1253 uint32_t nEncoded = encodeSequence(tmpBuffer, k);
1254 lastTime = tmpBuffer[k].getTimeStamp();
1255 lastRow = tmpBuffer[k].getRow();
1256 hdr->nADCsamples += nEncoded;
1262 for (int32_t
i = minhbf;
i <= maxhbf;
i++) {
1271 ZSfillEmpty(&
buffer[
j].back(), bcShiftInFirstHBF, rdh_utils::getFEEID(iSector * 10 +
j / 2,
j & 1, this->RAWLNK), orbitShift, this->RAWLNK);
1280size_t zsEncoderRun<T>::compare(std::vector<zsPage>*
buffer, std::vector<o2::tpc::Digit>& tmpBuffer)
1283 std::vector<o2::tpc::Digit> compareBuffer;
1284 compareBuffer.reserve(tmpBuffer.size());
1286 uint32_t firstOrbit =
ir ?
ir->
orbit : 0;
1287 for (uint32_t k = 0; k <
buffer[
j].size(); k++) {
1288 zsPage* decPage = &
buffer[
j][k];
1289 decodePage(compareBuffer, decPage,
j, firstOrbit);
1292 if (tmpBuffer.size() != compareBuffer.size()) {
1293 nErrors += tmpBuffer.size();
1294 printf(
"Number of clusters mismatch %d %d\n", (int32_t)tmpBuffer.size(), (int32_t)compareBuffer.size());
1296 for (uint32_t
j = 0;
j < tmpBuffer.size();
j++) {
1297 const float decodeBitsFactor = (1 << (encodeBits - 10));
1298 const float c = CAMath::Round(tmpBuffer[
j].getChargeFloat() * decodeBitsFactor) / decodeBitsFactor;
1299 int32_t ok =
c == compareBuffer[
j].getChargeFloat() && (int32_t)tmpBuffer[
j].
getTimeStamp() == (int32_t)compareBuffer[
j].
getTimeStamp() && (int32_t)tmpBuffer[
j].getPad() == (int32_t)compareBuffer[
j].getPad() && (int32_t)tmpBuffer[
j].getRow() == (int32_t)compareBuffer[
j].getRow();
1304 printf(
"%4u: OK %d: Charge %3d %3d Time %4d %4d Pad %3d %3d Row %3d %3d\n",
j, ok,
1305 (int32_t)
c, (int32_t)compareBuffer[
j].getChargeFloat(), (int32_t)tmpBuffer[
j].
getTimeStamp(), (int32_t)compareBuffer[
j].
getTimeStamp(), (int32_t)tmpBuffer[
j].getPad(), (int32_t)compareBuffer[
j].getPad(), (int32_t)tmpBuffer[
j].getRow(), (int32_t)compareBuffer[
j].getRow());
1316void GPUReconstructionConvert::RunZSEncoder(
const S& in, std::unique_ptr<uint64_t[]>* outBuffer, uint32_t* outSizes,
o2::raw::RawFileWriter* raw,
const o2::InteractionRecord*
ir,
const GPUParam&
param, int32_t
version,
bool verify,
float threshold,
bool padding, std::function<
void(std::vector<o2::tpc::Digit>&)> digitsFilter)
1320 if (((outBuffer ==
nullptr) ^ (outSizes ==
nullptr)) || ((raw !=
nullptr) && (
ir ==
nullptr)) || !((outBuffer ==
nullptr) ^ (raw ==
nullptr)) || (raw && verify)) {
1321 throw std::runtime_error(
"Invalid parameters");
1323#ifdef GPUCA_TPC_GEOMETRY_O2
1325 struct tmpReductionResult {
1326 uint32_t totalPages = 0;
1327 size_t totalSize = 0;
1329 size_t digitsInput = 0;
1330 size_t digitsEncoded = 0;
1332 auto reduced = tbb::parallel_reduce(tbb::blocked_range<uint32_t>(0,
NSECTORS), tmpReductionResult(), [&](
const auto range,
auto red) {
1334 std::vector<o2::tpc::Digit> tmpBuffer;
1335 red.digitsInput += ZSEncoderGetNDigits(in,
i);
1336 tmpBuffer.resize(ZSEncoderGetNDigits(in,
i));
1337 if (threshold > 0.f && !digitsFilter) {
1338 auto it = std::copy_if(ZSEncoderGetDigits(in,
i), ZSEncoderGetDigits(in,
i) + ZSEncoderGetNDigits(in,
i), tmpBuffer.begin(), [threshold](
auto&
v) { return v.getChargeFloat() >= threshold; });
1339 tmpBuffer.resize(std::distance(tmpBuffer.begin(), it));
1341 std::copy(ZSEncoderGetDigits(in,
i), ZSEncoderGetDigits(in,
i) + ZSEncoderGetNDigits(in,
i), tmpBuffer.begin());
1345 digitsFilter(tmpBuffer);
1346 if (threshold > 0.f) {
1347 std::vector<o2::tpc::Digit> tmpBuffer2 = std::move(tmpBuffer);
1348 tmpBuffer = std::vector<o2::tpc::Digit>(tmpBuffer2.size());
1349 auto it = std::copy_if(tmpBuffer2.begin(), tmpBuffer2.end(), tmpBuffer.begin(), [threshold](
auto&
v) { return v.getChargeFloat() >= threshold; });
1350 tmpBuffer.resize(std::distance(tmpBuffer.begin(), it));
1353 red.digitsEncoded += tmpBuffer.size();
1355 auto runZS = [&](
auto& encoder) {
1358 red.totalPages += encoder.run(
buffer[
i], tmpBuffer, &red.totalSize);
1360 red.nErrors += encoder.compare(
buffer[
i], tmpBuffer);
1364 if (
version >= ZSVersion::ZSVersionRowBased10BitADC &&
version <= ZSVersion::ZSVersionRowBased12BitADC) {
1365 zsEncoderRun<zsEncoderRow> enc{{{.iSector =
i, .raw = raw, .ir =
ir, .param = &
param, .padding =
padding}}};
1367 }
else if (
version >= ZSVersion::ZSVersionLinkBasedWithMeta &&
version <= ZSVersion::ZSVersionDenseLinkBasedV2) {
1369 if (
version == ZSVersion::ZSVersionLinkBasedWithMeta) {
1370 zsEncoderRun<zsEncoderImprovedLinkBased> enc{{{{.iSector =
i, .raw = raw, .ir =
ir, .param = &
param, .padding =
padding}}}};
1372 }
else if (
version >= ZSVersion::ZSVersionDenseLinkBased &&
version <= ZSVersion::ZSVersionDenseLinkBasedV2) {
1373 zsEncoderRun<zsEncoderDenseLinkBased> enc{{{{.iSector =
i, .raw = raw, .ir =
ir, .param = &
param, .padding =
padding}}}};
1377 throw std::runtime_error(
"Link based ZS encoding not supported in standalone build");
1383 return red; }, [&](
const auto& red1,
const auto& red2) {
1385 red.totalPages += red2.totalPages;
1386 red.totalSize += red2.totalSize;
1387 red.nErrors += red2.nErrors;
1388 red.digitsInput += red2.digitsInput;
1389 red.digitsEncoded += red2.digitsEncoded;
1403 if (reduced.nErrors) {
1404 GPUError(
"ERROR: %lu INCORRECT SAMPLES DURING ZS ENCODING VERIFICATION!!!", reduced.nErrors);
1405 }
else if (verify) {
1406 GPUInfo(
"ENCODING VERIFICATION PASSED");
1408 GPUInfo(
"TOTAL ENCODED SIZE: %lu (%lu of %lu digits encoded)", reduced.totalSize, reduced.digitsEncoded, reduced.digitsInput);
1412template void GPUReconstructionConvert::RunZSEncoder<GPUTrackingInOutDigits>(
const GPUTrackingInOutDigits&, std::unique_ptr<uint64_t[]>*, uint32_t*,
o2::raw::RawFileWriter*,
const o2::InteractionRecord*,
const GPUParam&, int32_t,
bool,
float,
bool, std::function<
void(std::vector<o2::tpc::Digit>&)> digitsFilter);
1414template void GPUReconstructionConvert::RunZSEncoder<DigitArray>(
const DigitArray&, std::unique_ptr<uint64_t[]>*, uint32_t*,
o2::raw::RawFileWriter*,
const o2::InteractionRecord*,
const GPUParam&, int32_t,
bool,
float,
bool, std::function<
void(std::vector<o2::tpc::Digit>&)> digitsFilter);
1434 if (
buffers[
i].get() != ptrs[
i] || nsb != ns) {
1435 throw std::runtime_error(
"Not owning digits");
1439 const float decodeBitsFactor = (1 << (decodeBits - 10));
1440 for (uint32_t k = 0; k < ns[
i]; k++) {
1441 if (
buffers[
i][k].getChargeFloat() >= threshold) {
1446 buffers[
i][
j].setCharge(CAMath::Round(
buffers[
i][
j].getChargeFloat() * decodeBitsFactor) / decodeBitsFactor);
1463 std::shared_ptr<T> enc = std::make_shared<T>();
1464 if (
param ==
nullptr) {
1466 param = &dummyParam;
1471 return [enc](std::vector<o2::tpc::Digit>& outBuffer,
const void* page, uint32_t firstTfOrbit, uint32_t triggerBC = 0) {
1476 if (o2::raw::RDHUtils::getDetectorField(rdh) != 2) {
1480 enc->iSector = cru.sector();
1481 int32_t
endpoint = cru.region() * 2 + o2::tpc::rdh_utils::getEndPoint(rdh);
1482 enc->decodePage(outBuffer, (
const zsPage*)page,
endpoint, firstTfOrbit, triggerBC);
1490 return o2::gpu::internal::GetDecoder_internal<zsEncoderRow>(
param,
version);
1492 return o2::gpu::internal::GetDecoder_internal<zsEncoderImprovedLinkBased>(
param,
version);
1494 return o2::gpu::internal::GetDecoder_internal<zsEncoderDenseLinkBased>(
param,
version);
1496 throw std::runtime_error(
"Invalid ZS version "s +
std::to_string(
version) +
", cannot create decoder"s);
Header to collect LHC related constants.
Utility class to write detectors data to (multiple) raw data file(s) respecting CRU format.
definitions to deal with the link based zero suppression format
Definitions of TPC Zero Suppression Data Headers.
static void RunZSEncoder(const S &in, std::unique_ptr< uint64_t[]> *outBuffer, uint32_t *outSizes, o2::raw::RawFileWriter *raw, const o2::InteractionRecord *ir, const GPUParam ¶m, int32_t version, bool verify, float threshold=0.f, bool padding=false, std::function< void(std::vector< o2::tpc::Digit > &)> digitsFilter=nullptr)
static void ConvertNativeToClusterData(o2::tpc::ClusterNativeAccess *native, std::unique_ptr< GPUTPCClusterData[]> *clusters, uint32_t *nClusters, const TPCFastTransform *transform, int32_t continuousMaxTimeBin=0)
static void RunZSFilter(std::unique_ptr< o2::tpc::Digit[]> *buffers, const o2::tpc::Digit *const *ptrs, size_t *nsb, const size_t *ns, const GPUParam ¶m, bool zs12bit, float threshold)
static std::function< void(std::vector< o2::tpc::Digit > &, const void *, uint32_t, uint32_t)> GetDecoder(int32_t version, const GPUParam *param)
static int32_t GetMaxTimeBin(const o2::tpc::ClusterNativeAccess &native)
static constexpr uint32_t NSECTORS
static void RunZSEncoderCreateMeta(const uint64_t *buffer, const uint32_t *sizes, void **ptrs, GPUTrackingInOutZS *out)
static void ConvertRun2RawToNative(o2::tpc::ClusterNativeAccess &native, std::unique_ptr< o2::tpc::ClusterNative[]> &nativeBuffer, const AliHLTTPCRawCluster **rawClusters, uint32_t *nRawClusters)
void addData(uint16_t feeid, uint16_t cru, uint8_t lnk, uint8_t endpoint, const IR &ir, const gsl::span< char > data, bool preformatted=false, uint32_t trigger=0, uint32_t detField=0)
unsigned char partition() const
unsigned char getSampaChannel() const
unsigned char getIndex() const
unsigned char getSampaChip() const
static Mapper & instance(const std::string mappingDir="")
static constexpr void getSampaAndChannelOnFEC(const int cruID, const size_t rawFECChannel, int &sampaOnFEC, int &channelOnSAMPA)
static constexpr int MAXSECTOR
std::array< gsl::span< const o2::tpc::Digit >, Sector::MAXSECTOR > DigitArray
GLuint GLsizei const GLuint const GLintptr const GLsizeiptr * sizes
GLboolean GLboolean GLboolean b
GLenum GLint GLenum GLsizei GLsizei GLsizei GLint GLsizei const void * bits
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
GLboolean GLboolean GLboolean GLboolean a
GLdouble GLdouble GLdouble z
uint8_t itsSharedClusterMap uint8_t
constexpr int LHCMaxBunches
const GBTLinkAttributes links[kNGBTLinks]
void dumpBuffer(gsl::span< const std::byte > buffer, std::ostream &out=std::cout, size_t maxbytes=std::numeric_limits< size_t >::max())
constexpr int LHCBCPERTIMEBIN
uint64_t getTimeStamp(o2::framework::ProcessingContext &pc)
Global TPC definitions and constants.
@ ZSVersionDenseLinkBased
@ ZSVersionLinkBasedWithMeta
@ ZSVersionDenseLinkBasedV2
@ ZSVersionRowBased10BitADC
@ ZSVersionRowBased12BitADC
unsigned short GlobalPadNumber
global pad number
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
std::string to_string(gsl::span< T, Size > span)
int32_t GetCharge() const
uint16_t GetFlags() const
float GetSigmaTime2() const
int32_t GetPadRow() const
float GetSigmaPad2() const
uint16_t bc
bunch crossing ID of interaction
const o2::tpc::Digit * tpcDigits[NSECTORS]
size_t nTPCDigits[NSECTORS]
const void *const * zsPtr[NENDPOINTS]
uint32_t count[NENDPOINTS]
const uint32_t * nZSPtr[NENDPOINTS]
GPUTrackingInOutZSSector sector[NSECTORS]
static constexpr uint32_t NENDPOINTS
static void setLinkID(H &rdh, uint8_t v, NOTPTR(H))
static void setDetectorField(H &rdh, uint32_t v, NOTPTR(H))
static constexpr int GBTWord128
static void setPageCounter(H &rdh, uint16_t v, NOTPTR(H))
static void setMemorySize(H &rdh, uint16_t v, NOTPTR(H))
static void setVersion(H &rdh, uint8_t v, NOTPTR(H))
static void setPacketCounter(H &rdh, uint8_t v, NOTPTR(H))
static void setHeartBeatOrbit(RDHv4 &rdh, uint32_t v)
static void setFEEID(RDHv4 &rdh, uint16_t v)
unsigned int nClusters[constants::MAXSECTOR][constants::MAXGLOBALPADROW]
unsigned int nClustersTotal
const ClusterNative * clusters[constants::MAXSECTOR][constants::MAXGLOBALPADROW]
unsigned int clusterOffset[constants::MAXSECTOR][constants::MAXGLOBALPADROW]
const ClusterNative * clustersLinear
@ payloadExtendsToNextPage
unsigned short nTimebinHeaders
unsigned short firstZSDataOffset
static constexpr bool TIGHTLY_PACKED_V3
static constexpr unsigned int TPC_ZS_NBITS_V34
static constexpr unsigned int SAMPLESPER64BIT
static constexpr unsigned int TRIGGER_WORD_SIZE
unsigned char nTimeBinSpan
unsigned short timeOffset
static constexpr unsigned int TPC_ZS_NBITS_V1
static constexpr unsigned int TPC_ZS_NBITS_V2
static constexpr size_t TPC_ZS_PAGE_SIZE
unsigned short nADCsamples
void compare(std::string_view s1, std::string_view s2)
o2::InteractionRecord ir(0, 0)
std::vector< Cluster > clusters
std::vector< Digit > digits