23#include "TGraphAsymmErrors.h"
41#include "GPUChainTrackingGetters.inc"
49#include "GPUParam.inc"
58#ifndef GPUCA_STANDALONE
69#include "TParticlePDG.h"
70#include "TDatabasePDG.h"
80#include <oneapi/tbb.h>
98template <
bool COUNT,
class T>
105 r.qpt = fabsf(mTracking->mIOPtrs.mergedTracks[
r.id].GetParam().GetQPt());
106 r.lowPt =
r.qpt * mTracking->GetParam().qptB5Scaler > mTracking->GetParam().rec.tpc.rejectQPtB5;
107 r.mev200 =
r.qpt > 5;
108 r.mergedLooperUnconnected = mTracking->mIOPtrs.mergedTracks[
r.id].MergedLooperUnconnected();
109 r.mergedLooperConnected = mTracking->mIOPtrs.mergedTracks[
r.id].MergedLooperConnected();
112 if constexpr (COUNT) {
117 if constexpr (COUNT) {
120 }
else if (
r.mergedLooperUnconnected) {
121 if constexpr (COUNT) {
122 counts->nMergedLooperUnconnected++;
124 }
else if (
r.mergedLooperConnected) {
125 if constexpr (COUNT) {
126 counts->nMergedLooperConnected++;
129 r.protect = !GPUTPCClusterRejection::GetRejectionStatus<COUNT>(attach,
r.physics, counts, &
r.mev200) && ((attach &
gputpcgmmergertypes::attachProtect) || !GPUTPCClusterRejection::IsTrackRejected(mTracking->mIOPtrs.mergedTracks[
r.id], mTracking->GetParam()));
136 static GPUSettingsQA defaultConfig;
138 return *
chain->mConfigQA;
140 return defaultConfig;
144static const constexpr float LOG_PT_MIN = -1.;
146static constexpr float Y_MAX = 40;
147static constexpr float Z_MAX = 100;
148static constexpr float PT_MIN = 0.01;
149static constexpr float PT_MIN_PRIM = 0.1;
150static constexpr float PT_MIN_CLUST = 0.01;
151static constexpr float PT_MAX = 20;
152static constexpr float ETA_MAX = 1.5;
153static constexpr float ETA_MAX2 = 0.9;
154static constexpr int32_t PADROW_CHECK_MINCLS = 50;
156static constexpr bool CLUST_HIST_INT_SUM =
false;
158static constexpr const int32_t COLORCOUNT = 12;
160static const constexpr char* EFF_TYPES[6] = {
"Rec",
"Clone",
"Fake",
"All",
"RecAndClone",
"MC"};
161static const constexpr char* FINDABLE_NAMES[2] = {
"All",
"Findable"};
162static const constexpr char* PRIM_NAMES[2] = {
"Prim",
"Sec"};
163static const constexpr char* PARAMETER_NAMES[5] = {
"Y",
"Z",
"#Phi",
"#lambda",
"Relative #it{p}_{T}"};
164static const constexpr char* PARAMETER_NAMES_NATIVE[5] = {
"Y",
"Z",
"sin(#Phi)",
"tan(#lambda)",
"q/#it{p}_{T} (curvature)"};
165static const constexpr char* VSPARAMETER_NAMES[6] = {
"Y",
"Z",
"Phi",
"Eta",
"Pt",
"Pt_log"};
166static const constexpr char* EFF_NAMES[3] = {
"Efficiency",
"Clone Rate",
"Fake Rate"};
167static const constexpr char* EFFICIENCY_TITLES[4] = {
"Efficiency (Primary Tracks, Findable)",
"Efficiency (Secondary Tracks, Findable)",
"Efficiency (Primary Tracks)",
"Efficiency (Secondary Tracks)"};
168static const constexpr double SCALE[5] = {10., 10., 1000., 1000., 100.};
169static const constexpr double SCALE_NATIVE[5] = {10., 10., 1000., 1000., 1.};
170static const constexpr char* XAXIS_TITLES[5] = {
"#it{y}_{mc} (cm)",
"#it{z}_{mc} (cm)",
"#Phi_{mc} (rad)",
"#eta_{mc}",
"#it{p}_{Tmc} (GeV/#it{c})"};
171static const constexpr char* AXIS_TITLES[5] = {
"#it{y}-#it{y}_{mc} (mm) (Resolution)",
"#it{z}-#it{z}_{mc} (mm) (Resolution)",
"#phi-#phi_{mc} (mrad) (Resolution)",
"#lambda-#lambda_{mc} (mrad) (Resolution)",
"(#it{p}_{T} - #it{p}_{Tmc}) / #it{p}_{Tmc} (%) (Resolution)"};
172static const constexpr char* AXIS_TITLES_NATIVE[5] = {
"#it{y}-#it{y}_{mc} (mm) (Resolution)",
"#it{z}-#it{z}_{mc} (mm) (Resolution)",
"sin(#phi)-sin(#phi_{mc}) (Resolution)",
"tan(#lambda)-tan(#lambda_{mc}) (Resolution)",
"q*(q/#it{p}_{T} - q/#it{p}_{Tmc}) (Resolution)"};
173static const constexpr char* AXIS_TITLES_PULL[5] = {
"#it{y}-#it{y}_{mc}/#sigma_{y} (Pull)",
"#it{z}-#it{z}_{mc}/#sigma_{z} (Pull)",
"sin(#phi)-sin(#phi_{mc})/#sigma_{sin(#phi)} (Pull)",
"tan(#lambda)-tan(#lambda_{mc})/#sigma_{tan(#lambda)} (Pull)",
174 "q*(q/#it{p}_{T} - q/#it{p}_{Tmc})/#sigma_{q/#it{p}_{T}} (Pull)"};
175static const constexpr char* CLUSTER_NAMES[GPUQA::N_CLS_HIST] = {
"Correctly attached clusters",
"Fake attached clusters",
"Attached + adjacent clusters",
"Fake adjacent clusters",
"Clusters of reconstructed tracks",
"Used in Physics",
"Protected",
"All clusters"};
176static const constexpr char* CLUSTER_TITLES[GPUQA::N_CLS_TYPE] = {
"Clusters Pt Distribution / Attachment",
"Clusters Pt Distribution / Attachment (relative to all clusters)",
"Clusters Pt Distribution / Attachment (integrated)"};
177static const constexpr char* CLUSTER_NAMES_SHORT[GPUQA::N_CLS_HIST] = {
"Attached",
"Fake",
"AttachAdjacent",
"FakeAdjacent",
"FoundTracks",
"Physics",
"Protected",
"All"};
178static const constexpr char* CLUSTER_TYPES[GPUQA::N_CLS_TYPE] = {
"",
"Ratio",
"Integral"};
179static const constexpr char* REJECTED_NAMES[3] = {
"All",
"Rejected",
"Fraction"};
180static const constexpr int32_t COLORS_HEX[COLORCOUNT] = {0xB03030, 0x00A000, 0x0000C0, 0x9400D3, 0x19BBBF, 0xF25900, 0x7F7F7F, 0xFFD700, 0x07F707, 0x07F7F7, 0xF08080, 0x000000};
182static const constexpr int32_t CONFIG_DASHED_MARKERS = 0;
184static const constexpr float AXES_MIN[5] = {-Y_MAX, -Z_MAX, 0.f, -ETA_MAX, PT_MIN};
185static const constexpr float AXES_MAX[5] = {Y_MAX, Z_MAX, 2.f * M_PI, ETA_MAX, PT_MAX};
186static const constexpr int32_t AXIS_BINS[5] = {51, 51, 144, 31, 50};
187static const constexpr int32_t RES_AXIS_BINS[] = {1017, 113};
188static const constexpr float RES_AXES[5] = {1., 1., 0.03, 0.03, 1.0};
189static const constexpr float RES_AXES_NATIVE[5] = {1., 1., 0.1, 0.1, 5.0};
190static const constexpr float PULL_AXIS = 10.f;
192std::vector<TColor*> GPUQA::mColors;
193int32_t GPUQA::initColors()
195 mColors.reserve(COLORCOUNT);
196 for (int32_t
i = 0;
i < COLORCOUNT;
i++) {
197 float f1 = (float)((COLORS_HEX[
i] >> 16) & 0xFF) / (
float)0xFF;
198 float f2 = (float)((COLORS_HEX[
i] >> 8) & 0xFF) / (
float)0xFF;
199 float f3 = (float)((COLORS_HEX[
i] >> 0) & 0xFF) / (
float)0xFF;
200 mColors.emplace_back(
new TColor(10000 +
i, f1, f2, f3));
204static constexpr Color_t defaultColorNums[COLORCOUNT] = {kRed, kBlue, kGreen, kMagenta, kOrange, kAzure, kBlack, kYellow, kGray, kTeal, kSpring, kPink};
206#define TRACK_EXPECTED_REFERENCE_X_DEFAULT 81
208static inline int32_t GPUQA_O2_ConvertFakeLabel(int32_t
label) {
return label >= 0x7FFFFFFE ? -1 :
label; }
209inline uint32_t GPUQA::GetNMCCollissions()
const {
return mMCInfosCol.size(); }
210inline uint32_t GPUQA::GetNMCTracks(int32_t iCol)
const {
return mMCInfosCol[iCol].num; }
211inline uint32_t GPUQA::GetNMCTracks(
const mcLabelI_t&
label)
const {
return mMCInfosCol[mMCEventOffset[
label.getSourceID()] +
label.getEventID()].num; }
212inline uint32_t GPUQA::GetNMCLabels()
const {
return mClNative->clustersMCTruth ? mClNative->clustersMCTruth->getIndexedSize() : 0; }
213inline const GPUQA::mcInfo_t& GPUQA::GetMCTrack(uint32_t iTrk, uint32_t iCol) {
return mMCInfos[mMCInfosCol[iCol].first + iTrk]; }
214inline const GPUQA::mcInfo_t& GPUQA::GetMCTrack(
const mcLabel_t&
label) {
return mMCInfos[mMCInfosCol[mMCEventOffset[
label.getSourceID()] +
label.getEventID()].first +
label.getTrackID()]; }
215inline GPUQA::mcLabels_t GPUQA::GetMCLabel(uint32_t
i) {
return mClNative->clustersMCTruth->getLabels(
i); }
216inline int32_t GPUQA::GetMCLabelNID(
const mcLabels_t&
label) {
return label.size(); }
217inline int32_t GPUQA::GetMCLabelNID(uint32_t
i) {
return mClNative->clustersMCTruth->getLabels(
i).size(); }
218inline GPUQA::mcLabel_t GPUQA::GetMCLabel(uint32_t
i, uint32_t
j) {
return mClNative->clustersMCTruth->getLabels(
i)[
j]; }
219inline int32_t GPUQA::GetMCLabelID(uint32_t
i, uint32_t
j) {
return GPUQA_O2_ConvertFakeLabel(mClNative->clustersMCTruth->getLabels(
i)[
j].getTrackID()); }
220inline int32_t GPUQA::GetMCLabelID(
const mcLabels_t&
label, uint32_t
j) {
return GPUQA_O2_ConvertFakeLabel(
label[
j].getTrackID()); }
221inline int32_t GPUQA::GetMCLabelID(
const mcLabel_t&
label) {
return GPUQA_O2_ConvertFakeLabel(
label.getTrackID()); }
222inline uint32_t GPUQA::GetMCLabelCol(uint32_t
i, uint32_t
j) {
return mMCEventOffset[mClNative->clustersMCTruth->getLabels(
i)[
j].getSourceID()] + mClNative->clustersMCTruth->getLabels(
i)[
j].getEventID(); }
223inline const auto& GPUQA::GetClusterLabels() {
return mClNative->clustersMCTruth; }
224inline float GPUQA::GetMCLabelWeight(uint32_t
i, uint32_t
j) {
return 1; }
225inline float GPUQA::GetMCLabelWeight(
const mcLabels_t&
label, uint32_t
j) {
return 1; }
226inline float GPUQA::GetMCLabelWeight(
const mcLabel_t&
label) {
return 1; }
227inline bool GPUQA::mcPresent() {
return !mConfig.noMC && mTracking && mClNative && mClNative->clustersMCTruth && mMCInfos.size(); }
228uint32_t GPUQA::GetMCLabelCol(
const mcLabel_t&
label)
const {
return !
label.isValid() ? 0 : (mMCEventOffset[
label.getSourceID()] +
label.getEventID()); }
230bool GPUQA::CompareIgnoreFake(
const mcLabelI_t& l1,
const mcLabelI_t& l2) {
return l1.compare(l2) >= 0; }
231#define TRACK_EXPECTED_REFERENCE_X 78
233inline GPUQA::mcLabelI_t::mcLabelI_t(
const GPUQA::mcLabel_t& l) : track(l.fMCID) {}
234inline bool GPUQA::mcLabelI_t::operator==(
const GPUQA::mcLabel_t& l) {
return AbsLabelID(track) == l.fMCID; }
235inline uint32_t GPUQA::GetNMCCollissions()
const {
return 1; }
236inline uint32_t GPUQA::GetNMCTracks(int32_t iCol)
const {
return mTracking->mIOPtrs.nMCInfosTPC; }
237inline uint32_t GPUQA::GetNMCTracks(
const mcLabelI_t&
label)
const {
return mTracking->mIOPtrs.nMCInfosTPC; }
238inline uint32_t GPUQA::GetNMCLabels()
const {
return mTracking->mIOPtrs.nMCLabelsTPC; }
239inline const GPUQA::mcInfo_t& GPUQA::GetMCTrack(uint32_t iTrk, uint32_t iCol) {
return mTracking->mIOPtrs.mcInfosTPC[AbsLabelID(iTrk)]; }
240inline const GPUQA::mcInfo_t& GPUQA::GetMCTrack(
const mcLabel_t&
label) {
return GetMCTrack(
label.fMCID, 0); }
241inline const GPUQA::mcInfo_t& GPUQA::GetMCTrack(
const mcLabelI_t&
label) {
return GetMCTrack(
label.track, 0); }
242inline const GPUQA::mcLabels_t& GPUQA::GetMCLabel(uint32_t
i) {
return mTracking->mIOPtrs.mcLabelsTPC[
i]; }
243inline const GPUQA::mcLabel_t& GPUQA::GetMCLabel(uint32_t
i, uint32_t
j) {
return mTracking->mIOPtrs.mcLabelsTPC[
i].fClusterID[
j]; }
244inline int32_t GPUQA::GetMCLabelNID(
const mcLabels_t&
label) {
return 3; }
245inline int32_t GPUQA::GetMCLabelNID(uint32_t
i) {
return 3; }
246inline int32_t GPUQA::GetMCLabelID(uint32_t
i, uint32_t
j) {
return mTracking->mIOPtrs.mcLabelsTPC[
i].fClusterID[
j].fMCID; }
247inline int32_t GPUQA::GetMCLabelID(
const mcLabels_t&
label, uint32_t
j) {
return label.fClusterID[
j].fMCID; }
248inline int32_t GPUQA::GetMCLabelID(
const mcLabel_t&
label) {
return label.fMCID; }
249inline uint32_t GPUQA::GetMCLabelCol(uint32_t
i, uint32_t
j) {
return 0; }
250inline const auto& GPUQA::GetClusterLabels() {
return mTracking->mIOPtrs.mcLabelsTPC; }
251inline float GPUQA::GetMCLabelWeight(uint32_t
i, uint32_t
j) {
return mTracking->mIOPtrs.mcLabelsTPC[
i].fClusterID[
j].fWeight; }
252inline float GPUQA::GetMCLabelWeight(
const mcLabels_t&
label, uint32_t
j) {
return label.fClusterID[
j].fWeight; }
253inline float GPUQA::GetMCLabelWeight(
const mcLabel_t&
label) {
return label.fWeight; }
254inline int32_t GPUQA::FakeLabelID(int32_t
id) {
return id < 0 ?
id : (-2 -
id); }
255inline int32_t GPUQA::AbsLabelID(int32_t
id) {
return id >= 0 ?
id : (-
id - 2); }
256inline bool GPUQA::mcPresent() {
return !mConfig.noMC && mTracking && GetNMCLabels() && GetNMCTracks(0); }
257uint32_t GPUQA::GetMCLabelCol(
const mcLabel_t&
label)
const {
return 0; }
259bool GPUQA::CompareIgnoreFake(
const mcLabelI_t& l1,
const mcLabelI_t& l2) {
return AbsLabelID(l1.track) == AbsLabelID(l2.track); }
260#define TRACK_EXPECTED_REFERENCE_X TRACK_EXPECTED_REFERENCE_X_DEFAULT
265 return obj[mMCEventOffset[l.getSourceID()] + l.getEventID()][l.getTrackID()];
269auto GPUQA::getHistArray<TH1F>()
271 return std::make_pair(mHist1D, &mHist1D_pos);
274auto GPUQA::getHistArray<TH2F>()
276 return std::make_pair(mHist2D, &mHist2D_pos);
279auto GPUQA::getHistArray<TH1D>()
281 return std::make_pair(mHist1Dd, &mHist1Dd_pos);
284auto GPUQA::getHistArray<TGraphAsymmErrors>()
286 return std::make_pair(mHistGraph, &mHistGraph_pos);
288template <
class T,
typename... Args>
289void GPUQA::createHist(T*&
h,
const char*
name, Args... args)
291 const auto&
p = getHistArray<T>();
292 if (mHaveExternalHists) {
293 if (
p.first->size() <=
p.second->size()) {
294 GPUError(
"Array sizes mismatch: Histograms %lu <= Positions %lu",
p.first->size(),
p.second->size());
295 throw std::runtime_error(
"Incoming histogram array incomplete");
297 if (strcmp((*
p.first)[
p.second->size()].GetName(),
name)) {
298 GPUError(
"Histogram name mismatch: in array %s, trying to create %s", (*
p.first)[
p.second->size()].GetName(),
name);
299 throw std::runtime_error(
"Incoming histogram has incorrect name");
302 if constexpr (std::is_same_v<T, TGraphAsymmErrors>) {
303 p.first->emplace_back();
304 p.first->back().SetName(
name);
306 p.first->emplace_back(
name, args...);
309 h = &((*
p.first)[
p.second->size()]);
310 p.second->emplace_back(&
h);
316 std::tuple<std::vector<std::unique_ptr<TCanvas>>, std::vector<std::unique_ptr<TLegend>>, std::vector<std::unique_ptr<TPad>>, std::vector<std::unique_ptr<TLatex>>, std::vector<std::unique_ptr<TH1D>>>
v;
320template <
class T,
typename... Args>
321T* GPUQA::createGarbageCollected(Args... args)
323 auto&
v = std::get<std::vector<std::unique_ptr<T>>>(mGarbageCollector->v);
324 v.emplace_back(std::make_unique<T>(args...));
325 return v.back().get();
327void GPUQA::clearGarbagageCollector()
329 std::get<std::vector<std::unique_ptr<TPad>>>(mGarbageCollector->v).
clear();
330 std::apply([](
auto&&... args) { ((args.clear()), ...); }, mGarbageCollector->v);
335 mMCEventOffset.resize(1, 0);
340 if (mQAInitialized && !mHaveExternalHists) {
346 clearGarbagageCollector();
351 const auto&
r = checkClusterState<false>(attach);
353 return r.protect ||
r.physics;
355 return (!
r.unattached && !
r.physics && !
r.protect);
359void GPUQA::SetAxisSize(T* e)
361 e->GetYaxis()->SetTitleOffset(1.0);
362 e->GetYaxis()->SetTitleSize(0.045);
363 e->GetYaxis()->SetLabelSize(0.045);
364 e->GetXaxis()->SetTitleOffset(1.03);
365 e->GetXaxis()->SetTitleSize(0.045);
366 e->GetXaxis()->SetLabelOffset(-0.005);
367 e->GetXaxis()->SetLabelSize(0.045);
370void GPUQA::SetLegend(TLegend* l,
bool bigText)
373 l->SetTextSize(bigText ? 0.03 : 0.016);
377double* GPUQA::CreateLogAxis(int32_t nbins,
float xmin,
float xmax)
379 float logxmin = std::log10(xmin);
380 float logxmax = std::log10(xmax);
381 float binwidth = (logxmax - logxmin) / nbins;
383 double* xbins =
new double[nbins + 1];
386 for (int32_t
i = 1;
i <= nbins;
i++) {
387 xbins[
i] = std::pow(10, logxmin +
i * binwidth);
392void GPUQA::ChangePadTitleSize(TPad* p,
float size)
395 TPaveText* pt = (TPaveText*)(
p->GetPrimitive(
"title"));
397 GPUError(
"Error changing title");
399 pt->SetTextSize(
size);
404void GPUQA::DrawHisto(TH1* histo,
char*
filename,
char* options)
408 histo->Draw(options);
412void GPUQA::doPerfFigure(
float x,
float y,
float size)
414 if (mConfig.perfFigure ==
"") {
417 static constexpr const char* str_perf_figure_1 =
"ALICE Performance";
418 static constexpr const char* str_perf_figure_2_mc =
"MC, Pb#minusPb, #sqrt{s_{NN}} = 5.36 TeV";
419 static constexpr const char* str_perf_figure_2_data =
"Pb#minusPb, #sqrt{s_{NN}} = 5.36 TeV";
420 const char* str_perf_figure_2 = (mConfig.perfFigure ==
"mc" || mConfig.perfFigure ==
"MC") ? str_perf_figure_2_mc : (mConfig.perfFigure ==
"data" ? str_perf_figure_2_data : mConfig.perfFigure.c_str());
422 TLatex* t = createGarbageCollected<TLatex>();
425 t->SetTextSize(
size);
426 t->DrawLatex(
x,
y, str_perf_figure_1);
427 t->SetTextSize(
size * 0.8);
428 t->DrawLatex(
x,
y - 0.01 -
size, str_perf_figure_2);
437int32_t GPUQA::InitQACreateHistograms()
439 char name[2048], fname[1024];
440 if (mQATasks & taskTrackingEff) {
442 for (int32_t
i = 0;
i < 6;
i++) {
443 for (int32_t
j = 0;
j < 2;
j++) {
444 for (int32_t k = 0; k < 2; k++) {
445 for (int32_t l = 0; l < 5; l++) {
446 snprintf(
name, 2048,
"%s%s%s%sVs%s",
"tracks", EFF_TYPES[
i], FINDABLE_NAMES[
j], PRIM_NAMES[k], VSPARAMETER_NAMES[l]);
448 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], k == 0 ? PT_MIN_PRIM : AXES_MIN[4], AXES_MAX[4])};
449 createHist(mEff[
i][
j][k][l],
name,
name, AXIS_BINS[l], binsPt.get());
451 createHist(mEff[
i][
j][k][l],
name,
name, AXIS_BINS[l], AXES_MIN[l], AXES_MAX[l]);
453 if (!mHaveExternalHists) {
454 mEff[
i][
j][k][l]->Sumw2();
456 strcat(
name,
"_eff");
458 createHist(mEffResult[
i][
j][k][l],
name);
467 if (mQATasks & taskTrackingRes) {
468 for (int32_t
i = 0;
i < 5;
i++) {
469 for (int32_t
j = 0;
j < 5;
j++) {
470 snprintf(
name, 2048,
"rms_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
471 snprintf(fname, 1024,
"mean_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
473 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], mConfig.resPrimaries == 1 ? PT_MIN_PRIM : AXES_MIN[4], AXES_MAX[4])};
474 createHist(mRes[
i][
j][0],
name,
name, AXIS_BINS[
j], binsPt.get());
475 createHist(mRes[
i][
j][1], fname, fname, AXIS_BINS[
j], binsPt.get());
477 createHist(mRes[
i][
j][0],
name,
name, AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
478 createHist(mRes[
i][
j][1], fname, fname, AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
480 snprintf(
name, 2048,
"res_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
481 const float* axis = mConfig.nativeFitResolutions ? RES_AXES_NATIVE : RES_AXES;
482 const int32_t nbins =
i == 4 && mConfig.nativeFitResolutions ? (10 * RES_AXIS_BINS[0]) : RES_AXIS_BINS[0];
484 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], mConfig.resPrimaries == 1 ? PT_MIN_PRIM : AXES_MIN[4], AXES_MAX[4])};
485 createHist(mRes2[
i][
j],
name,
name, nbins, -axis[
i], axis[
i], AXIS_BINS[
j], binsPt.get());
487 createHist(mRes2[
i][
j],
name,
name, nbins, -axis[
i], axis[
i], AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
494 if (mQATasks & taskTrackingResPull) {
495 for (int32_t
i = 0;
i < 5;
i++) {
496 for (int32_t
j = 0;
j < 5;
j++) {
497 snprintf(
name, 2048,
"pull_rms_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
498 snprintf(fname, 1024,
"pull_mean_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
500 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], AXES_MIN[4], AXES_MAX[4])};
501 createHist(mPull[
i][
j][0],
name,
name, AXIS_BINS[
j], binsPt.get());
502 createHist(mPull[
i][
j][1], fname, fname, AXIS_BINS[
j], binsPt.get());
504 createHist(mPull[
i][
j][0],
name,
name, AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
505 createHist(mPull[
i][
j][1], fname, fname, AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
507 snprintf(
name, 2048,
"pull_%s_vs_%s", VSPARAMETER_NAMES[
i], VSPARAMETER_NAMES[
j]);
509 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], AXES_MIN[4], AXES_MAX[4])};
510 createHist(mPull2[
i][
j],
name,
name, RES_AXIS_BINS[0], -PULL_AXIS, PULL_AXIS, AXIS_BINS[
j], binsPt.get());
512 createHist(mPull2[
i][
j],
name,
name, RES_AXIS_BINS[0], -PULL_AXIS, PULL_AXIS, AXIS_BINS[
j], AXES_MIN[
j], AXES_MAX[
j]);
519 if (mQATasks & taskClusterAttach) {
520 for (int32_t
i = 0;
i < N_CLS_TYPE * N_CLS_HIST - 1;
i++) {
521 int32_t ioffset =
i >= (2 * N_CLS_HIST - 1) ? (2 * N_CLS_HIST - 1) :
i >= N_CLS_HIST ? N_CLS_HIST : 0;
522 int32_t itype =
i >= (2 * N_CLS_HIST - 1) ? 2 :
i >= N_CLS_HIST ? 1 : 0;
523 snprintf(
name, 2048,
"clusters%s%s", CLUSTER_NAMES_SHORT[
i - ioffset], CLUSTER_TYPES[itype]);
524 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], PT_MIN_CLUST, PT_MAX)};
525 createHist(mClusters[
i],
name,
name, AXIS_BINS[4], binsPt.get());
534 if (mQATasks & taskTrackStatistics) {
536 for (int32_t
i = 0;
i < 2;
i++) {
537 snprintf(
name, 2048,
i ?
"nrows_with_cluster" :
"nclusters");
540 std::unique_ptr<double[]> binsPt{CreateLogAxis(AXIS_BINS[4], PT_MIN_CLUST, PT_MAX)};
541 createHist(mTrackPt,
"tracks_pt",
"tracks_pt", AXIS_BINS[4], binsPt.get());
542 const uint32_t maxTime = (mTracking && mTracking->GetParam().continuousMaxTimeBin > 0) ? mTracking->GetParam().continuousMaxTimeBin : constants::TPC_MAX_TIME_BIN_TRIGGERED;
543 createHist(mT0[0],
"tracks_t0",
"tracks_t0", (maxTime + 1) / 10, 0, maxTime);
544 createHist(mT0[1],
"tracks_t0_res",
"tracks_t0_res", 1000, -100, 100);
545 createHist(mClXY,
"clXY",
"clXY", 1000, -250, 250, 1000, -250, 250);
547 if (mQATasks & taskClusterRejection) {
549 for (int32_t
i = 0;
i < 3;
i++) {
550 snprintf(
name, 2048,
"clrej_%d",
i);
556 if ((mQATasks & taskClusterCounts) && mConfig.clusterRejectionHistograms) {
557 int32_t
num = DoClusterCounts(
nullptr, 2);
558 mHistClusterCount.resize(
num);
559 DoClusterCounts(
nullptr, 1);
562 for (uint32_t
i = 0;
i < mHist1D->size();
i++) {
563 *mHist1D_pos[
i] = &(*mHist1D)[
i];
565 for (uint32_t
i = 0;
i < mHist2D->size();
i++) {
566 *mHist2D_pos[
i] = &(*mHist2D)[
i];
568 for (uint32_t
i = 0;
i < mHist1Dd->size();
i++) {
569 *mHist1Dd_pos[
i] = &(*mHist1Dd)[
i];
571 for (uint32_t
i = 0;
i < mHistGraph->size();
i++) {
572 *mHistGraph_pos[
i] = &(*mHistGraph)[
i];
578int32_t GPUQA::loadHistograms(std::vector<TH1F>& i1, std::vector<TH2F>& i2, std::vector<TH1D>& i3, std::vector<TGraphAsymmErrors>& i4, int32_t tasks)
580 if (tasks == tasksAutomatic) {
583 if (mQAInitialized && (!mHaveExternalHists || tasks != mQATasks)) {
584 throw std::runtime_error(
"QA not initialized or initialized with different task array");
592 mHist1Dd_pos.clear();
593 mHistGraph_pos.clear();
594 mHaveExternalHists =
true;
599 if (InitQACreateHistograms()) {
602 mQAInitialized =
true;
612 uint32_t
n = mMCInfos.size();
613 fwrite(&
n,
sizeof(
n), 1, fp);
614 fwrite(mMCInfos.data(),
sizeof(mMCInfos[0]),
n, fp);
615 n = mMCInfosCol.size();
616 fwrite(&
n,
sizeof(
n), 1, fp);
617 fwrite(mMCInfosCol.data(),
sizeof(mMCInfosCol[0]),
n, fp);
618 n = mMCEventOffset.size();
619 fwrite(&
n,
sizeof(
n), 1, fp);
620 fwrite(mMCEventOffset.data(),
sizeof(mMCEventOffset[0]),
n, fp);
632 if ((
x = fread(&
n,
sizeof(
n), 1, fp)) != 1) {
637 if (fread(mMCInfos.data(),
sizeof(mMCInfos[0]),
n, fp) !=
n) {
641 if ((
x = fread(&
n,
sizeof(
n), 1, fp)) != 1) {
645 mMCInfosCol.resize(
n);
646 if (fread(mMCInfosCol.data(),
sizeof(mMCInfosCol[0]),
n, fp) !=
n) {
650 if ((
x = fread(&
n,
sizeof(
n), 1, fp)) != 1) {
654 mMCEventOffset.resize(
n);
655 if (fread(mMCEventOffset.data(),
sizeof(mMCEventOffset[0]),
n, fp) !=
n) {
659 if (mTracking && mTracking->GetProcessingSettings().debugLevel >= 2) {
660 printf(
"Read %ld bytes MC Infos\n", ftell(fp));
664 CopyO2MCtoIOPtr(&mTracking->mIOPtrs);
671 ptr->mcInfosTPC = mMCInfos.data();
672 ptr->nMCInfosTPC = mMCInfos.size();
673 ptr->mcInfosTPCCol = mMCInfosCol.data();
674 ptr->nMCInfosTPCCol = mMCInfosCol.size();
679#ifndef GPUCA_STANDALONE
680 if (!mO2MCDataLoaded) {
681 HighResTimer timer(mTracking && mTracking->GetProcessingSettings().debugLevel);
682 if (mTracking && mTracking->GetProcessingSettings().debugLevel) {
683 GPUInfo(
"Start reading O2 Track MC information");
685 static constexpr float PRIM_MAX_T = 0.01f;
688 std::vector<int32_t> refId;
691 const auto& evrec = dc->getEventRecords();
692 const auto& evparts = dc->getEventParts();
693 std::vector<std::vector<float>> evTimeBins(mcReader.getNSources());
694 for (uint32_t
i = 0;
i < evTimeBins.size();
i++) {
695 evTimeBins[
i].resize(mcReader.getNEvents(
i), -100.f);
697 for (uint32_t
i = 0;
i < evrec.size();
i++) {
698 const auto&
ir = evrec[
i];
699 for (uint32_t
j = 0;
j < evparts[
i].size();
j++) {
700 const int iSim = evparts[
i][
j].sourceID;
701 const int iEv = evparts[
i][
j].entryID;
705 if (evTimeBins[iSim][iEv] >= 0) {
706 throw std::runtime_error(
"Multiple time bins for same MC collision found");
708 evTimeBins[iSim][iEv] = timebin;
713 uint32_t nSimSources = mcReader.getNSources();
714 mMCEventOffset.resize(nSimSources);
715 uint32_t nSimTotalEvents = 0;
716 uint32_t nSimTotalTracks = 0;
717 for (uint32_t
i = 0;
i < nSimSources;
i++) {
718 mMCEventOffset[
i] = nSimTotalEvents;
719 nSimTotalEvents += mcReader.getNEvents(
i);
722 mMCInfosCol.resize(nSimTotalEvents);
723 for (int32_t iSim = 0; iSim < mcReader.getNSources(); iSim++) {
724 for (int32_t
i = 0;
i < mcReader.getNEvents(iSim);
i++) {
725 const float timebin = evTimeBins[iSim][
i];
727 const std::vector<o2::MCTrack>&
tracks = mcReader.getTracks(iSim,
i);
728 const std::vector<o2::TrackReference>& trackRefs = mcReader.getTrackRefsByEvent(iSim,
i);
730 refId.resize(
tracks.size());
731 std::fill(refId.begin(), refId.end(), -1);
732 for (uint32_t
j = 0;
j < trackRefs.size();
j++) {
734 int32_t trkId = trackRefs[
j].getTrackID();
735 if (refId[trkId] == -1) {
740 mMCInfosCol[mMCEventOffset[iSim] +
i].first = mMCInfos.size();
741 mMCInfosCol[mMCEventOffset[iSim] +
i].num =
tracks.size();
742 mMCInfos.resize(mMCInfos.size() +
tracks.size());
743 for (uint32_t
j = 0;
j <
tracks.size();
j++) {
744 auto&
info = mMCInfos[mMCInfosCol[mMCEventOffset[iSim] +
i].first +
j];
746 TParticlePDG* particle = TDatabasePDG::Instance()->GetParticle(trk.GetPdgCode());
748 if (abs(trk.GetPdgCode()) == kElectron) {
751 if (abs(trk.GetPdgCode()) == kMuonMinus) {
754 if (abs(trk.GetPdgCode()) == kPiPlus) {
757 if (abs(trk.GetPdgCode()) == kKPlus) {
760 if (abs(trk.GetPdgCode()) == kProton) {
764 info.charge = particle ? particle->Charge() : 0;
765 info.prim = trk.T() < PRIM_MAX_T;
766 info.primDaughters = 0;
767 if (trk.getFirstDaughterTrackId() != -1) {
768 for (int32_t k = trk.getFirstDaughterTrackId(); k <= trk.getLastDaughterTrackId(); k++) {
769 if (tracks[k].
T() < PRIM_MAX_T) {
770 info.primDaughters = 1;
778 const auto& trkRef = trackRefs[refId[
j]];
782 info.pX = trkRef.Px();
783 info.pY = trkRef.Py();
784 info.pZ = trkRef.Pz();
785 info.genRadius = std::sqrt(trk.GetStartVertexCoordinatesX() * trk.GetStartVertexCoordinatesX() + trk.GetStartVertexCoordinatesY() * trk.GetStartVertexCoordinatesY() + trk.GetStartVertexCoordinatesZ() * trk.GetStartVertexCoordinatesZ());
793 if (timer.IsRunning()) {
794 GPUInfo(
"Finished reading O2 Track MC information (%f seconds)", timer.GetCurrentElapsedTime());
796 mO2MCDataLoaded =
true;
799 CopyO2MCtoIOPtr(updateIOPtr);
806 if (mQAInitialized) {
807 throw std::runtime_error(
"QA already initialized");
809 if (tasks == tasksAutomatic) {
813 mHist1D =
new std::vector<TH1F>;
814 mHist2D =
new std::vector<TH2F>;
815 mHist1Dd =
new std::vector<TH1D>;
816 mHistGraph =
new std::vector<TGraphAsymmErrors>;
822 if (mTracking->GetProcessingSettings().qcRunFraction != 100.f && mQATasks != taskClusterCounts) {
823 throw std::runtime_error(
"QA with qcRunFraction only supported for taskClusterCounts");
827 mClNative = mTracking->mIOPtrs.clustersNative;
830 if (InitQACreateHistograms()) {
834 if (mConfig.enableLocalOutput) {
835 mkdir(mConfig.plotsDir.c_str(), S_IRWXU | S_IRWXG | S_IROTH | S_IXOTH);
838#ifndef GPUCA_STANDALONE
840 InitO2MCData(mTracking ? &mTracking->mIOPtrs : nullptr);
844 if (mConfig.matchMCLabels.size()) {
845 uint32_t nFiles = mConfig.matchMCLabels.size();
846 std::vector<std::unique_ptr<TFile>> files;
847 std::vector<std::vector<std::vector<int32_t>>*> labelsBuffer(nFiles);
848 std::vector<std::vector<std::vector<int32_t>>*> effBuffer(nFiles);
849 for (uint32_t
i = 0;
i < nFiles;
i++) {
850 files.emplace_back(std::make_unique<TFile>(mConfig.matchMCLabels[
i].c_str()));
851 labelsBuffer[
i] = (std::vector<std::vector<int32_t>>*)files[
i]->Get(
"mcLabelBuffer");
852 effBuffer[
i] = (std::vector<std::vector<int32_t>>*)files[
i]->Get(
"mcEffBuffer");
853 if (labelsBuffer[
i] ==
nullptr || effBuffer[
i] ==
nullptr) {
854 GPUError(
"Error opening / reading from labels file %u/%s: %p %p",
i, mConfig.matchMCLabels[
i].c_str(), (
void*)labelsBuffer[
i], (
void*)effBuffer[
i]);
859 mGoodTracks.resize(labelsBuffer[0]->
size());
860 mGoodHits.resize(labelsBuffer[0]->
size());
861 for (uint32_t iEvent = 0; iEvent < labelsBuffer[0]->size(); iEvent++) {
862 std::vector<bool> labelsOK((*effBuffer[0])[iEvent].
size());
863 for (uint32_t k = 0; k < (*effBuffer[0])[iEvent].
size(); k++) {
865 for (uint32_t l = 0; l < nFiles; l++) {
866 if ((*effBuffer[0])[iEvent][k] != (*effBuffer[l])[iEvent][k]) {
872 mGoodTracks[iEvent].resize((*labelsBuffer[0])[iEvent].size());
873 for (uint32_t k = 0; k < (*labelsBuffer[0])[iEvent].
size(); k++) {
874 if ((*labelsBuffer[0])[iEvent][k] == MC_LABEL_INVALID) {
877 mGoodTracks[iEvent][k] = labelsOK[abs((*labelsBuffer[0])[iEvent][k])];
881 mQAInitialized =
true;
887 if (!mQAInitialized) {
888 throw std::runtime_error(
"QA not initialized");
890 if (mTracking && mTracking->GetProcessingSettings().debugLevel >= 2) {
891 GPUInfo(
"Running QA - Mask %d, Efficiency %d, Resolution %d, Pulls %d, Cluster Attachment %d, Track Statistics %d, Cluster Counts %d", mQATasks, (int32_t)(mQATasks & taskTrackingEff), (int32_t)(mQATasks & taskTrackingRes), (int32_t)(mQATasks & taskTrackingResPull), (int32_t)(mQATasks & taskClusterAttach), (int32_t)(mQATasks & taskTrackStatistics), (int32_t)(mQATasks & taskClusterCounts));
893 if (!clNative && mTracking) {
894 clNative = mTracking->mIOPtrs.clustersNative;
896 mClNative = clNative;
899 uint32_t nSimEvents = GetNMCCollissions();
900 if (mTrackMCLabelsReverse.size() < nSimEvents) {
901 mTrackMCLabelsReverse.resize(nSimEvents);
903 if (mRecTracks.size() < nSimEvents) {
904 mRecTracks.resize(nSimEvents);
906 if (mFakeTracks.size() < nSimEvents) {
907 mFakeTracks.resize(nSimEvents);
909 if (mMCParam.size() < nSimEvents) {
910 mMCParam.resize(nSimEvents);
915 uint32_t nReconstructedTracks = 0;
916 if (tracksExternal) {
917#ifndef GPUCA_STANDALONE
918 nReconstructedTracks = tracksExternal->size();
921 nReconstructedTracks = mTracking->mIOPtrs.nMergedTracks;
923 mTrackMCLabels.resize(nReconstructedTracks);
924 for (uint32_t iCol = 0; iCol < GetNMCCollissions(); iCol++) {
925 mTrackMCLabelsReverse[iCol].resize(GetNMCTracks(iCol));
926 mRecTracks[iCol].resize(GetNMCTracks(iCol));
927 mFakeTracks[iCol].resize(GetNMCTracks(iCol));
928 mMCParam[iCol].resize(GetNMCTracks(iCol));
929 memset(mRecTracks[iCol].
data(), 0, mRecTracks[iCol].
size() *
sizeof(mRecTracks[iCol][0]));
930 memset(mFakeTracks[iCol].
data(), 0, mFakeTracks[iCol].
size() *
sizeof(mFakeTracks[iCol][0]));
931 for (
size_t i = 0;
i < mTrackMCLabelsReverse[iCol].size();
i++) {
932 mTrackMCLabelsReverse[iCol][
i] = -1;
935 if (mQATasks & taskClusterAttach && GetNMCLabels()) {
936 mClusterParam.resize(GetNMCLabels());
937 memset(mClusterParam.data(), 0, mClusterParam.size() *
sizeof(mClusterParam[0]));
942 if (mConfig.writeMCLabels) {
943 mcEffBuffer.resize(mNEvents);
944 mcLabelBuffer.resize(mNEvents);
945 mcEffBuffer[mNEvents - 1].resize(GetNMCTracks(0));
946 mcLabelBuffer[mNEvents - 1].resize(nReconstructedTracks);
949 bool mcAvail = mcPresent() || tracksExtMC;
952 if (tracksExternal) {
953#ifndef GPUCA_STANDALONE
954 for (uint32_t
i = 0;
i < tracksExternal->size();
i++) {
955 mTrackMCLabels[
i] = (*tracksExtMC)[
i];
959 tbb::parallel_for(tbb::blocked_range<uint32_t>(0, nReconstructedTracks, (
QA_DEBUG == 0) ? 32 : nReconstructedTracks), [&](const tbb::blocked_range<uint32_t>&
range) {
960 auto acc = GPUTPCTrkLbl<true, mcLabelI_t>(GetClusterLabels(), 1.f - mConfig.recThreshold);
965 std::vector<mcLabel_t>
labels;
966 for (uint32_t k = 0; k < track.NClusters(); k++) {
971 uint32_t hitId = mTracking->mIOPtrs.mergedTrackHits[track.FirstClusterRef() + k].num;
972 if (hitId >= GetNMCLabels()) {
973 GPUError(
"Invalid hit id %u > %d (nClusters %d)", hitId, GetNMCLabels(), clNative ? clNative->
nClustersTotal : 0);
974 throw std::runtime_error(
"qa error");
977 for (int32_t
j = 0;
j < GetMCLabelNID(hitId);
j++) {
978 if (GetMCLabelID(hitId,
j) >= (int32_t)GetNMCTracks(GetMCLabelCol(hitId,
j))) {
979 GPUError(
"Invalid label %d > %d (hit %d, label %d, col %d)", GetMCLabelID(hitId,
j), GetNMCTracks(GetMCLabelCol(hitId,
j)), hitId,
j, (int32_t)GetMCLabelCol(hitId,
j));
980 throw std::runtime_error(
"qa error");
982 if (GetMCLabelID(hitId,
j) >= 0) {
984 GPUInfo(
"Track %d Cluster %u Label %d: %d (%f)",
i, k,
j, GetMCLabelID(hitId,
j), GetMCLabelWeight(hitId,
j));
990 float maxweight, sumweight;
992 auto maxLabel = acc.computeLabel(&maxweight, &sumweight, &maxcount);
993 mTrackMCLabels[
i] = maxLabel;
994 if (
QA_DEBUG && track.OK() && GetNMCTracks(maxLabel) > (uint32_t)maxLabel.getTrackID()) {
995 const mcInfo_t& mc = GetMCTrack(maxLabel);
996 GPUInfo(
"Track %d label %d (fake %d) weight %f clusters %d (fitted %d) (%f%% %f%%) Pt %f",
i, maxLabel.getTrackID(), (int32_t)(maxLabel.isFake()), maxweight,
nClusters, track.NClustersFitted(), 100.f * maxweight / sumweight, 100.f * (
float)maxcount / (
float)
nClusters,
997 std::sqrt(mc.pX * mc.pX + mc.pY * mc.pY));
1002 if (timer.IsRunning()) {
1003 GPUInfo(
"QA Time: Assign Track Labels:\t\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1006 for (uint32_t
i = 0;
i < nReconstructedTracks;
i++) {
1008 mcLabelI_t
label = mTrackMCLabels[
i];
1009 if (mQATasks & taskClusterAttach) {
1014 if (!mTrackMCLabels[
i].
isValid()) {
1015 for (uint32_t k = 0; k < track->NClusters(); k++) {
1019 mClusterParam[mTracking->mIOPtrs.mergedTrackHits[track->FirstClusterRef() + k].num].fakeAttached++;
1023 if (mMCTrackMin == -1 || (
label.getTrackID() >= mMCTrackMin &&
label.getTrackID() < mMCTrackMax)) {
1024 for (uint32_t k = 0; k < track->NClusters(); k++) {
1028 int32_t hitId = mTracking->mIOPtrs.mergedTrackHits[track->FirstClusterRef() + k].num;
1029 bool correct =
false;
1030 for (int32_t
j = 0;
j < GetMCLabelNID(hitId);
j++) {
1031 if (
label == GetMCLabel(hitId,
j)) {
1037 mClusterParam[hitId].attached++;
1039 mClusterParam[hitId].fakeAttached++;
1045 if (mTrackMCLabels[
i].isFake()) {
1046 (GetMCTrackObj(mFakeTracks,
label))++;
1047 }
else if (tracksExternal || !track->MergedLooper()) {
1048 GetMCTrackObj(mRecTracks,
label)++;
1049 if (mMCTrackMin == -1 || (
label.getTrackID() >= mMCTrackMin &&
label.getTrackID() < mMCTrackMax)) {
1050 int32_t& revLabel = GetMCTrackObj(mTrackMCLabelsReverse,
label);
1051 if (tracksExternal) {
1052#ifndef GPUCA_STANDALONE
1053 if (revLabel == -1 || fabsf((*tracksExternal)[
i].getZ()) < fabsf((*tracksExternal)[revLabel].getZ())) {
1058 const auto* trks = mTracking->mIOPtrs.mergedTracks;
1060 if (revLabel == -1) {
1063 float shift1 = mTracking->GetTPCTransform()->convDeltaTimeToDeltaZinTimeFrame(trks[
i].CSide() *
GPUChainTracking::NSECTORS / 2, trks[
i].GetParam().GetTOffset());
1064 float shift2 = mTracking->GetTPCTransform()->convDeltaTimeToDeltaZinTimeFrame(trks[revLabel].CSide() *
GPUChainTracking::NSECTORS / 2, trks[revLabel].GetParam().GetTOffset());
1065 comp = fabsf(trks[
i].GetParam().GetZ() + shift1) < fabsf(trks[revLabel].GetParam().GetZ() + shift2);
1067 if (revLabel == -1 || !trks[revLabel].OK() || (trks[
i].OK() && comp)) {
1074 if ((mQATasks & taskClusterAttach) && !tracksExternal) {
1075 std::vector<uint8_t> lowestPadRow(mTracking->mIOPtrs.nMergedTracks);
1077 if (mTracking->mIOPtrs.mergedTrackHitAttachment) {
1078 for (uint32_t
i = 0;
i < GetNMCLabels();
i++) {
1079 if (mClusterParam[
i].attached == 0 && mClusterParam[
i].fakeAttached == 0) {
1080 int32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[
i];
1083 mcLabelI_t trackL = mTrackMCLabels[track];
1085 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1087 if (trackL == GetMCLabel(
i,
j)) {
1093 mClusterParam[
i].fakeAdjacent++;
1095 mClusterParam[
i].adjacent++;
1101 if (mTracking->mIOPtrs.nMergedTracks && clNative) {
1102 std::fill(lowestPadRow.begin(), lowestPadRow.end(), 255);
1107 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1108 uint32_t trackId = GetMCTrackObj(mTrackMCLabelsReverse, GetMCLabel(
i,
j));
1109 if (trackId < lowestPadRow.size() && lowestPadRow[trackId] > iRow) {
1110 lowestPadRow[trackId] = iRow;
1116 for (uint32_t
i = 0;
i < mTracking->mIOPtrs.nMergedTracks;
i++) {
1117 const auto& trk = mTracking->mIOPtrs.mergedTracks[
i];
1118 if (trk.OK() && lowestPadRow[
i] != 255 && trk.NClustersFitted() >= PADROW_CHECK_MINCLS && CAMath::Abs(trk.GetParam().GetQPt()) < 1.0) {
1119 const auto& lowestCl = mTracking->mIOPtrs.mergedTrackHits[trk.FirstClusterRef()].row < mTracking->mIOPtrs.mergedTrackHits[trk.FirstClusterRef() + trk.NClusters() - 1].row ? mTracking->mIOPtrs.mergedTrackHits[trk.FirstClusterRef()] : mTracking->mIOPtrs.mergedTrackHits[trk.FirstClusterRef() + trk.NClusters() - 1];
1120 const int32_t lowestRow = lowestCl.row;
1121 mPadRow[0]->Fill(lowestPadRow[
i], lowestRow, 1.f);
1122 mPadRow[1]->Fill(CAMath::ATan2(trk.GetParam().GetY(), trk.GetParam().GetX()), lowestRow, 1.f);
1123 if (lowestPadRow[
i] < 10 && lowestRow > lowestPadRow[
i] + 3) {
1126 mTracking->GetTPCTransform()->Transform(lowestCl.sector, lowestCl.row, cl.getPad(), cl.getTime(),
x,
y,
z, trk.GetParam().GetTOffset());
1127 float phi = CAMath::ATan2(
y,
x);
1128 mPadRow[2]->Fill(phi, lowestRow, 1.f);
1129 if (CAMath::Abs(phi) < 0.15) {
1130 const float time = cl.getTime();
1131 mPadRow[3]->Fill(mTracking->GetParam().GetUnscaledMult(
time), lowestRow, 1.f);
1139 if (mConfig.matchMCLabels.size()) {
1140 mGoodHits[mNEvents - 1].resize(GetNMCLabels());
1141 std::vector<bool> allowMCLabels(GetNMCTracks(0));
1142 for (uint32_t k = 0; k < GetNMCTracks(0); k++) {
1143 allowMCLabels[k] =
false;
1145 for (uint32_t
i = 0;
i < nReconstructedTracks;
i++) {
1146 if (!mGoodTracks[mNEvents - 1][
i]) {
1149 if (mConfig.matchDisplayMinPt > 0) {
1150 if (!mTrackMCLabels[
i].
isValid()) {
1153 const mcInfo_t&
info = GetMCTrack(mTrackMCLabels[
i]);
1154 if (
info.pX *
info.pX +
info.pY *
info.pY < mConfig.matchDisplayMinPt * mConfig.matchDisplayMinPt) {
1160 for (uint32_t
j = 0;
j < track.NClusters();
j++) {
1161 int32_t hitId = mTracking->mIOPtrs.mergedTrackHits[track.FirstClusterRef() +
j].num;
1162 if (GetMCLabelNID(hitId)) {
1163 int32_t mcID = GetMCLabelID(hitId, 0);
1165 allowMCLabels[mcID] =
true;
1170 for (uint32_t
i = 0;
i < GetNMCLabels();
i++) {
1171 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1172 int32_t mcID = GetMCLabelID(
i,
j);
1173 if (mcID >= 0 && allowMCLabels[mcID]) {
1174 mGoodHits[mNEvents - 1][
i] =
true;
1179 if (timer.IsRunning()) {
1180 GPUInfo(
"QA Time: Cluster attach status:\t\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1188 for (uint32_t iCol = 0; iCol < GetNMCCollissions(); iCol++) {
1189 for (uint32_t
i = 0;
i < GetNMCTracks(iCol);
i++) {
1190 mMCParam[iCol][
i].nWeightCls = 0.;
1193 for (uint32_t
i = 0;
i < GetNMCLabels();
i++) {
1194 float weightTotal = 0.f;
1195 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1196 if (GetMCLabelID(
i,
j) >= 0) {
1197 weightTotal += GetMCLabelWeight(
i,
j);
1200 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1201 if (GetMCLabelID(
i,
j) >= 0) {
1202 GetMCTrackObj(mMCParam, GetMCLabel(
i,
j)).nWeightCls += GetMCLabelWeight(
i,
j) / weightTotal;
1206 if (timer.IsRunning()) {
1207 GPUInfo(
"QA Time: Compute cluster label weights:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1211 tbb::parallel_for<uint32_t>(0, GetNMCCollissions(), [&](
auto iCol) {
1212 for (uint32_t
i = 0;
i < GetNMCTracks(iCol);
i++) {
1213 const mcInfo_t&
info = GetMCTrack(
i, iCol);
1214 additionalMCParameters& mc2 = mMCParam[iCol][
i];
1216 mc2.phi = M_PI + std::atan2(-
info.pY, -
info.pX);
1219 mc2.theta = mc2.eta = 0.f;
1221 mc2.theta =
info.pZ == 0 ? (M_PI / 2) : (
std::acos(
info.pZ /
std::sqrt(
p)));
1222 mc2.eta = -std::log(std::tan(0.5 * mc2.theta));
1224 if (mConfig.writeMCLabels) {
1225 std::vector<int32_t>& effBuffer = mcEffBuffer[mNEvents - 1];
1226 effBuffer[
i] = mRecTracks[iCol][
i] * 1000 + mFakeTracks[iCol][
i];
1229 }, tbb::simple_partitioner());
1230 if (timer.IsRunning()) {
1231 GPUInfo(
"QA Time: Compute track mc parameters:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1235 if (mQATasks & taskTrackingEff) {
1236 for (uint32_t iCol = 0; iCol < GetNMCCollissions(); iCol++) {
1237 for (uint32_t
i = 0;
i < GetNMCTracks(iCol);
i++) {
1238 if ((mMCTrackMin != -1 && (int32_t)
i < mMCTrackMin) || (mMCTrackMax != -1 && (int32_t)
i >= mMCTrackMax)) {
1241 const mcInfo_t&
info = GetMCTrack(
i, iCol);
1242 const additionalMCParameters& mc2 = mMCParam[iCol][
i];
1243 if (mc2.nWeightCls == 0.f) {
1246 const float& mcpt = mc2.pt;
1247 const float& mcphi = mc2.phi;
1248 const float& mceta = mc2.eta;
1250 if (
info.primDaughters) {
1253 if (mc2.nWeightCls < mConfig.minNClEff) {
1256 int32_t findable = mc2.nWeightCls >= mConfig.minNClFindable;
1260 if (
info.charge == 0.f) {
1263 if (mConfig.filterCharge &&
info.charge * mConfig.filterCharge < 0) {
1266 if (mConfig.filterPID >= 0 &&
info.pid != mConfig.filterPID) {
1270 if (fabsf(mceta) > ETA_MAX || mcpt < PT_MIN || mcpt > PT_MAX) {
1275 alpha /= M_PI / 9.f;
1277 alpha *= M_PI / 9.f;
1278 alpha += M_PI / 18.f;
1280 float c = std::cos(
alpha);
1281 float s = std::sin(
alpha);
1284 if (mConfig.dumpToROOT) {
1287 effdump.Fill(
alpha, localX, localY,
info.z, mcphi, mceta, mcpt, mRecTracks[iCol][
i], mFakeTracks[iCol][
i], findable,
info.prim, mc2.nWeightCls);
1290 for (int32_t
j = 0;
j < 6;
j++) {
1291 if (
j == 3 ||
j == 4) {
1294 for (int32_t k = 0; k < 2; k++) {
1295 if (k == 0 && findable == 0) {
1299 int32_t
val = (
j == 0) ? (mRecTracks[iCol][
i] ? 1 : 0) : (
j == 1) ? (mRecTracks[iCol][
i] ? mRecTracks[iCol][
i] - 1 : 0) : (
j == 2) ? mFakeTracks[iCol][
i] : 1;
1304 for (int32_t l = 0; l < 5; l++) {
1305 if (
info.prim && mcpt < PT_MIN_PRIM) {
1308 if (l != 3 && fabsf(mceta) > ETA_MAX2) {
1311 if (l < 4 && mcpt < 1.f / mConfig.qpt) {
1315 float pos = l == 0 ? localY : l == 1 ?
info.z : l == 2 ? mcphi : l == 3 ? mceta : mcpt;
1323 if (timer.IsRunning()) {
1324 GPUInfo(
"QA Time: Fill efficiency histograms:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1329 if (mQATasks & (taskTrackingRes | taskTrackingResPull)) {
1331 prop.SetMaxSinPhi(.999);
1332 prop.SetMaterialTPC();
1333 prop.SetPolynomialField(&mParam->polynomialField);
1335 for (uint32_t
i = 0;
i < mTrackMCLabels.size();
i++) {
1336 if (mConfig.writeMCLabels) {
1337 std::vector<int32_t>& labelBuffer = mcLabelBuffer[mNEvents - 1];
1338 labelBuffer[
i] = mTrackMCLabels[
i].getTrackID();
1340 if (mTrackMCLabels[
i].isFake()) {
1343 const mcInfo_t& mc1 = GetMCTrack(mTrackMCLabels[
i]);
1344 const additionalMCParameters& mc2 = GetMCTrackObj(mMCParam, mTrackMCLabels[
i]);
1346 if (mc1.primDaughters) {
1349 if (!tracksExternal) {
1350 if (!mTracking->mIOPtrs.mergedTracks[
i].OK()) {
1353 if (mTracking->mIOPtrs.mergedTracks[
i].MergedLooper()) {
1357 if ((mMCTrackMin != -1 && mTrackMCLabels[
i].getTrackID() < mMCTrackMin) || (mMCTrackMax != -1 && mTrackMCLabels[
i].getTrackID() >= mMCTrackMax)) {
1360 if (fabsf(mc2.eta) > ETA_MAX || mc2.pt < PT_MIN || mc2.pt > PT_MAX) {
1363 if (mc1.charge == 0.f) {
1370 if (mc1.t0 == -100.f) {
1374 if (mConfig.filterCharge && mc1.charge * mConfig.filterCharge < 0) {
1377 if (mConfig.filterPID >= 0 && mc1.pid != mConfig.filterPID) {
1380 if (mc2.nWeightCls < mConfig.minNClRes) {
1383 if (mConfig.resPrimaries == 1 && !mc1.prim) {
1385 }
else if (mConfig.resPrimaries == 2 && mc1.prim) {
1388 if (GetMCTrackObj(mTrackMCLabelsReverse, mTrackMCLabels[
i]) != (int32_t)
i) {
1395 if (tracksExternal) {
1396#ifndef GPUCA_STANDALONE
1397 for (int32_t k = 0; k < 5; k++) {
1398 param.Par()[k] = (*tracksExternal)[
i].getParams()[k];
1400 for (int32_t k = 0; k < 15; k++) {
1401 param.Cov()[k] = (*tracksExternal)[
i].getCov()[k];
1403 param.X() = (*tracksExternal)[
i].getX();
1404 param.TOffset() = (*tracksExternal)[
i].getTime0();
1405 alpha = (*tracksExternal)[
i].getAlpha();
1406 side = (*tracksExternal)[
i].hasBothSidesClusters() ? 2 : ((*tracksExternal)[
i].hasCSideClusters() ? 1 : 0);
1409 param = mTracking->mIOPtrs.mergedTracks[
i].GetParam();
1410 alpha = mTracking->mIOPtrs.mergedTracks[
i].GetAlpha();
1411 side = mTracking->mIOPtrs.mergedTracks[
i].CCE() ? 2 : (mTracking->mIOPtrs.mergedTracks[
i].CSide() ? 1 : 0);
1415 float c = std::cos(
alpha);
1416 float s = std::sin(
alpha);
1419 mclocal[0] =
x *
c +
y *
s;
1420 mclocal[1] = -
x *
s +
y *
c;
1423 mclocal[2] = px *
c + py *
s;
1424 mclocal[3] = -px *
s + py *
c;
1429 if (mclocal[0] >
param.GetX() + 20) {
1432 if (
param.GetX() > mConfig.maxResX) {
1436 auto getdz = [
this, &
param, &mc1, &
side, tracksExternal]() {
1437 if (tracksExternal) {
1438 return param.GetZ();
1440 if (!mParam->continuousMaxTimeBin) {
1441 return param.GetZ() - mc1.z;
1449 return param.GetZ() + shift - mc1.z;
1453 bool inFlyDirection = 0;
1454 if (mConfig.strict) {
1456 const float dy =
param.Y() - mclocal[1];
1457 const float dz = getdz();
1458 if (dx * dx + dy * dy + dz * dz > 5.f * 5.f) {
1463 if (prop.PropagateToXAlpha(mclocal[0],
alpha, inFlyDirection)) {
1466 if (fabsf(
param.Y() - mclocal[1]) > (mConfig.strict ? 1.f : 4.f) || fabsf(getdz()) > (mConfig.strict ? 1.f : 4.f)) {
1469 float charge = mc1.charge > 0 ? 1.f : -1.f;
1471 float deltaY =
param.GetY() - mclocal[1];
1472 float deltaZ = getdz();
1473 float deltaPhiNative =
param.GetSinPhi() - mclocal[3] / mc2.pt;
1474 float deltaPhi = std::asin(
param.GetSinPhi()) - std::atan2(mclocal[3], mclocal[2]);
1475 float deltaLambdaNative =
param.GetDzDs() - mc1.pZ / mc2.pt;
1476 float deltaLambda = std::atan(
param.GetDzDs()) - std::atan2(mc1.pZ, mc2.pt);
1478 float deltaPt = (fabsf(1.f /
param.GetQPt()) - mc2.pt) / mc2.pt;
1480 float paramval[5] = {mclocal[1], mc1.z, mc2.phi, mc2.eta, mc2.pt};
1481 float resval[5] = {deltaY, deltaZ, mConfig.nativeFitResolutions ? deltaPhiNative : deltaPhi, mConfig.nativeFitResolutions ? deltaLambdaNative : deltaLambda, mConfig.nativeFitResolutions ? deltaPtNative : deltaPt};
1482 float pullval[5] = {deltaY / std::sqrt(
param.GetErr2Y()), deltaZ / std::sqrt(
param.GetErr2Z()), deltaPhiNative / std::sqrt(
param.GetErr2SinPhi()), deltaLambdaNative / std::sqrt(
param.GetErr2DzDs()), deltaPtNative / std::sqrt(
param.GetErr2QPt())};
1484 for (int32_t
j = 0;
j < 5;
j++) {
1485 for (int32_t k = 0; k < 5; k++) {
1486 if (k != 3 && fabsf(mc2.eta) > ETA_MAX2) {
1489 if (k < 4 && mc2.pt < 1.f / mConfig.qpt) {
1492 if (mQATasks & taskTrackingRes) {
1493 mRes2[
j][k]->Fill(resval[
j], paramval[k]);
1495 if (mQATasks & taskTrackingResPull) {
1496 mPull2[
j][k]->Fill(pullval[
j], paramval[k]);
1501 if (timer.IsRunning()) {
1502 GPUInfo(
"QA Time: Fill resolution histograms:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1506 if ((mQATasks & taskClusterAttach) && !tracksExternal) {
1508 for (uint32_t iTrk = 0; iTrk < nReconstructedTracks; iTrk++) {
1513 if (!mTrackMCLabels[iTrk].
isValid()) {
1514 for (uint32_t k = 0; k < track.NClusters(); k++) {
1518 int32_t hitId = mTracking->mIOPtrs.mergedTrackHits[track.FirstClusterRef() + k].num;
1519 float totalWeight = 0.;
1520 for (int32_t
j = 0;
j < GetMCLabelNID(hitId);
j++) {
1521 if (GetMCLabelID(hitId,
j) >= 0 && GetMCTrackObj(mMCParam, GetMCLabel(hitId,
j)).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1522 totalWeight += GetMCLabelWeight(hitId,
j);
1525 int32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[hitId];
1526 const auto&
r = checkClusterState<false>(attach);
1527 if (totalWeight > 0) {
1528 float weight = 1.f / (totalWeight * (mClusterParam[hitId].attached + mClusterParam[hitId].fakeAttached));
1529 for (int32_t
j = 0;
j < GetMCLabelNID(hitId);
j++) {
1530 mcLabelI_t
label = GetMCLabel(hitId,
j);
1531 if (!
label.isFake() && GetMCTrackObj(mMCParam,
label).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1532 float pt = GetMCTrackObj(mMCParam,
label).pt;
1533 if (pt < PT_MIN_CLUST) {
1536 mClusters[CL_fake]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1537 mClusters[CL_att_adj]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1538 if (GetMCTrackObj(mRecTracks,
label)) {
1539 mClusters[CL_tracks]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1541 mClusters[CL_all]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1542 if (
r.protect ||
r.physics) {
1543 mClusters[CL_prot]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1546 mClusters[CL_physics]->Fill(pt, GetMCLabelWeight(hitId,
j) *
weight);
1551 float weight = 1.f / (mClusterParam[hitId].attached + mClusterParam[hitId].fakeAttached);
1552 mClusters[CL_fake]->Fill(0.f,
weight);
1553 mClusters[CL_att_adj]->Fill(0.f,
weight);
1554 mClusters[CL_all]->Fill(0.f,
weight);
1555 mClusterCounts.nUnaccessible +=
weight;
1556 if (
r.protect ||
r.physics) {
1557 mClusters[CL_prot]->Fill(0.f,
weight);
1560 mClusters[CL_physics]->Fill(0.f,
weight);
1566 mcLabelI_t
label = mTrackMCLabels[iTrk];
1567 if (mMCTrackMin != -1 && (
label.getTrackID() < mMCTrackMin ||
label.getTrackID() >= mMCTrackMax)) {
1570 for (uint32_t k = 0; k < track.NClusters(); k++) {
1574 int32_t hitId = mTracking->mIOPtrs.mergedTrackHits[track.FirstClusterRef() + k].num;
1575 float pt = GetMCTrackObj(mMCParam,
label).pt;
1576 if (pt < PT_MIN_CLUST) {
1579 float weight = 1.f / (mClusterParam[hitId].attached + mClusterParam[hitId].fakeAttached);
1580 bool correct =
false;
1581 for (int32_t
j = 0;
j < GetMCLabelNID(hitId);
j++) {
1582 if (
label == GetMCLabel(hitId,
j)) {
1588 mClusters[CL_attached]->Fill(pt,
weight);
1589 mClusters[CL_tracks]->Fill(pt,
weight);
1591 mClusters[CL_fake]->Fill(pt,
weight);
1593 mClusters[CL_att_adj]->Fill(pt,
weight);
1594 mClusters[CL_all]->Fill(pt,
weight);
1595 int32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[hitId];
1596 const auto&
r = checkClusterState<false>(attach);
1597 if (
r.protect ||
r.physics) {
1598 mClusters[CL_prot]->Fill(pt,
weight);
1601 mClusters[CL_physics]->Fill(pt,
weight);
1605 for (uint32_t
i = 0;
i < GetNMCLabels();
i++) {
1606 if ((mMCTrackMin != -1 && GetMCLabelID(
i, 0) < mMCTrackMin) || (mMCTrackMax != -1 && GetMCLabelID(
i, 0) >= mMCTrackMax)) {
1609 if (mClusterParam[
i].attached || mClusterParam[
i].fakeAttached) {
1612 int32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[
i];
1613 const auto&
r = checkClusterState<false>(attach);
1614 if (mClusterParam[
i].adjacent) {
1617 float totalWeight = 0.;
1618 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1619 mcLabelI_t labelT = GetMCLabel(
i,
j);
1620 if (!labelT.isFake() && GetMCTrackObj(mMCParam, labelT).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1621 totalWeight += GetMCLabelWeight(
i,
j);
1624 float weight = 1.f / totalWeight;
1625 if (totalWeight > 0) {
1626 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1627 mcLabelI_t labelT = GetMCLabel(
i,
j);
1628 if (!labelT.isFake() && GetMCTrackObj(mMCParam, labelT).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1629 float pt = GetMCTrackObj(mMCParam, labelT).pt;
1630 if (pt < PT_MIN_CLUST) {
1633 if (GetMCTrackObj(mRecTracks, labelT)) {
1634 mClusters[CL_tracks]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1636 mClusters[CL_att_adj]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1637 mClusters[CL_fakeAdj]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1638 mClusters[CL_all]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1639 if (
r.protect ||
r.physics) {
1640 mClusters[CL_prot]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1643 mClusters[CL_physics]->Fill(pt, GetMCLabelWeight(
i,
j) *
weight);
1648 mClusters[CL_att_adj]->Fill(0.f, 1.f);
1649 mClusters[CL_fakeAdj]->Fill(0.f, 1.f);
1650 mClusters[CL_all]->Fill(0.f, 1.f);
1651 mClusterCounts.nUnaccessible++;
1652 if (
r.protect ||
r.physics) {
1653 mClusters[CL_prot]->Fill(0.f, 1.f);
1656 mClusters[CL_physics]->Fill(0.f, 1.f);
1660 float pt = GetMCTrackObj(mMCParam, mTrackMCLabels[
label]).pt;
1661 if (pt < PT_MIN_CLUST) {
1664 mClusters[CL_att_adj]->Fill(pt, 1.f);
1665 mClusters[CL_tracks]->Fill(pt, 1.f);
1666 mClusters[CL_all]->Fill(pt, 1.f);
1667 if (
r.protect ||
r.physics) {
1668 mClusters[CL_prot]->Fill(pt, 1.f);
1671 mClusters[CL_physics]->Fill(pt, 1.f);
1675 float totalWeight = 0.;
1676 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1677 mcLabelI_t labelT = GetMCLabel(
i,
j);
1678 if (!labelT.isFake() && GetMCTrackObj(mMCParam, labelT).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1679 totalWeight += GetMCLabelWeight(
i,
j);
1682 if (totalWeight > 0) {
1683 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1684 mcLabelI_t
label = GetMCLabel(
i,
j);
1685 if (!
label.isFake() && GetMCTrackObj(mMCParam,
label).pt > 1.f / mTracking->GetParam().rec.maxTrackQPtB5) {
1686 float pt = GetMCTrackObj(mMCParam,
label).pt;
1687 if (pt < PT_MIN_CLUST) {
1690 float weight = GetMCLabelWeight(
i,
j) / totalWeight;
1691 if (mClusterParam[
i].fakeAdjacent) {
1692 mClusters[CL_fakeAdj]->Fill(pt,
weight);
1694 if (mClusterParam[
i].fakeAdjacent) {
1695 mClusters[CL_att_adj]->Fill(pt,
weight);
1697 if (GetMCTrackObj(mRecTracks,
label)) {
1698 mClusters[CL_tracks]->Fill(pt,
weight);
1700 mClusters[CL_all]->Fill(pt,
weight);
1701 if (
r.protect ||
r.physics) {
1702 mClusters[CL_prot]->Fill(pt,
weight);
1705 mClusters[CL_physics]->Fill(pt,
weight);
1710 if (mClusterParam[
i].fakeAdjacent) {
1711 mClusters[CL_fakeAdj]->Fill(0.f, 1.f);
1713 if (mClusterParam[
i].fakeAdjacent) {
1714 mClusters[CL_att_adj]->Fill(0.f, 1.f);
1716 mClusters[CL_all]->Fill(0.f, 1.f);
1717 mClusterCounts.nUnaccessible++;
1718 if (
r.protect ||
r.physics) {
1719 mClusters[CL_prot]->Fill(0.f, 1.f);
1722 mClusters[CL_physics]->Fill(0.f, 1.f);
1728 if (timer.IsRunning()) {
1729 GPUInfo(
"QA Time: Fill cluster histograms:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1732 }
else if (!mConfig.inputHistogramsOnly && !mConfig.noMC && (mQATasks & (taskTrackingEff | taskTrackingRes | taskTrackingResPull | taskClusterAttach))) {
1733 GPUWarning(
"No MC information available, only running partial TPC QA!");
1736 if ((mQATasks & taskTrackStatistics) && !tracksExternal) {
1738 std::vector<std::array<float, 3>> clusterAttachCounts;
1740 clusterAttachCounts.resize(GetNMCLabels(), {0.f, 0.f});
1742 for (uint32_t
i = 0;
i < nReconstructedTracks;
i++) {
1747 mTrackPt->Fill(1.f / fabsf(track.GetParam().GetQPt()));
1748 mNCl[0]->Fill(track.NClustersFitted());
1749 uint32_t nClCorrected = 0;
1750 const auto& trackClusters = mTracking->mIOPtrs.mergedTrackHits;
1752 for (uint32_t
j = 0;
j < track.NClusters();
j = jNext) {
1754 for (jNext =
j + 1;
j < track.NClusters(); jNext++) {
1755 if (trackClusters[track.FirstClusterRef() +
j].sector != trackClusters[track.FirstClusterRef() + jNext].sector || trackClusters[track.FirstClusterRef() +
j].row != trackClusters[track.FirstClusterRef() + jNext].row) {
1760 if (!track.MergedLooper() && rowClCount) {
1763 if (mcAvail && rowClCount) {
1764 for (uint32_t k =
j; k < jNext; k++) {
1765 const auto& cl = trackClusters[track.FirstClusterRef() + k];
1769 bool labelOk =
false, labelOkNonFake =
false;
1770 const mcLabelI_t& trkLabel = mTrackMCLabels[
i];
1771 if (trkLabel.isValid() && !trkLabel.isNoise()) {
1772 for (int32_t l = 0; l < GetMCLabelNID(cl.num); l++) {
1773 const mcLabelI_t& clLabel = GetMCLabel(cl.num, l);
1774 if (clLabel.isValid() && !clLabel.isNoise() && CompareIgnoreFake(trkLabel, clLabel)) {
1776 if (!trkLabel.isFake()) {
1777 labelOkNonFake =
true;
1783 clusterAttachCounts[cl.num][0] += 1.0f;
1784 clusterAttachCounts[cl.num][1] += (float)labelOk / rowClCount;
1785 clusterAttachCounts[cl.num][2] += (float)labelOkNonFake / rowClCount;
1790 mNCl[1]->Fill(nClCorrected);
1792 mT0[0]->Fill(track.GetParam().GetTOffset());
1794 if (mTrackMCLabels.size() && !mTrackMCLabels[
i].isFake() && !track.MergedLooper() && !track.CCE()) {
1795 const auto&
info = GetMCTrack(mTrackMCLabels[
i]);
1796 if (
info.t0 != -100.f) {
1797 mT0[1]->Fill(track.GetParam().GetTOffset() -
info.t0);
1802 if (mClNative && mTracking && mTracking->GetTPCTransform()) {
1805 for (uint32_t k = 0; k < mClNative->nClusters[
i][
j]; k++) {
1806 const auto& cl = mClNative->clusters[
i][
j][k];
1808 GPUTPCConvertImpl::convert(*mTracking->GetTPCTransform(), mTracking->GetParam(),
i,
j, cl.getPad(), cl.getTime(),
x,
y,
z);
1809 mTracking->GetParam().Sector2Global(
i,
x,
y,
z, &
x, &
y, &
z);
1816 double clusterAttachNormalizedCount = 0, clusterAttachNormalizedCountNonFake = 0;
1817 for (uint32_t
i = 0;
i < clusterAttachCounts.size();
i++) {
1818 if (clusterAttachCounts[
i][0]) {
1819 clusterAttachNormalizedCount += clusterAttachCounts[
i][1] / clusterAttachCounts[
i][0];
1820 clusterAttachNormalizedCountNonFake += clusterAttachCounts[
i][2] / clusterAttachCounts[
i][0];
1823 mClusterCounts.nCorrectlyAttachedNormalized = clusterAttachNormalizedCount;
1824 mClusterCounts.nCorrectlyAttachedNormalizedNonFake = clusterAttachNormalizedCountNonFake;
1825 clusterAttachCounts.clear();
1828 if (timer.IsRunning()) {
1829 GPUInfo(
"QA Time: Fill track statistics:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1833 uint32_t
nCl = clNative ? clNative->
nClustersTotal : mTracking->GetProcessors()->tpcMerger.NMaxClusters();
1834 mClusterCounts.nTotal +=
nCl;
1835 if (mQATasks & (taskClusterCounts | taskClusterRejection)) {
1840 int32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[
i];
1841 const auto&
r = checkClusterState<true>(attach, &mClusterCounts);
1843 if (mQATasks & taskClusterRejection) {
1845 float totalWeight = 0, weight400 = 0, weight40 = 0;
1846 for (int32_t
j = 0;
j < GetMCLabelNID(
i);
j++) {
1847 const auto&
label = GetMCLabel(
i,
j);
1848 if (GetMCLabelID(
label) >= 0) {
1849 totalWeight += GetMCLabelWeight(
label);
1850 if (GetMCTrackObj(mMCParam,
label).pt >= 0.4) {
1851 weight400 += GetMCLabelWeight(
label);
1853 if (GetMCTrackObj(mMCParam,
label).pt <= 0.04) {
1854 weight40 += GetMCLabelWeight(
label);
1858 if (totalWeight > 0 && 10.f * weight400 >= totalWeight) {
1859 if (!
r.unattached && !
r.protect && !
r.physics) {
1860 mClusterCounts.nFakeRemove400++;
1861 int32_t totalFake = weight400 < 0.9f * totalWeight;
1863 mClusterCounts.nFullFakeRemove400++;
1878 mClusterCounts.nAbove400++;
1880 if (totalWeight > 0 && weight40 >= 0.9 * totalWeight) {
1881 mClusterCounts.nBelow40++;
1882 if (
r.protect ||
r.physics) {
1883 mClusterCounts.nFakeProtect40++;
1889 mClusterCounts.nPhysics++;
1892 mClusterCounts.nProt++;
1895 mClusterCounts.nUnattached++;
1898 if (mQATasks & taskClusterRejection) {
1899 if (mTracking && clNative) {
1901 mClRej[0]->Fill(cl.getPad() - GPUTPCGeometry::NPads(iRow) / 2 + 0.5, iRow, 1.f);
1902 if (!
r.unattached && !
r.protect) {
1903 mClRej[1]->Fill(cl.getPad() - GPUTPCGeometry::NPads(iRow) / 2 + 0.5, iRow, 1.f);
1913 if ((mQATasks & taskClusterCounts) && mConfig.clusterRejectionHistograms) {
1914 DoClusterCounts(
nullptr);
1915 mClusterCounts = counts_t();
1918 if (timer.IsRunning()) {
1919 GPUInfo(
"QA Time: Cluster Counts:\t%6.0f us", timer.GetCurrentElapsedTime(
true) * 1e6);
1922 if (mConfig.dumpToROOT && !tracksExternal) {
1923 if (!clNative || !mTracking || !mTracking->mIOPtrs.mergedTrackHitAttachment || !mTracking->mIOPtrs.mergedTracks) {
1924 throw std::runtime_error(
"Cannot dump non o2::tpc::clusterNative clusters, need also hit attachmend and GPU tracks");
1929 for (uint32_t k = 0; k < mClNative->nClusters[
i][
j]; k++) {
1930 const auto& cl = mClNative->clusters[
i][
j][k];
1931 uint32_t attach = mTracking->mIOPtrs.mergedTrackHitAttachment[clid];
1932 float x = 0,
y = 0,
z = 0;
1935 const auto& trk = mTracking->mIOPtrs.mergedTracks[track];
1936 mTracking->GetTPCTransform()->Transform(
i,
j, cl.getPad(), cl.getTime(),
x,
y,
z, trk.GetParam().GetTOffset());
1937 mTracking->GetParam().Sector2Global(
i,
x,
y,
z, &
x, &
y, &
z);
1939 uint32_t extState = mTracking->mIOPtrs.mergedTrackHitStates ? mTracking->mIOPtrs.mergedTrackHitStates[clid] : 0;
1941 if (mConfig.dumpToROOT >= 2) {
1944 memset((
void*)&trk, 0,
sizeof(trk));
1945 memset((
void*)&trkHit, 0,
sizeof(trkHit));
1948 trk = mTracking->mIOPtrs.mergedTracks[track];
1949 for (uint32_t l = 0; l < trk.NClusters(); l++) {
1950 const auto& tmp = mTracking->mIOPtrs.mergedTrackHits[trk.FirstClusterRef() + l];
1951 if (tmp.num == clid) {
1957 static auto cldump =
GPUROOTDump<o2::tpc::ClusterNative, GPUTPCGMMergedTrack, GPUTPCGMMergedTrackHit, uint32_t, uint32_t, float, float, float, uint32_t, uint32_t, uint32_t>::getNew(
"cluster",
"track",
"trackHit",
"attach",
"extState",
"x",
"y",
"z",
"sector",
"row",
"nEv",
"clusterTree");
1958 cldump.Fill(cl, trk, trkHit, attach, extState,
x,
y,
z,
i,
j, mNEvents - 1);
1960 static auto cldump =
GPUROOTDump<o2::tpc::ClusterNative, uint32_t, uint32_t, float, float, float, uint32_t, uint32_t, uint32_t>::getNew(
"cluster",
"attach",
"extState",
"x",
"y",
"z",
"sector",
"row",
"nEv",
"clusterTree");
1961 cldump.Fill(cl, attach, extState,
x,
y,
z,
i,
j, mNEvents - 1);
1969 for (uint32_t
i = 0;
i < mTracking->mIOPtrs.nMergedTracks;
i++) {
1970 if (mTracking->mIOPtrs.mergedTracks[
i].OK()) {
1971 trkdump.Fill(mNEvents - 1, mTracking->mIOPtrs.mergedTracks[
i]);
1975 if (mTracking && mTracking->GetProcessingSettings().createO2Output) {
1977 for (uint32_t
i = 0;
i < mTracking->mIOPtrs.nOutputTracksTPCO2;
i++) {
1978 o2trkdump.Fill(mNEvents - 1, mTracking->mIOPtrs.outputTracksTPCO2[
i]);
1983 if (mConfig.compareTrackStatus) {
1984#ifdef GPUCA_DETERMINISTIC_MODE
1985 if (!mTracking || !mTracking->GetProcessingSettings().deterministicGPUReconstruction)
1988 throw std::runtime_error(
"Need deterministic processing to compare track status");
1990 std::vector<uint8_t> status(mTracking->mIOPtrs.nMergedTracks);
1991 for (uint32_t
i = 0;
i < mTracking->mIOPtrs.nMergedTracks;
i++) {
1992 const auto& trk = mTracking->mIOPtrs.mergedTracks[
i];
1993 status[
i] = trk.OK() && trk.NClusters() && trk.GetParam().GetNDF() > 0 && (mConfig.noMC || (mTrackMCLabels[
i].isValid() && !mTrackMCLabels[
i].isFake()));
1995 if (mConfig.compareTrackStatus == 1) {
1996 std::ofstream(
"track.status", std::ios::binary).write((
char*)status.data(), status.size() *
sizeof(status[0]));
1997 }
else if (mConfig.compareTrackStatus == 2) {
1998 std::ifstream
f(
"track.status", std::ios::binary | std::ios::ate);
1999 std::vector<uint8_t> comp(
f.tellg());
2001 f.read((
char*)comp.data(), comp.size());
2003 if (comp.size() != status.size()) {
2004 throw std::runtime_error(
"Number of tracks candidates in track fit in track.status and in current reconstruction differ");
2006 std::vector<uint32_t> missing, missingComp;
2007 for (uint32_t
i = 0;
i < status.size();
i++) {
2008 if (status[
i] && !comp[
i]) {
2009 missingComp.emplace_back(
i);
2011 if (comp[
i] && !status[
i]) {
2012 missing.emplace_back(
i);
2015 auto printer = [](std::vector<uint32_t>
m,
const char*
name) {
2017 printf(
"Missing in %s reconstruction: (%zu)\n",
name,
m.size());
2018 for (uint32_t
i = 0;
i <
m.size();
i++) {
2027 printer(missing,
"current");
2028 printer(missingComp,
"comparison");
2032 mTrackingScratchBuffer.clear();
2033 mTrackingScratchBuffer.shrink_to_fit();
2036void GPUQA::GetName(
char* fname, int32_t k,
bool noDash)
2038 const int32_t nNewInput = mConfig.inputHistogramsOnly ? 0 : 1;
2039 if (k || mConfig.inputHistogramsOnly || mConfig.name.size()) {
2040 if (!(mConfig.inputHistogramsOnly || k)) {
2041 snprintf(fname, 1024,
"%s%s", mConfig.name.c_str(), noDash ?
"" :
" - ");
2042 }
else if (mConfig.compareInputNames.size() > (
unsigned)(k - nNewInput)) {
2043 snprintf(fname, 1024,
"%s%s", mConfig.compareInputNames[k - nNewInput].c_str(), noDash ?
"" :
" - ");
2045 strcpy(fname, mConfig.compareInputs[k - nNewInput].c_str());
2046 if (strlen(fname) > 5 && strcmp(fname + strlen(fname) - 5,
".root") == 0) {
2047 fname[strlen(fname) - 5] = 0;
2050 strcat(fname,
" - ");
2059T* GPUQA::GetHist(T*& ee, std::vector<std::unique_ptr<TFile>>& tin, int32_t k, int32_t nNewInput)
2062 if ((mConfig.inputHistogramsOnly || k) && (e =
dynamic_cast<T*
>(tin[k - nNewInput]->Get(e->GetName()))) ==
nullptr) {
2063 GPUWarning(
"Missing histogram in input %s: %s", mConfig.compareInputs[k - nNewInput].c_str(), ee->GetName());
2070void GPUQA::DrawQAHistogramsCleanup()
2072 clearGarbagageCollector();
2075void GPUQA::resetHists()
2077 if (!mQAInitialized) {
2078 throw std::runtime_error(
"QA not initialized");
2080 if (mHaveExternalHists) {
2081 throw std::runtime_error(
"Cannot reset external hists");
2083 for (
auto&
h : *mHist1D) {
2086 for (
auto&
h : *mHist2D) {
2089 for (
auto&
h : *mHist1Dd) {
2092 for (
auto&
h : *mHistGraph) {
2093 h = TGraphAsymmErrors();
2095 mClusterCounts = counts_t();
2100 const auto oldRootIgnoreLevel = gErrorIgnoreLevel;
2101 gErrorIgnoreLevel = kWarning;
2102 if (!mQAInitialized) {
2103 throw std::runtime_error(
"QA not initialized");
2106 if (mTracking && mTracking->GetProcessingSettings().debugLevel >= 2) {
2107 printf(
"Creating QA Histograms\n");
2110 std::vector<Color_t> colorNums(COLORCOUNT);
2111 if (!(qcout || mConfig.writeFileExt ==
"root" || mConfig.writeFileExt ==
"C")) {
2112 [[maybe_unused]]
static int32_t initColorsInitialized = initColors();
2114 for (int32_t
i = 0;
i < COLORCOUNT;
i++) {
2115 colorNums[
i] = (qcout || mConfig.writeFileExt ==
"root" || mConfig.writeFileExt ==
"C") ? defaultColorNums[
i] : mColors[
i]->GetNumber();
2118 bool mcAvail = mcPresent();
2119 char name[2048], fname[1024];
2121 const int32_t nNewInput = mConfig.inputHistogramsOnly ? 0 : 1;
2122 const int32_t ConfigNumInputs = nNewInput + mConfig.compareInputs.size();
2124 std::vector<std::unique_ptr<TFile>> tin;
2125 for (uint32_t
i = 0;
i < mConfig.compareInputs.size();
i++) {
2126 tin.emplace_back(std::make_unique<TFile>(mConfig.compareInputs[
i].c_str()));
2128 std::unique_ptr<TFile> tout =
nullptr;
2129 if (mConfig.output.size()) {
2130 tout = std::make_unique<TFile>(mConfig.output.c_str(),
"RECREATE");
2133 if (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) {
2134 float legendSpacingString = 0.025;
2135 for (int32_t
i = 0;
i < ConfigNumInputs;
i++) {
2137 if (strlen(fname) * 0.006 > legendSpacingString) {
2138 legendSpacingString = strlen(fname) * 0.006;
2143 if (mQATasks & taskTrackingEff) {
2144 for (int32_t ii = 0; ii < 6; ii++) {
2145 snprintf(
name, 1024,
"eff_vs_%s_layout", VSPARAMETER_NAMES[ii]);
2146 mCEff[ii] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2149 mPEff[ii][0] = createGarbageCollected<TPad>(
"p0",
"", 0.0, dy * 0, 0.5, dy * 1);
2150 mPEff[ii][0]->Draw();
2151 mPEff[ii][0]->SetRightMargin(0.04);
2152 mPEff[ii][1] = createGarbageCollected<TPad>(
"p1",
"", 0.5, dy * 0, 1.0, dy * 1);
2153 mPEff[ii][1]->Draw();
2154 mPEff[ii][1]->SetRightMargin(0.04);
2155 mPEff[ii][2] = createGarbageCollected<TPad>(
"p2",
"", 0.0, dy * 1, 0.5, dy * 2 - .001);
2156 mPEff[ii][2]->Draw();
2157 mPEff[ii][2]->SetRightMargin(0.04);
2158 mPEff[ii][3] = createGarbageCollected<TPad>(
"p3",
"", 0.5, dy * 1, 1.0, dy * 2 - .001);
2159 mPEff[ii][3]->Draw();
2160 mPEff[ii][3]->SetRightMargin(0.04);
2161 mLEff[ii] = createGarbageCollected<TLegend>(0.92 - legendSpacingString * 1.45, 0.83 - (0.93 - 0.82) / 2. * (
float)ConfigNumInputs, 0.98, 0.849);
2162 SetLegend(mLEff[ii]);
2167 if (mQATasks & taskTrackingRes) {
2168 for (int32_t ii = 0; ii < 7; ii++) {
2170 snprintf(
name, 1024,
"res_integral_layout");
2172 snprintf(
name, 1024,
"res_vs_%s_layout", VSPARAMETER_NAMES[ii]);
2174 mCRes[ii] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2176 gStyle->SetOptFit(1);
2179 mPRes[ii][3] = createGarbageCollected<TPad>(
"p0",
"", 0.0, dy * 0, 0.5, dy * 1);
2180 mPRes[ii][3]->Draw();
2181 mPRes[ii][3]->SetRightMargin(0.04);
2182 mPRes[ii][4] = createGarbageCollected<TPad>(
"p1",
"", 0.5, dy * 0, 1.0, dy * 1);
2183 mPRes[ii][4]->Draw();
2184 mPRes[ii][4]->SetRightMargin(0.04);
2185 mPRes[ii][0] = createGarbageCollected<TPad>(
"p2",
"", 0.0, dy * 1, 1. / 3., dy * 2 - .001);
2186 mPRes[ii][0]->Draw();
2187 mPRes[ii][0]->SetRightMargin(0.04);
2188 mPRes[ii][0]->SetLeftMargin(0.15);
2189 mPRes[ii][1] = createGarbageCollected<TPad>(
"p3",
"", 1. / 3., dy * 1, 2. / 3., dy * 2 - .001);
2190 mPRes[ii][1]->Draw();
2191 mPRes[ii][1]->SetRightMargin(0.04);
2192 mPRes[ii][1]->SetLeftMargin(0.135);
2193 mPRes[ii][2] = createGarbageCollected<TPad>(
"p4",
"", 2. / 3., dy * 1, 1.0, dy * 2 - .001);
2194 mPRes[ii][2]->Draw();
2195 mPRes[ii][2]->SetRightMargin(0.06);
2196 mPRes[ii][2]->SetLeftMargin(0.135);
2198 mLRes[ii] = createGarbageCollected<TLegend>(0.9 - legendSpacingString * 1.45, 0.93 - (0.93 - 0.86) / 2. * (
float)ConfigNumInputs, 0.98, 0.949);
2199 SetLegend(mLRes[ii]);
2205 if (mQATasks & taskTrackingResPull) {
2206 for (int32_t ii = 0; ii < 7; ii++) {
2208 snprintf(
name, 1024,
"pull_integral_layout");
2210 snprintf(
name, 1024,
"pull_vs_%s_layout", VSPARAMETER_NAMES[ii]);
2212 mCPull[ii] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2214 gStyle->SetOptFit(1);
2217 mPPull[ii][3] = createGarbageCollected<TPad>(
"p0",
"", 0.0, dy * 0, 0.5, dy * 1);
2218 mPPull[ii][3]->Draw();
2219 mPPull[ii][3]->SetRightMargin(0.04);
2220 mPPull[ii][4] = createGarbageCollected<TPad>(
"p1",
"", 0.5, dy * 0, 1.0, dy * 1);
2221 mPPull[ii][4]->Draw();
2222 mPPull[ii][4]->SetRightMargin(0.04);
2223 mPPull[ii][0] = createGarbageCollected<TPad>(
"p2",
"", 0.0, dy * 1, 1. / 3., dy * 2 - .001);
2224 mPPull[ii][0]->Draw();
2225 mPPull[ii][0]->SetRightMargin(0.04);
2226 mPPull[ii][0]->SetLeftMargin(0.15);
2227 mPPull[ii][1] = createGarbageCollected<TPad>(
"p3",
"", 1. / 3., dy * 1, 2. / 3., dy * 2 - .001);
2228 mPPull[ii][1]->Draw();
2229 mPPull[ii][1]->SetRightMargin(0.04);
2230 mPPull[ii][1]->SetLeftMargin(0.135);
2231 mPPull[ii][2] = createGarbageCollected<TPad>(
"p4",
"", 2. / 3., dy * 1, 1.0, dy * 2 - .001);
2232 mPPull[ii][2]->Draw();
2233 mPPull[ii][2]->SetRightMargin(0.06);
2234 mPPull[ii][2]->SetLeftMargin(0.135);
2236 mLPull[ii] = createGarbageCollected<TLegend>(0.9 - legendSpacingString * 1.45, 0.93 - (0.93 - 0.86) / 2. * (
float)ConfigNumInputs, 0.98, 0.949);
2237 SetLegend(mLPull[ii]);
2243 if (mQATasks & taskClusterAttach) {
2244 for (int32_t
i = 0;
i < 3;
i++) {
2245 snprintf(
name, 1024,
"clusters_%s_layout", CLUSTER_TYPES[
i]);
2246 mCClust[
i] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2248 mPClust[
i] = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2250 float y1 =
i != 1 ? 0.77 : 0.27,
y2 =
i != 1 ? 0.9 : 0.42;
2251 mLClust[
i] = createGarbageCollected<TLegend>(
i == 2 ? 0.1 : (0.65 - legendSpacingString * 1.45),
y2 - (
y2 -
y1) * (ConfigNumInputs + (
i != 1) / 2.) + 0.005,
i == 2 ? (0.3 + legendSpacingString * 1.45) : 0.9,
y2);
2252 SetLegend(mLClust[
i]);
2257 if (mQATasks & taskTrackStatistics) {
2258 mCTrackPt = createGarbageCollected<TCanvas>(
"ctrackspt",
"ctrackspt", 0, 0, 700, 700. * 2. / 3.);
2260 mPTrackPt = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2262 mLTrackPt = createGarbageCollected<TLegend>(0.9 - legendSpacingString * 1.5, 0.93 - (0.93 - 0.86) / 2. * (
float)ConfigNumInputs, 0.98, 0.949);
2263 SetLegend(mLTrackPt,
true);
2265 for (int32_t
i = 0;
i < 2;
i++) {
2266 snprintf(
name, 2048,
"ctrackst0%d",
i);
2267 mCT0[
i] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2269 mPT0[
i] = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2271 mLT0[
i] = createGarbageCollected<TLegend>(0.9 - legendSpacingString * 1.45, 0.93 - (0.93 - 0.86) / 2. * (
float)ConfigNumInputs, 0.98, 0.949);
2274 snprintf(
name, 2048,
"cncl%d",
i);
2275 mCNCl[
i] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2277 mPNCl[
i] = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2279 mLNCl[
i] = createGarbageCollected<TLegend>(0.9 - legendSpacingString * 1.45, 0.93 - (0.93 - 0.86) / 2. * (
float)ConfigNumInputs, 0.98, 0.949);
2280 SetLegend(mLNCl[
i],
true);
2283 mCClXY = createGarbageCollected<TCanvas>(
"clxy",
"clxy", 0, 0, 700, 700. * 2. / 3.);
2285 mPClXY = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2289 if (mQATasks & taskClusterRejection) {
2290 for (int32_t
i = 0;
i < 3;
i++) {
2291 snprintf(
name, 2048,
"cnclrej%d",
i);
2292 mCClRej[
i] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2294 mPClRej[
i] = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2297 mCClRejP = createGarbageCollected<TCanvas>(
"cnclrejp",
"cnclrejp", 0, 0, 700, 700. * 2. / 3.);
2299 mPClRejP = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2303 if (mQATasks & taskClusterAttach) {
2304 for (int32_t
i = 0;
i < 4;
i++) {
2305 snprintf(
name, 2048,
"cpadrow%d",
i);
2306 mCPadRow[
i] = createGarbageCollected<TCanvas>(
name,
name, 0, 0, 700, 700. * 2. / 3.);
2308 mPPadRow[
i] = createGarbageCollected<TPad>(
"p0",
"", 0.0, 0.0, 1.0, 1.0);
2309 mPPadRow[
i]->Draw();
2314 if (mConfig.enableLocalOutput && !mConfig.inputHistogramsOnly && (mQATasks & taskTrackingEff) && mcPresent()) {
2315 GPUInfo(
"QA Stats: Eff: Tracks Prim %d (Eta %d, Pt %d) %f%% (%f%%) Sec %d (Eta %d, Pt %d) %f%% (%f%%) - Res: Tracks %d (Eta %d, Pt %d)", (int32_t)mEff[3][1][0][0]->GetEntries(), (int32_t)mEff[3][1][0][3]->GetEntries(), (int32_t)mEff[3][1][0][4]->GetEntries(),
2316 mEff[0][0][0][0]->GetSumOfWeights() / std::max(1., mEff[3][0][0][0]->GetSumOfWeights()), mEff[0][1][0][0]->GetSumOfWeights() / std::max(1., mEff[3][1][0][0]->GetSumOfWeights()), (int32_t)mEff[3][1][1][0]->GetEntries(), (int32_t)mEff[3][1][1][3]->GetEntries(),
2317 (int32_t)mEff[3][1][1][4]->GetEntries(), mEff[0][0][1][0]->GetSumOfWeights() / std::max(1., mEff[3][0][1][0]->GetSumOfWeights()), mEff[0][1][1][0]->GetSumOfWeights() / std::max(1., mEff[3][1][1][0]->GetSumOfWeights()), (int32_t)mRes2[0][0]->GetEntries(),
2318 (int32_t)mRes2[0][3]->GetEntries(), (int32_t)mRes2[0][4]->GetEntries());
2321 int32_t flagShowVsPtLog = (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) ? 1 : 0;
2323 if (mQATasks & taskTrackingEff) {
2325 for (int32_t ii = 0; ii < 5 + flagShowVsPtLog; ii++) {
2326 int32_t
i = ii == 5 ? 4 : ii;
2327 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2328 for (int32_t
j = 0;
j < 4;
j++) {
2329 if (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) {
2332 mPEff[ii][
j]->SetLogx();
2335 for (int32_t l = 0; l < 3; l++) {
2336 if (k == 0 && mConfig.inputHistogramsOnly == 0 && ii != 5) {
2339 auto oldLevel = gErrorIgnoreLevel;
2340 gErrorIgnoreLevel = kError;
2341 mEffResult[0][
j / 2][
j % 2][
i]->Divide(mEff[l][
j / 2][
j % 2][
i], mEff[5][
j / 2][
j % 2][
i],
"cl=0.683 b(1,1) mode");
2342 gErrorIgnoreLevel = oldLevel;
2343 mEff[3][
j / 2][
j % 2][
i]->Reset();
2344 mEff[3][
j / 2][
j % 2][
i]->Add(mEff[0][
j / 2][
j % 2][
i]);
2345 mEff[3][
j / 2][
j % 2][
i]->Add(mEff[1][
j / 2][
j % 2][
i]);
2346 mEff[3][
j / 2][
j % 2][
i]->Add(mEff[2][
j / 2][
j % 2][
i]);
2347 mEff[4][
j / 2][
j % 2][
i]->Reset();
2348 mEff[4][
j / 2][
j % 2][
i]->Add(mEff[0][
j / 2][
j % 2][
i]);
2349 mEff[4][
j / 2][
j % 2][
i]->Add(mEff[1][
j / 2][
j % 2][
i]);
2352 auto oldLevel = gErrorIgnoreLevel;
2353 gErrorIgnoreLevel = kError;
2354 mEffResult[l][
j / 2][
j % 2][
i]->Divide(mEff[l][
j / 2][
j % 2][
i], mEff[l == 1 ? 4 : 3][
j / 2][
j % 2][
i],
"cl=0.683 b(1,1) mode");
2355 gErrorIgnoreLevel = oldLevel;
2359 TGraphAsymmErrors* e = mEffResult[l][
j / 2][
j % 2][
i];
2361 if (!mConfig.inputHistogramsOnly && k == 0) {
2363 mEff[l][
j / 2][
j % 2][
i]->Write();
2366 mEff[3][
j / 2][
j % 2][
i]->Write();
2367 mEff[4][
j / 2][
j % 2][
i]->Write();
2370 }
else if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2373 e->SetTitle(EFFICIENCY_TITLES[
j]);
2374 e->GetYaxis()->SetTitle(
"(Efficiency)");
2375 e->GetXaxis()->SetTitle(XAXIS_TITLES[
i]);
2378 e->SetLineStyle(CONFIG_DASHED_MARKERS ? k + 1 : 1);
2380 if (qcout && !mConfig.shipToQCAsCanvas) {
2383 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2386 e->SetMarkerColor(kBlack);
2387 e->SetLineColor(colorNums[(k < 3 ? (l * 3 + k) : (k * 3 + l)) % COLORCOUNT]);
2388 e->GetHistogram()->GetYaxis()->SetRangeUser(-0.02, 1.02);
2389 e->Draw(k || l ?
"same P" :
"AP");
2392 mLEff[ii]->AddEntry(e, Form(
"%s%s", fname, EFF_NAMES[l]),
"l");
2395 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2399 ChangePadTitleSize(mPEff[ii][
j], 0.056);
2402 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2409 qcout->Add(mCEff[ii]);
2411 if (!mConfig.enableLocalOutput) {
2414 doPerfFigure(0.2, 0.295, 0.025);
2415 mCEff[ii]->Print(Form(
"%s/eff_vs_%s.pdf", mConfig.plotsDir.c_str(), VSPARAMETER_NAMES[ii]));
2416 if (mConfig.writeFileExt !=
"") {
2417 mCEff[ii]->Print(Form(
"%s/eff_vs_%s.%s", mConfig.plotsDir.c_str(), VSPARAMETER_NAMES[ii], mConfig.writeFileExt.c_str()));
2422 if (mQATasks & (taskTrackingRes | taskTrackingResPull)) {
2424 TH1D *resIntegral[5] = {}, *pullIntegral[5] = {};
2425 TCanvas* cfit =
nullptr;
2426 std::unique_ptr<TF1> customGaus = std::make_unique<TF1>(
"G",
"[0]*exp(-(x-[1])*(x-[1])/(2.*[2]*[2]))");
2427 for (int32_t p = 0;
p < 2;
p++) {
2428 if ((p == 0 && (mQATasks & taskTrackingRes) == 0) || (p == 1 && (mQATasks & taskTrackingResPull) == 0)) {
2431 for (int32_t ii = 0; ii < 5 + flagShowVsPtLog; ii++) {
2432 TCanvas* can =
p ? mCPull[ii] : mCRes[ii];
2433 TLegend* leg =
p ? mLPull[ii] : mLRes[ii];
2434 int32_t
i = ii == 5 ? 4 : ii;
2435 for (int32_t
j = 0;
j < 5;
j++) {
2436 TH2F*
src =
p ? mPull2[
j][
i] : mRes2[
j][
i];
2437 TH1F**
dst =
p ? mPull[
j][
i] : mRes[
j][
i];
2438 TH1D*& dstIntegral =
p ? pullIntegral[
j] : resIntegral[
j];
2439 TPad* pad =
p ? mPPull[ii][
j] : mPRes[ii][
j];
2441 if (!mConfig.inputHistogramsOnly && ii != 5) {
2442 if (cfit ==
nullptr) {
2443 cfit = createGarbageCollected<TCanvas>();
2447 TAxis* axis =
src->GetYaxis();
2448 int32_t nBins = axis->GetNbins();
2450 for (int32_t bin = 1; bin <= nBins; bin++) {
2451 int32_t bin0 = std::max(bin - integ, 0);
2452 int32_t bin1 = std::min(bin + integ, nBins);
2453 std::unique_ptr<TH1D> proj{
src->ProjectionX(
"proj", bin0, bin1)};
2454 proj->ClearUnderflowAndOverflow();
2455 if (proj->GetEntries()) {
2457 while (proj->GetMaximum() < 50 && rebin <
sizeof(RES_AXIS_BINS) /
sizeof(RES_AXIS_BINS[0])) {
2458 proj->Rebin(RES_AXIS_BINS[rebin - 1] / RES_AXIS_BINS[rebin]);
2462 if (proj->GetEntries() < 20 || proj->GetRMS() < 0.00001) {
2463 dst[0]->SetBinContent(bin, proj->GetRMS());
2464 dst[0]->SetBinError(bin, std::sqrt(proj->GetRMS()));
2465 dst[1]->SetBinContent(bin, proj->GetMean());
2466 dst[1]->SetBinError(bin, std::sqrt(proj->GetRMS()));
2468 proj->GetXaxis()->SetRange(0, 0);
2469 proj->GetXaxis()->SetRangeUser(std::max(proj->GetXaxis()->GetXmin(), proj->GetMean() - 3. * proj->GetRMS()), std::min(proj->GetXaxis()->GetXmax(), proj->GetMean() + 3. * proj->GetRMS()));
2470 bool forceLogLike = proj->GetMaximum() < 20;
2471 for (int32_t k = forceLogLike ? 2 : 0; k < 3; k++) {
2472 proj->Fit(
"gaus", forceLogLike || k == 2 ?
"sQl" : k ?
"sQww" :
"sQ");
2473 TF1* fitFunc = proj->GetFunction(
"gaus");
2475 if (k && !forceLogLike) {
2476 customGaus->SetParameters(fitFunc->GetParameter(0), fitFunc->GetParameter(1), fitFunc->GetParameter(2));
2477 proj->Fit(customGaus.get(),
"sQ");
2478 fitFunc = customGaus.get();
2481 const float sigma = fabs(fitFunc->GetParameter(2));
2482 dst[0]->SetBinContent(bin, sigma);
2483 dst[1]->SetBinContent(bin, fitFunc->GetParameter(1));
2484 dst[0]->SetBinError(bin, fitFunc->GetParError(2));
2485 dst[1]->SetBinError(bin, fitFunc->GetParError(1));
2487 const bool fail1 = sigma <= 0.f;
2488 const bool fail2 = fabs(proj->GetMean() -
dst[1]->GetBinContent(bin)) > std::min<float>(p ? PULL_AXIS : mConfig.nativeFitResolutions ? RES_AXES_NATIVE[
j] : RES_AXES[
j], 3.f * proj->GetRMS());
2489 const bool fail3 =
dst[0]->GetBinContent(bin) > 3.f * proj->GetRMS() ||
dst[0]->GetBinError(bin) > 1 ||
dst[1]->GetBinError(bin) > 1;
2490 const bool fail4 = fitFunc->GetParameter(0) < proj->GetMaximum() / 5.;
2491 const bool fail = fail1 || fail2 || fail3 || fail4;
2496 }
else if (k >= 2) {
2497 dst[0]->SetBinContent(bin, proj->GetRMS());
2498 dst[0]->SetBinError(bin, std::sqrt(proj->GetRMS()));
2499 dst[1]->SetBinContent(bin, proj->GetMean());
2500 dst[1]->SetBinError(bin, std::sqrt(proj->GetRMS()));
2505 dst[0]->SetBinContent(bin, 0.f);
2506 dst[0]->SetBinError(bin, 0.f);
2507 dst[1]->SetBinContent(bin, 0.f);
2508 dst[1]->SetBinError(bin, 0.f);
2512 dstIntegral =
src->ProjectionX(mConfig.nativeFitResolutions ? PARAMETER_NAMES_NATIVE[
j] : PARAMETER_NAMES[
j], 0, nBins + 1);
2514 while (dstIntegral->GetMaximum() < 50 && rebin <
sizeof(RES_AXIS_BINS) /
sizeof(RES_AXIS_BINS[0])) {
2515 dstIntegral->Rebin(RES_AXIS_BINS[rebin - 1] / RES_AXIS_BINS[rebin]);
2521 if (mConfig.inputHistogramsOnly) {
2522 dstIntegral = createGarbageCollected<TH1D>();
2524 dstIntegral->SetName(Form(p ?
"IntPull%s" :
"IntRes%s", VSPARAMETER_NAMES[
j]));
2525 dstIntegral->SetTitle(Form(p ?
"%s Pull" :
"%s Resolution",
p || mConfig.nativeFitResolutions ? PARAMETER_NAMES_NATIVE[
j] : PARAMETER_NAMES[
j]));
2527 if (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) {
2530 int32_t numColor = 0;
2531 float tmpMax = -1000.;
2532 float tmpMin = 1000.;
2534 for (int32_t l = 0; l < 2; l++) {
2535 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2537 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2540 if (nNewInput && k == 0 && ii != 5) {
2542 e->Scale(mConfig.nativeFitResolutions ? SCALE_NATIVE[
j] : SCALE[
j]);
2546 e->GetXaxis()->SetRangeUser(0.2, PT_MAX);
2547 }
else if (LOG_PT_MIN > 0 && ii == 5) {
2548 e->GetXaxis()->SetRangeUser(LOG_PT_MIN, PT_MAX);
2549 }
else if (ii == 5) {
2550 e->GetXaxis()->SetRange(1, 0);
2552 e->SetMinimum(-1111);
2553 e->SetMaximum(-1111);
2555 if (e->GetMaximum() > tmpMax) {
2556 tmpMax = e->GetMaximum();
2558 if (e->GetMinimum() < tmpMin) {
2559 tmpMin = e->GetMinimum();
2565 tmpSpan = tmpMax - tmpMin;
2566 tmpMax += tmpSpan * .02;
2567 tmpMin -= tmpSpan * .02;
2568 if (
j == 2 &&
i < 3) {
2569 tmpMax += tmpSpan * 0.13 * ConfigNumInputs;
2572 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2573 for (int32_t l = 0; l < 2; l++) {
2575 if (!mConfig.inputHistogramsOnly && k == 0) {
2576 e->SetTitle(Form(p ?
"%s Pull" :
"%s Resolution",
p || mConfig.nativeFitResolutions ? PARAMETER_NAMES_NATIVE[
j] : PARAMETER_NAMES[
j]));
2577 e->SetStats(kFALSE);
2580 mRes2[
j][
i]->SetOption(
"colz");
2581 mRes2[
j][
i]->Write();
2585 }
else if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2588 e->SetMaximum(tmpMax);
2589 e->SetMinimum(tmpMin);
2591 e->SetLineStyle(CONFIG_DASHED_MARKERS ? k + 1 : 1);
2593 e->GetYaxis()->SetTitle(p ? AXIS_TITLES_PULL[
j] : mConfig.nativeFitResolutions ? AXIS_TITLES_NATIVE[
j] : AXIS_TITLES[
j]);
2594 e->GetXaxis()->SetTitle(XAXIS_TITLES[
i]);
2595 if (LOG_PT_MIN > 0 && ii == 5) {
2596 e->GetXaxis()->SetRangeUser(LOG_PT_MIN, PT_MAX);
2600 e->GetYaxis()->SetTitleOffset(1.5);
2602 e->GetYaxis()->SetTitleOffset(1.4);
2604 if (qcout && !mConfig.shipToQCAsCanvas) {
2607 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2611 e->SetMarkerColor(kBlack);
2612 e->SetLineColor(colorNums[numColor++ % COLORCOUNT]);
2613 e->Draw(k || l ?
"same" :
"");
2616 leg->AddEntry(e, Form(
"%s%s", fname, l ?
"Mean" : (
p ?
"Pull" :
"Resolution")),
"l");
2620 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2629 ChangePadTitleSize(pad, 0.056);
2632 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2641 if (!mConfig.enableLocalOutput) {
2644 doPerfFigure(0.2, 0.295, 0.025);
2645 can->Print(Form(p ?
"%s/pull_vs_%s.pdf" :
"%s/res_vs_%s.pdf", mConfig.plotsDir.c_str(), VSPARAMETER_NAMES[ii]));
2646 if (mConfig.writeFileExt !=
"") {
2647 can->Print(Form(p ?
"%s/pull_vs_%s.%s" :
"%s/res_vs_%s.%s", mConfig.plotsDir.c_str(), VSPARAMETER_NAMES[ii], mConfig.writeFileExt.c_str()));
2653 for (int32_t p = 0;
p < 2;
p++) {
2654 if ((p == 0 && (mQATasks & taskTrackingRes) == 0) || (p == 1 && (mQATasks & taskTrackingResPull) == 0)) {
2657 TCanvas* can =
p ? mCPull[6] : mCRes[6];
2658 for (int32_t
i = 0;
i < 5;
i++) {
2659 TPad* pad =
p ? mPPull[6][
i] : mPRes[6][
i];
2660 TH1D* hist =
p ? pullIntegral[
i] : resIntegral[
i];
2661 int32_t numColor = 0;
2662 if (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) {
2665 if (!mConfig.inputHistogramsOnly && mcAvail) {
2667 if (e && e->GetEntries()) {
2668 e->Fit(
"gaus",
"sQ");
2673 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2675 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2678 e->SetMaximum(-1111);
2679 if (e->GetMaximum() > tmpMax) {
2680 tmpMax = e->GetMaximum();
2684 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2686 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2689 e->SetMaximum(tmpMax * 1.02);
2690 e->SetMinimum(tmpMax * -0.02);
2691 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
2694 if (qcout && !mConfig.shipToQCAsCanvas) {
2697 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2701 e->SetLineColor(colorNums[numColor++ % COLORCOUNT]);
2702 e->Draw(k == 0 ?
"" :
"same");
2704 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2712 if (!mConfig.enableLocalOutput) {
2716 can->Print(Form(p ?
"%s/pull_integral.pdf" :
"%s/res_integral.pdf", mConfig.plotsDir.c_str()));
2717 if (mConfig.writeFileExt !=
"") {
2718 can->Print(Form(p ?
"%s/pull_integral.%s" :
"%s/res_integral.%s", mConfig.plotsDir.c_str(), mConfig.writeFileExt.c_str()));
2723 uint64_t attachClusterCounts[N_CLS_HIST];
2724 if (mQATasks & taskClusterAttach) {
2726 if (mConfig.inputHistogramsOnly == 0) {
2727 for (int32_t
i = N_CLS_HIST;
i < N_CLS_TYPE * N_CLS_HIST - 1;
i++) {
2728 mClusters[
i]->Sumw2(
true);
2730 double totalVal = 0;
2731 if (!CLUST_HIST_INT_SUM) {
2732 for (int32_t
j = 0;
j < mClusters[N_CLS_HIST - 1]->GetXaxis()->GetNbins() + 2;
j++) {
2733 totalVal += mClusters[N_CLS_HIST - 1]->GetBinContent(
j);
2736 if (totalVal == 0.) {
2739 for (int32_t
i = 0;
i < N_CLS_HIST;
i++) {
2741 for (int32_t
j = 0;
j < mClusters[
i]->GetXaxis()->GetNbins() + 2;
j++) {
2742 val += mClusters[
i]->GetBinContent(
j);
2743 mClusters[2 * N_CLS_HIST - 1 +
i]->SetBinContent(
j,
val / totalVal);
2745 attachClusterCounts[
i] =
val;
2748 if (!CLUST_HIST_INT_SUM) {
2749 for (int32_t
i = 0;
i < N_CLS_HIST;
i++) {
2750 mClusters[2 * N_CLS_HIST - 1 +
i]->SetMaximum(1.02);
2751 mClusters[2 * N_CLS_HIST - 1 +
i]->SetMinimum(-0.02);
2755 for (int32_t
i = 0;
i < N_CLS_HIST - 1;
i++) {
2756 auto oldLevel = gErrorIgnoreLevel;
2757 gErrorIgnoreLevel = kError;
2758 mClusters[N_CLS_HIST +
i]->Divide(mClusters[
i], mClusters[N_CLS_HIST - 1], 1, 1,
"B");
2759 gErrorIgnoreLevel = oldLevel;
2760 mClusters[N_CLS_HIST +
i]->SetMinimum(-0.02);
2761 mClusters[N_CLS_HIST +
i]->SetMaximum(1.02);
2765 float tmpMax[2] = {0, 0}, tmpMin[2] = {0, 0};
2766 for (int32_t l = 0; l <= CLUST_HIST_INT_SUM; l++) {
2767 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2768 TH1* e = mClusters[l ? (N_CLS_TYPE * N_CLS_HIST - 2) : (N_CLS_HIST - 1)];
2769 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2772 e->SetMinimum(-1111);
2773 e->SetMaximum(-1111);
2775 e->GetXaxis()->SetRange(2, AXIS_BINS[4]);
2777 if (e->GetMaximum() > tmpMax[l]) {
2778 tmpMax[l] = e->GetMaximum();
2780 if (e->GetMinimum() < tmpMin[l]) {
2781 tmpMin[l] = e->GetMinimum();
2784 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2785 for (int32_t
i = 0;
i < N_CLS_HIST;
i++) {
2786 TH1* e = mClusters[l ? (2 * N_CLS_HIST - 1 +
i) :
i];
2787 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2790 e->SetMaximum(tmpMax[l] * 1.02);
2791 e->SetMinimum(tmpMax[l] * -0.02);
2796 for (int32_t
i = 0;
i < N_CLS_TYPE;
i++) {
2797 if (mConfig.enableLocalOutput || mConfig.shipToQCAsCanvas) {
2799 mPClust[
i]->SetLogx();
2801 int32_t
begin =
i == 2 ? (2 * N_CLS_HIST - 1) :
i == 1 ? N_CLS_HIST : 0;
2802 int32_t
end =
i == 2 ? (3 * N_CLS_HIST - 1) :
i == 1 ? (2 * N_CLS_HIST - 1) : N_CLS_HIST;
2803 int32_t numColor = 0;
2804 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2806 TH1* e = mClusters[
j];
2807 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2811 e->SetTitle(mConfig.plotsNoTitle ?
"" : CLUSTER_TITLES[
i]);
2812 e->GetYaxis()->SetTitle(
i == 0 ?
"Number of TPC clusters" :
i == 1 ?
"Fraction of TPC clusters" : CLUST_HIST_INT_SUM ?
"Total TPC clusters (integrated)" :
"Fraction of TPC clusters (integrated)");
2813 e->GetXaxis()->SetTitle(
"#it{p}_{Tmc} (GeV/#it{c})");
2814 e->GetXaxis()->SetTitleOffset(1.1);
2815 e->GetXaxis()->SetLabelOffset(-0.005);
2816 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
2819 e->SetStats(kFALSE);
2821 e->SetLineStyle(CONFIG_DASHED_MARKERS ?
j + 1 : 1);
2823 e->GetXaxis()->SetRange(2, AXIS_BINS[4]);
2825 if (qcout && !mConfig.shipToQCAsCanvas) {
2828 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2832 e->SetMarkerColor(kBlack);
2833 e->SetLineColor(colorNums[numColor++ % COLORCOUNT]);
2834 e->Draw(
j ==
end - 1 && k == 0 ?
"" :
"same");
2836 mLClust[
i]->AddEntry(e, Form(
"%s%s", fname, CLUSTER_NAMES[
j - begin]),
"l");
2839 if (ConfigNumInputs == 1) {
2840 TH1* e =
reinterpret_cast<TH1F*
>(mClusters[
begin + CL_att_adj]->Clone());
2841 e->Add(mClusters[begin + CL_prot], -1);
2842 if (qcout && !mConfig.shipToQCAsCanvas) {
2845 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2849 e->SetLineColor(colorNums[numColor++ % COLORCOUNT]);
2851 mLClust[
i]->AddEntry(e,
"Removed (Strategy A)",
"l");
2853 if (!mConfig.enableLocalOutput && !mConfig.shipToQCAsCanvas) {
2860 qcout->Add(mCClust[
i]);
2862 if (!mConfig.enableLocalOutput) {
2865 doPerfFigure(
i == 0 ? 0.37 : (
i == 1 ? 0.34 : 0.6), 0.295, 0.030);
2867 mCClust[
i]->Print(Form(
i == 2 ?
"%s/clusters_integral.pdf" :
i == 1 ?
"%s/clusters_relative.pdf" :
"%s/clusters.pdf", mConfig.plotsDir.c_str()));
2868 if (mConfig.writeFileExt !=
"") {
2869 mCClust[
i]->Print(Form(
i == 2 ?
"%s/clusters_integral.%s" :
i == 1 ?
"%s/clusters_relative.%s" :
"%s/clusters.%s", mConfig.plotsDir.c_str(), mConfig.writeFileExt.c_str()));
2873 for (int32_t
i = 0;
i < 4;
i++) {
2874 auto* e = mPadRow[
i];
2875 if (tout && !mConfig.inputHistogramsOnly) {
2879 e->SetOption(
"colz");
2880 std::string title =
"First Track Pad Row (p_{T} > 1GeV, N_{Cl} #geq " +
std::to_string(PADROW_CHECK_MINCLS);
2882 title +=
", row_{trk} > row_{MC} + 3, row_{MC} < 10";
2885 title +=
", #Phi_{Cl} < 0.15";
2889 e->SetTitle(mConfig.plotsNoTitle ?
"" : title.c_str());
2890 e->GetXaxis()->SetTitle(
i == 3 ?
"Local Occupancy" : (
i ?
"#Phi_{Cl} (sector)" :
"First MC Pad Row"));
2891 e->GetYaxis()->SetTitle(
"First Pad Row");
2894 static const constexpr char* PADROW_NAMES[4] = {
"MC",
"Phi",
"Phi1",
"Occ"};
2895 mCPadRow[
i]->Print(Form(
"%s/padRow%s.pdf", mConfig.plotsDir.c_str(), PADROW_NAMES[
i]));
2896 if (mConfig.writeFileExt !=
"") {
2897 mCPadRow[
i]->Print(Form(
"%s/padRow%s.%s", mConfig.plotsDir.c_str(), PADROW_NAMES[
i], mConfig.writeFileExt.c_str()));
2903 if ((mQATasks & taskClusterCounts) && !mHaveExternalHists && !mConfig.clusterRejectionHistograms && !mConfig.inputHistogramsOnly) {
2904 DoClusterCounts(attachClusterCounts);
2906 if ((qcout || tout) && (mQATasks & taskClusterCounts) && mConfig.clusterRejectionHistograms) {
2907 for (uint32_t
i = 0;
i < mHistClusterCount.size();
i++) {
2909 mHistClusterCount[
i]->Write();
2912 qcout->Add(mHistClusterCount[
i]);
2917 if (mQATasks & taskTrackStatistics) {
2920 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2922 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2925 e->SetMaximum(-1111);
2926 if (e->GetMaximum() > tmpMax) {
2927 tmpMax = e->GetMaximum();
2931 mPTrackPt->SetLogx();
2932 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2934 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2937 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
2940 e->SetMaximum(tmpMax * 1.02);
2941 e->SetMinimum(tmpMax * -0.02);
2942 e->SetStats(kFALSE);
2944 e->SetTitle(mConfig.plotsNoTitle ?
"" :
"Number of Tracks vs #it{p}_{T}");
2945 e->GetYaxis()->SetTitle(
"Number of Tracks");
2946 e->GetXaxis()->SetTitle(
"#it{p}_{T} (GeV/#it{c})");
2947 e->GetXaxis()->SetTitleOffset(1.2);
2951 e->SetMarkerColor(kBlack);
2952 e->SetLineColor(colorNums[k % COLORCOUNT]);
2953 e->Draw(k == 0 ?
"" :
"same");
2954 GetName(fname, k, mConfig.inputHistogramsOnly);
2955 mLTrackPt->AddEntry(e, Form(mConfig.inputHistogramsOnly ?
"%s" :
"%sTrack #it{p}_{T}", fname),
"l");
2958 doPerfFigure(0.63, 0.7, 0.030);
2960 mCTrackPt->Print(Form(
"%s/tracks.pdf", mConfig.plotsDir.c_str()));
2961 if (mConfig.writeFileExt !=
"") {
2962 mCTrackPt->Print(Form(
"%s/tracks.%s", mConfig.plotsDir.c_str(), mConfig.writeFileExt.c_str()));
2965 for (int32_t
i = 0;
i < 2;
i++) {
2967 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2969 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2972 e->SetMaximum(-1111);
2973 if (e->GetMaximum() > tmpMax) {
2974 tmpMax = e->GetMaximum();
2978 for (int32_t k = 0; k < ConfigNumInputs; k++) {
2980 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
2983 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
2986 e->SetMaximum(tmpMax * 1.02);
2987 e->SetMinimum(tmpMax * -0.02);
2988 e->SetStats(kFALSE);
2990 e->SetTitle(mConfig.plotsNoTitle ?
"" : (
i ?
"Track t_{0} resolution" :
"Track t_{0} distribution"));
2991 e->GetYaxis()->SetTitle(
"a.u.");
2992 e->GetXaxis()->SetTitle(
i ?
"t_{0} - t_{0, mc}" :
"t_{0}");
2996 e->SetMarkerColor(kBlack);
2997 e->SetLineColor(colorNums[k % COLORCOUNT]);
2998 e->Draw(k == 0 ?
"" :
"same");
2999 GetName(fname, k, mConfig.inputHistogramsOnly);
3000 mLT0[
i]->AddEntry(e, Form(mConfig.inputHistogramsOnly ?
"%s (%s)" :
"%sTrack t_{0} %s", fname,
i ?
"" :
"resolution"),
"l");
3003 doPerfFigure(0.63, 0.7, 0.030);
3005 mCT0[
i]->Print(Form(
"%s/t0%s.pdf", mConfig.plotsDir.c_str(),
i ?
"_res" :
""));
3006 if (mConfig.writeFileExt !=
"") {
3007 mCT0[
i]->Print(Form(
"%s/t0%s.%s", mConfig.plotsDir.c_str(),
i ?
"_res" :
"", mConfig.writeFileExt.c_str()));
3011 for (int32_t k = 0; k < ConfigNumInputs; k++) {
3013 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
3016 e->SetMaximum(-1111);
3017 if (e->GetMaximum() > tmpMax) {
3018 tmpMax = e->GetMaximum();
3022 for (int32_t k = 0; k < ConfigNumInputs; k++) {
3024 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
3027 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
3030 e->SetMaximum(tmpMax * 1.02);
3031 e->SetMinimum(tmpMax * -0.02);
3032 e->SetStats(kFALSE);
3034 e->SetTitle(mConfig.plotsNoTitle ?
"" : (
i ?
"Number of Rows with attached Cluster" :
"Number of Clusters"));
3035 e->GetYaxis()->SetTitle(
"a.u.");
3036 e->GetXaxis()->SetTitle(
i ?
"N_{Rows with Clusters}" :
"N_{Clusters}");
3040 e->SetMarkerColor(kBlack);
3041 e->SetLineColor(colorNums[k % COLORCOUNT]);
3042 e->Draw(k == 0 ?
"" :
"same");
3043 GetName(fname, k, mConfig.inputHistogramsOnly);
3044 mLNCl[
i]->AddEntry(e, Form(mConfig.inputHistogramsOnly ?
"%s" : (
i ?
"%sN_{Clusters}" :
"%sN_{Rows with Clusters}"), fname),
"l");
3047 doPerfFigure(0.6, 0.7, 0.030);
3049 mCNCl[
i]->Print(Form(
"%s/nClusters%s.pdf", mConfig.plotsDir.c_str(),
i ?
"_corrected" :
""));
3050 if (mConfig.writeFileExt !=
"") {
3051 mCNCl[
i]->Print(Form(
"%s/nClusters%s.%s", mConfig.plotsDir.c_str(),
i ?
"_corrected" :
"", mConfig.writeFileExt.c_str()));
3056 mClXY->SetOption(
"colz");
3059 mCClXY->Print(Form(
"%s/clustersXY.pdf", mConfig.plotsDir.c_str()));
3060 if (mConfig.writeFileExt !=
"") {
3061 mCClXY->Print(Form(
"%s/clustersXY.%s", mConfig.plotsDir.c_str(), mConfig.writeFileExt.c_str()));
3065 if (mQATasks & taskClusterRejection) {
3066 mClRej[2]->Divide(mClRej[1], mClRej[0]);
3068 for (int32_t
i = 0;
i < 3;
i++) {
3069 if (tout && !mConfig.inputHistogramsOnly) {
3073 mClRej[
i]->SetTitle(mConfig.plotsNoTitle ?
"" : REJECTED_NAMES[
i]);
3074 mClRej[
i]->SetOption(
"colz");
3077 mCClRej[
i]->Print(Form(
"%s/clustersRej%d%s.pdf", mConfig.plotsDir.c_str(),
i, REJECTED_NAMES[
i]));
3078 if (mConfig.writeFileExt !=
"") {
3079 mCClRej[
i]->Print(Form(
"%s/clustersRej%d%s.%s", mConfig.plotsDir.c_str(),
i, REJECTED_NAMES[
i], mConfig.writeFileExt.c_str()));
3084 for (int32_t k = 0; k < ConfigNumInputs; k++) {
3085 auto* tmp = mClRej[0];
3086 if (GetHist(tmp, tin, k, nNewInput) ==
nullptr) {
3089 TH1D* proj1 = tmp->ProjectionY(Form(
"clrejptmp1%d", k));
3090 proj1->SetDirectory(
nullptr);
3092 if (GetHist(tmp, tin, k, nNewInput) ==
nullptr) {
3095 TH1D* proj2 = tmp->ProjectionY(Form(
"clrejptmp2%d", k));
3096 proj2->SetDirectory(
nullptr);
3099 if (GetHist(e, tin, k, nNewInput) ==
nullptr) {
3102 e->Divide(proj2, proj1);
3103 if (tout && !mConfig.inputHistogramsOnly && k == 0) {
3108 e->SetMinimum(-0.02);
3109 e->SetMaximum(0.22);
3110 e->SetTitle(mConfig.plotsNoTitle ?
"" :
"Rejected Clusters");
3111 e->GetXaxis()->SetTitle(
"Pad Row");
3112 e->GetYaxis()->SetTitle(
"Rejected Clusters (fraction)");
3113 e->Draw(k == 0 ?
"" :
"same");
3115 mPClRejP->Print(Form(
"%s/clustersRejProjected.pdf", mConfig.plotsDir.c_str()));
3116 if (mConfig.writeFileExt !=
"") {
3117 mPClRejP->Print(Form(
"%s/clustersRejProjected.%s", mConfig.plotsDir.c_str(), mConfig.writeFileExt.c_str()));
3121 if (tout && !mConfig.inputHistogramsOnly && mConfig.writeMCLabels) {
3122 gInterpreter->GenerateDictionary(
"vector<vector<int32_t>>",
"");
3123 tout->WriteObject(&mcEffBuffer,
"mcEffBuffer");
3124 tout->WriteObject(&mcLabelBuffer,
"mcLabelBuffer");
3125 remove(
"AutoDict_vector_vector_int__.cxx");
3126 remove(
"AutoDict_vector_vector_int___cxx_ACLiC_dict_rdict.pcm");
3127 remove(
"AutoDict_vector_vector_int___cxx.d");
3128 remove(
"AutoDict_vector_vector_int___cxx.so");
3134 for (uint32_t
i = 0;
i < mConfig.compareInputs.size();
i++) {
3138 clearGarbagageCollector();
3140 GPUInfo(
"GPU TPC QA histograms have been written to pdf%s%s files", mConfig.writeFileExt ==
"" ?
"" :
" and ", mConfig.writeFileExt.c_str());
3141 gErrorIgnoreLevel = oldRootIgnoreLevel;
3145void GPUQA::PrintClusterCount(int32_t
mode, int32_t&
num,
const char*
name, uint64_t
n, uint64_t normalization)
3149 }
else if (
mode == 1) {
3151 snprintf(name2, 128,
"clusterCount%d_",
num);
3152 char*
ptr = name2 + strlen(name2);
3153 for (uint32_t
i = 0;
i < strlen(
name);
i++) {
3159 createHist(mHistClusterCount[
num], name2,
name, 1000, 0, mConfig.histMaxNClusters, 1000, 0, 100);
3160 }
else if (
mode == 0) {
3161 if (normalization && mConfig.enableLocalOutput) {
3162 for (uint32_t
i = 0;
i < 1 + (mTextDump !=
nullptr);
i++) {
3163 fprintf(
i ? mTextDump : stdout,
"\t%40s: %'12" PRIu64
" (%6.2f%%)\n",
name,
n, 100.f *
n / normalization);
3166 if (mConfig.clusterRejectionHistograms) {
3167 float ratio = 100.f *
n / std::max<uint64_t>(normalization, 1);
3168 mHistClusterCount[
num]->Fill(normalization, ratio, 1);
3174int32_t GPUQA::DoClusterCounts(uint64_t* attachClusterCounts, int32_t
mode)
3176 if (mConfig.enableLocalOutput && !mConfig.inputHistogramsOnly && mConfig.plotsDir !=
"") {
3177 mTextDump = fopen((mConfig.plotsDir +
"/clusterCounts.txt").c_str(),
"w+");
3180 if (mcPresent() && (mQATasks & taskClusterAttach) && attachClusterCounts) {
3181 for (int32_t
i = 0;
i < N_CLS_HIST;
i++) {
3182 PrintClusterCount(
mode,
num, CLUSTER_NAMES[
i], attachClusterCounts[
i], mClusterCounts.nTotal);
3184 PrintClusterCount(
mode,
num,
"Unattached", attachClusterCounts[N_CLS_HIST - 1] - attachClusterCounts[CL_att_adj], mClusterCounts.nTotal);
3185 PrintClusterCount(
mode,
num,
"Removed (Strategy A)", attachClusterCounts[CL_att_adj] - attachClusterCounts[CL_prot], mClusterCounts.nTotal);
3186 PrintClusterCount(
mode,
num,
"Unaccessible", mClusterCounts.nUnaccessible, mClusterCounts.nTotal);
3188 PrintClusterCount(
mode,
num,
"All Clusters", mClusterCounts.nTotal, mClusterCounts.nTotal);
3189 PrintClusterCount(
mode,
num,
"Used in Physics", mClusterCounts.nPhysics, mClusterCounts.nTotal);
3190 PrintClusterCount(
mode,
num,
"Protected", mClusterCounts.nProt, mClusterCounts.nTotal);
3191 PrintClusterCount(
mode,
num,
"Unattached", mClusterCounts.nUnattached, mClusterCounts.nTotal);
3192 PrintClusterCount(
mode,
num,
"Removed (Strategy A)", mClusterCounts.nTotal - mClusterCounts.nUnattached - mClusterCounts.nProt, mClusterCounts.nTotal);
3193 PrintClusterCount(
mode,
num,
"Removed (Strategy B)", mClusterCounts.nTotal - mClusterCounts.nProt, mClusterCounts.nTotal);
3196 PrintClusterCount(
mode,
num,
"Merged Loopers (Track Merging)", mClusterCounts.nMergedLooperConnected, mClusterCounts.nTotal);
3197 PrintClusterCount(
mode,
num,
"Merged Loopers (Afterburner)", mClusterCounts.nMergedLooperUnconnected, mClusterCounts.nTotal);
3198 PrintClusterCount(
mode,
num,
"Looping Legs (other)", mClusterCounts.nLoopers, mClusterCounts.nTotal);
3199 PrintClusterCount(
mode,
num,
"High Inclination Angle", mClusterCounts.nHighIncl, mClusterCounts.nTotal);
3200 PrintClusterCount(
mode,
num,
"Rejected", mClusterCounts.nRejected, mClusterCounts.nTotal);
3201 PrintClusterCount(
mode,
num,
"Tube (> 200 MeV)", mClusterCounts.nTube, mClusterCounts.nTotal);
3202 PrintClusterCount(
mode,
num,
"Tube (< 200 MeV)", mClusterCounts.nTube200, mClusterCounts.nTotal);
3203 PrintClusterCount(
mode,
num,
"Low Pt < 50 MeV", mClusterCounts.nLowPt, mClusterCounts.nTotal);
3204 PrintClusterCount(
mode,
num,
"Low Pt < 200 MeV", mClusterCounts.n200MeV, mClusterCounts.nTotal);
3206 if (mcPresent() && (mQATasks & taskClusterAttach)) {
3207 PrintClusterCount(
mode,
num,
"Tracks > 400 MeV", mClusterCounts.nAbove400, mClusterCounts.nTotal);
3208 PrintClusterCount(
mode,
num,
"Fake Removed (> 400 MeV)", mClusterCounts.nFakeRemove400, mClusterCounts.nAbove400);
3209 PrintClusterCount(
mode,
num,
"Full Fake Removed (> 400 MeV)", mClusterCounts.nFullFakeRemove400, mClusterCounts.nAbove400);
3210 PrintClusterCount(
mode,
num,
"Tracks < 40 MeV", mClusterCounts.nBelow40, mClusterCounts.nTotal);
3211 PrintClusterCount(
mode,
num,
"Fake Protect (< 40 MeV)", mClusterCounts.nFakeProtect40, mClusterCounts.nBelow40);
3213 if (mcPresent() && (mQATasks & taskTrackStatistics)) {
3214 PrintClusterCount(
mode,
num,
"Correctly Attached all-trk normalized", mClusterCounts.nCorrectlyAttachedNormalized, mClusterCounts.nTotal);
3215 PrintClusterCount(
mode,
num,
"Correctly Attached non-fake normalized", mClusterCounts.nCorrectlyAttachedNormalizedNonFake, mClusterCounts.nTotal);
3219 mTextDump =
nullptr;
3226 mTrackingScratchBuffer.resize((nBytes +
sizeof(mTrackingScratchBuffer[0]) - 1) /
sizeof(mTrackingScratchBuffer[0]));
3227 return mTrackingScratchBuffer.data();
std::vector< std::string > labels
A const (ready only) version of MCTruthContainer.
#define TRACK_EXPECTED_REFERENCE_X_DEFAULT
#define TRACK_EXPECTED_REFERENCE_X
Definition of the MCTrack class.
Definition of the Names Generator class.
Class for time synchronization of RawReader instances.
static const HBFUtils & Instance()
static constexpr int32_t NSECTORS
int32_t ReadO2MCData(const char *filename)
bool clusterRemovable(int32_t attach, bool prot) const
void * AllocateScratchBuffer(size_t nBytes)
void SetMCTrackRange(int32_t min, int32_t max)
mcLabelI_t GetMCTrackLabel(uint32_t trackId) const
int32_t DrawQAHistograms()
int32_t InitQA(int32_t tasks=0)
void DumpO2MCData(const char *filename) const
void RunQA(bool matchOnly=false)
static GPUROOTDump< T, Args... > getNew(const char *name1, Names... names)
static constexpr uint32_t NROWS
static constexpr uint32_t NSECTORS
static DigitizationContext * loadFromFile(std::string_view filename="")
GLfloat GLfloat GLfloat alpha
GLuint GLfloat GLfloat GLfloat GLfloat y1
GLuint const GLchar * name
GLuint GLuint GLfloat weight
GLuint GLsizei const GLchar * label
GLdouble GLdouble GLdouble z
@ tasksDefaultPostprocess
int int int float float float int nCl
constexpr int LHCBCPERTIMEBIN
Enum< T >::Iterator begin(Enum< T >)
struct o2::upgrades_utils::@469 tracks
structure to keep trigger-related info
std::string to_string(gsl::span< T, Size > span)
bool isValid(std::string alias)
int64_t differenceInBC(const InteractionRecord &other) const
bool mergedLooperConnected
bool mergedLooperUnconnected
std::tuple< std::vector< std::unique_ptr< TCanvas > >, std::vector< std::unique_ptr< TLegend > >, std::vector< std::unique_ptr< TPad > >, std::vector< std::unique_ptr< TLatex > >, std::vector< std::unique_ptr< TH1D > > > v
IR getFirstIRofTF(const IR &rec) const
get 1st IR of TF corresponding to the 1st sampled orbit (in MC)
unsigned int nClusters[constants::MAXSECTOR][constants::MAXGLOBALPADROW]
unsigned int nClustersTotal
unsigned int clusterOffset[constants::MAXSECTOR][constants::MAXGLOBALPADROW]
const ClusterNative * clustersLinear
o2::InteractionRecord ir(0, 0)
o2::InteractionRecord ir0(3, 5)