26#if (defined(WITH_OPENMP) || defined(_OPENMP))
30o2::tpc::IDCFactorization::IDCFactorization(
const std::array<unsigned char, Mapper::NREGIONS>& groupPads,
const std::array<unsigned char, Mapper::NREGIONS>& groupRows,
const std::array<unsigned char, Mapper::NREGIONS>& groupLastRowsThreshold,
const std::array<unsigned char, Mapper::NREGIONS>& groupLastPadsThreshold,
const unsigned int groupNotnPadsSectorEdges,
const unsigned int timeFrames,
const unsigned int timeframesDeltaIDC,
const std::vector<uint32_t>& crus)
31 :
IDCGroupHelperSector{groupPads, groupRows, groupLastRowsThreshold, groupLastPadsThreshold, groupNotnPadsSectorEdges}, mTimeFrames{timeFrames}, mTimeFramesDeltaIDC{timeframesDeltaIDC}, mCRUs{crus}
34 if (mSides.size() == 1) {
38 for (
int i = 0;
i < mSides.size(); ++
i) {
39 mIDCDelta.emplace_back(std::vector<
IDCDelta<float>>{timeFrames / timeframesDeltaIDC + (timeFrames % timeframesDeltaIDC != 0)});
42 mIDCZero.resize(mSides.size());
43 mIDCOne.resize(mSides.size());
45 for (
auto& idc : mIDCs) {
46 idc.resize(mTimeFrames);
61 std::unordered_map<o2::tpc::Side, bool> map;
62 for (
auto cru : crus) {
66 std::vector<o2::tpc::Side> sides;
78 TFile fOut(outFileName,
"RECREATE");
79 fOut.WriteObject(
this, outName);
85 LOGP(info,
"Writing object to file using {} threads", sNThreads);
86 ROOT::EnableThreadSafety();
87 std::vector<std::unique_ptr<TFile>> fOutPar;
88 fOutPar.reserve(sNThreads);
89 for (
int i = 0;
i < sNThreads; ++
i) {
90 const std::string outFile = fmt::format(
"{}_{}", outFileName,
i);
91 fOutPar.emplace_back(TFile::Open(outFile.data(),
"RECREATE"));
94#pragma omp parallel for num_threads(sNThreads)
95 for (
int iTF = 0; iTF < mTimeFrames; ++iTF) {
99 idcTmp.
idcs[icru] = mIDCs[icru][iTF];
102 const int ithread = omp_get_thread_num();
104 const int ithread = 0;
106 fOutPar[ithread]->WriteObject(&idcTmp, fmt::format(
"IDCs_TF{}", iTF).
data());
110 TFile fOut(outFileName,
"RECREATE");
112 idcFacTmp.
setRun(getRun());
114 fOut.WriteObject(&idcFacTmp, outName);
115 std::vector<int> nFiles{sNThreads};
116 fOut.WriteObject(&nFiles,
"out_files");
122 LOGP(info,
"Reading object from file using {} threads", sNThreads);
123 TFile fMeta(inpFileName,
"READ");
124 if (fMeta.IsZombie()) {
125 LOGP(warning,
"Input file {} not found! Returning", inpFileName);
129 std::unique_ptr<IDCFactorization> idc;
131 idc->setRun(idcTmp->
getRun());
133 std::vector<int>* out_files = (std::vector<int>*)fMeta.Get(
"out_files");
134 const int nFiles = out_files->front();
138 std::vector<std::unique_ptr<TFile>> fInp;
139 fInp.reserve(nFiles);
140 for (
int i = 0;
i < nFiles; ++
i) {
141 const std::string inpFile = fmt::format(
"{}_{}", inpFileName,
i);
142 fInp.emplace_back(TFile::Open(inpFile.data()));
143 if (fInp.front() ==
nullptr) {
144 LOGP(warning,
"Input file {} not found!", inpFile.data());
148 ROOT::EnableThreadSafety();
151#pragma omp parallel for num_threads(sNThreads)
152 for (
int iFile = 0; iFile < nFiles; ++iFile) {
153 const auto* keys = fInp[iFile]->GetListOfKeys();
154 const auto nKeys = keys->GetEntries();
155 for (
int iKey = 0; iKey < nKeys; ++iKey) {
156 const auto key =
dynamic_cast<TKey*
>(keys->At(iKey));
158 fInp[iFile]->GetObject(
key->GetName(), idcTmp);
160 const int relTF = idcTmp->
relTF;
161 if (relTF >= idc->getNTimeframes()) {
162 LOGP(warning,
"stored TF {} is larger than max TF {}", relTF, idc->getNTimeframes());
166 auto idcVec = idcTmp->
idcs[icru];
167 idc->setIDCs(std::move(idcVec), icru, relTF);
172 for (
auto&
file : fInp) {
180 TFile fOut(outFileName,
"RECREATE");
181 fOut.WriteObject(&mIDCZero[mSideIndex[
side]], outName);
187 TFile fOut(outFileName,
"RECREATE");
188 fOut.WriteObject(&mIDCOne[mSideIndex[
side]], outName);
194 if (mIDCZero[mSideIndex[
side]].mIDCZero.empty() || mIDCOne[mSideIndex[
side]].mIDCOne.empty()) {
195 LOGP(info,
"IDC0, IDC1 is empty! Returning");
212 std::vector<double> timestamp;
214 for (
int i = 0;
i < getIDCOneVec(
Side::A).size(); ++
i) {
221 <<
"timestamp=" << timestamp
229 if (chunk >= getTimeFramesDeltaIDC()) {
230 LOGP(info,
"index {} is larger than maximum index {} for IDCDelta", chunk, getTimeFramesDeltaIDC() - 1);
243 for (
auto& idcZero : mIDCZero) {
245 idcZero.resize(nIDCsSide);
248#pragma omp parallel for num_threads(sNThreads)
249 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
250 const unsigned int cru = mCRUs[cruInd];
253 const unsigned int region = cruTmp.
region();
255 for (
unsigned int timeframe = 0; timeframe < mTimeFrames; ++timeframe) {
256 for (
unsigned int idcs = 0; idcs < mIDCs[cru][timeframe].size(); ++idcs) {
257 if ((mIDCs[cru][timeframe][idcs] == -1) || (mIDCs[cru][timeframe][idcs] == 0)) {
263 const unsigned int indexGlob = (idcs % mNIDCsPerCRU[region]) + factorIndexGlob;
264 mIDCZero[mSideIndex[
side]].fillValueIDCZero(mIDCs[cru][timeframe][idcs], indexGlob);
270#pragma omp parallel for num_threads(sNThreads)
271 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
272 const unsigned int cru = mCRUs[cruInd];
275 const unsigned int region = cruTmp.
region();
277 const auto indexEnd = indexStart + mNIDCsPerCRU[region];
279 const auto normFac = getNIntegrationIntervals(cru);
281 LOGP(info,
"number of integration intervals is zero for CRU {}! Skipping normalization of IDC0", cru);
285 std::transform(mIDCZero[mSideIndex[
side]].mIDCZero.begin() + indexStart, mIDCZero[mSideIndex[
side]].mIDCZero.begin() + indexEnd, mIDCZero[mSideIndex[
side]].mIDCZero.begin() + indexStart, [normVal = normFac](
auto&
val) { return val / normVal; });
291#pragma omp parallel for num_threads(sNThreads)
292 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
293 const unsigned int cru = mCRUs[cruInd];
296 const int region = cruTmp.
region();
297 const int sector = cruTmp.
sector();
298 std::vector<unsigned int> padTmp;
303 const unsigned int integrationInterval = 0;
306 const auto index = getIndexUngrouped(sector, region, lrow, pad, integrationInterval);
307 const auto idcZeroVal = mIDCZero[mSideIndex[
side]].mIDCZero[
index];
308 if (idcZeroVal > 0) {
310 const float gain = (mGainMap ? mGainMap->getValue(sector, globalPad) : 1);
311 idcRow += idcZeroVal / gain;
314 padTmp.emplace_back(pad);
318 for (
const auto pad : padTmp) {
319 const float meanIDC0 = idcRow /
count;
321 const float gain = (mGainMap ? mGainMap->getValue(sector, globalPad) : 1);
322 const float idcZero = gain * meanIDC0;
323 const auto index = getIndexUngrouped(sector, region, lrow, pad, integrationInterval);
324 mIDCZero[mSideIndex[
side]].setValueIDCZero(idcZero,
index);
333 const unsigned int integrationIntervals = getNIntegrationIntervals();
334 for (
auto& idcOne : mIDCOne) {
335 idcOne.mIDCOne.clear();
336 idcOne.mIDCOneMedian.clear();
337 idcOne.mIDCOneRMS.clear();
338 idcOne.mIDCOne.resize(integrationIntervals);
339 idcOne.mIDCOneMedian.resize(integrationIntervals);
340 idcOne.mIDCOneRMS.resize(integrationIntervals);
341 std::fill(idcOne.mIDCOne.begin(), idcOne.mIDCOne.end(), 1);
344#pragma omp parallel for num_threads(sNThreads)
345 for (
unsigned int timeframe = 0; timeframe < mTimeFrames; ++timeframe) {
347 std::vector<std::vector<std::vector<float>>> idcOneSafe(mSides.size());
350 for (
auto& interavalvec : idcOneSafe) {
351 interavalvec.resize(mIntegrationIntervalsPerTF[timeframe]);
352 for (
auto& idcsPerInteraval : interavalvec) {
353 idcsPerInteraval.reserve(nIDCsSide);
358 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
359 const unsigned int cru = mCRUs[cruInd];
361 const unsigned int region = cruTmp.
region();
363 const unsigned int indexSide = mSideIndex[
side];
365 unsigned int integrationIntervalOffset = 0;
366 calcIDCOne(mIDCs[cru][timeframe], mNIDCsPerCRU[region], integrationIntervalOffset, factorIndexGlob, cru, idcOneSafe[indexSide], &mIDCZero[indexSide], mInputGrouped ?
nullptr : mPadFlagsMap.get(), mUsePadStatusMap);
370 int offsInterval = 0;
371 for (
unsigned int tf = 0;
tf < timeframe; ++
tf) {
372 offsInterval += mIntegrationIntervalsPerTF[
tf];
376 const unsigned int indexSide = mSideIndex[
side];
377 for (
int interval = 0; interval < mIntegrationIntervalsPerTF[timeframe]; ++interval) {
380 const int intervalGlobal = interval + offsInterval;
381 if (!idcOneSafe[indexSide].
empty()) {
382 RobustAverage average(std::move(idcOneSafe[indexSide][interval]));
384 const float median = average.
getMedian();
387 mIDCOne[indexSide].mIDCOne[intervalGlobal] = mean;
389 mIDCOne[indexSide].mIDCOneMedian[intervalGlobal] = median;
390 mIDCOne[indexSide].mIDCOneRMS[intervalGlobal] = rms;
397template <
typename DataVec>
400 for (
unsigned int idcs = 0; idcs < idcsData.size(); ++idcs) {
401 if ((idcsData[idcs] == -1) || (idcsData[idcs] == 0)) {
405 const unsigned int integrationInterval = idcs / idcsPerCRU + integrationIntervalOffset;
406 const unsigned int localPad = (idcs % idcsPerCRU);
407 const unsigned int indexGlob = localPad + indexOffset;
408 const auto idcZeroVal = idcZero ? idcZero->
mIDCZero[indexGlob] : 1;
411 if (usePadStatusMap && flagMap) {
418 if (idcOneTmp.size() <= integrationInterval) {
419 LOGP(error,
"integrationInterval {} is larger than maximum: {}", integrationInterval, idcOneTmp.size());
422 const double epsilon = 0.001;
423 if (std::abs(idcZeroVal) > epsilon) {
424 idcOneTmp[integrationInterval].emplace_back(idcsData[idcs] / idcZeroVal);
432 for (
auto side : mSides) {
433 for (
unsigned int i = 0;
i < getNChunks(
side); ++
i) {
434 const auto idcsSide = nIDCsSide * getNIntegrationIntervalsInChunk(
i);
435 mIDCDelta[mSideIndex[
side]][
i].getIDCDelta().clear();
436 mIDCDelta[mSideIndex[
side]][
i].getIDCDelta().resize(idcsSide);
440#pragma omp parallel for num_threads(sNThreads)
441 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
442 const unsigned int cru = mCRUs[cruInd];
444 const unsigned int region = cruTmp.
region();
447 unsigned int integrationIntervallast = 0;
448 unsigned int integrationIntervallastLocal = 0;
449 unsigned int lastChunk = 0;
451 for (
unsigned int timeframe = 0; timeframe < mTimeFrames; ++timeframe) {
452 const unsigned int chunk = getChunk(timeframe);
453 if (lastChunk != chunk) {
454 integrationIntervallastLocal = 0;
457 for (
unsigned int idcs = 0; idcs < mIDCs[cru][timeframe].size(); ++idcs) {
458 if ((mIDCs[cru][timeframe][idcs] == -1) || (mIDCs[cru][timeframe][idcs] == 0)) {
461 const unsigned int intervallocal = idcs / mNIDCsPerCRU[region];
462 const unsigned int integrationIntervalGlobal = intervallocal + integrationIntervallast;
463 const unsigned int integrationIntervalLocal = intervallocal + integrationIntervallastLocal;
464 const unsigned int localPad = idcs % mNIDCsPerCRU[region];
465 const unsigned int indexGlob = localPad + factorIndexGlob;
466 const auto idcZero = mIDCZero[mSideIndex[
side]].mIDCZero[indexGlob];
467 const auto idcOne = mIDCOne[mSideIndex[
side]].mIDCOne[integrationIntervalGlobal];
468 const auto mult = idcZero * idcOne;
469 auto val = (mult != 0) ? mIDCs[cru][timeframe][idcs] / mult - 1 : 0;
470 if (mUsePadStatusMap && !mInputGrouped && mPadFlagsMap) {
471 const o2::tpc::PadFlags flag = mPadFlagsMap->getCalArray(cru).getValue(localPad);
476 mIDCDelta[mSideIndex[
side]][chunk].setIDCDelta(indexGlob + integrationIntervalLocal * nIDCsSide,
val);
479 const unsigned int intervals = mIntegrationIntervalsPerTF[timeframe];
480 integrationIntervallast += intervals;
481 integrationIntervallastLocal += intervals;
489 createStatusMapOutlier();
491 checkNeighbourOutliers();
499 std::array<std::vector<o2::tpc::CRU>,
SIDES> crus;
501 crus[
Side::A].reserve(crusPerSide);
502 crus[
Side::C].reserve(crusPerSide);
503 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
505 crus[cru.
side()].emplace_back(cru);
509 const auto stacksMedian = getStackMedian();
510 const std::array<int, Mapper::NREGIONS> stackInSector{0, 0, 0, 0, 1, 1, 2, 2, 3, 3};
512 for (
const auto side : mSides) {
513 std::vector<o2::tpc::RobustAverage> average;
519 for (
int iter = 0; iter < 2; ++iter) {
520 std::vector<std::pair<float, float>> mean;
524 mean.emplace_back(average[
row].getFilteredAverage());
529 for (
int row = 0;
row < average.size(); ++
row) {
530 std::vector<float>
values = average[
row].getValues();
531 int nValues = average[
row].getValues().size();
532 float median = average[
row].getMedian();
535 <<
"mean=" << mean[
row]
537 <<
"nValues=" << nValues
538 <<
"median=" << median
544 for (
const auto& cru : crus[
side]) {
545 const int region = cru.region();
546 const int sector = cru.sector();
551 const unsigned int integrationInterval = 0;
556 const auto index = getIndexUngrouped(sector, region, lrow, pad, integrationInterval);
557 auto idcZeroVal = mIDCZero[mSideIndex[
side]].mIDCZero[
index];
562 const float gain = mGainMap->getValue(sector, globalPad);
566 const bool checkSimple = (idcZeroVal != -1) && (idcZeroVal != 0);
574 average[globalRow].addValue(idcZeroVal);
577 const int relPad =
static_cast<int>(pad) -
static_cast<int>(
Mapper::PADSPERROW[region][lrow] / 2);
579 const int nPadsCentre = 2;
580 const bool isSecCentreEdge = ((relPad >= -nPadsCentre) && (relPad < nPadsCentre)) || ((pad == 0) || (pad ==
Mapper::PADSPERROW[region][lrow] - 1));
581 const float idc0MedianMin = isSecCentreEdge ? paramIDCGroup.minmaxIDC0MedianCentreEdge : paramIDCGroup.minIDC0Median;
582 const float idc0MedianMax = isSecCentreEdge ? paramIDCGroup.minmaxIDC0MedianCentreEdge : paramIDCGroup.maxIDC0Median;
586 if ((idcZeroVal == -1) || (idcZeroVal == 0)) {
588 }
else if (idcZeroVal > mean[globalRow].
first + mean[globalRow].second * idc0MedianMax) {
590 }
else if (idcZeroVal < mean[globalRow].
first - mean[globalRow].second * idc0MedianMin) {
593 mPadFlagsMap->getCalArray(cru).setValue(padInRegion, flag);
612 createStatusMapOutlier();
615 std::array<int, FECInfo::FECsTotal> nOutliersPerFEC{0};
617 for (
int iter = 0; iter < 2; ++iter) {
633 ++nOutliersPerFEC[fecIndex];
642 mPadFlagsMap->getCalArray(cru).setValue(padInRegion, flag);
654 for (
int iter = 0; iter < maxIter; ++iter) {
655 int countTotOutlier = 0;
660 for (
int lrow = 0; lrow <= maxRow; ++lrow) {
662 const int nPadsInRow = 1;
663 const int nPadsInPad = 2;
664 const int rowStart = std::clamp(lrow - nPadsInRow, 0, maxRow);
665 const int rowEnd = std::clamp(lrow + nPadsInRow, 0, maxRow);
669 int countOutliers = 0;
671 for (
int rowCl = rowStart; rowCl <= rowEnd; ++rowCl) {
674 const int diffAddPads = addPadsCl - addPadsStart;
675 const int padClCentre = pad + diffAddPads;
678 const int padStart = std::clamp(padClCentre - nPadsInPad, 0, maxPad);
679 const int padEnd = std::clamp(padClCentre + nPadsInPad, 0, maxPad);
680 for (
int padCl = padStart; padCl <= padEnd; ++padCl) {
682 if ((padCl == pad) && (rowCl == lrow)) {
694 if (countOutliers >= nOutliersNeighbours) {
706 mPadFlagsMap->getCalArray(cru).setValue(padInRegion, flag);
712 LOGP(
debug,
"iter: {} count total outlier: {}", iter, countTotOutlier);
713 if (countTotOutlier == 0) {
722 LOGP(info,
"getMeanZ(): Status map not set returning");
730 unsigned int timeFrame = 0;
731 unsigned int interval = 0;
732 getTF(integrationInterval, timeFrame, interval);
737 const int index = interval * mNIDCsPerCRU[region] + mOffsRow[region][getGroupedRow(region, urow)] + getGroupedPad(region, urow, upad) + getOffsetForEdgePad(upad, urow, region);
741void o2::tpc::IDCFactorization::getTF(
unsigned int integrationInterval,
unsigned int& timeFrame,
unsigned int& interval)
const
743 const auto& intervalsPerTF = mIntegrationIntervalsPerTF.empty() ? getAllIntegrationIntervalsPerTF() : mIntegrationIntervalsPerTF;
744 unsigned int nintervals = 0;
745 unsigned int intervalTmp = 0;
746 for (
unsigned int tf = 0;
tf < mTimeFrames; ++
tf) {
747 nintervals += intervalsPerTF[
tf];
748 if (integrationInterval < nintervals) {
750 interval = integrationInterval - intervalTmp;
753 intervalTmp = nintervals;
760 const int nMaxTFs = 3;
761 const int nTFs = (mTimeFrames >= nMaxTFs) ? nMaxTFs : mTimeFrames;
764 std::vector<unsigned int> ordering;
765 ordering.reserve(nTFs);
766 unsigned int order = 0;
770 for (
int iTF = 0; iTF < mTimeFrames; ++iTF) {
772 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
773 const unsigned int cru = mCRUs[cruInd];
774 if (!mIDCs[cru][iTF].
empty()) {
775 order = mIDCs[cru][iTF].size() / mNIDCsPerCRU[
CRU(cru).
region()];
776 ordering.emplace_back(order);
781 if (ordering.size() == nTFs) {
782 firstTF = iTF - nTFs + 1;
791 if ((nTFs == mTimeFrames) && (ordering.size() == nTFs)) {
795 if (ordering.size() != nTFs) {
796 LOGP(warning,
"Couldnt find {} consecutive TFs with data", nTFs);
797 ordering = std::vector<unsigned int>(nTFs, order);
800 std::sort(ordering.begin(), ordering.end());
803 std::vector<unsigned int> integrationIntervalsPerTF(mTimeFrames);
804 for (
int iter = 0; iter < 2; ++iter) {
805 const int end = (iter == 0) ? (mTimeFrames - firstTF) : firstTF;
806 for (
int iTFLoop = 0; iTFLoop <
end; ++iTFLoop) {
807 const int iTF = (iter == 0) ? (firstTF + iTFLoop) : (firstTF - iTFLoop - 1);
808 int intervalsInTF = -1;
809 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
810 const unsigned int cru = mCRUs[cruInd];
811 if (!mIDCs[cru][iTF].
empty()) {
812 intervalsInTF = mIDCs[cru][iTF].size() / mNIDCsPerCRU[
CRU(cru).
region()];
817 if (intervalsInTF == -1) {
818 if ((ordering.size() != nTFs)) {
820 intervalsInTF = ordering.front();
821 }
else if (iTF == 1) {
822 intervalsInTF = ordering[1];
825 const int m1Val = (iter == 0) ? -1 : +1;
826 const int intervalsLastm1 = integrationIntervalsPerTF[iTF + m1Val];
827 const int intervalsLastm2 = integrationIntervalsPerTF[iTF + 2 * m1Val];
828 if ((intervalsLastm1 == ordering[1]) && (intervalsLastm2 == ordering[2])) {
829 intervalsInTF = ordering[0];
831 intervalsInTF = ordering[1];
836 if (intervalsInTF < -1) {
837 LOGP(warning,
"interval is smaller than 0!");
841 integrationIntervalsPerTF[iTF] = intervalsInTF;
844 return integrationIntervalsPerTF;
849 using timer = std::chrono::high_resolution_clock;
851 LOGP(info,
"Using {} threads for factorization of IDCs", sNThreads);
852 LOGP(info,
"Checking received IDCs for consistency");
855 LOGP(info,
"Calculating IDC0");
857 auto start = timer::now();
859 auto stop = timer::now();
860 std::chrono::duration<float>
time = stop -
start;
861 float totalTime =
time.count();
862 LOGP(info,
"IDCZero time: {}",
time.count());
864 if (!mInputGrouped) {
865 LOGP(info,
"Creating pad status map");
866 start = timer::now();
870 LOGP(info,
"Pad-by-pad status map time: {}",
time.count());
871 totalTime +=
time.count();
874 start = timer::now();
875 mIntegrationIntervalsPerTF = getAllIntegrationIntervalsPerTF();
878 totalTime =
time.count();
879 LOGP(info,
"Getting integration intervals for all TFs time: {}",
time.count());
881 LOGP(info,
"Calculating IDC1");
882 start = timer::now();
886 LOGP(info,
"IDC1 time: {}",
time.count());
887 totalTime +=
time.count();
890 LOGP(info,
"Calculating IDCDelta");
891 start = timer::now();
895 LOGP(info,
"IDCDelta time: {}",
time.count());
896 totalTime +=
time.count();
899 LOGP(info,
"Factorization done. Total time: {}", totalTime);
904 const auto firstTF = chunk * getNTFsPerChunk(0);
905 const auto& intervalsPerTF = mIntegrationIntervalsPerTF.empty() ? getAllIntegrationIntervalsPerTF() : mIntegrationIntervalsPerTF;
906 const auto intervals = std::reduce(intervalsPerTF.begin() + firstTF, intervalsPerTF.begin() + firstTF + getNTFsPerChunk(chunk));
912 const auto chunkTF = chunk * getNTFsPerChunk(0);
913 const auto& intervalsPerTF = mIntegrationIntervalsPerTF.empty() ? getAllIntegrationIntervalsPerTF() : mIntegrationIntervalsPerTF;
914 const auto intervals = std::reduce(intervalsPerTF.begin(), intervalsPerTF.begin() + chunkTF);
921 for (
auto&& idcsTF : mIDCs[cru]) {
922 sum += idcsTF.size();
929 const auto& intervalsPerTF = mIntegrationIntervalsPerTF.empty() ? getAllIntegrationIntervalsPerTF() : mIntegrationIntervalsPerTF;
930 const auto intervals = std::reduce(intervalsPerTF.begin(), intervalsPerTF.end());
936 const auto& intervalsPerTF = mIntegrationIntervalsPerTF.empty() ? getAllIntegrationIntervalsPerTF() : mIntegrationIntervalsPerTF;
937 unsigned int nintervals = 0;
938 unsigned int nitervalsChunk = 0;
939 unsigned int globalTF = 0;
940 for (
unsigned int ichunk = 0; ichunk < getNChunks(mSides.front()); ++ichunk) {
941 const auto nTFsPerChunk = getNTFsPerChunk(ichunk);
942 for (
unsigned int tf = 0;
tf < nTFsPerChunk; ++
tf) {
943 nintervals += intervalsPerTF[globalTF];
944 if (integrationInterval < nintervals) {
945 chunk = getChunk(globalTF);
946 localintegrationInterval = integrationInterval - nitervalsChunk;
951 nitervalsChunk = nintervals;
955unsigned int o2::tpc::IDCFactorization::getChunk(
const unsigned int timeframe)
const
957 return timeframe / mTimeFramesDeltaIDC;
960unsigned int o2::tpc::IDCFactorization::getNTFsPerChunk(
const unsigned int chunk)
const
962 const unsigned int remain = mTimeFrames % mTimeFramesDeltaIDC;
963 return ((chunk == getNChunks(mSides.front()) - 1) && remain) ? remain : mTimeFramesDeltaIDC;
968 const uint32_t cruTmp = (cru < 0) ? mCRUs.front() : cru;
969 const auto region =
CRU(cruTmp).
region();
970 std::vector<unsigned int> integrationIntervalsPerTF;
971 integrationIntervalsPerTF.reserve(mTimeFrames);
972 for (
unsigned int tf = 0;
tf < mTimeFrames; ++
tf) {
973 integrationIntervalsPerTF.emplace_back(mIDCs[cruTmp][
tf].
size() / mNIDCsPerCRU[region]);
975 return integrationIntervalsPerTF;
980 for (
auto&
tf : mIDCs) {
981 for (
auto& idcs :
tf) {
987void o2::tpc::IDCFactorization::drawIDCDeltaHelper(
const bool type,
const Sector sector,
const unsigned int integrationInterval,
const IDCDeltaCompression compression,
const std::string
filename,
const float minZ,
const float maxZ)
const
989 std::function<float(
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int)> idcFunc;
991 unsigned int chunk = 0;
992 unsigned int localintegrationInterval = 0;
993 getLocalIntegrationInterval(integrationInterval, chunk, localintegrationInterval);
997 switch (compression) {
1000 idcFunc = [
this, chunk, localintegrationInterval](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad) {
1001 return this->getIDCDeltaVal(sector, region, irow, pad, chunk, localintegrationInterval);
1004 type ?
IDCDrawHelper::drawSide(drawFun, sector.side(), zAxisTitle,
filename,
minZ,
maxZ) : IDCDrawHelper::drawSector(drawFun, 0, Mapper::NREGIONS, sector, zAxisTitle,
filename,
minZ,
maxZ);
1008 const auto idcDeltaMedium = this->getIDCDeltaMediumCompressed(chunk, sector.side());
1009 idcFunc = [
this, &idcDeltaMedium, localintegrationInterval = localintegrationInterval](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad) {
1010 return idcDeltaMedium.getValue(this->getIndexUngrouped(sector, region, irow, pad, localintegrationInterval));
1013 type ?
IDCDrawHelper::drawSide(drawFun, sector.side(), zAxisTitle,
filename,
minZ,
maxZ) : IDCDrawHelper::drawSector(drawFun, 0, Mapper::NREGIONS, sector, zAxisTitle,
filename,
minZ,
maxZ);
1017 const auto idcDeltaHigh = this->getIDCDeltaHighCompressed(chunk, sector.side());
1018 idcFunc = [
this, &idcDeltaHigh, localintegrationInterval](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad) {
1019 return idcDeltaHigh.getValue(this->getIndexUngrouped(sector, region, irow, pad, localintegrationInterval));
1022 type ?
IDCDrawHelper::drawSide(drawFun, sector.side(), zAxisTitle,
filename,
minZ,
maxZ) : IDCDrawHelper::drawSector(drawFun, 0, Mapper::NREGIONS, sector, zAxisTitle,
filename,
minZ,
maxZ);
1028void o2::tpc::IDCFactorization::drawIDCZeroHelper(
const bool type,
const Sector sector,
const std::string
filename,
const float minZ,
const float maxZ)
const
1030 std::function<
float(
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int)> idcFunc = [
this](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad) {
1031 return this->getIDCZeroVal(sector, region, irow, pad);
1034 IDCDrawHelper::IDCDraw drawFun;
1038 const float maxZTmp = (
maxZ == -1) ? (3 * getMeanZ(sector.side())) :
maxZ;
1039 type ?
IDCDrawHelper::drawSide(drawFun, sector.side(), zAxisTitle,
filename,
minZ, maxZTmp) : IDCDrawHelper::drawSector(drawFun, 0, Mapper::NREGIONS, sector, zAxisTitle,
filename,
minZ, maxZTmp);
1042void o2::tpc::IDCFactorization::drawIDCHelper(
const bool type,
const Sector sector,
const unsigned int integrationInterval,
const std::string
filename,
const float minZ,
const float maxZ,
const bool drawGIF,
const int run)
const
1044 const float maxZTmp = (
maxZ == -1) ? (3 * getMeanZ(sector.side())) :
maxZ;
1047 std::function<
float(
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int)> idcFunc = [
this, integrationInterval](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad) {
1048 return this->getIDCValUngrouped(sector, region, irow, pad, integrationInterval);
1051 IDCDrawHelper::IDCDraw drawFun;
1052 drawFun.mIDCFunc = idcFunc;
1054 type ?
IDCDrawHelper::drawSide(drawFun, sector.side(), zAxisTitleDraw,
filename,
minZ, maxZTmp) : IDCDrawHelper::drawSector(drawFun, 0, Mapper::NREGIONS, sector, zAxisTitleDraw,
filename,
minZ, maxZTmp);
1056 std::function<
float(
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int)> idcFunc = [
this](
const unsigned int sector,
const unsigned int region,
const unsigned int irow,
const unsigned int pad,
const unsigned int slice) {
1057 return this->getIDCValUngrouped(sector, region, irow, pad, slice);
1060 if (mIDCOne.front().mIDCOne.empty()) {
1061 LOGP(warning,
"Factorised IDCs are missing!");
1066 return this->getIDCOneVec(
side)[slice];
1069 IDCDrawHelper::IDCDrawGIF drawFun;
1070 drawFun.mIDCFunc = idcFunc;
1071 drawFun.mIDCOneFunc = idcOneFunc;
1072 const int nSlices = (integrationInterval == 0) ? getNIntegrationIntervals() : integrationInterval;
1079 TFile
f(inpFile,
"READ");
1081 f.GetObject(mapName, gainMap);
1084 LOGP(info,
"GainMap {} not found returning", mapName);
1087 setGainMap(*gainMap);
1093 mGainMap = std::make_unique<CalDet<float>>(gainmap);
1098 TFile
f(inpFile,
"READ");
1100 f.GetObject(mapName, statusmap);
1103 LOGP(info,
"Pad flag map {} not found returning", mapName);
1106 setPadFlagMap(*statusmap);
1112 mPadFlagsMap = std::make_unique<CalDet<PadFlags>>(flagmap);
1115void o2::tpc::IDCFactorization::drawPadFlagMap(
const bool type,
const Sector sector,
const std::string
filename,
const PadFlags flag)
const
1117 if (!mPadFlagsMap) {
1118 LOGP(info,
"Status map not set returning");
1122 std::function<float(
const unsigned int,
const unsigned int,
const unsigned int,
const unsigned int)> idcFunc = [
this, flag](
const unsigned int sector,
const unsigned int region,
const unsigned int row,
const unsigned int pad) {
1124 const auto flagDraw = mPadFlagsMap->getCalArray(region + sector *
Mapper::NREGIONS).getValue(padInRegion);
1125 if ((flagDraw & flag) == flag) {
1132 IDCDrawHelper::IDCDraw drawFun;
1133 drawFun.mIDCFunc = idcFunc;
1134 const std::string zAxisTitle =
"status flag";
1140 if (!mPadFlagsMap) {
1141 LOGP(info,
"Status map not set returning");
1143 TFile fOut(outFile,
"RECREATE");
1144 fOut.WriteObject(mPadFlagsMap.get(), mapName);
1148void o2::tpc::IDCFactorization::normIDCZero(
const int type)
1150 if ((
type == 0) && !mGainMap) {
1151 LOGP(info,
"Gain map not set returning");
1155 const std::array<int, Mapper::NREGIONS> stackInSector{0, 0, 0, 0, 1, 1, 2, 2, 3, 3};
1156 std::array<float, o2::tpc::GEMSTACKS> stacksMedian;
1158 stacksMedian = getStackMedian();
1161#pragma omp parallel for num_threads(sNThreads)
1162 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
1163 const unsigned int cru = mCRUs[cruInd];
1166 const int region = cruTmp.region();
1167 const int sector = cruTmp.sector();
1169 const unsigned int integrationInterval = 0;
1171 const auto index = getIndexUngrouped(sector, region, lrow, pad, integrationInterval);
1175 normVal = mGainMap->getValue(sector, globalPad);
1176 }
else if (
type == 1) {
1179 mIDCZero[mSideIndex[
side]].mIDCZero[
index] /= normVal;
1187 std::array<float, GEMSTACKS> median;
1188 for (
const auto side : mSides) {
1191 std::copy(tmpMedian.begin(), tmpMedian.end(), median.begin() + offs);
1196bool o2::tpc::IDCFactorization::checkReceivedIDCs()
1198 bool idcsGood =
true;
1199 for (
unsigned int timeframe = 0; timeframe < mTimeFrames; ++timeframe) {
1201 std::unordered_map<int, std::vector<int>> receivedIDCsSize;
1202 for (
unsigned int cruInd = 0; cruInd < mCRUs.size(); ++cruInd) {
1203 const unsigned int cru = mCRUs[cruInd];
1205 const unsigned int region = cruTmp.region();
1206 const int nSlices = mIDCs[cru][timeframe].size() / mNIDCsPerCRU[region];
1208 if (receivedIDCsSize[nSlices].
empty()) {
1209 receivedIDCsSize[nSlices].reserve(mCRUs.size());
1211 receivedIDCsSize[nSlices].emplace_back(cru);
1215 if (receivedIDCsSize.size() > 1) {
1216 LOGP(warning,
"Received inconsistent IDCs!");
1219 std::pair<int, int> nSlicesMostCRUs = {-1, -1};
1220 for (
auto nSlices : receivedIDCsSize) {
1221 const int nCRUs = nSlices.second.size();
1223 if (nCRUs > nSlicesMostCRUs.second) {
1224 nSlicesMostCRUs = {nSlices.first, nCRUs};
1226 for (
auto cru : nSlices.second) {
1227 LOGP(info,
"Received {} slices for CRU {}", nSlices.first, cru);
1230 LOGP(info,
"Setting {} slices for {} valid CRUs", nSlicesMostCRUs.first, nSlicesMostCRUs.second);
1231 for (
auto nSlices : receivedIDCsSize) {
1232 if (nSlices.first != nSlicesMostCRUs.first) {
1233 for (
auto cru : nSlices.second) {
1234 LOGP(info,
"Clearing IDCs for CRU {}", cru);
1235 mIDCs[cru][timeframe].clear();
helper class for accessing IDCs from CCDB
helper class for drawing IDCs per region/side
class for aggregating IDCs for the full TPC (all sectors) and factorization of aggregated IDCs
Header to collect LHC related constants.
class for performing robust averaging and outlier filtering
static const ParameterIDCGroup & Instance()
unsigned char region() const
const Sector sector() const
const T getValue(const size_t channel) const
const CalType & getCalArray(size_t position) const
static constexpr int FECsPerSector
static constexpr int ChannelsPerFEC
unsigned char getIndex() const
void setIDCZero(IDCZero *idcZero, const Side side=Side::A)
setting the 0D-IDCs
void setIDCDelta(IDCDelta< DataT > *idcDelta, const Side side=Side::A)
setting the IDCDelta class member
void setIDCOne(IDCOne *idcOne, const Side side=Side::A)
setting the 1D-IDCs
static float getMeanIDC0(const Side side, const IDCZero &idcZero, const CalDet< PadFlags > *outlierMap)
void dumpToTree(const Side side, const char *outFileName="IDCCCDBTree.root") const
void setPadStatusMap(const CalDet< PadFlags > &outliermap)
static float getStackMedian(const IDCZero &idczero, const Sector sector, const GEMstack stack)
void setGroupingParameter(IDCGroupHelperSector *helperSector, const Side side=Side::A)
setting the grouping parameters
void dumpToTreeIDCDelta(const Side side, const char *outFileName="IDCCCDBTreeDeltaIDC.root") const
static void drawSideGIF(const IDCDrawGIF &idcs, const unsigned int slices, const std::string zAxisTitle, const std::string filename="IDCs", const float minZ=0, const float maxZ=-1, const int run=-1)
make a GIF of IDCs for A and C side and the 1D IDCs
static std::string getZAxisTitle(const IDCType type, const IDCDeltaCompression compression=IDCDeltaCompression::NO)
static void drawSide(const IDCDraw &idc, const o2::tpc::Side side, const std::string zAxisTitle, const std::string filename, const float minZ=0, const float maxZ=-1)
std::array< float, o2::tpc::GEMSTACKS > getStackMedian() const
void setGainMap(const char *inpFile, const char *mapName)
const std::vector< Side > & getSides() const
void calcIDCOne()
calculate I_1(t) = <I(r,\phi,t) / I_0(r,\phi)>_{r,\phi}
void dumpIDCDeltaToTree(const Side side, const int chunk=0, const char *outFileName="IDCDeltaTree.root")
dump IDCDelta to a TTree
unsigned int getNTimeframes() const
void setRun(const int run)
long getTimeStamp() const
void reset()
resetting aggregated IDCs
void fillIDCZeroDeadPads()
fill I_0 values in case of dead pads,FECs etc.
unsigned long getNIntegrationIntervalsInChunk(const unsigned int chunk) const
void dumpToTree(const Side side, const char *outFileName="IDCTree.root")
unsigned int getTimeFramesDeltaIDC() const
IDCFactorization()=default
default constructor for ROOT I/O
unsigned long getNIntegrationIntervalsToChunk(const unsigned int chunk) const
void factorizeIDCs(const bool norm, const bool calcDeltas)
void dumpPadFlagMap(const char *outFile, const char *mapName)
void dumpIDCZeroToFile(const Side side, const char *outFileName="IDCZero.root", const char *outName="IDC0") const
dump the IDC0 to file
float getIDCValUngrouped(const unsigned int sector, const unsigned int region, unsigned int urow, unsigned int upad, unsigned int integrationInterval) const
void checkNeighbourOutliers(const int maxIter=10, const int nOutliersNeighbours=8)
std::vector< unsigned int > getIntegrationIntervalsPerTF(const int cru=-1) const
~IDCFactorization()
destructor
std::vector< unsigned int > getAllIntegrationIntervalsPerTF() const
unsigned long getNIntegrationIntervals() const
void dumpToFile(const char *outFileName="IDCFactorized.root", const char *outName="IDCFactorized") const
void calcIDCZero(const bool norm)
static std::unique_ptr< IDCFactorization > getLargeObjectFromFile(const char *inpFileName="IDCFactorized.root", const char *inName="IDCFactorized")
void dumpLargeObjectToFile(const char *outFileName="IDCFactorized.root", const char *outName="IDCFactorized") const
void calcIDCDelta()
calculate \Delta I(r,\phi,t) = I(r,\phi,t) / ( I_0(r,\phi) * I_1(t) )
void checkFECs(const float maxOutliersPerFEC=0.7)
void getLocalIntegrationInterval(const unsigned int integrationInterval, unsigned int &chunk, unsigned int &localintegrationInterval) const
float getMeanZ(const Side side) const
void dumpToTreeIDC1(const float integrationTimeOrbits=12, const char *outFileName="IDC1Tree.root") const
void dumpIDCOneToFile(const Side side, const char *outFileName="IDCOne.root", const char *outName="IDC1") const
dump the IDC1 to file
void createStatusMapOutlier(const bool debug=false)
void setTimeStamp(const long timeStamp)
void setPadFlagMap(const char *inpFile, const char *mapName)
Helper class for accessing grouped pads for one sector.
std::array< unsigned int, Mapper::NREGIONS > mNIDCsPerCRU
total number of IDCs per region per integration interval
static constexpr unsigned int ROWOFFSET[NREGIONS]
offset to calculate local row from global row
static GlobalPadNumber getGlobalPadNumber(const unsigned int lrow, const unsigned int pad, const unsigned int region)
static const std::vector< unsigned int > ADDITIONALPADSPERROW[NREGIONS]
additional pads per row compared to first row
static const std::vector< unsigned int > PADSPERROW[NREGIONS]
number of pads per row in region
static const std::vector< unsigned int > OFFSETCRULOCAL[NREGIONS]
row offset in cru for given local pad row
const FECInfo & fecInfo(GlobalPadNumber padNumber) const
static constexpr float INVPADAREA[NREGIONS]
inverse size of the pad area padwidth*padLength
static Mapper & instance(const std::string mappingDir="")
static constexpr unsigned int ROWSPERREGION[NREGIONS]
number of pad rows for region
static constexpr unsigned int NSECTORS
total number of sectors in the TPC
static constexpr unsigned int NREGIONS
total number of regions in one sector
static constexpr unsigned int PADROWS
total number of pad rows
static constexpr unsigned int PADSPERREGION[NREGIONS]
number of pads per CRU
static constexpr unsigned REGION[PADROWS]
region for global pad row
float getTrunctedMean(float low, float high)
GLint GLint GLsizei GLint GLenum GLenum type
GLenum GLsizei GLsizei GLint * values
GLfloat GLfloat GLfloat GLfloat GLfloat GLfloat maxZ
constexpr double LHCOrbitMUS
std::vector< T, o2::pmr::polymorphic_allocator< T > > vector
constexpr unsigned char SECTORSPERSIDE
@ IDC
integrated and grouped IDCs
@ IDCZero
IDC0: I_0(r,\phi) = <I(r,\phi,t)>_t.
@ IDCDelta
IDCDelta: \Delta I(r,\phi,t) = I(r,\phi,t) / ( I_0(r,\phi) * I_1(t) )
constexpr unsigned char SIDES
@ Region
Regions (up to 36*10)
IDCDeltaCompression
IDC Delta IDC Compression types.
@ HIGH
high compression using char (data compression ratio ~5.5 when stored in CCDB)
@ NO
no compression using floats
@ MEDIUM
medium compression using short (data compression ratio 2 when stored in CCDB)
@ flagLowPad
flag for pad with extremly low IDC value
@ flagNeighbour
flag if n neighbouring pads are outlier
@ flagGoodPad
flag for a good pad binary 0001
@ flagSkip
flag for defining a pad which is just ignored during the calculation of I1 and IDCDelta
@ flagDeadPad
flag for a dead pad binary 0010
@ flagHighPad
flag for pad with extremly high IDC value
@ flagFEC
flag for a whole masked FEC
constexpr unsigned short GEMSTACKSPERSECTOR
DataT sum(const Vector< DataT, N > &a)
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
std::unique_ptr< GPUReconstructionTimeframe > tf
struct to access and set Delta IDCs
std::function< float(const unsigned int, const unsigned int, const unsigned int, const unsigned int)> mIDCFunc
function returning the value which will be drawn for sector, region, row, pad
std::array< std::vector< float >, CRU::MaxCRU > idcs
struct containing the ITPC0 values (integrated TPC clusters)
std::vector< float > mIDCZero
I_0(r,\phi) = <I(r,\phi,t)>_t.