67 TPCTimeSeries(std::shared_ptr<o2::base::GRPGeomRequest> req,
const bool disableWriter,
const o2::base::Propagator::MatCorrType matType,
const bool enableUnbinnedWriter,
const bool tpcOnly, std::shared_ptr<o2::globaltracking::DataRequest> dr) : mCCDBRequest(req), mDisableWriter(disableWriter), mMatType(matType), mUnbinnedWriter(enableUnbinnedWriter), mTPCOnly(tpcOnly), mDataRequest(dr) {};
72 mNMaxTracks = ic.options().get<
int>(
"max-tracks");
73 mMinMom = ic.options().get<
float>(
"min-momentum");
74 mMinNCl = ic.options().get<
int>(
"min-cluster");
75 mMaxTgl = ic.options().get<
float>(
"max-tgl");
76 mMaxQPt = ic.options().get<
float>(
"max-qPt");
77 mCoarseStep = ic.options().get<
float>(
"coarse-step");
78 mFineStep = ic.options().get<
float>(
"fine-step");
79 mCutDCA = ic.options().get<
float>(
"cut-DCA-median");
80 mCutRMS = ic.options().get<
float>(
"cut-DCA-RMS");
81 mRefXSec = ic.options().get<
float>(
"refX-for-sector");
82 mTglBins = ic.options().get<
int>(
"tgl-bins");
83 mPhiBins = ic.options().get<
int>(
"phi-bins");
84 mQPtBins = ic.options().get<
int>(
"qPt-bins");
85 mNThreads = ic.options().get<
int>(
"threads");
86 maxITSTPCDCAr = ic.options().get<
float>(
"max-ITS-TPC-DCAr");
87 maxITSTPCDCAz = ic.options().get<
float>(
"max-ITS-TPC-DCAz");
88 maxITSTPCDCAr_comb = ic.options().get<
float>(
"max-ITS-TPC-DCAr_comb");
89 maxITSTPCDCAz_comb = ic.options().get<
float>(
"max-ITS-TPC-DCAz_comb");
90 mTimeWindowMUS = ic.options().get<
float>(
"time-window-mult-mus");
91 mMIPdEdx = ic.options().get<
float>(
"MIP-dedx");
92 mMaxSnp = ic.options().get<
float>(
"max-snp");
93 mXCoarse = ic.options().get<
float>(
"mX-coarse");
94 mSqrt = ic.options().get<
float>(
"sqrts");
95 mMultBins = ic.options().get<
int>(
"mult-bins");
96 mMultMax = ic.options().get<
int>(
"mult-max");
97 mMinTracksPerVertex = ic.options().get<
int>(
"min-tracks-per-vertex");
98 mMaxdEdxRatio = ic.options().get<
float>(
"max-dedx-ratio");
99 mMaxdEdxRegionRatio = ic.options().get<
float>(
"max-dedx-region-ratio");
100 mSamplingFactor = ic.options().get<
float>(
"sampling-factor");
101 mSampleTsallis = ic.options().get<
bool>(
"sample-unbinned-tsallis");
102 mXOuterMatching = ic.options().get<
float>(
"refX-for-outer-ITS");
103 mUseMinBiasTrigger = !ic.options().get<
bool>(
"disable-min-bias-trigger");
104 mMaxOccupancyHistBins = ic.options().get<
int>(
"max-occupancy-bins");
106 if (mUnbinnedWriter) {
107 for (
int iThread = 0; iThread < mNThreads; ++iThread) {
108 mGenerator.emplace_back(std::mt19937(std::random_device{}()));
111 mBufferVals.resize(mNThreads);
112 mBufferDCA.
setBinning(mPhiBins, mTglBins, mQPtBins, mMultBins, mMaxTgl, mMaxQPt, mMultMax);
113 if (mUnbinnedWriter) {
114 std::string outfile = ic.options().get<std::string>(
"out-file-unbinned");
116 ROOT::EnableThreadSafety();
118 mStreamer.resize(mNThreads);
119 for (
int iThread = 0; iThread < mNThreads; ++iThread) {
120 std::string outfileThr = outfile;
121 outfileThr.replace(outfileThr.length() - 5, outfileThr.length(), fmt::format(
"_{}.root", iThread));
122 LOGP(info,
"Writing unbinned data to: {}", outfileThr);
123 mStreamer[iThread] = std::make_unique<o2::utils::TreeStreamRedirector>(outfileThr.data(),
"recreate");
136 LOGP(info,
"Updated reference drift velocity to: {}", mVDrift);
138 pc.inputs().get<TTree*>(
"tpcSecFlucInfo");
141 const int nBins = getNBins();
150 if (mAvgADCAr.size() != nBins) {
152 mAvgADCAr.resize(nBins);
153 mAvgCDCAr.resize(nBins);
154 mAvgADCAz.resize(nBins);
155 mAvgCDCAz.resize(nBins);
156 mAvgMeffA.resize(nBins);
157 mAvgMeffC.resize(nBins);
158 mAvgChi2MatchA.resize(nBins);
159 mAvgChi2MatchC.resize(nBins);
160 mMIPdEdxRatioQMaxA.resize(nBins);
161 mMIPdEdxRatioQMaxC.resize(nBins);
162 mMIPdEdxRatioQTotA.resize(nBins);
163 mMIPdEdxRatioQTotC.resize(nBins);
164 mTPCChi2A.resize(nBins);
165 mTPCChi2C.resize(nBins);
166 mTPCNClA.resize(nBins);
167 mTPCNClC.resize(nBins);
168 mLogdEdxQTotA.resize(nBins);
169 mLogdEdxQTotC.resize(nBins);
170 mLogdEdxQMaxA.resize(nBins);
171 mLogdEdxQMaxC.resize(nBins);
172 mITSPropertiesA.resize(nBins);
173 mITSPropertiesC.resize(nBins);
174 mITSTPCDeltaPA.resize(nBins);
175 mITSTPCDeltaPC.resize(nBins);
176 mSigmaYZA.resize(nBins);
177 mSigmaYZC.resize(nBins);
185 auto tracksITSTPC = mTPCOnly ? gsl::span<o2::dataformats::TrackTPCITS>() : recoData.
getTPCITSTracks();
186 auto tracksITS = mTPCOnly ? gsl::span<o2::its::TrackITS>() : recoData.
getITSTracks();
189 auto vertices = mTPCOnly ? gsl::span<o2::dataformats::PrimaryVertex>() : recoData.
getPrimaryVertices();
200 const auto& tofClusters = mTPCOnly ? gsl::span<o2::tof::Cluster>() : recoData.
getTOFClusters();
202 LOGP(info,
"Processing {} vertices, {} primary matched vertices, {} TPC tracks, {} ITS tracks, {} ITS-TPC tracks, {} TOF clusters", vertices.size(), primMatchedTracks.size(), tracksTPC.size(), tracksITS.size(), tracksITSTPC.size(), tofClusters.size());
205 auto indicesITSTPC_vtx = processVertices(vertices, primMatchedTracks, primMatchedTracksRef, recoData);
208 std::unordered_map<unsigned int, std::array<int, 2>> indicesITSTPC;
210 for (
int i = 0;
i < tracksITSTPC.size(); ++
i) {
211 auto it = indicesITSTPC_vtx.find(
i);
213 const auto idxVtx = (it != indicesITSTPC_vtx.end()) ? (it->second) : -1;
215 indicesITSTPC[tracksITSTPC[
i].getRefTPC().getIndex()] = {
i, idxVtx};
218 std::vector<std::tuple<int, float, float, o2::track::TrackLTIntegral, double, float, unsigned int, unsigned short>> idxTPCTrackToTOFCluster;
221 if (mUnbinnedWriter) {
225 idxTPCTrackToTOFCluster = std::vector<std::tuple<int, float, float, o2::track::TrackLTIntegral, double, float, unsigned int, unsigned short>>(tracksTPC.size(), {-1, -999, -999, defLT, 0, 0, 0, 0});
230 std::map<ULong64_t, short> t0array;
231 for (
const auto&
t0 : ft0rec) {
232 if (!(
t0.isValidTime(1) &&
t0.isValidTime(2))) {
236 auto bclong =
t0.mIntRecord.differenceInBC(recoData.
startIR);
237 if (t0array.find(bclong) == t0array.end()) {
238 t0array.emplace(std::make_pair(bclong,
t0.getCollisionTime(0)));
245 for (
const auto& tofMatch : tofMatches) {
246 for (
const auto& tpctofmatch : tofMatch) {
248 if (refTPC.isIndexSet()) {
250 ULong64_t bclongtof = (tpctofmatch.getSignal() - 10000) * BC_TIME_INPS_INV;
252 unsigned int mask = 0;
253 if (!(t0array.find(bclongtof) == t0array.end())) {
254 t0 += t0array.find(bclongtof)->second;
258 double signal = tpctofmatch.getSignal() -
t0;
259 float deltaT = tpctofmatch.getDeltaT();
261 float dy = tpctofmatch.getDYatTOF();
262 bool isMultiHitZ = tpctofmatch.getHitPatternUpDown();
263 bool isMultiHitX = tpctofmatch.getHitPatternLeftRight();
264 bool isMultiStripMatch = tpctofmatch.getChi2() < 1E-9;
265 float chi2 = tpctofmatch.getChi2();
266 bool hasT0_1BCbefore = (t0array.find(bclongtof - 1) != t0array.end());
267 bool hasT0_2BCbefore = (t0array.find(bclongtof - 2) != t0array.end());
275 if (fabs(dy) > 0.5) {
278 if (isMultiStripMatch) {
290 if (hasT0_1BCbefore) {
293 if (hasT0_2BCbefore) {
297 idxTPCTrackToTOFCluster[refTPC] = {tpctofmatch.getIdxTOFCl(), tpctofmatch.getDXatTOF(), tpctofmatch.getDZatTOF(), ltIntegral, signal, deltaT,
mask, tpctofmatch.getChannel() % 8736};
304 findNearesVertex(tracksTPC, vertices);
307 if (mUnbinnedWriter) {
308 mTPCTrackClIdx = pc.inputs().get<gsl::span<o2::tpc::TPCClRefElem>>(
"trackTPCClRefs");
313 findNNeighbourTracks(tracksTPC);
316 for (
int i = 0;
i < nBins; ++
i) {
318 mAvgADCAr[
i][
type].clear();
319 mAvgCDCAr[
i][
type].clear();
320 mAvgADCAz[
i][
type].clear();
321 mAvgCDCAz[
i][
type].clear();
323 for (
int type = 0;
type < mMIPdEdxRatioQMaxA[
i].size(); ++
type) {
324 mMIPdEdxRatioQMaxA[
i][
type].clear();
325 mMIPdEdxRatioQMaxC[
i][
type].clear();
326 mMIPdEdxRatioQTotA[
i][
type].clear();
327 mMIPdEdxRatioQTotC[
i][
type].clear();
328 mTPCChi2A[
i][
type].clear();
329 mTPCChi2C[
i][
type].clear();
330 mTPCNClA[
i][
type].clear();
331 mTPCNClC[
i][
type].clear();
332 mMIPdEdxRatioQMaxA[
i][
type].setUseWeights(
false);
333 mMIPdEdxRatioQMaxC[
i][
type].setUseWeights(
false);
334 mMIPdEdxRatioQTotA[
i][
type].setUseWeights(
false);
335 mMIPdEdxRatioQTotC[
i][
type].setUseWeights(
false);
336 mTPCChi2A[
i][
type].setUseWeights(
false);
337 mTPCChi2C[
i][
type].setUseWeights(
false);
338 mTPCNClA[
i][
type].setUseWeights(
false);
339 mTPCNClC[
i][
type].setUseWeights(
false);
342 mLogdEdxQTotA[
i][
type].clear();
343 mLogdEdxQTotC[
i][
type].clear();
344 mLogdEdxQMaxA[
i][
type].clear();
345 mLogdEdxQMaxC[
i][
type].clear();
346 mLogdEdxQTotA[
i][
type].setUseWeights(
false);
347 mLogdEdxQTotC[
i][
type].setUseWeights(
false);
348 mLogdEdxQMaxA[
i][
type].setUseWeights(
false);
349 mLogdEdxQMaxC[
i][
type].setUseWeights(
false);
351 for (
int j = 0;
j < mITSPropertiesA[
i].size(); ++
j) {
352 mITSPropertiesA[
i][
j].clear();
353 mITSPropertiesC[
i][
j].clear();
354 mITSPropertiesA[
i][
j].setUseWeights(
false);
355 mITSPropertiesC[
i][
j].setUseWeights(
false);
358 for (
int j = 0;
j < mITSTPCDeltaPA[
i].size(); ++
j) {
359 mITSTPCDeltaPA[
i][
j].clear();
360 mITSTPCDeltaPC[
i][
j].clear();
361 mITSTPCDeltaPA[
i][
j].setUseWeights(
false);
362 mITSTPCDeltaPC[
i][
j].setUseWeights(
false);
365 for (
int j = 0;
j < mSigmaYZA[
i].size(); ++
j) {
366 mSigmaYZA[
i][
j].clear();
367 mSigmaYZC[
i][
j].clear();
368 mSigmaYZA[
i][
j].setUseWeights(
false);
369 mSigmaYZC[
i][
j].setUseWeights(
false);
372 for (
int j = 0;
j < mAvgMeffA[
i].size(); ++
j) {
373 mAvgMeffA[
i][
j].clear();
374 mAvgMeffC[
i][
j].clear();
375 mAvgChi2MatchA[
i][
j].clear();
376 mAvgChi2MatchC[
i][
j].clear();
377 mAvgMeffA[
i][
j].setUseWeights(
false);
378 mAvgMeffC[
i][
j].setUseWeights(
false);
379 mAvgChi2MatchA[
i][
j].setUseWeights(
false);
380 mAvgChi2MatchC[
i][
j].setUseWeights(
false);
384 for (
int i = 0;
i < mNThreads; ++
i) {
385 mBufferVals[
i].front().clear();
386 mBufferVals[
i].back().clear();
390 const auto nTracks = tracksTPC.size();
391 const size_t loopEnd = (mNMaxTracks < 0) ? nTracks : ((mNMaxTracks > nTracks) ? nTracks : size_t(mNMaxTracks));
394 for (
int i = 0;
i < nBins; ++
i) {
401 if (
i < lastIdxPhi) {
402 resMem = loopEnd / mPhiBins;
403 }
else if (
i < lastIdxTgl) {
404 resMem = loopEnd / mTglBins;
405 }
else if (
i < lastIdxQPt) {
406 resMem = loopEnd / mQPtBins;
407 }
else if (
i < lastIdxMult) {
408 resMem = loopEnd / mMultBins;
415 mAvgADCAr[
i][
type].reserve(resMem);
416 mAvgCDCAr[
i][
type].reserve(resMem);
417 mAvgADCAz[
i][
type].reserve(resMem);
418 mAvgCDCAz[
i][
type].reserve(resMem);
420 for (
int type = 0;
type < mMIPdEdxRatioQMaxA[
i].size(); ++
type) {
421 mMIPdEdxRatioQMaxA[
i][
type].reserve(resMem);
422 mMIPdEdxRatioQMaxC[
i][
type].reserve(resMem);
423 mMIPdEdxRatioQTotA[
i][
type].reserve(resMem);
424 mMIPdEdxRatioQTotC[
i][
type].reserve(resMem);
425 mTPCChi2A[
i][
type].reserve(resMem);
426 mTPCChi2C[
i][
type].reserve(resMem);
427 mTPCNClA[
i][
type].reserve(resMem);
428 mTPCNClC[
i][
type].reserve(resMem);
430 for (
int j = 0;
j < mAvgMeffA[
i].size(); ++
j) {
431 mAvgMeffA[
i][
j].reserve(resMem);
432 mAvgMeffC[
i][
j].reserve(resMem);
433 mAvgChi2MatchA[
i][
j].reserve(resMem);
434 mAvgChi2MatchC[
i][
j].reserve(resMem);
436 for (
int j = 0;
j < mITSPropertiesA[
i].size(); ++
j) {
437 mITSPropertiesA[
i][
j].reserve(resMem);
438 mITSPropertiesC[
i][
j].reserve(resMem);
440 for (
int j = 0;
j < mITSTPCDeltaPA[
i].size(); ++
j) {
441 mITSTPCDeltaPA[
i][
j].reserve(resMem);
442 mITSTPCDeltaPC[
i][
j].reserve(resMem);
444 for (
int j = 0;
j < mSigmaYZA[
i].size(); ++
j) {
445 mSigmaYZA[
i][
j].reserve(resMem);
446 mSigmaYZC[
i][
j].reserve(resMem);
448 for (
int j = 0;
j < mAvgMeffA[
i].size(); ++
j) {
449 mLogdEdxQTotA[
i][
j].reserve(resMem);
450 mLogdEdxQTotC[
i][
j].reserve(resMem);
451 mLogdEdxQMaxA[
i][
j].reserve(resMem);
452 mLogdEdxQMaxC[
i][
j].reserve(resMem);
455 for (
int iThread = 0; iThread < mNThreads; ++iThread) {
456 const int resMem = (mNThreads > 0) ? loopEnd / mNThreads : loopEnd;
457 mBufferVals[iThread].front().reserve(loopEnd, 1);
458 mBufferVals[iThread].back().reserve(loopEnd, 0);
461 using timer = std::chrono::high_resolution_clock;
462 auto startTotal = timer::now();
465 if (loopEnd < nTracks) {
467 std::vector<size_t> ind(nTracks);
468 std::iota(ind.begin(), ind.end(), 0);
469 std::minstd_rand rng(std::time(
nullptr));
470 std::shuffle(ind.begin(), ind.end(), rng);
472 auto myThread = [&](
int iThread) {
473 for (
size_t i = iThread;
i < loopEnd;
i += mNThreads) {
474 if (acceptTrack(tracksTPC[
i])) {
475 fillDCA(tracksTPC, tracksITSTPC, vertices,
i, iThread, indicesITSTPC, tracksITS, idxTPCTrackToTOFCluster, tofClusters);
480 std::vector<std::thread> threads(mNThreads);
481 for (
int i = 0;
i < mNThreads;
i++) {
482 threads[
i] = std::thread(myThread,
i);
485 for (
auto& th : threads) {
489 auto myThread = [&](
int iThread) {
490 for (
size_t i = iThread;
i < loopEnd;
i += mNThreads) {
491 if (acceptTrack(tracksTPC[
i])) {
492 fillDCA(tracksTPC, tracksITSTPC, vertices,
i, iThread, indicesITSTPC, tracksITS, idxTPCTrackToTOFCluster, tofClusters);
497 std::vector<std::thread> threads(mNThreads);
498 for (
int i = 0;
i < mNThreads;
i++) {
499 threads[
i] = std::thread(myThread,
i);
502 for (
auto& th : threads) {
508 for (
const auto& vals : mBufferVals) {
511 const auto nPoints =
val.side.size();
512 for (
int i = 0;
i < nPoints; ++
i) {
513 const auto tglBin =
val.tglBin[
i];
514 const auto phiBin =
val.phiBin[
i];
515 const auto qPtBin =
val.qPtBin[
i];
516 const auto multBin =
val.multBin[
i];
517 const auto dcar =
val.dcar[
i];
518 const auto dcaz =
val.dcaz[
i];
519 const auto dcarW =
val.dcarW[
i];
520 const int binInt = nBins - 1;
521 const bool fillCombDCA = ((
type == 1) && (
val.dcarcomb[
i] != -1) && (
val.dcazcomb[
i] != -1));
522 const bool fillDCAR = (
type == 1) ? (dcar != -999) :
true;
523 const std::array<int, 5>
bins{tglBin, phiBin, qPtBin, multBin, binInt};
525 for (
auto bin :
bins) {
528 mAvgCDCAr[bin][
type].addValue(dcar, dcarW);
531 mAvgCDCAr[bin][2].addValue(
val.dcarcomb[
i], dcarW);
532 mAvgCDCAz[bin][2].addValue(
val.dcazcomb[
i], dcarW);
536 mAvgCDCAz[bin][
type].addValue(dcaz, dcarW);
540 mAvgADCAr[bin][
type].addValue(dcar, dcarW);
543 mAvgADCAr[bin][2].addValue(
val.dcarcomb[
i], dcarW);
544 mAvgADCAz[bin][2].addValue(
val.dcazcomb[
i], dcarW);
548 mAvgADCAz[bin][
type].addValue(dcaz, dcarW);
560 for (
int slice = 0; slice < nBins; ++slice) {
563 const auto dcaAr = mAvgADCAr[slice][
type].filterPointsMedian(mCutDCA, mCutRMS);
564 bufferDCA.mDCAr_A_Median[slice] = std::get<0>(dcaAr);
565 bufferDCA.mDCAr_A_WeightedMean[slice] = std::get<1>(dcaAr);
566 bufferDCA.mDCAr_A_RMS[slice] = std::get<2>(dcaAr);
567 bufferDCA.mDCAr_A_NTracks[slice] = std::get<3>(dcaAr);
569 const auto dcaAz = mAvgADCAz[slice][
type].filterPointsMedian(mCutDCA, mCutRMS);
570 bufferDCA.mDCAz_A_Median[slice] = std::get<0>(dcaAz);
571 bufferDCA.mDCAz_A_WeightedMean[slice] = std::get<1>(dcaAz);
572 bufferDCA.mDCAz_A_RMS[slice] = std::get<2>(dcaAz);
573 bufferDCA.mDCAz_A_NTracks[slice] = std::get<3>(dcaAz);
575 const auto dcaCr = mAvgCDCAr[slice][
type].filterPointsMedian(mCutDCA, mCutRMS);
576 bufferDCA.mDCAr_C_Median[slice] = std::get<0>(dcaCr);
577 bufferDCA.mDCAr_C_WeightedMean[slice] = std::get<1>(dcaCr);
578 bufferDCA.mDCAr_C_RMS[slice] = std::get<2>(dcaCr);
579 bufferDCA.mDCAr_C_NTracks[slice] = std::get<3>(dcaCr);
581 const auto dcaCz = mAvgCDCAz[slice][
type].filterPointsMedian(mCutDCA, mCutRMS);
582 bufferDCA.mDCAz_C_Median[slice] = std::get<0>(dcaCz);
583 bufferDCA.mDCAz_C_WeightedMean[slice] = std::get<1>(dcaCz);
584 bufferDCA.mDCAz_C_RMS[slice] = std::get<2>(dcaCz);
585 bufferDCA.mDCAz_C_NTracks[slice] = std::get<3>(dcaCz);
588 const auto dcaArComb = mAvgADCAr[slice][2].filterPointsMedian(mCutDCA, mCutRMS);
592 const auto dcaAzCom = mAvgADCAz[slice][2].filterPointsMedian(mCutDCA, mCutRMS);
596 const auto dcaCrComb = mAvgCDCAr[slice][2].filterPointsMedian(mCutDCA, mCutRMS);
600 const auto dcaCzComb = mAvgCDCAz[slice][2].filterPointsMedian(mCutDCA, mCutRMS);
608 for (
const auto& vals : mBufferVals) {
609 const auto&
val = vals.front();
610 const auto nPoints =
val.side.size();
611 for (
int i = 0;
i < nPoints; ++
i) {
612 const auto tglBin =
val.tglBin[
i];
613 const auto phiBin =
val.phiBin[
i];
614 const auto qPtBin =
val.qPtBin[
i];
615 const auto multBin =
val.multBin[
i];
616 const auto dcar =
val.dcar[
i];
617 const auto dcaz =
val.dcaz[
i];
618 const auto hasITS =
val.hasITS[
i];
619 const auto chi2Match =
val.chi2Match[
i];
620 const auto dedxRatioqMax =
val.dedxRatioqMax[
i];
621 const auto dedxRatioqTot =
val.dedxRatioqTot[
i];
622 const auto sqrtChi2TPC =
val.sqrtChi2TPC[
i];
623 const auto nClTPC =
val.nClTPC[
i];
624 const int binInt = nBins - 1;
631 auto& mAvgEff = isCSide ? mAvgMeffC : mAvgMeffA;
632 auto& mAvgChi2Match = isCSide ? mAvgChi2MatchC : mAvgChi2MatchA;
633 auto& mAvgmMIPdEdxRatioqMax = isCSide ? mMIPdEdxRatioQMaxC : mMIPdEdxRatioQMaxA;
634 auto& mAvgmMIPdEdxRatioqTot = isCSide ? mMIPdEdxRatioQTotC : mMIPdEdxRatioQTotA;
635 auto& mAvgmTPCChi2 = isCSide ? mTPCChi2C : mTPCChi2A;
636 auto& mAvgmTPCNCl = isCSide ? mTPCNClC : mTPCNClA;
637 auto& mAvgmdEdxRatioQMax = isCSide ? mLogdEdxQMaxC : mLogdEdxQMaxA;
638 auto& mAvgmdEdxRatioQTot = isCSide ? mLogdEdxQTotC : mLogdEdxQTotA;
639 auto& mITSProperties = isCSide ? mITSPropertiesC : mITSPropertiesA;
640 auto& mSigmaYZ = isCSide ? mSigmaYZC : mSigmaYZA;
641 auto& mITSTPCDeltaP = isCSide ? mITSTPCDeltaPC : mITSTPCDeltaPA;
643 const std::array<int, 5>
bins{tglBin, phiBin, qPtBin, multBin, binInt};
645 for (
auto bin :
bins) {
647 if ((std::abs(dcar - bufferDCAMedR[bin]) < (bufferDCARMSR[bin] * mCutRMS)) && (std::abs(dcaz - bufferDCAMedZ[bin]) < (bufferDCARMSZ[bin] * mCutRMS))) {
648 const auto gID =
val.gID[
i];
649 mAvgEff[bin][0].addValue(hasITS);
652 mAvgEff[bin][1].addValue(hasITS);
653 mAvgEff[bin][2].addValue(hasITS);
657 mAvgEff[bin][1].addValue(hasITS);
659 mAvgEff[bin][2].addValue(hasITS);
662 mAvgChi2Match[bin][0].addValue(chi2Match);
664 mAvgChi2Match[bin][1].addValue(chi2Match);
666 mAvgChi2Match[bin][2].addValue(chi2Match);
669 if (dedxRatioqMax > 0) {
670 mAvgmMIPdEdxRatioqMax[bin][0].addValue(dedxRatioqMax);
672 if (dedxRatioqTot > 0) {
673 mAvgmMIPdEdxRatioqTot[bin][0].addValue(dedxRatioqTot);
675 mAvgmTPCChi2[bin][0].addValue(sqrtChi2TPC);
676 mAvgmTPCNCl[bin][0].addValue(nClTPC);
678 if (dedxRatioqMax > 0) {
679 mAvgmMIPdEdxRatioqMax[bin][1].addValue(dedxRatioqMax);
681 if (dedxRatioqTot > 0) {
682 mAvgmMIPdEdxRatioqTot[bin][1].addValue(dedxRatioqTot);
684 mAvgmTPCChi2[bin][1].addValue(sqrtChi2TPC);
685 mAvgmTPCNCl[bin][1].addValue(nClTPC);
688 float dedxNormQMax =
val.dedxValsqMax[
i].dedxNorm;
689 if (dedxNormQMax > 0) {
690 mAvgmdEdxRatioQMax[bin][0].addValue(dedxNormQMax);
693 float dedxNormQTot =
val.dedxValsqTot[
i].dedxNorm;
694 if (dedxNormQTot > 0) {
695 mAvgmdEdxRatioQTot[bin][0].addValue(dedxNormQTot);
698 float dedxIROCQMax =
val.dedxValsqMax[
i].dedxIROC;
699 if (dedxIROCQMax > 0) {
700 mAvgmdEdxRatioQMax[bin][1].addValue(dedxIROCQMax);
703 float dedxIROCQTot =
val.dedxValsqTot[
i].dedxIROC;
704 if (dedxIROCQTot > 0) {
705 mAvgmdEdxRatioQTot[bin][1].addValue(dedxIROCQTot);
708 float dedxOROC1QMax =
val.dedxValsqMax[
i].dedxOROC1;
709 if (dedxOROC1QMax > 0) {
710 mAvgmdEdxRatioQMax[bin][2].addValue(dedxOROC1QMax);
713 float dedxOROC1QTot =
val.dedxValsqTot[
i].dedxOROC1;
714 if (dedxOROC1QTot > 0) {
715 mAvgmdEdxRatioQTot[bin][2].addValue(dedxOROC1QTot);
718 float dedxOROC2QMax =
val.dedxValsqMax[
i].dedxOROC2;
719 if (dedxOROC2QMax > 0) {
720 mAvgmdEdxRatioQMax[bin][3].addValue(dedxOROC2QMax);
723 float dedxOROC2QTot =
val.dedxValsqTot[
i].dedxOROC2;
724 if (dedxOROC2QTot > 0) {
725 mAvgmdEdxRatioQTot[bin][3].addValue(dedxOROC2QTot);
728 float dedxOROC3QMax =
val.dedxValsqMax[
i].dedxOROC3;
729 if (dedxOROC3QMax > 0) {
730 mAvgmdEdxRatioQMax[bin][4].addValue(dedxOROC3QMax);
733 float dedxOROC3QTot =
val.dedxValsqTot[
i].dedxOROC3;
734 if (dedxOROC3QTot > 0) {
735 mAvgmdEdxRatioQTot[bin][4].addValue(dedxOROC3QTot);
738 float nClITS =
val.nClITS[
i];
740 mITSProperties[bin][0].addValue(nClITS);
742 float chi2ITS =
val.chi2ITS[
i];
744 mITSProperties[bin][1].addValue(chi2ITS);
747 float sigmay2 =
val.sigmaY2[
i];
749 mSigmaYZ[bin][0].addValue(sigmay2);
751 float sigmaz2 =
val.sigmaZ2[
i];
753 mSigmaYZ[bin][1].addValue(sigmaz2);
756 float deltaP2 =
val.deltaP2[
i];
757 if (deltaP2 != -999) {
758 mITSTPCDeltaP[bin][0].addValue(deltaP2);
761 float deltaP3 =
val.deltaP3[
i];
762 if (deltaP3 != -999) {
763 mITSTPCDeltaP[bin][1].addValue(deltaP3);
766 float deltaP4 =
val.deltaP4[
i];
767 if (deltaP4 != -999) {
768 mITSTPCDeltaP[bin][2].addValue(deltaP4);
776 for (
int slice = 0; slice < nBins; ++slice) {
777 for (
int i = 0;
i < mAvgMeffA[slice].size(); ++
i) {
779 itsBuf.mITSTPC_A_MatchEff[slice] = mAvgMeffA[slice][
i].getMean();
780 itsBuf.mITSTPC_C_MatchEff[slice] = mAvgMeffC[slice][
i].getMean();
781 itsBuf.mITSTPC_A_Chi2Match[slice] = mAvgChi2MatchA[slice][
i].getMean();
782 itsBuf.mITSTPC_C_Chi2Match[slice] = mAvgChi2MatchC[slice][
i].getMean();
786 for (
int i = 0;
i < mMIPdEdxRatioQMaxC[slice].size(); ++
i) {
789 buff.mMIPdEdxRatioQMaxC[slice] = mMIPdEdxRatioQMaxC[slice][
i].getMean();
790 buff.mMIPdEdxRatioQTotA[slice] = mMIPdEdxRatioQTotA[slice][
i].getMean();
791 buff.mMIPdEdxRatioQTotC[slice] = mMIPdEdxRatioQTotC[slice][
i].getMean();
792 buff.mTPCChi2C[slice] = mTPCChi2C[slice][
i].getMean();
793 buff.mTPCChi2A[slice] = mTPCChi2A[slice][
i].getMean();
794 buff.mTPCNClC[slice] = mTPCNClC[slice][
i].getMean();
795 buff.mTPCNClA[slice] = mTPCNClA[slice][
i].getMean();
800 auto& logdEdxA = (
type == 0) ? mLogdEdxQMaxA : mLogdEdxQTotA;
804 buffer.mLogdEdx_A_RMS[slice] = logdEdxA[slice][0].getStdDev();
805 buffer.mLogdEdx_A_IROC_Median[slice] = logdEdxA[slice][1].getMedian();
806 buffer.mLogdEdx_A_IROC_RMS[slice] = logdEdxA[slice][1].getStdDev();
807 buffer.mLogdEdx_A_OROC1_Median[slice] = logdEdxA[slice][2].getMedian();
808 buffer.mLogdEdx_A_OROC1_RMS[slice] = logdEdxA[slice][2].getStdDev();
809 buffer.mLogdEdx_A_OROC2_Median[slice] = logdEdxA[slice][3].getMedian();
810 buffer.mLogdEdx_A_OROC2_RMS[slice] = logdEdxA[slice][3].getStdDev();
811 buffer.mLogdEdx_A_OROC3_Median[slice] = logdEdxA[slice][4].getMedian();
812 buffer.mLogdEdx_A_OROC3_RMS[slice] = logdEdxA[slice][4].getStdDev();
814 auto& logdEdxC = (
type == 0) ? mLogdEdxQMaxC : mLogdEdxQTotC;
815 buffer.mLogdEdx_C_Median[slice] = logdEdxC[slice][0].getMedian();
816 buffer.mLogdEdx_C_RMS[slice] = logdEdxC[slice][0].getStdDev();
817 buffer.mLogdEdx_C_IROC_Median[slice] = logdEdxC[slice][1].getMedian();
818 buffer.mLogdEdx_C_IROC_RMS[slice] = logdEdxC[slice][1].getStdDev();
819 buffer.mLogdEdx_C_OROC1_Median[slice] = logdEdxC[slice][2].getMedian();
820 buffer.mLogdEdx_C_OROC1_RMS[slice] = logdEdxC[slice][2].getStdDev();
821 buffer.mLogdEdx_C_OROC2_Median[slice] = logdEdxC[slice][3].getMedian();
822 buffer.mLogdEdx_C_OROC2_RMS[slice] = logdEdxC[slice][3].getStdDev();
823 buffer.mLogdEdx_C_OROC3_Median[slice] = logdEdxC[slice][4].getMedian();
824 buffer.mLogdEdx_C_OROC3_RMS[slice] = logdEdxC[slice][4].getStdDev();
830 mBufferDCA.
mITS_A_NCl_RMS[slice] = mITSPropertiesA[slice][0].getStdDev();
835 mBufferDCA.
mITS_C_NCl_RMS[slice] = mITSPropertiesC[slice][0].getStdDev();
862 auto stop = timer::now();
863 std::chrono::duration<float>
time = stop - startTotal;
864 LOGP(info,
"Time series creation took {}",
time.count());
872 for (
auto& streamer : mStreamer) {
875 eos.services().get<
ControlService>().readyToQuit(QuitRequest::Me);
884 LOGP(info,
"Updating TPC sector edge fluctuation info");
886 LOGP(info,
"Loaded sector edge fluctuation information with {} intervals for {} runs", mSecEdgeFlucInfo.
size(), mSecEdgeFlucInfo.
getNRuns());
901 void reserve(
int n,
int type)
911 dedxRatioqTot.reserve(
n);
912 dedxRatioqMax.reserve(
n);
913 sqrtChi2TPC.reserve(
n);
917 chi2Match.reserve(
n);
919 dedxValsqTot.reserve(
n);
920 dedxValsqMax.reserve(
n);
928 }
else if (
type == 0) {
946 dedxRatioqTot.clear();
947 dedxRatioqMax.clear();
955 dedxValsqTot.clear();
956 dedxValsqMax.clear();
964 void emplace_back(
Side sideTmp,
int tglBinTmp,
int phiBinTmp,
int qPtBinTmp,
int multBinTmp,
float dcarTmp,
float dcazTmp,
float dcarWTmp,
float dedxRatioqTotTmp,
float dedxRatioqMaxTmp,
float sqrtChi2TPCTmp,
float nClTPCTmp,
o2::dataformats::GlobalTrackID::Source gIDTmp,
float chi2MatchTmp,
int hasITSTmp,
int nClITSTmp,
float chi2ITSTmp,
const ValsdEdx& dedxValsqTotTmp,
const ValsdEdx& dedxValsqMaxTmp,
float sigmaY2Tmp,
float sigmaZ2Tmp)
966 side.emplace_back(sideTmp);
967 tglBin.emplace_back(tglBinTmp);
968 phiBin.emplace_back(phiBinTmp);
969 qPtBin.emplace_back(qPtBinTmp);
970 multBin.emplace_back(multBinTmp);
971 dcar.emplace_back(dcarTmp);
972 dcaz.emplace_back(dcazTmp);
973 dcarW.emplace_back(dcarWTmp);
974 dedxRatioqTot.emplace_back(dedxRatioqTotTmp);
975 dedxRatioqMax.emplace_back(dedxRatioqMaxTmp);
976 sqrtChi2TPC.emplace_back(sqrtChi2TPCTmp);
977 nClTPC.emplace_back(nClTPCTmp);
978 chi2Match.emplace_back(chi2MatchTmp);
979 hasITS.emplace_back(hasITSTmp);
980 gID.emplace_back(gIDTmp);
981 dedxValsqMax.emplace_back(dedxValsqMaxTmp);
982 dedxValsqTot.emplace_back(dedxValsqTotTmp);
983 nClITS.emplace_back(nClITSTmp);
984 chi2ITS.emplace_back(chi2ITSTmp);
985 sigmaY2.emplace_back(sigmaY2Tmp);
986 sigmaZ2.emplace_back(sigmaZ2Tmp);
987 deltaP2.emplace_back(-999);
988 deltaP3.emplace_back(-999);
989 deltaP4.emplace_back(-999);
992 void setDeltaParam(
float deltaP2Tmp,
float deltaP3Tmp,
float deltaP4Tmp)
994 if (!deltaP2.empty()) {
995 deltaP2.back() = deltaP2Tmp;
996 deltaP3.back() = deltaP3Tmp;
997 deltaP4.back() = deltaP4Tmp;
1001 void emplace_back_ITSTPC(
Side sideTmp,
int tglBinTmp,
int phiBinTmp,
int qPtBinTmp,
int multBinTmp,
float dcarTmp,
float dcazTmp,
float dcarWTmp,
float dedxRatioqTotTmp,
float dedxRatioqMaxTmp,
float sqrtChi2TPCTmp,
float nClTPCTmp,
float dcarCombTmp,
float dcazCombTmp)
1003 side.emplace_back(sideTmp);
1004 tglBin.emplace_back(tglBinTmp);
1005 phiBin.emplace_back(phiBinTmp);
1006 qPtBin.emplace_back(qPtBinTmp);
1007 multBin.emplace_back(multBinTmp);
1008 dcar.emplace_back(dcarTmp);
1009 dcaz.emplace_back(dcazTmp);
1010 dcarW.emplace_back(dcarWTmp);
1011 dedxRatioqTot.emplace_back(dedxRatioqTotTmp);
1012 dedxRatioqMax.emplace_back(dedxRatioqMaxTmp);
1013 sqrtChi2TPC.emplace_back(sqrtChi2TPCTmp);
1014 nClTPC.emplace_back(nClTPCTmp);
1015 dcarcomb.emplace_back(dcarCombTmp);
1016 dcazcomb.emplace_back(dcazCombTmp);
1019 std::vector<Side>
side;
1020 std::vector<int> tglBin;
1021 std::vector<int> phiBin;
1022 std::vector<int> qPtBin;
1023 std::vector<int> multBin;
1024 std::vector<float> dcar;
1025 std::vector<float> dcaz;
1026 std::vector<float> dcarW;
1027 std::vector<bool> hasITS;
1028 std::vector<float> chi2Match;
1029 std::vector<float> dedxRatioqTot;
1030 std::vector<float> dedxRatioqMax;
1031 std::vector<float> sqrtChi2TPC;
1032 std::vector<float> nClTPC;
1033 std::vector<float> dcarcomb;
1034 std::vector<float> dcazcomb;
1035 std::vector<ValsdEdx> dedxValsqTot;
1036 std::vector<ValsdEdx> dedxValsqMax;
1037 std::vector<int> nClITS;
1038 std::vector<float> chi2ITS;
1039 std::vector<o2::dataformats::GlobalTrackID::Source> gID;
1040 std::vector<float> deltaP2;
1041 std::vector<float> deltaP3;
1042 std::vector<float> deltaP4;
1043 std::vector<float> sigmaY2;
1044 std::vector<float> sigmaZ2;
1046 std::shared_ptr<o2::base::GRPGeomRequest> mCCDBRequest;
1047 const bool mDisableWriter{
false};
1049 const bool mUnbinnedWriter{
false};
1050 const bool mTPCOnly{
false};
1051 std::shared_ptr<o2::globaltracking::DataRequest> mDataRequest;
1055 TimeSeriesITSTPC mBufferDCA;
1056 std::vector<std::array<RobustAverage, 3>> mAvgADCAr;
1057 std::vector<std::array<RobustAverage, 3>> mAvgCDCAr;
1058 std::vector<std::array<RobustAverage, 3>> mAvgADCAz;
1059 std::vector<std::array<RobustAverage, 3>> mAvgCDCAz;
1060 std::vector<std::array<RobustAverage, 2>> mMIPdEdxRatioQMaxA;
1061 std::vector<std::array<RobustAverage, 2>> mMIPdEdxRatioQMaxC;
1062 std::vector<std::array<RobustAverage, 2>> mMIPdEdxRatioQTotA;
1063 std::vector<std::array<RobustAverage, 2>> mMIPdEdxRatioQTotC;
1064 std::vector<std::array<RobustAverage, 2>> mTPCChi2A;
1065 std::vector<std::array<RobustAverage, 2>> mTPCChi2C;
1066 std::vector<std::array<RobustAverage, 2>> mTPCNClA;
1067 std::vector<std::array<RobustAverage, 2>> mTPCNClC;
1068 std::vector<std::array<RobustAverage, 3>> mAvgMeffA;
1069 std::vector<std::array<RobustAverage, 3>> mAvgMeffC;
1070 std::vector<std::array<RobustAverage, 3>> mAvgChi2MatchA;
1071 std::vector<std::array<RobustAverage, 3>> mAvgChi2MatchC;
1072 std::vector<std::array<RobustAverage, 10>> mLogdEdxQTotA;
1073 std::vector<std::array<RobustAverage, 10>> mLogdEdxQTotC;
1074 std::vector<std::array<RobustAverage, 10>> mLogdEdxQMaxA;
1075 std::vector<std::array<RobustAverage, 10>> mLogdEdxQMaxC;
1076 std::vector<std::array<RobustAverage, 2>> mITSPropertiesA;
1077 std::vector<std::array<RobustAverage, 2>> mITSPropertiesC;
1078 std::vector<std::array<RobustAverage, 3>> mITSTPCDeltaPA;
1079 std::vector<std::array<RobustAverage, 3>> mITSTPCDeltaPC;
1080 std::vector<std::array<RobustAverage, 2>> mSigmaYZA;
1081 std::vector<std::array<RobustAverage, 2>> mSigmaYZC;
1082 int mNMaxTracks{-1};
1087 float mCoarseStep{1};
1088 float mFineStep{0.005};
1091 float mRefXSec{108.475};
1093 float maxITSTPCDCAr{0.2};
1094 float maxITSTPCDCAz{10};
1095 float maxITSTPCDCAr_comb{0.2};
1096 float maxITSTPCDCAz_comb{0.2};
1097 gsl::span<const TPCClRefElem> mTPCTrackClIdx{};
1098 std::vector<std::array<FillVals, 2>> mBufferVals;
1099 uint32_t mFirstTFOrbit{0};
1100 float mTimeWindowMUS{50};
1102 std::vector<int> mNTracksWindow;
1103 std::vector<int> mNearestVtxTPC;
1105 float mVDrift{2.64};
1106 float mMaxSnp{0.85};
1110 int mMultMax{80000};
1112 int mMinTracksPerVertex{5};
1113 float mMaxdEdxRatio{0.3};
1114 float mMaxdEdxRegionRatio{0.5};
1115 float mSamplingFactor{0.1};
1116 bool mSampleTsallis{
false};
1117 std::vector<std::mt19937> mGenerator;
1118 std::vector<std::unique_ptr<o2::utils::TreeStreamRedirector>> mStreamer;
1119 float mXOuterMatching{60};
1120 bool mUseMinBiasTrigger{
false};
1123 int mMaxOccupancyHistBins{912};
1124 PressureTemperatureHelper mPTHelper;
1128 bool acceptTrack(
const TrackTPC& track)
const {
return std::abs(
track.getTgl()) < mMaxTgl; }
1130 bool checkTrack(
const TrackTPC& track)
const
1132 const bool isGoodTrack = ((
track.getNClusters() < mMinNCl) || (
track.getP() < mMinMom)) ? false :
true;
1136 void fillDCA(
const gsl::span<const TrackTPC> tracksTPC,
const gsl::span<const o2::dataformats::TrackTPCITS> tracksITSTPC,
const gsl::span<const o2::dataformats::PrimaryVertex> vertices,
const int iTrk,
const int iThread,
const std::unordered_map<
unsigned int, std::array<int, 2>>& indicesITSTPC,
const gsl::span<const o2::its::TrackITS> tracksITS,
const std::vector<std::tuple<int, float, float, o2::track::TrackLTIntegral, double, float, unsigned int, unsigned short>>& idxTPCTrackToTOFCluster,
const gsl::span<const o2::tof::Cluster> tofClusters)
1138 const auto& trackFull = tracksTPC[iTrk];
1139 const bool isGoodTrack = checkTrack(trackFull);
1142 bool minBiasOk =
false;
1143 const float factorMinBias = 0.1 * mSamplingFactor;
1144 if (mUnbinnedWriter && mUseMinBiasTrigger) {
1145 std::uniform_real_distribution<>
distr(0., 1.);
1146 if (
distr(mGenerator[iThread]) < factorMinBias) {
1152 if (!isGoodTrack && !minBiasOk) {
1162 std::array<float, 2> dca;
1166 if (!propagator->PropagateToXBxByBz(track, mXCoarse, mMaxSnp, mCoarseStep, mMatType)) {
1171 if (!propagator->propagateToDCA(refPoint, track, propagator->getNominalBz(), mFineStep, mMatType, &dca)) {
1178 if (!propagator->propagateTo(trackTmp, mRefXSec,
false, mMaxSnp, mCoarseStep, mMatType)) {
1182 const int tglBin = mTglBins * std::abs(trackTmp.getTgl()) / mMaxTgl + mPhiBins;
1185 const int offsQPtBin = mPhiBins + mTglBins;
1186 const int qPtBin = offsQPtBin + mQPtBins * (trackTmp.getQ2Pt() + mMaxQPt) / (2 * mMaxQPt);
1187 const int localMult = mNTracksWindow[iTrk];
1189 const int offsMult = offsQPtBin + mQPtBins;
1190 const int multBin = offsMult + mMultBins * localMult / mMultMax;
1191 const int nBins = getNBins();
1193 if ((phiBin < 0) || (phiBin > mPhiBins) || (tglBin < mPhiBins) || (tglBin > offsQPtBin) || (qPtBin < offsQPtBin) || (qPtBin > offsMult) || (multBin < offsMult) || (multBin > offsMult + mMultBins)) {
1201 auto it = indicesITSTPC.find(iTrk);
1202 const auto idxITSTPC = (it != indicesITSTPC.end()) ? (it->second) : std::array<int, 2>{-1, -1};
1205 const auto vertex = (idxITSTPC.back() != -1) ? vertices[idxITSTPC.back()] : ((mNearestVtxTPC[iTrk] != -1) ? vertices[mNearestVtxTPC[iTrk]] :
o2::dataformats::PrimaryVertex{});
1208 const float signSide = trackFull.hasCSideClustersOnly() ? -1 : 1;
1209 const float dcaZFromDeltaTime = (
vertex.getTimeStamp().getTimeStamp() == 0) ? 0 : (
o2::tpc::ParameterElectronics::Instance().ZbinWidth * trackFull.getTime0() -
vertex.
getTimeStamp().
getTimeStamp()) * mVDrift + signSide *
vertex.getZ();
1214 const float div = (resCl *
track.getPt());
1219 const float fB = 0.2 / div;
1220 const float fA = 0.15 + 0.15;
1221 const float dcarW = 1. / std::sqrt(fA * fA + fB * fB);
1224 const bool hasITSTPC = idxITSTPC.front() != -1;
1227 const float chi2 = hasITSTPC ? tracksITSTPC[idxITSTPC.front()].getChi2Match() : -1;
1231 const auto src = tracksITSTPC[idxITSTPC.front()].getRefITS().getSource();
1239 const float chi2Match = (chi2 > 0) ? std::sqrt(chi2) : -1;
1240 const float sqrtChi2TPC = (trackFull.getChi2() > 0) ? std::sqrt(trackFull.getChi2()) : 0;
1241 const float nClTPC = trackFull.getNClusters();
1244 const float dedxRatioqTot = (trackFull.getdEdx().dEdxTotTPC > 0) ? (mMIPdEdx / trackFull.getdEdx().dEdxTotTPC) : -1;
1245 const float dedxRatioqMax = (trackFull.getdEdx().dEdxMaxTPC > 0) ? (mMIPdEdx / trackFull.getdEdx().dEdxMaxTPC) : -1;
1247 const auto dedxQTotVars = getdEdxVars(0, trackFull);
1248 const auto dedxQMaxVars = getdEdxVars(1, trackFull);
1252 const bool idxITSCheck = (idxITSTrack != -1);
1254 const int nClITS = idxITSCheck ? tracksITS[idxITSTrack].getNClusters() : -1;
1255 float chi2ITS = idxITSCheck ? tracksITS[idxITSTrack].getChi2() : -1;
1256 if ((chi2ITS > 0) && (nClITS > 0)) {
1257 chi2ITS = std::sqrt(chi2ITS / nClITS);
1259 sigmaY2 =
track.getSigmaY2();
1260 sigmaZ2 =
track.getSigmaZ2();
1262 if (trackFull.hasCSideClustersOnly()) {
1263 mBufferVals[iThread].front().emplace_back(
Side::C, tglBin, phiBin, qPtBin, multBin, dca[0], dcaZFromDeltaTime, dcarW, dedxRatioqTot, dedxRatioqMax, sqrtChi2TPC, nClTPC, gID, chi2Match, hasITSTPC, nClITS, chi2ITS, dedxQTotVars, dedxQMaxVars, sigmaY2, sigmaZ2);
1264 }
else if (trackFull.hasASideClustersOnly()) {
1265 mBufferVals[iThread].front().emplace_back(
Side::A, tglBin, phiBin, qPtBin, multBin, dca[0], dcaZFromDeltaTime, dcarW, dedxRatioqTot, dedxRatioqMax, sqrtChi2TPC, nClTPC, gID, chi2Match, hasITSTPC, nClITS, chi2ITS, dedxQTotVars, dedxQMaxVars, sigmaY2, sigmaZ2);
1271 std::array<float, 2> dcaITSTPC{0, 0};
1272 float deltaP0 = -999;
1273 float deltaP1 = -999;
1274 float deltaP2 = -999;
1275 float deltaP3 = -999;
1276 float deltaP4 = -999;
1277 float phiITSTPCAtVertex = -999;
1278 float dcaTPCAtVertex = -999;
1281 auto trackITSTPCTmp = tracksITSTPC[idxITSTPC.front()];
1283 if (propagator->propagateToDCA(refPoint, trackITSTPCTmp, propagator->getNominalBz(), mFineStep, mMatType, &dcaITSTPC)) {
1285 if ((std::abs(dcaITSTPC[0]) < maxITSTPCDCAr) && (std::abs(dcaITSTPC[1]) < maxITSTPCDCAz)) {
1288 const bool contributeToVertex = (idxITSTPC.back() != -1);
1289 std::array<float, 2> dcaITSTPCTmp{-1, -1};
1291 if (contributeToVertex) {
1292 if (propagator->propagateToDCA(
vertex.getXYZ(), trackITSTPCTmp, propagator->getNominalBz(), mFineStep, mMatType, &dcaITSTPCTmp)) {
1293 phiITSTPCAtVertex = trackITSTPCTmp.getPhi();
1294 dcaITSTPC = dcaITSTPCTmp;
1298 std::array<float, 2> dcaTPCTmp{-1, -1};
1299 if (propagator->propagateToDCA(
vertex.getXYZ(), track, propagator->getNominalBz(), mFineStep, mMatType, &dcaTPCTmp)) {
1300 dcaTPCAtVertex = dcaTPCTmp[0];
1305 if ((std::abs(dcaITSTPCTmp[0]) < maxITSTPCDCAr_comb) && (std::abs(dcaITSTPCTmp[1]) < maxITSTPCDCAz_comb)) {
1307 if (idxITSTrack >= 0 &&
track.rotate(tracksITS[idxITSTrack].getAlpha()) && propagator->propagateTo(track, trackITSTPCTmp.getX(),
false, mMaxSnp, mFineStep, mMatType)) {
1309 const bool propITSOk = propagator->propagateTo(trackITS, trackITSTPCTmp.getX(),
false, mMaxSnp, mFineStep, mMatType);
1311 deltaP0 =
track.getParam(0) - trackITS.getParam(0);
1312 deltaP1 =
track.getParam(1) - trackITS.getParam(1);
1313 deltaP2 =
track.getParam(2) - trackITS.getParam(2);
1314 deltaP3 =
track.getParam(3) - trackITS.getParam(3);
1315 deltaP4 =
track.getParam(4) - trackITS.getParam(4);
1316 mBufferVals[iThread].front().setDeltaParam(deltaP2, deltaP3, deltaP4);
1320 dcaITSTPCTmp[0] = -1;
1321 dcaITSTPCTmp[1] = -1;
1325 if (trackFull.hasCSideClustersOnly()) {
1326 mBufferVals[iThread].back().emplace_back_ITSTPC(
Side::C, tglBin, phiBin, qPtBin, multBin, dcaTPCAtVertex, dcaZFromDeltaTime, dcarW, dedxRatioqTot, dedxRatioqMax, sqrtChi2TPC, nClTPC, dcaITSTPCTmp[0], dcaITSTPCTmp[1]);
1327 }
else if (trackFull.hasASideClustersOnly()) {
1328 mBufferVals[iThread].back().emplace_back_ITSTPC(
Side::A, tglBin, phiBin, qPtBin, multBin, dcaTPCAtVertex, dcaZFromDeltaTime, dcarW, dedxRatioqTot, dedxRatioqMax, sqrtChi2TPC, nClTPC, dcaITSTPCTmp[0], dcaITSTPCTmp[1]);
1335 if (mUnbinnedWriter && mStreamer[iThread]) {
1336 const float factorPt = mSamplingFactor;
1337 bool writeData =
true;
1339 if (mSampleTsallis) {
1340 std::uniform_real_distribution<>
distr(0., 1.);
1343 if (writeData || minBiasOk) {
1344 auto clusterMask = makeClusterBitMask(trackFull);
1345 const auto& trkOrig = tracksTPC[iTrk];
1346 const bool isNearestVtx = (idxITSTPC.back() == -1);
1347 const float mx_ITS = hasITSTPC ? tracksITSTPC[idxITSTPC.front()].getX() : -1;
1348 const float pt_ITS = hasITSTPC ? tracksITSTPC[idxITSTPC.front()].getQ2Pt() : -1;
1349 const float chi2match_ITSTPC = hasITSTPC ? tracksITSTPC[idxITSTPC.front()].getChi2Match() : -1;
1350 const int nClITS = idxITSCheck ? tracksITS[idxITSTrack].getNClusters() : -1;
1351 const int chi2ITS = idxITSCheck ? tracksITS[idxITSTrack].getChi2() : -1;
1353 if (trackFull.hasASideClustersOnly()) {
1355 }
else if (trackFull.hasCSideClustersOnly()) {
1360 bool hasTOFCluster = (std::get<0>(idxTPCTrackToTOFCluster[iTrk]) != -1);
1361 auto tofCl = hasTOFCluster ? tofClusters[std::get<0>(idxTPCTrackToTOFCluster[iTrk])] :
o2::tof::Cluster();
1363 float tpcYDeltaAtTOF = -999;
1364 float tpcZDeltaAtTOF = -999;
1365 if (hasTOFCluster) {
1367 if (trackTmpOut.rotate(
o2::math_utils::sector2Angle(tofCl.getSector())) && propagator->propagateTo(trackTmpOut, tofCl.getX(),
false, mMaxSnp, mFineStep, mMatType)) {
1368 tpcYDeltaAtTOF = trackTmpOut.getY() - tofCl.getY();
1374 float deltaTPCParamInOutTgl = trackFull.getTgl() - trackFull.getParamOut().getTgl();
1375 float deltaTPCParamInOutQPt = trackFull.getQ2Pt() - trackFull.getParamOut().getQ2Pt();
1378 float deltaP0OuterITS = -999;
1379 float deltaP1OuterITS = -999;
1380 float deltaP2OuterITS = -999;
1381 float deltaP3OuterITS = -999;
1382 float deltaP4OuterITS = -999;
1385 const bool propITSOk = propagator->propagateTo(trackTmpOut, mXOuterMatching,
false, mMaxSnp, mCoarseStep, mMatType);
1386 if (propITSOk && trackTmp.rotate(trackTmpOut.getAlpha())) {
1387 const bool propTPCOk = propagator->propagateTo(trackTmp, mXOuterMatching,
false, mMaxSnp, mCoarseStep, mMatType);
1390 deltaP0OuterITS = trackTmp.getParam(0) - trackTmpOut.getParam(0);
1391 deltaP1OuterITS = trackTmp.getParam(1) - trackTmpOut.getParam(1);
1392 deltaP2OuterITS = trackTmp.getParam(2) - trackTmpOut.getParam(2);
1393 deltaP3OuterITS = trackTmp.getParam(3) - trackTmpOut.getParam(3);
1394 deltaP4OuterITS = trackTmp.getParam(4) - trackTmpOut.getParam(4);
1398 const int triggerMask = 0x1 * minBiasOk + 0x2 * writeData;
1400 float deltaP2ConstrVtx = -999;
1401 float deltaP3ConstrVtx = -999;
1402 float deltaP4ConstrVtx = -999;
1405 float covTPCConstrVtxP2 = -999;
1406 float covTPCConstrVtxP3 = -999;
1407 float covTPCConstrVtxP4 = -999;
1410 float covITSTPCConstrVtxP2 = -999;
1411 float covITSTPCConstrVtxP3 = -999;
1412 float covITSTPCConstrVtxP4 = -999;
1414 float covTPCAtVertex0 = -999;
1415 float covTPCAtVertex1 = -999;
1417 const bool contributeToVertex = (idxITSTPC.back() != -1);
1418 if (hasITSTPC && contributeToVertex) {
1420 std::array<float, 2> dcaITSTPCTmp{-1, -1};
1421 if (propagator->propagateToDCA(
vertex.getXYZ(), trackITSTPCTmp, propagator->getNominalBz(), mFineStep, mMatType, &dcaITSTPCTmp)) {
1423 if (trackTPC.rotate(trackITSTPCTmp.getAlpha()) && propagator->propagateTo(trackTPC, trackITSTPCTmp.getX(),
false, mMaxSnp, mFineStep, mMatType)) {
1425 covTPCAtVertex0 = trackTPC.getCovarElem(0, 0);
1426 covTPCAtVertex1 = trackTPC.getCovarElem(1, 1);
1429 deltaP2ConstrVtx = trackTPC.getParam(2) - trackITSTPCTmp.getParam(2);
1430 deltaP3ConstrVtx = trackTPC.getParam(3) - trackITSTPCTmp.getParam(3);
1431 deltaP4ConstrVtx = trackTPC.getParam(4) - trackITSTPCTmp.getParam(4);
1432 covTPCConstrVtxP2 = trackTPC.getCovarElem(2, 2);
1433 covTPCConstrVtxP3 = trackTPC.getCovarElem(3, 3);
1434 covTPCConstrVtxP4 = trackTPC.getCovarElem(4, 4);
1435 covITSTPCConstrVtxP2 = trackITSTPCTmp.getCovarElem(2, 2);
1436 covITSTPCConstrVtxP3 = trackITSTPCTmp.getCovarElem(3, 3);
1437 covITSTPCConstrVtxP4 = trackITSTPCTmp.getCovarElem(4, 4);
1441 double vertexTime =
vertex.getTimeStamp().getTimeStamp();
1442 double trackTime0 = trackFull.getTime0();
1443 *mStreamer[iThread] <<
"treeTimeSeries"
1445 <<
"triggerMask=" << triggerMask
1446 <<
"factorMinBias=" << factorMinBias
1447 <<
"factorPt=" << factorPt
1449 <<
"dcar_tpc_vertex=" << dcaTPCAtVertex
1450 <<
"dcar_tpc=" << dca[0]
1451 <<
"dcaz_tpc=" << dca[1]
1452 <<
"dcar_itstpc=" << dcaITSTPC[0]
1453 <<
"dcaz_itstpc=" << dcaITSTPC[1]
1454 <<
"dcarW=" << dcarW
1455 <<
"dcaZFromDeltaTime=" << dcaZFromDeltaTime
1456 <<
"hasITSTPC=" << hasITSTPC
1458 <<
"vertex_x=" <<
vertex.getX()
1459 <<
"vertex_y=" <<
vertex.getY()
1460 <<
"vertex_z=" <<
vertex.getZ()
1461 <<
"vertex_time=" <<
vertex.getTimeStamp().getTimeStamp()
1462 <<
"vertex_nContributors=" <<
vertex.getNContributors()
1463 <<
"isNearestVertex=" << isNearestVtx
1465 <<
"pt=" << trkOrig.getPt()
1466 <<
"qpt_ITSTPC=" << pt_ITS
1467 <<
"tpc_timebin=" << trkOrig.getTime0()
1468 <<
"qpt=" << trkOrig.getParam(4)
1469 <<
"ncl=" << trkOrig.getNClusters()
1470 <<
"ncl_shared=" << trkOrig.getNClusters()
1471 <<
"tgl=" << trkOrig.getTgl()
1472 <<
"side_type=" << typeSide
1473 <<
"phi=" << trkOrig.getPhi()
1474 <<
"clusterMask=" << clusterMask
1475 <<
"dedxTPC=" << trkOrig.getdEdx()
1476 <<
"chi2=" << trkOrig.getChi2()
1477 <<
"mX=" << trkOrig.getX()
1478 <<
"mX_ITS=" << mx_ITS
1479 <<
"nClITS=" << nClITS
1480 <<
"chi2ITS=" << chi2ITS
1481 <<
"chi2match_ITSTPC=" << chi2match_ITSTPC
1482 <<
"PID=" << trkOrig.getPID().getID()
1484 <<
"covTPCAtVertex0=" << covTPCAtVertex0
1485 <<
"covTPCAtVertex1=" << covTPCAtVertex1
1487 <<
"covTPCConstrVtxP2=" << covTPCConstrVtxP2
1488 <<
"covTPCConstrVtxP3=" << covTPCConstrVtxP3
1489 <<
"covTPCConstrVtxP4=" << covTPCConstrVtxP4
1491 <<
"covITSTPCConstrVtxP2=" << covITSTPCConstrVtxP2
1492 <<
"covITSTPCConstrVtxP3=" << covITSTPCConstrVtxP3
1493 <<
"covITSTPCConstrVtxP4=" << covITSTPCConstrVtxP4
1495 <<
"deltaP2ConstrVtx=" << deltaP2ConstrVtx
1496 <<
"deltaP3ConstrVtx=" << deltaP3ConstrVtx
1497 <<
"deltaP4ConstrVtx=" << deltaP4ConstrVtx
1499 <<
"deltaPar0=" << deltaP0
1500 <<
"deltaPar1=" << deltaP1
1501 <<
"deltaPar2=" << deltaP2
1502 <<
"deltaPar3=" << deltaP3
1503 <<
"deltaPar4=" << deltaP4
1504 <<
"sigmaY2=" << sigmaY2
1505 <<
"sigmaZ2=" << sigmaZ2
1507 <<
"mult=" << mNTracksWindow[iTrk]
1508 <<
"time_window_mult=" << mTimeWindowMUS
1509 <<
"firstTFOrbit=" << mFirstTFOrbit
1510 <<
"timeMS=" << mTimeMS
1512 <<
"mVDrift=" << mVDrift
1513 <<
"its_flag=" <<
int(gID)
1514 <<
"sqrtChi2Match=" << chi2Match
1516 <<
"tpcYDeltaAtTOF=" << tpcYDeltaAtTOF
1517 <<
"tpcZDeltaAtTOF=" << tpcZDeltaAtTOF
1518 <<
"mDXatTOF=" << std::get<1>(idxTPCTrackToTOFCluster[iTrk])
1519 <<
"mDZatTOF=" << std::get<2>(idxTPCTrackToTOFCluster[iTrk])
1520 <<
"mTOFLength=" << std::get<3>(idxTPCTrackToTOFCluster[iTrk])
1521 <<
"mTOFSignal=" << std::get<4>(idxTPCTrackToTOFCluster[iTrk])
1522 <<
"mDeltaTTOFTPC=" << std::get<5>(idxTPCTrackToTOFCluster[iTrk])
1523 <<
"vertexTime=" << vertexTime
1524 <<
"trackTime0=" << trackTime0
1525 <<
"TOFmask=" << std::get<6>(idxTPCTrackToTOFCluster[iTrk])
1526 <<
"TOFchannel=" << std::get<7>(idxTPCTrackToTOFCluster[iTrk])
1528 <<
"deltaTPCParamInOutTgl=" << deltaTPCParamInOutTgl
1529 <<
"deltaTPCParamInOutQPt=" << deltaTPCParamInOutQPt
1531 <<
"deltaP0OuterITS=" << deltaP0OuterITS
1532 <<
"deltaP1OuterITS=" << deltaP1OuterITS
1533 <<
"deltaP2OuterITS=" << deltaP2OuterITS
1534 <<
"deltaP3OuterITS=" << deltaP3OuterITS
1535 <<
"deltaP4OuterITS=" << deltaP4OuterITS
1536 <<
"mXOuterMatching=" << mXOuterMatching
1538 <<
"phiITSTPCAtVertex=" << phiITSTPCAtVertex
1546 mBufferDCA.mTSTPC.setStartTime(mTimeMS);
1547 mBufferDCA.mTSITSTPC.setStartTime(mTimeMS);
1550 if (!mDisableWriter) {
1558 void findNearesVertex(
const gsl::span<const TrackTPC> tracksTPC,
const gsl::span<const o2::dataformats::PrimaryVertex> vertices)
1561 const int nVertices = vertices.size();
1563 const int nTracks = tracksTPC.size();
1564 mNearestVtxTPC.clear();
1565 mNearestVtxTPC.resize(nTracks);
1569 std::fill(mNearestVtxTPC.begin(), mNearestVtxTPC.end(), -1);
1574 std::vector<float> times_vtx;
1575 times_vtx.reserve(nVertices);
1576 for (
const auto& vtx : vertices) {
1577 times_vtx.emplace_back(vtx.getTimeStamp().getTimeStamp());
1581 auto myThread = [&](
int iThread) {
1582 for (
int i = iThread;
i < nTracks;
i += mNThreads) {
1584 const auto lower = std::lower_bound(times_vtx.begin(), times_vtx.end(), timeTrack);
1585 int closestVtx = std::distance(times_vtx.begin(), lower);
1587 if (closestVtx == nVertices) {
1589 }
else if (closestVtx > 0) {
1591 double diff1 = std::abs(timeTrack - *lower);
1592 double diff2 = std::abs(timeTrack - *(lower - 1));
1593 if (diff2 < diff1) {
1597 mNearestVtxTPC[
i] = closestVtx;
1601 std::vector<std::thread> threads(mNThreads);
1602 for (
int i = 0;
i < mNThreads;
i++) {
1603 threads[
i] = std::thread(myThread,
i);
1607 for (
auto& th : threads) {
1613 std::vector<bool> makeClusterBitMask(
const TrackTPC& track)
const
1616 const int nCl =
track.getNClusterReferences();
1617 for (
int j = 0;
j <
nCl; ++
j) {
1619 uint32_t clusterIndexInRow;
1620 track.getClusterReference(mTPCTrackClIdx,
j, sector,
padrow, clusterIndexInRow);
1621 tpcClusterMask[
padrow] =
true;
1623 return tpcClusterMask;
1627 void findNNeighbourTracks(
const gsl::span<const TrackTPC> tracksTPC)
1630 const float windowTimeBins = mTimeWindowMUS / tpcTBinMUS;
1633 std::vector<float>
times;
1634 const int nTracks = tracksTPC.size();
1635 times.reserve(nTracks);
1636 for (
const auto& trk : tracksTPC) {
1637 times.emplace_back(trk.getTime0());
1641 mNTracksWindow.clear();
1642 mNTracksWindow.resize(nTracks);
1645 auto myThread = [&](
int iThread) {
1646 for (
int i = iThread;
i < nTracks;
i += mNThreads) {
1647 const float t0 = tracksTPC[
i].getTime0();
1648 const auto upperV0 = std::upper_bound(
times.begin(),
times.end(),
t0 + windowTimeBins);
1649 const auto lowerV0 = std::lower_bound(
times.begin(),
times.end(),
t0 - windowTimeBins);
1650 const int nMult = std::distance(
times.begin(), upperV0) - std::distance(
times.begin(), lowerV0);
1651 mNTracksWindow[
i] = nMult;
1655 std::vector<std::thread> threads(mNThreads);
1656 for (
int i = 0;
i < mNThreads;
i++) {
1657 threads[
i] = std::thread(myThread,
i);
1661 for (
auto& th : threads) {
1666 std::unordered_map<unsigned int, int> processVertices(
const gsl::span<const o2::dataformats::PrimaryVertex> vertices,
const gsl::span<const o2::dataformats::VtxTrackIndex> primMatchedTracks,
const gsl::span<const o2::dataformats::VtxTrackRef> primMatchedTracksRef,
const RecoContainer& recoData)
1669 std::unordered_map<unsigned int, int> indicesITSTPC_vtx;
1671 std::unordered_map<int, int> nContributors_ITS;
1672 std::unordered_map<int, int> nContributors_ITSTPC;
1675 if (!vertices.empty()) {
1676 for (
const auto&
ref : primMatchedTracksRef) {
1678 const std::array<TrkSrc, 5>
sources = {TrkSrc::ITSTPC, TrkSrc::ITSTPCTRD, TrkSrc::ITSTPCTOF, TrkSrc::ITSTPCTRDTOF, TrkSrc::ITS};
1680 const int vID =
ref.getVtxID();
1681 const int firstEntry =
ref.getFirstEntryOfSource(
source);
1682 const int nEntries =
ref.getEntriesOfSource(
source);
1684 for (
int i = 0;
i < nEntries; ++
i) {
1685 const auto& matchedTrk = primMatchedTracks[
i + firstEntry];
1686 bool pvCont = matchedTrk.isPVContributor();
1690 if (refITSTPC.isIndexSet()) {
1691 indicesITSTPC_vtx[refITSTPC] = vID;
1692 ++nContributors_ITSTPC[vID];
1694 ++nContributors_ITS[vID];
1703 std::array<RobustAverage, 4> avgVtxITS;
1704 std::array<RobustAverage, 4> avgVtxITSTPC;
1705 for (
int i = 0;
i < avgVtxITS.size(); ++
i) {
1706 avgVtxITS[
i].reserve(vertices.size());
1707 avgVtxITSTPC[
i].reserve(vertices.size());
1709 for (
int ivtx = 0; ivtx < vertices.size(); ++ivtx) {
1710 const auto& vtx = vertices[ivtx];
1711 const float itsFrac = nContributors_ITS[ivtx] /
static_cast<float>(vtx.getNContributors());
1712 const float itstpcFrac = (nContributors_ITS[ivtx] + nContributors_ITSTPC[ivtx]) /
static_cast<float>(vtx.getNContributors());
1714 const float itsMin = 0.2;
1715 const float itsMax = 0.8;
1716 if ((itsFrac > itsMin) && (itsFrac < itsMax)) {
1717 avgVtxITS[0].addValue(vtx.getX());
1718 avgVtxITS[1].addValue(vtx.getY());
1719 avgVtxITS[2].addValue(vtx.getZ());
1720 avgVtxITS[3].addValue(vtx.getNContributors());
1723 const float itstpcMax = 0.95;
1724 if (itstpcFrac < itstpcMax) {
1725 avgVtxITSTPC[0].addValue(vtx.getX());
1726 avgVtxITSTPC[1].addValue(vtx.getY());
1727 avgVtxITSTPC[2].addValue(vtx.getZ());
1728 avgVtxITSTPC[3].addValue(vtx.getNContributors());
1733 mBufferDCA.nPrimVertices_ITS.front() = avgVtxITS[3].getValues().size();
1734 mBufferDCA.nVertexContributors_ITS_Median.front() = avgVtxITS[3].getMedian();
1735 mBufferDCA.nVertexContributors_ITS_RMS.front() = avgVtxITS[3].getStdDev();
1736 mBufferDCA.vertexX_ITS_Median.front() = avgVtxITS[0].getMedian();
1737 mBufferDCA.vertexY_ITS_Median.front() = avgVtxITS[1].getMedian();
1738 mBufferDCA.vertexZ_ITS_Median.front() = avgVtxITS[2].getMedian();
1739 mBufferDCA.vertexX_ITS_RMS.front() = avgVtxITS[0].getStdDev();
1740 mBufferDCA.vertexY_ITS_RMS.front() = avgVtxITS[1].getStdDev();
1741 mBufferDCA.vertexZ_ITS_RMS.front() = avgVtxITS[2].getStdDev();
1744 mBufferDCA.nPrimVertices_ITSTPC.front() = avgVtxITSTPC[3].getValues().size();
1745 mBufferDCA.nVertexContributors_ITSTPC_Median.front() = avgVtxITSTPC[3].getMedian();
1746 mBufferDCA.nVertexContributors_ITSTPC_RMS.front() = avgVtxITSTPC[3].getStdDev();
1747 mBufferDCA.vertexX_ITSTPC_Median.front() = avgVtxITSTPC[0].getMedian();
1748 mBufferDCA.vertexY_ITSTPC_Median.front() = avgVtxITSTPC[1].getMedian();
1749 mBufferDCA.vertexZ_ITSTPC_Median.front() = avgVtxITSTPC[2].getMedian();
1750 mBufferDCA.vertexX_ITSTPC_RMS.front() = avgVtxITSTPC[0].getStdDev();
1751 mBufferDCA.vertexY_ITSTPC_RMS.front() = avgVtxITSTPC[1].getStdDev();
1752 mBufferDCA.vertexZ_ITSTPC_RMS.front() = avgVtxITSTPC[2].getStdDev();
1755 RobustAverage avg(vertices.size(),
false);
1756 for (
const auto& vtx : vertices) {
1757 if (vtx.getNContributors() > mMinTracksPerVertex) {
1759 avg.addValue(std::sqrt(vtx.getNContributors()));
1764 int sizeQ = mBufferDCA.nVertexContributors_Quantiles.size();
1765 const int nBinsQ = 20;
1766 if (sizeQ >= (nBinsQ + 3)) {
1767 for (
int iq = 0; iq < nBinsQ; ++iq) {
1768 const float quantile = (iq + 1) /
static_cast<float>(nBinsQ);
1769 const float val = avg.getQuantile(quantile, 1);
1770 mBufferDCA.nVertexContributors_Quantiles[iq] =
val *
val;
1772 const float tr0 = avg.getTrunctedMean(0.05, 0.95);
1773 const float tr1 = avg.getTrunctedMean(0.1, 0.9);
1774 const float tr2 = avg.getTrunctedMean(0.2, 0.8);
1775 mBufferDCA.nVertexContributors_Quantiles[sizeQ - 3] = tr0 * tr0;
1776 mBufferDCA.nVertexContributors_Quantiles[sizeQ - 2] = tr1 * tr1;
1777 mBufferDCA.nVertexContributors_Quantiles[sizeQ - 1] = tr2 * tr2;
1779 mBufferDCA.nPrimVertices.front() = vertices.size();
1781 return indicesITSTPC_vtx;
1785 int getNBins()
const {
return mBufferDCA.mTSTPC.getNBins(); }
1787 ValsdEdx getdEdxVars(
bool useQMax,
const TrackTPC& track)
const
1789 const float dedx = useQMax ?
track.getdEdx().dEdxMaxTPC :
track.getdEdx().dEdxTotTPC;
1793 if (std::abs(dedxNorm) > mMaxdEdxRatio) {
1798 float dedxIROC = -1;
1799 float dedxOROC1 = -1;
1800 float dedxOROC2 = -1;
1801 float dedxOROC3 = -1;
1803 dedxIROC = useQMax ?
track.getdEdx().dEdxMaxIROC :
track.getdEdx().dEdxTotIROC;
1804 dedxOROC1 = useQMax ?
track.getdEdx().dEdxMaxOROC1 :
track.getdEdx().dEdxTotOROC1;
1805 dedxOROC2 = useQMax ?
track.getdEdx().dEdxMaxOROC2 :
track.getdEdx().dEdxTotOROC2;
1806 dedxOROC3 = useQMax ?
track.getdEdx().dEdxMaxOROC3 :
track.getdEdx().dEdxTotOROC3;
1811 dedxIROC = (dedxIROC > 0) ? std::log(dedxIROC) : -1;
1812 dedxOROC1 = (dedxOROC1 > 0) ? std::log(dedxOROC1) : -1;
1813 dedxOROC2 = (dedxOROC2 > 0) ? std::log(dedxOROC2) : -1;
1814 dedxOROC3 = (dedxOROC3 > 0) ? std::log(dedxOROC3) : -1;
1817 if (std::abs(dedxIROC) > mMaxdEdxRegionRatio) {
1820 if (std::abs(dedxOROC1) > mMaxdEdxRegionRatio) {
1823 if (std::abs(dedxOROC2) > mMaxdEdxRegionRatio) {
1826 if (std::abs(dedxOROC3) > mMaxdEdxRegionRatio) {
1830 return ValsdEdx{dedxNorm, dedxIROC, dedxOROC1, dedxOROC2, dedxOROC3};