Belle II Software development
KLMTimeAlgorithm.cc
1/**************************************************************************
2 * basf2 (Belle II Analysis Software Framework) *
3 * Author: The Belle II Collaboration *
4 * *
5 * See git log for contributors and copyright holders. *
6 * This file is licensed under LGPL-3.0, see LICENSE.md. *
7 **************************************************************************/
8
9/* Own header. */
10#include <klm/calibration/KLMTimeAlgorithm.h>
11
12/* KLM headers. */
13#include <klm/dataobjects/bklm/BKLMElementNumbers.h>
14
15/* Basf2 headers. */
16#include <framework/database/Database.h>
17#include <framework/database/DBObjPtr.h>
18#include <framework/database/DBStore.h>
19#include <framework/dataobjects/EventMetaData.h>
20#include <framework/gearbox/Const.h>
21#include <framework/logging/Logger.h>
22
23/* ROOT headers. */
24#include <Math/MinimizerOptions.h>
25#include <Math/Vector3D.h>
26#include <TFile.h>
27#include <TFitResult.h>
28#include <TMinuit.h>
29#include <TROOT.h>
30#include <TString.h>
31#include <TTree.h>
32
33/* C++ headers. */
34#include <functional>
35#include <set>
36
37using namespace Belle2;
38using namespace ROOT::Math;
39
41const int c_NBinsTime = 100;
42
44const int c_NBinsDistance = 100;
45
47static double s_BinnedData[c_NBinsTime][c_NBinsDistance];
48
50static double s_LowerTimeBoundary = 0;
51
53static double s_UpperTimeBoundary = 0;
54
56static double s_StripLength = 0;
57
58static bool compareEventNumber(const std::pair<KLMChannelNumber, unsigned int>& pair1,
59 const std::pair<KLMChannelNumber, unsigned int>& pair2)
60{
61 return pair1.second < pair2.second;
62}
63
64static double timeDensity(const double x[2], const double* par)
65{
66 double polynomial, t0, gauss;
67 polynomial = par[0];
68 t0 = par[2] + par[4] * x[1];
69 gauss = par[1] / (sqrt(2.0 * M_PI) * par[3]) *
70 exp(-0.5 * pow((x[0] - t0) / par[3], 2));
71 return fabs(polynomial + gauss);
72}
73
74static void fcn(int& npar, double* grad, double& fval, double* par, int iflag)
75{
76 (void)npar;
77 (void)grad;
78 (void)iflag;
79 double x[2];
80 fval = 0;
81 for (int i = 0; i < c_NBinsTime; ++i) {
82 x[0] = s_LowerTimeBoundary +
83 (s_UpperTimeBoundary - s_LowerTimeBoundary) *
84 (double(i) + 0.5) / c_NBinsTime;
85 for (int j = 0; j < c_NBinsDistance; ++j) {
86 x[1] = s_StripLength * (double(j) + 0.5) / c_NBinsDistance;
87 double f = timeDensity(x, par);
88 if (s_BinnedData[i][j] == 0)
89 fval = fval + 2.0 * f;
90 else
91 fval = fval + 2.0 * (f - s_BinnedData[i][j] *
92 (1.0 - log(s_BinnedData[i][j] / f)));
93 }
94 }
95}
96
98 CalibrationAlgorithm("KLMTimeCollector")
99{
101 m_minimizerOptions = ROOT::Math::MinimizerOptions();
102}
103
107
109{
110 const std::vector<Calibration::ExpRun>& runs = getRunList();
111 int firstExperiment = runs[0].first;
112 int lastExperiment = runs[runs.size() - 1].first;
113 if (firstExperiment != lastExperiment) {
114 B2FATAL("Runs from different experiments are used "
115 "for KLM time calibration (single algorithm run).");
116 }
117 /* DataStore. */
119 StoreObjPtr<EventMetaData> eventMetaData;
120 eventMetaData.registerInDataStore();
122 /* Database. */
123 if (eventMetaData.isValid()) {
124 if (eventMetaData->getExperiment() != firstExperiment) {
125 B2FATAL("Runs from different experiments are used "
126 "for KLM time calibration (consecutive algorithm runs).");
127 }
128 eventMetaData->setExperiment(firstExperiment);
129 eventMetaData->setRun(runs[0].second);
130 } else {
131 eventMetaData.construct(1, runs[0].second, firstExperiment);
132 }
133 DBStore& dbStore = DBStore::Instance();
134 dbStore.update();
135 dbStore.updateEvent();
136 /*
137 * For calibration algorithms, the database is not initialized on class
138 * creation. Do not move the database object to class members.
139 */
140 DBObjPtr<BKLMGeometryPar> bklmGeometry;
143}
144
145// Version 1: Initial check only (don't load data)
147{
148 B2INFO("Read tree entries (initial data check only).");
149 std::shared_ptr<TTree> timeCalibrationData;
150 timeCalibrationData = getObjectPtr<TTree>("time_calibration_data");
151
152 int n = timeCalibrationData->GetEntries();
153 B2INFO(LogVar("Total number of digits:", n));
154
155 if (n < m_MinimalDigitNumber)
157
159}
160
161void KLMTimeAlgorithm::readCalibrationDataCounts(std::map<KLMChannelNumber, unsigned int>& eventCounts)
162{
163 B2INFO("Counting events per channel (lightweight scan)...");
164 Event event;
165 std::shared_ptr<TTree> timeCalibrationData;
166 timeCalibrationData = getObjectPtr<TTree>("time_calibration_data");
167 timeCalibrationData->SetBranchAddress("channelId", &event.channelId);
168
169 eventCounts.clear();
170
171 int n = timeCalibrationData->GetEntries();
172 for (int i = 0; i < n; ++i) {
173 timeCalibrationData->GetEntry(i);
174 eventCounts[event.channelId]++;
175 }
176
177 B2INFO("Event counting complete." << LogVar("Total events", n) << LogVar("Unique channels", eventCounts.size()));
178}
179
181 const std::vector<std::pair<KLMChannelNumber, unsigned int>>& channelsBKLM,
182 const std::vector<std::pair<KLMChannelNumber, unsigned int>>& channelsEKLM)
183{
184 B2INFO("Loading data for 2D fit (top 1000 channels from BKLM and EKLM)...");
185 Event event;
186 std::shared_ptr<TTree> timeCalibrationData;
187 timeCalibrationData = getObjectPtr<TTree>("time_calibration_data");
188 timeCalibrationData->SetBranchAddress("Run", &event.Run);
189 timeCalibrationData->SetBranchAddress("Event", &event.Events);
190 timeCalibrationData->SetBranchAddress("nTrack", &event.nTrack);
191 timeCalibrationData->SetBranchAddress("Track_Charge", &event.Track_Charge);
192
193 timeCalibrationData->SetBranchAddress("t0", &event.t0);
194 timeCalibrationData->SetBranchAddress("t0_uc", &event.t0_uc);
195 timeCalibrationData->SetBranchAddress("flyTime", &event.flyTime);
196 timeCalibrationData->SetBranchAddress("recTime", &event.recTime);
197 timeCalibrationData->SetBranchAddress("dist", &event.dist);
198 timeCalibrationData->SetBranchAddress("diffDistX", &event.diffDistX);
199 timeCalibrationData->SetBranchAddress("diffDistY", &event.diffDistY);
200 timeCalibrationData->SetBranchAddress("diffDistZ", &event.diffDistZ);
201 timeCalibrationData->SetBranchAddress("eDep", &event.eDep);
202 timeCalibrationData->SetBranchAddress("nPE", &event.nPE);
203 timeCalibrationData->SetBranchAddress("channelId", &event.channelId);
204 timeCalibrationData->SetBranchAddress("inRPC", &event.inRPC);
205 timeCalibrationData->SetBranchAddress("isFlipped", &event.isFlipped);
206 timeCalibrationData->SetBranchAddress("isGood", &event.isGood);
207 timeCalibrationData->SetBranchAddress("getADCcount", &event.getADCcount);
208
209 m_evts.clear();
210
211 // Build set of channels we need for 2D fit (top 1000 from each)
212 std::set<KLMChannelNumber> neededChannels;
213 int maxChannels = 1000;
214
215 for (size_t i = 0; i < channelsBKLM.size() && i < static_cast<size_t>(maxChannels); ++i) {
216 neededChannels.insert(channelsBKLM[i].first);
217 }
218 for (size_t i = 0; i < channelsEKLM.size() && i < static_cast<size_t>(maxChannels); ++i) {
219 neededChannels.insert(channelsEKLM[i].first);
220 }
221
222 int n = timeCalibrationData->GetEntries();
223 int loadedEvents = 0;
224
225 for (int i = 0; i < n; ++i) {
226 timeCalibrationData->GetEntry(i);
227
228 if (neededChannels.find(event.channelId) != neededChannels.end()) {
229 m_evts[event.channelId].push_back(event);
230 loadedEvents++;
231 }
232 }
233
234 B2INFO("2D fit data loaded." << LogVar("Events", loadedEvents) << LogVar("Channels", m_evts.size()));
235}
236
237void KLMTimeAlgorithm::readCalibrationDataBatch(std::function<bool(const KLMChannelIndex&)> channelFilter)
238{
239 B2INFO("Loading calibration data batch...");
240 Event event;
241 std::shared_ptr<TTree> timeCalibrationData;
242 timeCalibrationData = getObjectPtr<TTree>("time_calibration_data");
243 timeCalibrationData->SetBranchAddress("Run", &event.Run);
244 timeCalibrationData->SetBranchAddress("Event", &event.Events);
245 timeCalibrationData->SetBranchAddress("nTrack", &event.nTrack);
246 timeCalibrationData->SetBranchAddress("Track_Charge", &event.Track_Charge);
247
248 timeCalibrationData->SetBranchAddress("t0", &event.t0);
249 timeCalibrationData->SetBranchAddress("t0_uc", &event.t0_uc);
250 timeCalibrationData->SetBranchAddress("flyTime", &event.flyTime);
251 timeCalibrationData->SetBranchAddress("recTime", &event.recTime);
252 timeCalibrationData->SetBranchAddress("dist", &event.dist);
253 timeCalibrationData->SetBranchAddress("diffDistX", &event.diffDistX);
254 timeCalibrationData->SetBranchAddress("diffDistY", &event.diffDistY);
255 timeCalibrationData->SetBranchAddress("diffDistZ", &event.diffDistZ);
256 timeCalibrationData->SetBranchAddress("eDep", &event.eDep);
257 timeCalibrationData->SetBranchAddress("nPE", &event.nPE);
258 timeCalibrationData->SetBranchAddress("channelId", &event.channelId);
259 timeCalibrationData->SetBranchAddress("inRPC", &event.inRPC);
260 timeCalibrationData->SetBranchAddress("isFlipped", &event.isFlipped);
261 timeCalibrationData->SetBranchAddress("isGood", &event.isGood);
262 timeCalibrationData->SetBranchAddress("getADCcount", &event.getADCcount);
263
264 m_evts.clear();
265
266 int n = timeCalibrationData->GetEntries();
267 int loadedEvents = 0;
268
269 for (int i = 0; i < n; ++i) {
270 timeCalibrationData->GetEntry(i);
271
272 // Convert channel number to KLMChannelIndex using channelNumberToElementNumbers
273 int subdetector, section, sector, layer, plane, strip;
274 m_ElementNumbers->channelNumberToElementNumbers(
275 event.channelId, &subdetector, &section, &sector, &layer, &plane, &strip);
276 KLMChannelIndex klmChannel(subdetector, section, sector, layer, plane, strip);
277
278 if (channelFilter(klmChannel)) {
279 m_evts[event.channelId].push_back(event);
280 loadedEvents++;
281 }
282 }
283
284 B2INFO("Batch loaded." << LogVar("Events", loadedEvents) << LogVar("Channels", m_evts.size()));
285}
286
288{
289 if (m_mc) {
296 } else {
297 m_LowerTimeBoundaryRPC = -800.0;
298 m_UpperTimeBoundaryRPC = -600.0;
303 }
304
305 // Create directory structure for per-channel histograms
306 // This must be done here (not in setupDatabase) because m_outFile is created just before this function is called
308 TDirectory* dir_channels = m_outFile->mkdir("channels", "Per-channel histograms", true);
309
310 // BKLM directories
311 TDirectory* dir_bklm = dir_channels->mkdir("BKLM", "", true);
312 TString sectionName[2] = {"Backward", "Forward"};
313 TString planeName[2] = {"Z", "Phi"};
314
315 for (int iF = 0; iF < 2; ++iF) {
316 TDirectory* dir_section = dir_bklm->mkdir(sectionName[iF].Data(), "", true);
317 for (int iS = 0; iS < 8; ++iS) {
318 TDirectory* dir_sector = dir_section->mkdir(Form("Sector_%d", iS + 1), "", true);
319 for (int iL = 0; iL < 15; ++iL) {
320 TDirectory* dir_layer = dir_sector->mkdir(Form("Layer_%d", iL + 1), "", true);
321 for (int iP = 0; iP < 2; ++iP) {
322 m_channelHistDir_BKLM[iF][iS][iL][iP] = dir_layer->mkdir(Form("Plane_%s", planeName[iP].Data()), "", true);
323 }
324 }
325 }
326 }
327
328 // EKLM directories
329 TDirectory* dir_eklm = dir_channels->mkdir("EKLM", "", true);
330 for (int iF = 0; iF < 2; ++iF) {
331 TDirectory* dir_section = dir_eklm->mkdir(sectionName[iF].Data(), "", true);
332 for (int iS = 0; iS < 4; ++iS) {
333 TDirectory* dir_sector = dir_section->mkdir(Form("Sector_%d", iS + 1), "", true);
334 int maxLayer = 12 + 2 * iF; // 12 for backward, 14 for forward
335 for (int iL = 0; iL < maxLayer; ++iL) {
336 TDirectory* dir_layer = dir_sector->mkdir(Form("Layer_%d", iL + 1), "", true);
337 for (int iP = 0; iP < 2; ++iP) {
338 m_channelHistDir_EKLM[iF][iS][iL][iP] = dir_layer->mkdir(Form("Plane_%d", iP + 1), "", true);
339 }
340 }
341 }
342 }
343
344 m_outFile->cd(); // Return to root directory
345 B2INFO("Created directory structure for per-channel histograms.");
346 }
347
348 int nBin = 80;
349 int nBin_scint = 80;
350
351 TString iFstring[2] = {"Backward", "Forward"};
352 TString iPstring[2] = {"ZReadout", "PhiReadout"};
353 TString hn, ht;
354
355 h_diff = new TH1F("h_diff", "Position difference between bklmHit2d and extHit;position difference", 100, 0, 10);
356 h_calibrated = new TH1I("h_calibrated_summary", "h_calibrated_summary;calibrated or not", 3, 0, 3);
357 hc_calibrated = new TH1I("hc_calibrated_summary", "hc_calibrated_summary;calibrated or not", 3, 0, 3);
358
359 gre_time_channel_scint = new TGraphErrors();
360 gre_time_channel_rpc = new TGraphErrors();
361 gre_time_channel_scint_end = new TGraphErrors();
362
363 gr_timeShift_channel_scint = new TGraph();
364 gr_timeShift_channel_rpc = new TGraph();
365 gr_timeShift_channel_scint_end = new TGraph();
366
367 gre_ctime_channel_scint = new TGraphErrors();
368 gre_ctime_channel_rpc = new TGraphErrors();
369 gre_ctime_channel_scint_end = new TGraphErrors();
370
371 gr_timeRes_channel_scint = new TGraph();
372 gr_timeRes_channel_rpc = new TGraph();
373 gr_timeRes_channel_scint_end = new TGraph();
374
375 double maximalPhiStripLengthBKLM =
376 m_BKLMGeometry->getMaximalPhiStripLength();
377 double maximalZStripLengthBKLM =
378 m_BKLMGeometry->getMaximalZStripLength();
379 double maximalStripLengthEKLM =
380 m_EKLMGeometry->getMaximalStripLength() / CLHEP::cm * Unit::cm;
381
382 m_ProfileRpcPhi = new TProfile("hprf_rpc_phi_effC",
383 "Time over propagation length for RPCs (Phi_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]", 50, 0.0,
384 400.0);
385 m_ProfileRpcZ = new TProfile("hprf_rpc_z_effC",
386 "Time over propagation length for RPCs (Z_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]", 50, 0.0,
387 400.0);
388 m_ProfileBKLMScintillatorPhi = new TProfile("hprf_scint_phi_effC",
389 "Time over propagation length for scintillators (Phi_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
390 50, 0.0, maximalPhiStripLengthBKLM);
391 m_ProfileBKLMScintillatorZ = new TProfile("hprf_scint_z_effC",
392 "Time over propagation length for scintillators (Z_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
393 50, 0.0, maximalZStripLengthBKLM);
394 m_ProfileEKLMScintillatorPlane1 = new TProfile("hprf_scint_plane1_effC_end",
395 "Time over propagation length for scintillators (plane1, Endcap); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
396 50, 0.0, maximalStripLengthEKLM);
397 m_ProfileEKLMScintillatorPlane2 = new TProfile("hprf_scint_plane2_effC_end",
398 "Time over propagation length for scintillators (plane2, Endcap); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
399 50, 0.0, maximalStripLengthEKLM);
400
401 m_Profile2RpcPhi = new TProfile("hprf2_rpc_phi_effC",
402 "Time over propagation length for RPCs (Phi_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]", 50, 0.0,
403 400.0);
404 m_Profile2RpcZ = new TProfile("hprf2_rpc_z_effC",
405 "Time over propagation length for RPCs (Z_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]", 50, 0.0,
406 400.0);
407 m_Profile2BKLMScintillatorPhi = new TProfile("hprf2_scint_phi_effC",
408 "Time over propagation length for scintillators (Phi_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
409 50, 0.0, maximalPhiStripLengthBKLM);
410 m_Profile2BKLMScintillatorZ = new TProfile("hprf2_scint_z_effC",
411 "Time over propagation length for scintillators (Z_Readout); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
412 50, 0.0, maximalZStripLengthBKLM);
413 m_Profile2EKLMScintillatorPlane1 = new TProfile("hprf2_scint_plane1_effC_end",
414 "Time over propagation length for scintillators (plane1, Endcap); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
415 50, 0.0, maximalStripLengthEKLM);
416 m_Profile2EKLMScintillatorPlane2 = new TProfile("hprf2_scint_plane2_effC_end",
417 "Time over propagation length for scintillators (plane2, Endcap); propagation distance[cm]; T_rec-T_0-T_fly-'T_calibration'[ns]",
418 50, 0.0, maximalStripLengthEKLM);
419
420 h_time_rpc_tc = new TH1F("h_time_rpc_tc", "time distribution for RPC", nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
421 h_time_scint_tc = new TH1F("h_time_scint_tc", "time distribution for Scintillator", nBin_scint,
423 h_time_scint_tc_end = new TH1F("h_time_scint_tc_end", "time distribution for Scintillator (Endcap)", nBin_scint,
426
428 h_time_rpc = new TH1F("h_time_rpc", "time distribution for RPC; T_rec-T_0-T_fly-T_propagation[ns]", nBin, m_LowerTimeBoundaryRPC,
430 h_time_scint = new TH1F("h_time_scint", "time distribution for Scintillator; T_rec-T_0-T_fly-T_propagation[ns]", nBin_scint,
432 h_time_scint_end = new TH1F("h_time_scint_end", "time distribution for Scintillator (Endcap); T_rec-T_0-T_fly-T_propagation[ns]",
434
435 hc_time_rpc = new TH1F("hc_time_rpc", "Calibrated time distribution for RPC; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
437 hc_time_scint = new TH1F("hc_time_scint",
438 "Calibrated time distribution for Scintillator; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", nBin_scint,
441 hc_time_scint_end = new TH1F("hc_time_scint_end",
442 "Calibrated time distribution for Scintillator (Endcap); T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", nBin_scint,
444
445 int nBin_t0 = 100;
446 h_eventT0_rpc = new TH1F("h_eventT0_rpc",
447 "RPC: Event T0; T_{0}[ns]", nBin_t0, -100.0, 100.0);
448 h_eventT0_scint = new TH1F("h_eventT0_scint",
449 "BKLM scintillator: Event T0; T_{0}[ns]", nBin_t0, -100.0, 100.0);
450 h_eventT0_scint_end = new TH1F("h_eventT0_scint_end",
451 "EKLM scintillator: Event T0; T_{0}[ns]", nBin_t0, -100.0, 100.0);
452
453 // Corrected EventT0 distributions between 2 muon tracks in an event.
454 hc_eventT0_rpc = new TH1F("hc_eventT0_rpc",
455 "RPC: corrected Event T0; T_{0}^{+} - T_{0}^{-} [ns]", nBin_t0, -40.0, 40.0);
456 hc_eventT0_scint = new TH1F("hc_eventT0_scint",
457 "BKLM scintillator: corrected Event T0; T_{0}^{+} - T_{0}^{-} [ns]", nBin_t0, -40.0, 40.0);
458 hc_eventT0_scint_end = new TH1F("hc_eventT0_scint_end",
459 "EKLM scintillator: corrected Event T0; T_{0}^{+} - T_{0}^{-} [ns]", nBin_t0, -20.0, 20.0);
460
461 // ==== NEW DIAGNOSTIC PLOTS ====
462
463 // Hit multiplicity distributions
464 h_nHits_plus_rpc = new TH1F("h_nHits_plus_rpc",
465 "RPC: #mu^{+} hit multiplicity;N_{hits};Events", 100, 0, 100);
466 h_nHits_minus_rpc = new TH1F("h_nHits_minus_rpc",
467 "RPC: #mu^{-} hit multiplicity;N_{hits};Events", 100, 0, 100);
468 h_nHits_plus_scint = new TH1F("h_nHits_plus_scint",
469 "BKLM Scint: #mu^{+} hit multiplicity;N_{hits};Events", 30, 0, 30);
470 h_nHits_minus_scint = new TH1F("h_nHits_minus_scint",
471 "BKLM Scint: #mu^{-} hit multiplicity;N_{hits};Events", 30, 0, 30);
472 h_nHits_plus_scint_end = new TH1F("h_nHits_plus_scint_end",
473 "EKLM Scint: #mu^{+} hit multiplicity;N_{hits};Events", 30, 0, 30);
474 h_nHits_minus_scint_end = new TH1F("h_nHits_minus_scint_end",
475 "EKLM Scint: #mu^{-} hit multiplicity;N_{hits};Events", 30, 0, 30);
476
477 // ΔT0 vs variance weight v = 1/N+ + 1/N-
478 h2_deltaT0_vs_v_rpc = new TH2F("h2_deltaT0_vs_v_rpc",
479 "RPC: #DeltaT_{0} vs variance weight;v = 1/N^{+} + 1/N^{-};#DeltaT_{0} [ns]",
480 50, 0, 2.0, 100, -40, 40);
481 h2_deltaT0_vs_v_scint = new TH2F("h2_deltaT0_vs_v_scint",
482 "BKLM Scint: #DeltaT_{0} vs variance weight;v = 1/N^{+} + 1/N^{-};#DeltaT_{0} [ns]",
483 50, 0, 1.0, 100, -40, 40);
484 h2_deltaT0_vs_v_scint_end = new TH2F("h2_deltaT0_vs_v_scint_end",
485 "EKLM Scint: #DeltaT_{0} vs variance weight;v = 1/N^{+} + 1/N^{-};#DeltaT_{0} [ns]",
486 50, 0, 1.0, 100, -20, 20);
487
488 // Profile: RMS(ΔT0) vs v (should scale as √v if model is correct)
489 prof_deltaT0_rms_vs_v_rpc = new TProfile("prof_deltaT0_rms_vs_v_rpc",
490 "RPC: RMS(#DeltaT_{0}) vs v;v = 1/N^{+} + 1/N^{-};RMS(#DeltaT_{0}) [ns]",
491 20, 0, 2.0, "s"); // "s" option for RMS
492 prof_deltaT0_rms_vs_v_scint = new TProfile("prof_deltaT0_rms_vs_v_scint",
493 "BKLM Scint: RMS(#DeltaT_{0}) vs v;v = 1/N^{+} + 1/N^{-};RMS(#DeltaT_{0}) [ns]",
494 20, 0, 1.0, "s");
495 prof_deltaT0_rms_vs_v_scint_end = new TProfile("prof_deltaT0_rms_vs_v_scint_end",
496 "EKLM Scint: RMS(#DeltaT_{0}) vs v;v = 1/N^{+} + 1/N^{-};RMS(#DeltaT_{0}) [ns]",
497 20, 0, 1.0, "s");
498
499 // ΔT0 vs total hits
500 h2_deltaT0_vs_nhits_rpc = new TH2F("h2_deltaT0_vs_nhits_rpc",
501 "RPC: #DeltaT_{0} vs total hits;N^{+} + N^{-};#DeltaT_{0} [ns]",
502 50, 0, 200, 100, -40, 40);
503 h2_deltaT0_vs_nhits_scint = new TH2F("h2_deltaT0_vs_nhits_scint",
504 "BKLM Scint: #DeltaT_{0} vs total hits;N^{+} + N^{-};#DeltaT_{0} [ns]",
505 40, 0, 40, 100, -40, 40);
506 h2_deltaT0_vs_nhits_scint_end = new TH2F("h2_deltaT0_vs_nhits_scint_end",
507 "EKLM Scint: #DeltaT_{0} vs total hits;N^{+} + N^{-};#DeltaT_{0} [ns]",
508 40, 0, 40, 100, -20, 20);
509
510 // ΔT0 separated by hit multiplicity bins
511 hc_eventT0_rpc_lowN = new TH1F("hc_eventT0_rpc_lowN",
512 "RPC: #DeltaT_{0} (N^{+}+N^{-} < 10);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
513 hc_eventT0_rpc_midN = new TH1F("hc_eventT0_rpc_midN",
514 "RPC: #DeltaT_{0} (10 #leq N^{+}+N^{-} < 30);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
515 hc_eventT0_rpc_highN = new TH1F("hc_eventT0_rpc_highN",
516 "RPC: #DeltaT_{0} (N^{+}+N^{-} #geq 30);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
517
518 hc_eventT0_scint_lowN = new TH1F("hc_eventT0_scint_lowN",
519 "BKLM Scint: #DeltaT_{0} (N^{+}+N^{-} < 5);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
520 hc_eventT0_scint_midN = new TH1F("hc_eventT0_scint_midN",
521 "BKLM Scint: #DeltaT_{0} (5 #leq N^{+}+N^{-} < 15);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
522 hc_eventT0_scint_highN = new TH1F("hc_eventT0_scint_highN",
523 "BKLM Scint: #DeltaT_{0} (N^{+}+N^{-} #geq 15);#DeltaT_{0} [ns]", nBin_t0, -40.0, 40.0);
524
525 hc_eventT0_scint_end_lowN = new TH1F("hc_eventT0_scint_end_lowN",
526 "EKLM Scint: #DeltaT_{0} (N^{+}+N^{-} < 5);#DeltaT_{0} [ns]", nBin_t0, -20.0, 20.0);
527 hc_eventT0_scint_end_midN = new TH1F("hc_eventT0_scint_end_midN",
528 "EKLM Scint: #DeltaT_{0} (5 #leq N^{+}+N^{-} < 15);#DeltaT_{0} [ns]", nBin_t0, -20.0, 20.0);
529 hc_eventT0_scint_end_highN = new TH1F("hc_eventT0_scint_end_highN",
530 "EKLM Scint: #DeltaT_{0} (N^{+}+N^{-} #geq 15);#DeltaT_{0} [ns]", nBin_t0, -20.0, 20.0);
531
532 if (!m_saveAllPlots) {
533 B2INFO("Skipping debug histogram allocation (m_saveAllPlots = false)");
534 return; // Skip all debugging histogram allocation
535 }
536
537 for (int iF = 0; iF < 2; ++iF) {
538 hn = Form("h_timeF%d_rpc", iF);
539 ht = Form("Time distribution for RPC of %s; T_rec-T_0-T_fly-T_propagation[ns]", iFstring[iF].Data());
540 h_timeF_rpc[iF] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
541 hn = Form("h_timeF%d_scint", iF);
542 ht = Form("Time distribution for Scintillator of %s; T_rec-T_0-T_fly-T_propagation[ns]", iFstring[iF].Data());
543 h_timeF_scint[iF] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
545 hn = Form("h_timeF%d_scint_end", iF);
546 ht = Form("Time distribution for Scintillator of %s (Endcap); T_rec-T_0-T_fly-T_propagation[ns]", iFstring[iF].Data());
547 h_timeF_scint_end[iF] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
549
550 hn = Form("h2_timeF%d_rpc", iF);
551 ht = Form("Time distribution for RPC of %s; Sector Index; T_rec-T_0-T_fly-T_propagation[ns]", iFstring[iF].Data());
552 h2_timeF_rpc[iF] = new TH2F(hn.Data(), ht.Data(), 8, 0, 8, nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
553 hn = Form("h2_timeF%d_scint", iF);
554 ht = Form("Time distribution for Scintillator of %s; Sector Index; T_rec-T_0-T_fly-T_propagation[ns]", iFstring[iF].Data());
555 h2_timeF_scint[iF] = new TH2F(hn.Data(), ht.Data(), 8, 0, 8, nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
557 hn = Form("h2_timeF%d_scint_end", iF);
558 ht = Form("Time distribution for Scintillator of %s (Endcap); Sector Index; T_rec-T_0-T_fly-T_propagation[ns]",
559 iFstring[iF].Data());
560 h2_timeF_scint_end[iF] = new TH2F(hn.Data(), ht.Data(), 4, 0, 4, nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
562
563 hn = Form("hc_timeF%d_rpc", iF);
564 ht = Form("Calibrated time distribution for RPC of %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", iFstring[iF].Data());
565 hc_timeF_rpc[iF] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryCalibratedRPC, m_UpperTimeBoundaryCalibratedRPC);
566 hn = Form("hc_timeF%d_scint", iF);
567 ht = Form("Calibrated time distribution for Scintillator of %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
568 iFstring[iF].Data());
569 hc_timeF_scint[iF] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
571 hn = Form("hc_timeF%d_scint_end", iF);
572 ht = Form("Calibrated time distribution for Scintillator of %s (Endcap); T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
573 iFstring[iF].Data());
574 hc_timeF_scint_end[iF] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
576
577 hn = Form("h2c_timeF%d_rpc", iF);
578 ht = Form("Calibrated time distribution for RPC of %s; Sector Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
579 iFstring[iF].Data());
580 h2c_timeF_rpc[iF] = new TH2F(hn.Data(), ht.Data(), 8, 0, 8, nBin, m_LowerTimeBoundaryCalibratedRPC,
582 hn = Form("h2c_timeF%d_scint", iF);
583 ht = Form("Calibrated time distribution for Scintillator of %s; Sector Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
584 iFstring[iF].Data());
585 h2c_timeF_scint[iF] = new TH2F(hn.Data(), ht.Data(), 8, 0, 8, nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
587 hn = Form("h2c_timeF%d_scint_end", iF);
588 ht = Form("Calibrated time distribution for Scintillator of %s (Endcap) ; Sector Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
589 iFstring[iF].Data());
590 h2c_timeF_scint_end[iF] = new TH2F(hn.Data(), ht.Data(), 4, 0, 4, nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
592
593 for (int iS = 0; iS < 8; ++iS) {
594 // Barrel parts
595 hn = Form("h_timeF%d_S%d_scint", iF, iS);
596 ht = Form("Time distribution for Scintillator of Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]", iS, iFstring[iF].Data());
597 h_timeFS_scint[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
599 hn = Form("h_timeF%d_S%d_rpc", iF, iS);
600 ht = Form("Time distribution for RPC of Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]", iS, iFstring[iF].Data());
601 h_timeFS_rpc[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
602 hn = Form("h2_timeF%d_S%d", iF, iS);
603 ht = Form("Time distribution of Sector%d, %s; Layer Index; T_rec-T_0-T_fly-T_propagation[ns]", iS, iFstring[iF].Data());
604 h2_timeFS[iF][iS] = new TH2F(hn.Data(), ht.Data(), 15, 0, 15, nBin_scint, m_LowerTimeBoundaryRPC,
606
607 hn = Form("hc_timeF%d_S%d_scint", iF, iS);
608 ht = Form("Calibrated time distribution for Scintillator of Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", iS,
609 iFstring[iF].Data());
610 hc_timeFS_scint[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
612 hn = Form("hc_timeF%d_S%d_rpc", iF, iS);
613 ht = Form("Calibrated time distribution for RPC of Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", iS,
614 iFstring[iF].Data());
615 hc_timeFS_rpc[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedRPC,
617 hn = Form("h2c_timeF%d_S%d", iF, iS);
618 ht = Form("Calibrated time distribution of Sector%d, %s; Layer Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", iS,
619 iFstring[iF].Data());
620 h2c_timeFS[iF][iS] = new TH2F(hn.Data(), ht.Data(), 15, 0, 15, nBin_scint, m_LowerTimeBoundaryCalibratedRPC,
622
623 // Inner 2 layers --> Scintillators
624 for (int iL = 0; iL < 2; ++iL) {
625 hn = Form("h_timeF%d_S%d_L%d", iF, iS, iL);
626 ht = Form("Time distribution for Scintillator of Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]", iL, iS,
627 iFstring[iF].Data());
628 h_timeFSL[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
630 hn = Form("hc_timeF%d_S%d_L%d", iF, iS, iL);
631 ht = Form("Calibrated time distribution for Scintillator of Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
632 iL, iS, iFstring[iF].Data());
633 hc_timeFSL[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
635
636 for (int iP = 0; iP < 2; ++iP) {
637 hn = Form("h_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
638 ht = Form("Time distribution for Scintillator of %s, Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]",
639 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
640 h_timeFSLP[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
642 hn = Form("h2_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
643 ht = Form("Time distribution for Scintillator of %s, Layer%d, Sector%d, %s; Channel Index; T_rec-T_0-T_fly-T_propagation[ns]",
644 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
645 h2_timeFSLP[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 54, 0, 54, nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
647
648 hn = Form("hc_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
649 ht = Form("Calibrated time distribution for Scintillator of %s, Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
650 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
651 hc_timeFSLP[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
653 hn = Form("h2c_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
654 ht = Form("Calibrated time distribution for Scintillator of %s, Layer%d, Sector%d, %s; Channel Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
655 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
656 h2c_timeFSLP[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 54, 0, 54, nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
658 }
659 }
660
661 for (int iL = 2; iL < 15; ++iL) {
662 hn = Form("h_timeF%d_S%d_L%d", iF, iS, iL);
663 ht = Form("time distribution for RPC of Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]", iL, iS, iFstring[iF].Data());
664 h_timeFSL[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
665
666 hn = Form("hc_timeF%d_S%d_L%d", iF, iS, iL);
667 ht = Form("Calibrated time distribution for RPC of Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]", iL, iS,
668 iFstring[iF].Data());
669 hc_timeFSL[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryCalibratedRPC, m_UpperTimeBoundaryCalibratedRPC);
670
671 for (int iP = 0; iP < 2; ++iP) {
672 hn = Form("h_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
673 ht = Form("time distribution for RPC of %s, Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation[ns]", iPstring[iP].Data(), iL, iS,
674 iFstring[iF].Data());
675 h_timeFSLP[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
676
677 hn = Form("h2_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
678 ht = Form("time distribution for RPC of %s, Layer%d, Sector%d, %s; Channel Index; T_rec-T_0-T_fly-T_propagation[ns]",
679 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
680 h2_timeFSLP[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 48, 0, 48, nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
681
682 hn = Form("hc_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
683 ht = Form("Calibrated time distribution for RPC of %s, Layer%d, Sector%d, %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
684 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
685 hc_timeFSLP[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryCalibratedRPC,
687
688 hn = Form("h2c_timeF%d_S%d_L%d_P%d", iF, iS, iL, iP);
689 ht = Form("Calibrated time distribution for RPC of %s, Layer%d, Sector%d, %s; Channel Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
690 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
691 h2c_timeFSLP[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 48, 0, 48, nBin, m_LowerTimeBoundaryCalibratedRPC,
693 }
694 }
695 }
696 // Endcap part
697 int maxLay = 12 + 2 * iF;
698 for (int iS = 0; iS < 4; ++iS) {
699 hn = Form("h_timeF%d_S%d_scint_end", iF, iS);
700 ht = Form("Time distribution for Scintillator of Sector%d, %s (Endcap); T_rec-T_0-T_fly-T_propagation[ns]", iS,
701 iFstring[iF].Data());
702 h_timeFS_scint_end[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
704 hn = Form("h2_timeF%d_S%d_end", iF, iS);
705 ht = Form("Time distribution of Sector%d, %s (Endcap); Layer Index; T_rec-T_0-T_fly-T_propagation[ns]", iS, iFstring[iF].Data());
706 h2_timeFS_end[iF][iS] = new TH2F(hn.Data(), ht.Data(), maxLay, 0, maxLay, nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
708 hn = Form("hc_timeF%d_S%d_scint_end", iF, iS);
709 ht = Form("Calibrated time distribution for Scintillator of Sector%d (Endcap), %s; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
710 iS, iFstring[iF].Data());
711 hc_timeFS_scint_end[iF][iS] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
713 hn = Form("h2c_timeF%d_S%d_end", iF, iS);
714 ht = Form("Calibrated time distribution of Sector%d, %s (Endcap); Layer Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
715 iS, iFstring[iF].Data());
716 h2c_timeFS_end[iF][iS] = new TH2F(hn.Data(), ht.Data(), maxLay, 0, maxLay, nBin_scint,
719
720 for (int iL = 0; iL < maxLay; ++iL) {
721 hn = Form("h_timeF%d_S%d_L%d_end", iF, iS, iL);
722 ht = Form("Time distribution for Scintillator of Layer%d, Sector%d, %s (Endcap); T_rec-T_0-T_fly-T_propagation[ns]", iL, iS,
723 iFstring[iF].Data());
724 h_timeFSL_end[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
726 hn = Form("hc_timeF%d_S%d_L%d_end", iF, iS, iL);
727 ht = Form("Calibrated time distribution for Scintillator of Layer%d, Sector%d, %s (Endcap); T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
728 iL, iS, iFstring[iF].Data());
729 hc_timeFSL_end[iF][iS][iL] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
731
732 for (int iP = 0; iP < 2; ++iP) {
733 hn = Form("h_timeF%d_S%d_L%d_P%d_end", iF, iS, iL, iP);
734 ht = Form("Time distribution for Scintillator of %s, Layer%d, Sector%d, %s (Endcap); T_rec-T_0-T_fly-T_propagation[ns]",
735 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
736 h_timeFSLP_end[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
738
739 hn = Form("h2_timeF%d_S%d_L%d_P%d_end", iF, iS, iL, iP);
740 ht = Form("Time distribution for Scintillator of %s, Layer%d, Sector%d, %s (Endcap); Channel Index; T_rec-T_0-T_fly-T_propagation[ns]",
741 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
742 h2_timeFSLP_end[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 75, 0, 75, nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
744
745 hn = Form("hc_timeF%d_S%d_L%d_P%d_end", iF, iS, iL, iP);
746 ht = Form("Calibrated time distribution for Scintillator of %s, Layer%d, Sector%d, %s (Endcap); T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
747 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
748 hc_timeFSLP_end[iF][iS][iL][iP] = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
750
751 hn = Form("h2c_timeF%d_S%d_L%d_P%d_end", iF, iS, iL, iP);
752 ht = Form("Calibrated time distribution for Scintillator of %s, Layer%d, Sector%d, %s (Endcap); Channel Index; T_rec-T_0-T_fly-T_propagation-T_calibration[ns]",
753 iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
754 h2c_timeFSLP_end[iF][iS][iL][iP] = new TH2F(hn.Data(), ht.Data(), 75, 0, 75, nBin_scint,
757 }
758 }
759 }
760 }
761
762 // NOTE: Directory structure for per-channel histograms is created in createHistograms()
763 // because m_outFile is not yet created when setupDatabase() is called.
764}
765
767 TProfile* profileRpcPhi, TProfile* profileRpcZ,
768 TProfile* profileBKLMScintillatorPhi, TProfile* profileBKLMScintillatorZ,
769 TProfile* profileEKLMScintillatorPlane1,
770 TProfile* profileEKLMScintillatorPlane2, bool fill2dHistograms)
771{
772 B2INFO("Filling time-distance profiles" << (fill2dHistograms ? " with 2D histograms" : "") << " (batched processing)...");
773
774 TString iFstring[2] = {"Backward", "Forward"};
775 TString iPstring[2] = {"ZReadout", "PhiReadout"};
776
777 // Define the 6 batches (same as in calibrate())
778 auto isRPCBackward = [](const KLMChannelIndex & ch) {
779 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
780 ch.getLayer() >= BKLMElementNumbers::c_FirstRPCLayer &&
781 ch.getSection() == BKLMElementNumbers::c_BackwardSection;
782 };
783
784 auto isRPCForward = [](const KLMChannelIndex & ch) {
785 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
786 ch.getLayer() >= BKLMElementNumbers::c_FirstRPCLayer &&
787 ch.getSection() == BKLMElementNumbers::c_ForwardSection;
788 };
789
790 auto isBKLMScintillatorBackward = [](const KLMChannelIndex & ch) {
791 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
792 ch.getLayer() < BKLMElementNumbers::c_FirstRPCLayer &&
793 ch.getSection() == BKLMElementNumbers::c_BackwardSection;
794 };
795
796 auto isBKLMScintillatorForward = [](const KLMChannelIndex & ch) {
797 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
798 ch.getLayer() < BKLMElementNumbers::c_FirstRPCLayer &&
799 ch.getSection() == BKLMElementNumbers::c_ForwardSection;
800 };
801
802 auto isEKLMScintillatorBackward = [](const KLMChannelIndex & ch) {
803 return ch.getSubdetector() == KLMElementNumbers::c_EKLM &&
804 ch.getSection() == EKLMElementNumbers::c_BackwardSection;
805 };
806
807 auto isEKLMScintillatorForward = [](const KLMChannelIndex & ch) {
808 return ch.getSubdetector() == KLMElementNumbers::c_EKLM &&
809 ch.getSection() == EKLMElementNumbers::c_ForwardSection;
810 };
811
812 std::vector<std::pair<std::string, std::function<bool(const KLMChannelIndex&)>>> batches = {
813 {"RPC Backward", isRPCBackward},
814 {"RPC Forward", isRPCForward},
815 {"BKLM Scintillator Backward", isBKLMScintillatorBackward},
816 {"BKLM Scintillator Forward", isBKLMScintillatorForward},
817 {"EKLM Scintillator Backward", isEKLMScintillatorBackward},
818 {"EKLM Scintillator Forward", isEKLMScintillatorForward}
819 };
820
821 // Process each batch
822 for (const auto& batch : batches) {
823 B2INFO("Processing batch for profiles: " << batch.first);
824 readCalibrationDataBatch(batch.second);
825
826 // Temporary storage for per-channel 2D histograms (only if fill2dHistograms is true)
827 // Store pairs of (histogram, target directory) so we can write to correct folder
828 std::map<KLMChannelNumber, std::pair<TH2F*, TDirectory*>> tempHistBKLM;
829 std::map<KLMChannelNumber, std::pair<TH2F*, TDirectory*>> tempHistEKLM;
830
831 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
832 KLMChannelNumber channel = klmChannel.getKLMChannelNumber();
833
834 // Skip if not in current batch
835 if (!batch.second(klmChannel))
836 continue;
837
838 if (m_cFlag[channel] == ChannelCalibrationStatus::c_NotEnoughData)
839 continue;
840
841 if (m_evts.find(channel) == m_evts.end())
842 continue;
843
844 std::vector<struct Event> eventsChannel = m_evts[channel];
845 int iSub = klmChannel.getSubdetector();
846
847 // Create 2D histogram for this channel if needed
848 TH2F* hist2d = nullptr;
849 if (fill2dHistograms) {
850 if (iSub == KLMElementNumbers::c_BKLM) {
851 int iF = klmChannel.getSection();
852 int iS = klmChannel.getSector() - 1;
853 int iL = klmChannel.getLayer() - 1;
854 int iP = klmChannel.getPlane();
855 int iC = klmChannel.getStrip() - 1;
856
857 // Only create for scintillators (layers 0-1)
858 if (iL < 2) {
859 TString hn = Form("time_length_bklm_F%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
860 double stripLength = 200;
861 hist2d = new TH2F(hn.Data(),
862 "Time versus propagation length; "
863 "propagation distance[cm]; "
864 "T_rec-T_0-T_fly-'T_calibration'[ns]",
865 50, 0.0, stripLength,
868 tempHistBKLM[channel] = std::make_pair(hist2d, m_channelHistDir_BKLM[iF][iS][iL][iP]);
869 }
870 } else { // EKLM
871 int iF = klmChannel.getSection() - 1;
872 int iS = klmChannel.getSector() - 1;
873 int iL = klmChannel.getLayer() - 1;
874 int iP = klmChannel.getPlane() - 1;
875 int iC = klmChannel.getStrip() - 1;
876
877 TString hn = Form("time_length_eklm_F%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
878 double stripLength = m_EKLMGeometry->getStripLength(iC + 1) / CLHEP::cm * Unit::cm;
879 hist2d = new TH2F(hn.Data(),
880 "Time versus propagation length; "
881 "propagation distance[cm]; "
882 "T_rec-T_0-T_fly-'T_calibration'[ns]",
883 50, 0.0, stripLength,
886 tempHistEKLM[channel] = std::make_pair(hist2d, m_channelHistDir_EKLM[iF][iS][iL][iP]);
887 }
888 }
889
890 // Fill histograms
891 for (const Event& event : eventsChannel) {
892 double timeHit = event.time() - m_timeShift[channel];
893 if (m_useEventT0)
894 timeHit = timeHit - event.t0;
895 double distHit = event.dist;
896
897 if (timeHit <= -400e3)
898 continue;
899
900 if (iSub == KLMElementNumbers::c_BKLM) {
901 int iL = klmChannel.getLayer() - 1;
902 int iP = klmChannel.getPlane();
903
904 if (iL > 1) {
905 // RPC
906 if (iP) {
907 profileRpcPhi->Fill(distHit, timeHit);
908 } else {
909 profileRpcZ->Fill(distHit, timeHit);
910 }
911 } else {
912 // Scintillator
914 uint16_t Charge = event.getADCcount;
915 if (Charge <= 30 || Charge >= 320) {
916 continue;
917 }
918 }
919
920 if (hist2d)
921 hist2d->Fill(distHit, timeHit);
922
923 if (iP) {
924 profileBKLMScintillatorPhi->Fill(distHit, timeHit);
925 } else {
926 profileBKLMScintillatorZ->Fill(distHit, timeHit);
927 }
928 }
929 } else {
930 // EKLM
932 uint16_t Charge = event.getADCcount;
933 if (Charge <= 40 || Charge >= 350) {
934 continue;
935 }
936 }
937
938 int iP = klmChannel.getPlane() - 1;
939
940 if (hist2d)
941 hist2d->Fill(distHit, timeHit);
942
943 if (iP) {
944 profileEKLMScintillatorPlane1->Fill(distHit, timeHit);
945 } else {
946 profileEKLMScintillatorPlane2->Fill(distHit, timeHit);
947 }
948 }
949 }
950 }
951
952 // Write and delete 2D histograms for this batch
953 // Use m_saveChannelHists (not m_saveAllPlots) since these are per-channel histograms
954 // and the directories are created based on m_saveChannelHists
955 if (fill2dHistograms) {
956 for (auto& entry : tempHistBKLM) {
957 writeThenDelete_(entry.second.first, m_saveChannelHists, entry.second.second);
958 }
959 for (auto& entry : tempHistEKLM) {
960 writeThenDelete_(entry.second.first, m_saveChannelHists, entry.second.second);
961 }
962 }
963
964 m_evts.clear();
965 B2INFO("Batch processed and cleared: " << batch.first);
966 }
967
968 B2INFO("Time-distance profile filling complete.");
969}
970
972 const std::vector< std::pair<KLMChannelNumber, unsigned int> >& channels,
973 double& delay, double& delayError)
974{
975 std::vector<struct Event> eventsChannel;
976 int nFits = 1000;
977 int nConvergedFits = 0;
978 delay = 0;
979 delayError = 0;
980 if (nFits > (int)channels.size())
981 nFits = channels.size();
982 for (int i = 0; i < nFits; ++i) {
983 int subdetector, section, sector, layer, plane, strip;
984 m_ElementNumbers->channelNumberToElementNumbers(
985 channels[i].first, &subdetector, &section, &sector, &layer, &plane,
986 &strip);
987 if (subdetector == KLMElementNumbers::c_BKLM) {
988 s_LowerTimeBoundary = m_LowerTimeBoundaryScintillatorsBKLM;
989 s_UpperTimeBoundary = m_UpperTimeBoundaryScintillatorsBKLM;
990 const bklm::Module* module =
991 m_BKLMGeometry->findModule(section, sector, layer);
992 s_StripLength = module->getStripLength(plane, strip);
993 } else {
994 s_LowerTimeBoundary = m_LowerTimeBoundaryScintillatorsEKLM;
995 s_UpperTimeBoundary = m_UpperTimeBoundaryScintillatorsEKLM;
996 s_StripLength = m_EKLMGeometry->getStripLength(strip) /
997 CLHEP::cm * Unit::cm;
998 }
999 for (int j = 0; j < c_NBinsTime; ++j) {
1000 for (int k = 0; k < c_NBinsDistance; ++k)
1001 s_BinnedData[j][k] = 0;
1002 }
1003 eventsChannel = m_evts[channels[i].first];
1004 double averageTime = 0;
1005 for (const Event& event : eventsChannel) {
1006 double timeHit = event.time();
1007 if (m_useEventT0)
1008 timeHit = timeHit - event.t0;
1009
1010 if (timeHit <= -400e3) {
1011 continue;
1012 }
1013
1014 averageTime = averageTime + timeHit;
1015 int timeBin = std::floor((timeHit - s_LowerTimeBoundary) * c_NBinsTime /
1016 (s_UpperTimeBoundary - s_LowerTimeBoundary));
1017 if (timeBin < 0 || timeBin >= c_NBinsTime)
1018 continue;
1019 int distanceBin = std::floor(event.dist * c_NBinsDistance / s_StripLength);
1020 if (distanceBin < 0 || distanceBin >= c_NBinsDistance) {
1021 B2ERROR("The distance to SiPM is greater than the strip length.");
1022 continue;
1023 }
1024 s_BinnedData[timeBin][distanceBin] += 1;
1025 }
1026 averageTime = averageTime / eventsChannel.size();
1027 TMinuit minuit(5);
1028 minuit.SetPrintLevel(-1);
1029 int minuitResult;
1030 minuit.mnparm(0, "P0", 1, 0.001, 0, 0, minuitResult);
1031 minuit.mnparm(1, "N", 10, 0.001, 0, 0, minuitResult);
1032 minuit.mnparm(2, "T0", averageTime, 0.001, 0, 0, minuitResult);
1033 minuit.mnparm(3, "SIGMA", 10, 0.001, 0, 0, minuitResult);
1034 minuit.mnparm(4, "DELAY", 0.0, 0.001, 0, 0, minuitResult);
1035 minuit.SetFCN(fcn);
1036 minuit.mncomd("FIX 2 3 4 5", minuitResult);
1037 minuit.mncomd("MIGRAD 10000", minuitResult);
1038 minuit.mncomd("RELEASE 2", minuitResult);
1039 minuit.mncomd("MIGRAD 10000", minuitResult);
1040 minuit.mncomd("RELEASE 3", minuitResult);
1041 minuit.mncomd("MIGRAD 10000", minuitResult);
1042 minuit.mncomd("RELEASE 4", minuitResult);
1043 minuit.mncomd("MIGRAD 10000", minuitResult);
1044 minuit.mncomd("RELEASE 5", minuitResult);
1045 minuit.mncomd("MIGRAD 10000", minuitResult);
1046 /* Require converged fit with accurate error matrix. */
1047 if (minuit.fISW[1] != 3)
1048 continue;
1049 nConvergedFits++;
1050 double channelDelay, channelDelayError;
1051 minuit.GetParameter(4, channelDelay, channelDelayError);
1052 delay = delay + channelDelay;
1053 delayError = delayError + channelDelayError * channelDelayError;
1054 }
1055 delay = delay / nConvergedFits;
1056 delayError = sqrt(delayError) / (nConvergedFits - 1);
1057}
1058
1059void KLMTimeAlgorithm::writeThenDelete_(TH1* h, bool write, TDirectory* dir)
1060{
1061 if (h == nullptr)
1062 return;
1063 if (write && m_outFile) {
1064 // Follow the pattern from saveHist(): cd into directory, then SetDirectory, then Write
1065 if (dir) {
1066 dir->cd();
1067 h->SetDirectory(dir);
1068 } else {
1069 m_outFile->cd();
1070 h->SetDirectory(m_outFile);
1071 }
1072 h->Write();
1073 m_outFile->cd(); // Return to root directory
1074 }
1075 delete h;
1076}
1077
1078void KLMTimeAlgorithm::writeThenDelete_(TH2* h, bool write, TDirectory* dir)
1079{
1080 if (h == nullptr)
1081 return;
1082 if (write && m_outFile) {
1083 // Follow the pattern from saveHist(): cd into directory, then SetDirectory, then Write
1084 if (dir) {
1085 dir->cd();
1086 h->SetDirectory(dir);
1087 } else {
1088 m_outFile->cd();
1089 h->SetDirectory(m_outFile);
1090 }
1091 h->Write();
1092 m_outFile->cd(); // Return to root directory
1093 }
1094 delete h;
1095}
1096
1097bool KLMTimeAlgorithm::passesADCCut(const Event& event, int subdetector, int layer_0indexed) const
1098{
1099 // If charge restriction is disabled, accept all hits
1101 return true;
1102 }
1103
1104 uint16_t charge = event.getADCcount;
1105
1106 if (subdetector == KLMElementNumbers::c_BKLM) {
1107 // FIXED: Use constant for RPC check (0-indexed comparison)
1108 if (layer_0indexed >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1109 return true; // RPC - no ADC cut
1110 } else {
1111 return (charge > 30 && charge < 320); // Scintillator
1112 }
1113 } else {
1114 return (charge > 40 && charge < 350); // EKLM
1115 }
1116}
1117
1119{
1120 int channelId;
1121 gROOT->SetBatch(kTRUE);
1122 setupDatabase();
1127
1128 fcn_gaus = new TF1("fcn_gaus", "gaus");
1129 fcn_land = new TF1("fcn_land", "landau");
1130 fcn_pol1 = new TF1("fcn_pol1", "pol1");
1131 fcn_const = new TF1("fcn_const", "pol0");
1132
1133 // Initial validation only - DON'T load all data yet
1135 if (result != CalibrationAlgorithm::c_OK)
1136 return result;
1137
1138 /* Choose non-existing file name. */
1139 std::string name = "time_calibration.root";
1140 int i = 1;
1141 while (1) {
1142 struct stat buffer;
1143 if (stat(name.c_str(), &buffer) != 0)
1144 break;
1145 name = "time_calibration_" + std::to_string(i) + ".root";
1146 i = i + 1;
1147 if (i < 0)
1148 break;
1149 }
1150 m_outFile = new TFile(name.c_str(), "recreate");
1152
1153 std::vector<struct Event> eventsChannel;
1154 eventsChannel.clear();
1155 m_cFlag.clear();
1156 m_minimizerOptions.SetDefaultStrategy(2);
1157
1158 B2INFO("Counting events per channel...");
1159 std::map<KLMChannelNumber, unsigned int> eventCounts;
1160 readCalibrationDataCounts(eventCounts);
1161
1162 /* Sort channels by number of events and initialize flags. */
1163 std::vector< std::pair<KLMChannelNumber, unsigned int> > channelsBKLM;
1164 std::vector< std::pair<KLMChannelNumber, unsigned int> > channelsEKLM;
1165 KLMChannelIndex klmChannels;
1166
1167 for (KLMChannelIndex& klmChannel : klmChannels) {
1168 KLMChannelNumber channel = klmChannel.getKLMChannelNumber();
1169 m_cFlag[channel] = ChannelCalibrationStatus::c_NotEnoughData;
1170
1171 if (eventCounts.find(channel) == eventCounts.end())
1172 continue;
1173
1174 int nEvents = eventCounts[channel];
1175 if (nEvents < m_lower_limit_counts) {
1176 B2WARNING("Not enough calibration data collected."
1177 << LogVar("channel", channel)
1178 << LogVar("number of digit", nEvents));
1179 continue;
1180 }
1181
1182 m_cFlag[channel] = ChannelCalibrationStatus::c_FailedFit;
1183
1184 if (klmChannel.getSubdetector() == KLMElementNumbers::c_BKLM &&
1185 klmChannel.getLayer() < BKLMElementNumbers::c_FirstRPCLayer) {
1186 channelsBKLM.push_back(std::pair<KLMChannelNumber, unsigned int>(channel, nEvents));
1187 }
1188 if (klmChannel.getSubdetector() == KLMElementNumbers::c_EKLM) {
1189 channelsEKLM.push_back(std::pair<KLMChannelNumber, unsigned int>(channel, nEvents));
1190 }
1191 }
1192
1193 std::sort(channelsBKLM.begin(), channelsBKLM.end(), compareEventNumber);
1194 std::sort(channelsEKLM.begin(), channelsEKLM.end(), compareEventNumber);
1195
1196 /* Two-dimensional fit using top channels only. */
1197 double delayBKLM, delayBKLMError;
1198 double delayEKLM, delayEKLMError;
1199
1200 /* Load data for 2D fit channels only. */
1201 readCalibrationDataFor2DFit(channelsBKLM, channelsEKLM);
1202 timeDistance2dFit(channelsBKLM, delayBKLM, delayBKLMError);
1203 timeDistance2dFit(channelsEKLM, delayEKLM, delayEKLMError);
1204 m_evts.clear();
1205
1206 B2INFO("2D fits complete, data cleared.");
1207
1208 /* Define processing batches for channel calibration. */
1209 auto isRPCBackward = [](const KLMChannelIndex & ch) {
1210 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
1211 ch.getLayer() >= BKLMElementNumbers::c_FirstRPCLayer &&
1212 ch.getSection() == BKLMElementNumbers::c_BackwardSection;
1213 };
1214
1215 auto isRPCForward = [](const KLMChannelIndex & ch) {
1216 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
1217 ch.getLayer() >= BKLMElementNumbers::c_FirstRPCLayer &&
1218 ch.getSection() == BKLMElementNumbers::c_ForwardSection;
1219 };
1220
1221 auto isBKLMScintillatorBackward = [](const KLMChannelIndex & ch) {
1222 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
1223 ch.getLayer() < BKLMElementNumbers::c_FirstRPCLayer &&
1224 ch.getSection() == BKLMElementNumbers::c_BackwardSection;
1225 };
1226
1227 auto isBKLMScintillatorForward = [](const KLMChannelIndex & ch) {
1228 return ch.getSubdetector() == KLMElementNumbers::c_BKLM &&
1229 ch.getLayer() < BKLMElementNumbers::c_FirstRPCLayer &&
1230 ch.getSection() == BKLMElementNumbers::c_ForwardSection;
1231 };
1232
1233 auto isEKLMScintillatorBackward = [](const KLMChannelIndex & ch) {
1234 return ch.getSubdetector() == KLMElementNumbers::c_EKLM &&
1235 ch.getSection() == EKLMElementNumbers::c_BackwardSection;
1236 };
1237
1238 auto isEKLMScintillatorForward = [](const KLMChannelIndex & ch) {
1239 return ch.getSubdetector() == KLMElementNumbers::c_EKLM &&
1240 ch.getSection() == EKLMElementNumbers::c_ForwardSection;
1241 };
1242
1243 std::vector<std::pair<std::string, std::function<bool(const KLMChannelIndex&)>>> batches = {
1244 {"RPC Backward", isRPCBackward},
1245 {"RPC Forward", isRPCForward},
1246 {"BKLM Scintillator Backward", isBKLMScintillatorBackward},
1247 {"BKLM Scintillator Forward", isBKLMScintillatorForward},
1248 {"EKLM Scintillator Backward", isEKLMScintillatorBackward},
1249 {"EKLM Scintillator Forward", isEKLMScintillatorForward}
1250 };
1251
1252 /**********************************
1253 * FIRST LOOP (BATCHED)
1254 * Fill global histograms to compute global means
1255 **********************************/
1256 B2INFO("First loop: Computing global statistics (batched processing)...");
1257
1258 TString iFstring[2] = {"Backward", "Forward"};
1259 TString iPstring[2] = {"ZReadout", "PhiReadout"};
1260 int nBin = 80;
1261 int nBin_scint = 80;
1262
1263 for (const auto& batch : batches) {
1264 B2INFO("Processing batch for global stats: " << batch.first);
1265 readCalibrationDataBatch(batch.second);
1266
1267 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1268 channelId = klmChannel.getKLMChannelNumber();
1269
1270 if (!batch.second(klmChannel))
1271 continue;
1272
1273 if (m_cFlag[channelId] == ChannelCalibrationStatus::c_NotEnoughData)
1274 continue;
1275
1276 if (m_evts.find(channelId) == m_evts.end())
1277 continue;
1278
1279 eventsChannel = m_evts[channelId];
1280 int iSub = klmChannel.getSubdetector();
1281 int iL = (iSub == KLMElementNumbers::c_BKLM) ? klmChannel.getLayer() - 1 : -1;
1282
1283 // Fill global histograms only
1284 for (const Event& event : eventsChannel) {
1285 // Apply ADC cut early
1286 if (!passesADCCut(event, iSub, iL))
1287 continue;
1288
1289 XYZVector diffD = XYZVector(event.diffDistX, event.diffDistY, event.diffDistZ);
1290 h_diff->Fill(diffD.R());
1291
1292 double timeHit = event.time();
1293 if (m_useEventT0)
1294 timeHit = timeHit - event.t0;
1295
1296 if (timeHit <= -400e3)
1297 continue;
1298
1299 if (iSub == KLMElementNumbers::c_BKLM) {
1300 // FIXED: Use constant instead of hardcoded value
1301 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1302 h_time_rpc_tc->Fill(timeHit);
1303 } else {
1304 h_time_scint_tc->Fill(timeHit);
1305 }
1306 } else {
1307 h_time_scint_tc_end->Fill(timeHit);
1308 }
1309 }
1310 }
1311
1312 m_evts.clear();
1313 B2INFO("Batch processed and cleared: " << batch.first);
1314 }
1315
1316 // Compute global means
1317 m_timeShift.clear();
1318 double tmpMean_rpc_global = h_time_rpc_tc->GetMean();
1319 double tmpMean_scint_global = h_time_scint_tc->GetMean();
1320 double tmpMean_scint_global_end = h_time_scint_tc_end->GetMean();
1321
1322 B2INFO("Global Mean for Raw." << LogVar("RPC", tmpMean_rpc_global)
1323 << LogVar("Scint BKLM", tmpMean_scint_global)
1324 << LogVar("Scint EKLM", tmpMean_scint_global_end));
1325
1326 /**********************************
1327 * SECOND PASS (BATCHED)
1328 * Compute per-channel time shifts
1329 **********************************/
1330 B2INFO("Second pass: Computing per-channel time shifts (batched processing)...");
1331
1332 for (const auto& batch : batches) {
1333 B2INFO("Processing batch for time shifts: " << batch.first);
1334 readCalibrationDataBatch(batch.second);
1335
1336 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1337 channelId = klmChannel.getKLMChannelNumber();
1338
1339 if (!batch.second(klmChannel))
1340 continue;
1341
1342 if (m_cFlag[channelId] == ChannelCalibrationStatus::c_NotEnoughData)
1343 continue;
1344
1345 if (m_evts.find(channelId) == m_evts.end())
1346 continue;
1347
1348 eventsChannel = m_evts[channelId];
1349 int iSub = klmChannel.getSubdetector();
1350 int iF, iS, iL, iP, iC;
1351
1352 if (iSub == KLMElementNumbers::c_BKLM) {
1353 iF = klmChannel.getSection();
1354 iS = klmChannel.getSector() - 1;
1355 iL = klmChannel.getLayer() - 1;
1356 iP = klmChannel.getPlane();
1357 iC = klmChannel.getStrip() - 1;
1358 } else {
1359 iF = klmChannel.getSection() - 1;
1360 iS = klmChannel.getSector() - 1;
1361 iL = klmChannel.getLayer() - 1;
1362 iP = klmChannel.getPlane() - 1;
1363 iC = klmChannel.getStrip() - 1;
1364 }
1365
1366 // Create and fill temp histogram
1367 TString hn, ht;
1368 TH1F* h_temp_tc = nullptr;
1369
1370 if (iSub == KLMElementNumbers::c_BKLM) {
1371 // FIXED: Use constant instead of hardcoded value
1372 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1373 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d_tc", iF, iS, iL, iP, iC);
1374 ht = Form("Time distribution for RPC of Channel%d, %s, Layer%d, Sector%d, %s",
1375 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1376 h_temp_tc = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
1377 } else {
1378 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d_tc", iF, iS, iL, iP, iC);
1379 ht = Form("time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s",
1380 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1381 h_temp_tc = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
1383 }
1384 } else {
1385 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d_tc_end", iF, iS, iL, iP, iC);
1386 ht = Form("Time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s (Endcap)",
1387 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1388 h_temp_tc = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
1390 }
1391
1392 for (const Event& event : eventsChannel) {
1393 // Add ADC cut
1394 if (!passesADCCut(event, iSub, iL))
1395 continue;
1396
1397 double timeHit = event.time();
1398 if (m_useEventT0)
1399 timeHit = timeHit - event.t0;
1400 if (timeHit <= -400e3)
1401 continue;
1402 h_temp_tc->Fill(timeHit);
1403 }
1404
1405 h_temp_tc->Fit(fcn_gaus, "LESQ");
1406 double tmpMean_channel = fcn_gaus->GetParameter(1);
1407
1408 if (iSub == KLMElementNumbers::c_BKLM) {
1409 // FIXED: Use constant instead of hardcoded value
1410 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1411 m_timeShift[channelId] = tmpMean_channel - tmpMean_rpc_global;
1412 } else {
1413 m_timeShift[channelId] = tmpMean_channel - tmpMean_scint_global;
1414 }
1415 } else {
1416 m_timeShift[channelId] = tmpMean_channel - tmpMean_scint_global_end;
1417 }
1418
1419 delete h_temp_tc;
1420 }
1421
1422 m_evts.clear();
1423 B2INFO("Batch processed and cleared: " << batch.first);
1424 }
1425
1426 delete h_time_scint_tc;
1427 delete h_time_scint_tc_end;
1428 delete h_time_rpc_tc;
1429 B2INFO("Effective Light m_timeShift obtained.");
1430
1431 // NOTE: fillTimeDistanceProfiles also needs batching - user will handle separately
1436
1437 B2INFO("Effective light speed fitting.");
1438
1439 // Fit the RPC profiles (for diagnostics), but use fixed values if configured
1440 m_ProfileRpcPhi->Fit("fcn_pol1", "EMQ");
1441 double fittedDelayRPCPhi = fcn_pol1->GetParameter(1);
1442 double e_slope_rpc_phi = fcn_pol1->GetParError(1);
1443
1444 m_ProfileRpcZ->Fit("fcn_pol1", "EMQ");
1445 double fittedDelayRPCZ = fcn_pol1->GetParameter(1);
1446 double e_slope_rpc_z = fcn_pol1->GetParError(1);
1447
1448 // Use fixed RPC delay if configured (per BELLE2-NOTE-TE-2021-015: c_eff = 0.5c)
1449 double delayRPCPhi, delayRPCZ;
1450 if (m_useFixedRPCDelay) {
1451 delayRPCPhi = m_fixedRPCDelay;
1452 delayRPCZ = m_fixedRPCDelay;
1453 B2INFO("Using fixed RPC propagation delay: " << m_fixedRPCDelay << " ns/cm (c_eff = 0.5c)"
1454 << LogVar("Fitted phi (not used)", fittedDelayRPCPhi)
1455 << LogVar("Fitted Z (not used)", fittedDelayRPCZ));
1456 } else {
1457 delayRPCPhi = fittedDelayRPCPhi;
1458 delayRPCZ = fittedDelayRPCZ;
1459 }
1460
1461 m_ProfileBKLMScintillatorPhi->Fit("fcn_pol1", "EMQ");
1462 double slope_scint_phi = fcn_pol1->GetParameter(1);
1463 double e_slope_scint_phi = fcn_pol1->GetParError(1);
1464
1465 m_ProfileBKLMScintillatorZ->Fit("fcn_pol1", "EMQ");
1466 double slope_scint_z = fcn_pol1->GetParameter(1);
1467 double e_slope_scint_z = fcn_pol1->GetParError(1);
1468
1469 m_ProfileEKLMScintillatorPlane1->Fit("fcn_pol1", "EMQ");
1470 double slope_scint_plane1_end = fcn_pol1->GetParameter(1);
1471 double e_slope_scint_plane1_end = fcn_pol1->GetParError(1);
1472
1473 m_ProfileEKLMScintillatorPlane2->Fit("fcn_pol1", "EMQ");
1474 double slope_scint_plane2_end = fcn_pol1->GetParameter(1);
1475 double e_slope_scint_plane2_end = fcn_pol1->GetParError(1);
1476
1477 TString logStr_phi, logStr_z;
1478 if (m_useFixedRPCDelay) {
1479 logStr_phi = Form("%.4f ns/cm (fixed)", delayRPCPhi);
1480 logStr_z = Form("%.4f ns/cm (fixed)", delayRPCZ);
1481 B2INFO("Delay in RPCs (using fixed value):"
1482 << LogVar("Used Value (phi readout)", logStr_phi.Data())
1483 << LogVar("Used Value (z readout)", logStr_z.Data()));
1484 } else {
1485 logStr_phi = Form("%.4f +/- %.4f ns/cm", delayRPCPhi, e_slope_rpc_phi);
1486 logStr_z = Form("%.4f +/- %.4f ns/cm", delayRPCZ, e_slope_rpc_z);
1487 B2INFO("Delay in RPCs:"
1488 << LogVar("Fitted Value (phi readout)", logStr_phi.Data())
1489 << LogVar("Fitted Value (z readout)", logStr_z.Data()));
1490 }
1491 logStr_phi = Form("%.4f +/- %.4f ns/cm", slope_scint_phi, e_slope_scint_phi);
1492 logStr_z = Form("%.4f +/- %.4f ns/cm", slope_scint_z, e_slope_scint_z);
1493 B2INFO("Delay in BKLM scintillators:"
1494 << LogVar("Fitted Value (phi readout) ", logStr_phi.Data())
1495 << LogVar("Fitted Value (z readout) ", logStr_z.Data()));
1496 logStr_phi = Form("%.4f +/- %.4f ns/cm", slope_scint_plane1_end,
1497 e_slope_scint_plane1_end);
1498 logStr_z = Form("%.4f +/- %.4f ns/cm", slope_scint_plane2_end,
1499 e_slope_scint_plane2_end);
1500 B2INFO("Delay in EKLM scintillators:"
1501 << LogVar("Fitted Value (plane1 readout) ", logStr_phi.Data())
1502 << LogVar("Fitted Value (plane2 readout) ", logStr_z.Data()));
1503
1504 logStr_z = Form("%.4f +/- %.4f ns/cm", delayBKLM, delayBKLMError);
1505 B2INFO("Delay in BKLM scintillators:"
1506 << LogVar("Fitted Value (2d fit) ", logStr_z.Data()));
1507 logStr_z = Form("%.4f +/- %.4f ns/cm", delayEKLM, delayEKLMError);
1508 B2INFO("Delay in EKLM scintillators:"
1509 << LogVar("Fitted Value (2d fit) ", logStr_z.Data()));
1510
1511 m_timeConstants->setDelay(delayEKLM, KLMTimeConstants::c_EKLM);
1512 m_timeConstants->setDelay(delayBKLM, KLMTimeConstants::c_BKLM);
1513 m_timeConstants->setDelay(delayRPCPhi, KLMTimeConstants::c_RPCPhi);
1514 m_timeConstants->setDelay(delayRPCZ, KLMTimeConstants::c_RPCZ);
1515
1516 /**********************************
1517 * THIRD LOOP (BATCHED)
1518 * Fill per-channel distributions and fit
1519 **********************************/
1520 B2INFO("Third loop: Time distribution filling (batched processing)...");
1521
1522 for (const auto& batch : batches) {
1523 B2INFO("Processing batch: " << batch.first);
1524 readCalibrationDataBatch(batch.second);
1525
1526 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1527 channelId = klmChannel.getKLMChannelNumber();
1528
1529 if (!batch.second(klmChannel))
1530 continue;
1531
1532 if (m_cFlag[channelId] == ChannelCalibrationStatus::c_NotEnoughData)
1533 continue;
1534
1535 if (m_evts.find(channelId) == m_evts.end())
1536 continue;
1537
1538 eventsChannel = m_evts[channelId];
1539 int iSub = klmChannel.getSubdetector();
1540 int iF, iS, iL, iP, iC;
1541
1542 if (iSub == KLMElementNumbers::c_BKLM) {
1543 iF = klmChannel.getSection();
1544 iS = klmChannel.getSector() - 1;
1545 iL = klmChannel.getLayer() - 1;
1546 iP = klmChannel.getPlane();
1547 iC = klmChannel.getStrip() - 1;
1548 } else {
1549 iF = klmChannel.getSection() - 1;
1550 iS = klmChannel.getSector() - 1;
1551 iL = klmChannel.getLayer() - 1;
1552 iP = klmChannel.getPlane() - 1;
1553 iC = klmChannel.getStrip() - 1;
1554 }
1555
1556 // Create per-channel histogram
1557 TString hn, ht;
1558 TH1F* h_temp = nullptr;
1559
1560 if (iSub == KLMElementNumbers::c_BKLM) {
1561 // FIXED: Use constant instead of hardcoded value
1562 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1563 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
1564 ht = Form("Time distribution for RPC of Channel%d, %s, Layer%d, Sector%d, %s",
1565 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1566 h_temp = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryRPC, m_UpperTimeBoundaryRPC);
1567 } else {
1568 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
1569 ht = Form("time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s",
1570 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1571 h_temp = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsBKLM,
1573 }
1574 } else {
1575 hn = Form("h_timeF%d_S%d_L%d_P%d_C%d_end", iF, iS, iL, iP, iC);
1576 ht = Form("Time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s (Endcap)",
1577 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1578 h_temp = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryScintillatorsEKLM,
1580 }
1581
1582 // Fill histogram
1583 for (const Event& event : eventsChannel) {
1584 // Add ADC cut
1585 if (!passesADCCut(event, iSub, iL))
1586 continue;
1587
1588 double timeHit = event.time();
1589 if (m_useEventT0)
1590 timeHit = timeHit - event.t0;
1591 if (timeHit <= -400e3)
1592 continue;
1593
1594 if (iSub == KLMElementNumbers::c_BKLM) {
1595 // FIXED: Use constant instead of hardcoded value
1596 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1597 double propgationT;
1599 propgationT = event.dist * delayRPCZ;
1600 else
1601 propgationT = event.dist * delayRPCPhi;
1602 double time = timeHit - propgationT;
1603
1604 h_time_rpc->Fill(time);
1605 h_temp->Fill(time);
1606
1607 if (m_saveAllPlots) {
1608 h_timeF_rpc[iF]->Fill(time);
1609 h_timeFS_rpc[iF][iS]->Fill(time);
1610 h_timeFSL[iF][iS][iL]->Fill(time);
1611 h_timeFSLP[iF][iS][iL][iP]->Fill(time);
1612 h2_timeF_rpc[iF]->Fill(iS, time);
1613 h2_timeFS[iF][iS]->Fill(iL, time);
1614 h2_timeFSLP[iF][iS][iL][iP]->Fill(iC, time);
1615 }
1616 } else {
1617 double propgationT = event.dist * delayBKLM;
1618 double time = timeHit - propgationT;
1619
1620 h_time_scint->Fill(time);
1621 h_temp->Fill(time);
1622
1623 if (m_saveAllPlots) {
1624 h_timeF_scint[iF]->Fill(time);
1625 h_timeFS_scint[iF][iS]->Fill(time);
1626 h_timeFSL[iF][iS][iL]->Fill(time);
1627 h_timeFSLP[iF][iS][iL][iP]->Fill(time);
1628 h2_timeF_scint[iF]->Fill(iS, time);
1629 h2_timeFS[iF][iS]->Fill(iL, time);
1630 h2_timeFSLP[iF][iS][iL][iP]->Fill(iC, time);
1631 }
1632 }
1633 } else {
1634 double propgationT = event.dist * delayEKLM;
1635 double time = timeHit - propgationT;
1636
1637 h_time_scint_end->Fill(time);
1638 h_temp->Fill(time);
1639
1640 if (m_saveAllPlots) {
1641 h_timeF_scint_end[iF]->Fill(time);
1642 h_timeFS_scint_end[iF][iS]->Fill(time);
1643 h_timeFSL_end[iF][iS][iL]->Fill(time);
1644 h_timeFSLP_end[iF][iS][iL][iP]->Fill(time);
1645 h2_timeF_scint_end[iF]->Fill(iS, time);
1646 h2_timeFS_end[iF][iS]->Fill(iL, time);
1647 h2_timeFSLP_end[iF][iS][iL][iP]->Fill(iC, time);
1648 }
1649 }
1650 }
1651
1652 TFitResultPtr r = h_temp->Fit(fcn_gaus, "LESQ");
1653 if (int(r) == 0) {
1654 m_cFlag[channelId] = ChannelCalibrationStatus::c_SuccessfulCalibration;
1655 m_time_channel[channelId] = fcn_gaus->GetParameter(1);
1656 m_etime_channel[channelId] = fcn_gaus->GetParError(1);
1657 }
1658
1659 // Get the appropriate directory for this channel
1660 TDirectory* dir = (iSub == KLMElementNumbers::c_BKLM) ?
1661 m_channelHistDir_BKLM[iF][iS][iL][iP] :
1662 m_channelHistDir_EKLM[iF][iS][iL][iP];
1664 }
1665
1666 m_evts.clear();
1667 B2INFO("Batch processed and cleared: " << batch.first);
1668 }
1669
1670 B2INFO("Original filling done.");
1671
1672 // Fill TGraphs with extracted parameters
1673 int iChannel_rpc = 0;
1674 int iChannel = 0;
1675 int iChannel_end = 0;
1676 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1677 channelId = klmChannel.getKLMChannelNumber();
1678 if (m_cFlag[channelId] != ChannelCalibrationStatus::c_SuccessfulCalibration)
1679 continue;
1680
1681 int iSub = klmChannel.getSubdetector();
1682 if (iSub == KLMElementNumbers::c_BKLM) {
1683 int iL = klmChannel.getLayer() - 1;
1684 // FIXED: Use constant instead of hardcoded value
1685 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1686 gre_time_channel_rpc->SetPoint(iChannel_rpc, channelId, m_time_channel[channelId]);
1687 gre_time_channel_rpc->SetPointError(iChannel_rpc, 0., m_etime_channel[channelId]);
1688 iChannel_rpc++;
1689 } else {
1690 gre_time_channel_scint->SetPoint(iChannel, channelId, m_time_channel[channelId]);
1691 gre_time_channel_scint->SetPointError(iChannel, 0., m_etime_channel[channelId]);
1692 iChannel++;
1693 }
1694 } else {
1695 gre_time_channel_scint_end->SetPoint(iChannel_end, channelId, m_time_channel[channelId]);
1696 gre_time_channel_scint_end->SetPointError(iChannel_end, 0., m_etime_channel[channelId]);
1697 iChannel_end++;
1698 }
1699 }
1700
1701 gre_time_channel_scint->Fit("fcn_const", "EMQ");
1702 m_time_channelAvg_scint = fcn_const->GetParameter(0);
1703 m_etime_channelAvg_scint = fcn_const->GetParError(0);
1704
1705 gre_time_channel_scint_end->Fit("fcn_const", "EMQ");
1706 m_time_channelAvg_scint_end = fcn_const->GetParameter(0);
1707 m_etime_channelAvg_scint_end = fcn_const->GetParError(0);
1708
1709 gre_time_channel_rpc->Fit("fcn_const", "EMQ");
1710 m_time_channelAvg_rpc = fcn_const->GetParameter(0);
1711 m_etime_channelAvg_rpc = fcn_const->GetParError(0);
1712
1713 B2INFO("Channel's time distribution fitting done.");
1714 B2DEBUG(20, LogVar("Average time (RPC)", m_time_channelAvg_rpc)
1715 << LogVar("Average time (BKLM scintillators)", m_time_channelAvg_scint)
1716 << LogVar("Average time (EKLM scintillators)", m_time_channelAvg_scint_end));
1717
1718 B2INFO("Calibrated channel's time distribution filling begins.");
1719
1720 m_timeShift.clear();
1721 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1722 channelId = klmChannel.getKLMChannelNumber();
1723 h_calibrated->Fill(m_cFlag[channelId]);
1724 if (m_time_channel.find(channelId) == m_time_channel.end())
1725 continue;
1726 double timeShift = m_time_channel[channelId];
1727 m_timeShift[channelId] = timeShift;
1728 m_timeCableDelay->setTimeDelay(channelId, m_timeShift[channelId]);
1729 }
1730
1731 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1732 channelId = klmChannel.getKLMChannelNumber();
1733 if (m_timeShift.find(channelId) != m_timeShift.end())
1734 continue;
1735 m_timeShift[channelId] = esti_timeShift(klmChannel);
1736 m_timeCableDelay->setTimeDelay(channelId, m_timeShift[channelId]);
1737 B2DEBUG(20, "Uncalibrated Estimation " << LogVar("Channel", channelId) << LogVar("Estimated value", m_timeShift[channelId]));
1738 }
1739
1740 iChannel_rpc = 0;
1741 iChannel = 0;
1742 iChannel_end = 0;
1743 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1744 channelId = klmChannel.getKLMChannelNumber();
1745 if (m_timeShift.find(channelId) == m_timeShift.end()) {
1746 B2ERROR("!!! Not All Channels Calibration Constant Set. Error Happened on " << LogVar("Channel", channelId));
1747 continue;
1748 }
1749 int iSub = klmChannel.getSubdetector();
1750 if (iSub == KLMElementNumbers::c_BKLM) {
1751 // FIXED: Use 0-indexed layer and constant
1752 int iL = klmChannel.getLayer() - 1;
1753 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1754 gr_timeShift_channel_rpc->SetPoint(iChannel_rpc, channelId, m_timeShift[channelId]);
1755 iChannel_rpc++;
1756 } else {
1757 gr_timeShift_channel_scint->SetPoint(iChannel, channelId, m_timeShift[channelId]);
1758 iChannel++;
1759 }
1760 } else {
1761 gr_timeShift_channel_scint_end->SetPoint(iChannel_end, channelId, m_timeShift[channelId]);
1762 iChannel_end++;
1763 }
1764 }
1765
1766 // NOTE: This also needs batching
1771
1772 /**********************************
1773 * FOURTH LOOP (BATCHED)
1774 * Fill calibrated per-channel histograms
1775 **********************************/
1776 B2INFO("Fourth loop: Calibrated time distribution filling (batched processing)...");
1777
1778 for (const auto& batch : batches) {
1779 B2INFO("Processing batch: " << batch.first);
1780 readCalibrationDataBatch(batch.second);
1781
1782 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1783 channelId = klmChannel.getKLMChannelNumber();
1784
1785 if (!batch.second(klmChannel))
1786 continue;
1787
1788 if (m_evts.find(channelId) == m_evts.end())
1789 continue;
1790
1791 eventsChannel = m_evts[channelId];
1792 int iSub = klmChannel.getSubdetector();
1793 int iF, iS, iL, iP, iC;
1794
1795 if (iSub == KLMElementNumbers::c_BKLM) {
1796 iF = klmChannel.getSection();
1797 iS = klmChannel.getSector() - 1;
1798 iL = klmChannel.getLayer() - 1;
1799 iP = klmChannel.getPlane();
1800 iC = klmChannel.getStrip() - 1;
1801 } else {
1802 iF = klmChannel.getSection() - 1;
1803 iS = klmChannel.getSector() - 1;
1804 iL = klmChannel.getLayer() - 1;
1805 iP = klmChannel.getPlane() - 1;
1806 iC = klmChannel.getStrip() - 1;
1807 }
1808
1809 TString hn, ht;
1810 TH1F* hc_temp = nullptr;
1811
1812 if (iSub == KLMElementNumbers::c_BKLM) {
1813 // FIXED: Use constant instead of hardcoded value
1814 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1815 hn = Form("hc_timeF%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
1816 ht = Form("Calibrated time distribution for RPC of Channel%d, %s, Layer%d, Sector%d, %s",
1817 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1818 hc_temp = new TH1F(hn.Data(), ht.Data(), nBin, m_LowerTimeBoundaryCalibratedRPC,
1820 } else {
1821 hn = Form("hc_timeF%d_S%d_L%d_P%d_C%d", iF, iS, iL, iP, iC);
1822 ht = Form("Calibrated time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s",
1823 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1824 hc_temp = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsBKLM,
1826 }
1827 } else {
1828 hn = Form("hc_timeF%d_S%d_L%d_P%d_C%d_end", iF, iS, iL, iP, iC);
1829 ht = Form("Calibrated time distribution for Scintillator of Channel%d, %s, Layer%d, Sector%d, %s (Endcap)",
1830 iC, iPstring[iP].Data(), iL, iS, iFstring[iF].Data());
1831 hc_temp = new TH1F(hn.Data(), ht.Data(), nBin_scint, m_LowerTimeBoundaryCalibratedScintillatorsEKLM,
1833 }
1834
1835 for (const Event& event : eventsChannel) {
1836 // Add ADC cut
1837 if (!passesADCCut(event, iSub, iL))
1838 continue;
1839
1840 double timeHit = event.time();
1841 if (m_useEventT0)
1842 timeHit = timeHit - event.t0;
1843 if (timeHit <= -400e3)
1844 continue;
1845
1846 if (iSub == KLMElementNumbers::c_BKLM) {
1847 // FIXED: Use constant instead of hardcoded value
1848 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1849 double propgationT;
1851 propgationT = event.dist * delayRPCZ;
1852 else
1853 propgationT = event.dist * delayRPCPhi;
1854 double time = timeHit - propgationT - m_timeShift[channelId];
1855
1856 hc_time_rpc->Fill(time);
1857 hc_temp->Fill(time);
1858
1859 if (m_saveAllPlots) {
1860 hc_timeF_rpc[iF]->Fill(time);
1861 hc_timeFS_rpc[iF][iS]->Fill(time);
1862 hc_timeFSL[iF][iS][iL]->Fill(time);
1863 hc_timeFSLP[iF][iS][iL][iP]->Fill(time);
1864 h2c_timeF_rpc[iF]->Fill(iS, time);
1865 h2c_timeFS[iF][iS]->Fill(iL, time);
1866 h2c_timeFSLP[iF][iS][iL][iP]->Fill(iC, time);
1867 }
1868 } else {
1869 double propgationT = event.dist * delayBKLM;
1870 double time = timeHit - propgationT - m_timeShift[channelId];
1871
1872 hc_time_scint->Fill(time);
1873 hc_temp->Fill(time);
1874
1875 if (m_saveAllPlots) {
1876 hc_timeF_scint[iF]->Fill(time);
1877 hc_timeFS_scint[iF][iS]->Fill(time);
1878 hc_timeFSL[iF][iS][iL]->Fill(time);
1879 hc_timeFSLP[iF][iS][iL][iP]->Fill(time);
1880 h2c_timeF_scint[iF]->Fill(iS, time);
1881 h2c_timeFS[iF][iS]->Fill(iL, time);
1882 h2c_timeFSLP[iF][iS][iL][iP]->Fill(iC, time);
1883 }
1884 }
1885 } else {
1886 double propgationT = event.dist * delayEKLM;
1887 double time = timeHit - propgationT - m_timeShift[channelId];
1888
1889 hc_time_scint_end->Fill(time);
1890 hc_temp->Fill(time);
1891
1892 if (m_saveAllPlots) {
1893 hc_timeF_scint_end[iF]->Fill(time);
1894 hc_timeFS_scint_end[iF][iS]->Fill(time);
1895 hc_timeFSL_end[iF][iS][iL]->Fill(time);
1896 hc_timeFSLP_end[iF][iS][iL][iP]->Fill(time);
1897 h2c_timeF_scint_end[iF]->Fill(iS, time);
1898 h2c_timeFS_end[iF][iS]->Fill(iL, time);
1899 h2c_timeFSLP_end[iF][iS][iL][iP]->Fill(iC, time);
1900 }
1901 }
1902 }
1903
1904 if (m_cFlag[channelId] == ChannelCalibrationStatus::c_NotEnoughData) {
1905 delete hc_temp;
1906 continue;
1907 }
1908
1909 TFitResultPtr rc = hc_temp->Fit(fcn_gaus, "LESQ");
1910 if (int(rc) == 0) {
1911 m_cFlag[channelId] = ChannelCalibrationStatus::c_SuccessfulCalibration;
1912 m_ctime_channel[channelId] = fcn_gaus->GetParameter(1);
1913 mc_etime_channel[channelId] = fcn_gaus->GetParError(1);
1914 }
1915
1916 // Get the appropriate directory for this channel
1917 TDirectory* dir = (iSub == KLMElementNumbers::c_BKLM) ?
1918 m_channelHistDir_BKLM[iF][iS][iL][iP] :
1919 m_channelHistDir_EKLM[iF][iS][iL][iP];
1920 writeThenDelete_(hc_temp, m_saveChannelHists, dir);
1921 }
1922
1923 m_evts.clear();
1924 B2INFO("Batch processed and cleared: " << batch.first);
1925 }
1926
1927 // Fill TGraphs with calibrated parameters
1928 int icChannel_rpc = 0;
1929 int icChannel = 0;
1930 int icChannel_end = 0;
1931 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1932 channelId = klmChannel.getKLMChannelNumber();
1933 if (m_cFlag[channelId] != ChannelCalibrationStatus::c_SuccessfulCalibration)
1934 continue;
1935
1936 int iSub = klmChannel.getSubdetector();
1937 if (iSub == KLMElementNumbers::c_BKLM) {
1938 int iL = klmChannel.getLayer() - 1;
1939 // FIXED: Use constant instead of hardcoded value
1940 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
1941 gre_ctime_channel_rpc->SetPoint(icChannel_rpc, channelId, m_ctime_channel[channelId]);
1942 gre_ctime_channel_rpc->SetPointError(icChannel_rpc, 0., mc_etime_channel[channelId]);
1943 icChannel_rpc++;
1944 } else {
1945 gre_ctime_channel_scint->SetPoint(icChannel, channelId, m_ctime_channel[channelId]);
1946 gre_ctime_channel_scint->SetPointError(icChannel, 0., mc_etime_channel[channelId]);
1947 icChannel++;
1948 }
1949 } else {
1950 gre_ctime_channel_scint_end->SetPoint(icChannel_end, channelId, m_ctime_channel[channelId]);
1951 gre_ctime_channel_scint_end->SetPointError(icChannel_end, 0., mc_etime_channel[channelId]);
1952 icChannel_end++;
1953 }
1954 }
1955
1956 gre_ctime_channel_scint->Fit("fcn_const", "EMQ");
1957 m_ctime_channelAvg_scint = fcn_const->GetParameter(0);
1958 mc_etime_channelAvg_scint = fcn_const->GetParError(0);
1959
1960 gre_ctime_channel_scint_end->Fit("fcn_const", "EMQ");
1961 m_ctime_channelAvg_scint_end = fcn_const->GetParameter(0);
1962 mc_etime_channelAvg_scint_end = fcn_const->GetParError(0);
1963
1964 gre_ctime_channel_rpc->Fit("fcn_const", "EMQ");
1965 m_ctime_channelAvg_rpc = fcn_const->GetParameter(0);
1966 mc_etime_channelAvg_rpc = fcn_const->GetParError(0);
1967
1968 B2INFO("Channel's time distribution fitting done.");
1969 B2DEBUG(20, LogVar("Average calibrated time (RPC)", m_ctime_channelAvg_rpc)
1970 << LogVar("Average calibrated time (BKLM scintillators)", m_ctime_channelAvg_scint)
1971 << LogVar("Average calibrated time (EKLM scintillators)", m_ctime_channelAvg_scint_end));
1972
1973 B2INFO("Calibrated channel's time distribution filling begins.");
1974
1975 m_timeRes.clear();
1976 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1977 channelId = klmChannel.getKLMChannelNumber();
1978 hc_calibrated->Fill(m_cFlag[channelId]);
1979 if (m_ctime_channel.find(channelId) == m_ctime_channel.end())
1980 continue;
1981 double timeRes = m_ctime_channel[channelId];
1982 m_timeRes[channelId] = timeRes;
1983 m_timeResolution->setTimeResolution(channelId, m_timeRes[channelId]);
1984 }
1985
1986 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1987 channelId = klmChannel.getKLMChannelNumber();
1988 if (m_timeRes.find(channelId) != m_timeRes.end())
1989 continue;
1990 m_timeRes[channelId] = esti_timeRes(klmChannel);
1991 m_timeResolution->setTimeResolution(channelId, m_timeRes[channelId]);
1992 B2DEBUG(20, "Calibrated Estimation " << LogVar("Channel", channelId) << LogVar("Estimated value", m_timeRes[channelId]));
1993 }
1994
1995 icChannel_rpc = 0;
1996 icChannel = 0;
1997 icChannel_end = 0;
1998 for (KLMChannelIndex klmChannel = m_klmChannels.begin(); klmChannel != m_klmChannels.end(); ++klmChannel) {
1999 channelId = klmChannel.getKLMChannelNumber();
2000 if (m_timeRes.find(channelId) == m_timeRes.end()) {
2001 B2ERROR("!!! Not All Channels Calibration Constant Set. Error Happened on " << LogVar("Channel", channelId));
2002 continue;
2003 }
2004 int iSub = klmChannel.getSubdetector();
2005 if (iSub == KLMElementNumbers::c_BKLM) {
2006 // FIXED: Use 0-indexed layer and constant
2007 int iL = klmChannel.getLayer() - 1;
2008 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
2009 gr_timeRes_channel_rpc->SetPoint(icChannel_rpc, channelId, m_timeRes[channelId]);
2010 icChannel_rpc++;
2011 } else {
2012 gr_timeRes_channel_scint->SetPoint(icChannel, channelId, m_timeRes[channelId]);
2013 icChannel++;
2014 }
2015 } else {
2016 gr_timeRes_channel_scint_end->SetPoint(icChannel_end, channelId, m_timeRes[channelId]);
2017 icChannel_end++;
2018 }
2019 }
2020
2021 // ===================================================================
2022 // FIFTH PASS: Di-muon EventT0 analysis for hit resolution calibration
2023 // ===================================================================
2024 B2INFO("Fifth pass: Computing di-muon ΔT0 for EventT0 hit resolution calibration...");
2025
2026 // Data structure for per-track T0 accumulation
2027 struct TrackT0Info {
2028 int charge;
2029 int nHits_BKLM_Scint;
2030 int nHits_BKLM_RPC_Phi; // Split RPC by readout direction
2031 int nHits_BKLM_RPC_Z;
2032 int nHits_EKLM_Scint;
2033 double sumT0_BKLM_Scint;
2034 double sumT0_BKLM_RPC_Phi;
2035 double sumT0_BKLM_RPC_Z;
2036 double sumT0_EKLM_Scint;
2037
2038 TrackT0Info() : charge(0),
2039 nHits_BKLM_Scint(0),
2040 nHits_BKLM_RPC_Phi(0),
2041 nHits_BKLM_RPC_Z(0),
2042 nHits_EKLM_Scint(0),
2043 sumT0_BKLM_Scint(0.0),
2044 sumT0_BKLM_RPC_Phi(0.0),
2045 sumT0_BKLM_RPC_Z(0.0),
2046 sumT0_EKLM_Scint(0.0) {}
2047 };
2048
2049 // Map: (Run, Event) -> (nTrack -> TrackT0Info)
2050 std::map<std::pair<int, int>, std::map<int, TrackT0Info>> eventTrackMap;
2051
2052 // Process all data in batches to build event-track map
2053 for (const auto& batch : batches) {
2054 B2INFO("Processing batch for di-muon analysis: " << batch.first);
2055 readCalibrationDataBatch(batch.second);
2056
2057 for (const auto& channelPair : m_evts) {
2058 KLMChannelNumber chId = channelPair.first;
2059 const std::vector<Event>& chEvents = channelPair.second;
2060
2061 // Get channel geometry
2062 int subdetector, section, sector, layer, plane, strip;
2063 m_ElementNumbers->channelNumberToElementNumbers(
2064 chId, &subdetector, &section, &sector, &layer, &plane, &strip);
2065
2066 // Convert to 0-indexed immediately (consistent with loops 1-4)
2067 int iSub = subdetector;
2068 int iL = layer - 1;
2069
2070 for (const Event& event : chEvents) {
2071 // Apply ADC cut with 0-indexed layer
2072 if (!passesADCCut(event, iSub, iL))
2073 continue;
2074
2075 // Event and track identification
2076 std::pair<int, int> eventKey(event.Run, event.Events);
2077 int trackIdx = event.nTrack;
2078 int charge = event.Track_Charge;
2079
2080 // Compute calibrated time for this hit
2081 double timeHit = event.time() - m_timeShift[chId];
2082
2083 if (timeHit <= -400e3)
2084 continue;
2085
2086 // Apply propagation correction (using 0-indexed layer)
2087 double propT = 0.0;
2088 if (iSub == KLMElementNumbers::c_BKLM) {
2089 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
2090 // RPC
2091 if (plane == BKLMElementNumbers::c_ZPlane)
2092 propT = event.dist * delayRPCZ;
2093 else
2094 propT = event.dist * delayRPCPhi;
2095 } else {
2096 // Scintillator
2097 propT = event.dist * delayBKLM;
2098 }
2099 } else {
2100 // EKLM
2101 propT = event.dist * delayEKLM;
2102 }
2103
2104 double t0_estimate = timeHit - propT;
2105
2106 // Accumulate per-track T0
2107 TrackT0Info& trackInfo = eventTrackMap[eventKey][trackIdx];
2108 trackInfo.charge = charge;
2109
2110 if (iSub == KLMElementNumbers::c_BKLM) {
2111 if (iL >= (BKLMElementNumbers::c_FirstRPCLayer - 1)) {
2112 // RPC - split by readout direction
2113 if (plane == BKLMElementNumbers::c_ZPlane) {
2114 trackInfo.nHits_BKLM_RPC_Z++;
2115 trackInfo.sumT0_BKLM_RPC_Z += t0_estimate;
2116 } else {
2117 trackInfo.nHits_BKLM_RPC_Phi++;
2118 trackInfo.sumT0_BKLM_RPC_Phi += t0_estimate;
2119 }
2120 } else {
2121 // Scintillator
2122 trackInfo.nHits_BKLM_Scint++;
2123 trackInfo.sumT0_BKLM_Scint += t0_estimate;
2124 }
2125 } else {
2126 trackInfo.nHits_EKLM_Scint++;
2127 trackInfo.sumT0_EKLM_Scint += t0_estimate;
2128 }
2129 }
2130 }
2131
2132 m_evts.clear();
2133 }
2134
2135 B2INFO("Event-track map built. Processing events for EventT0 histograms...");
2136
2137 // Accumulators for variance calculation using the Gaussian MLE
2138 // Model: ΔT0_i ~ N(0, σ_hit^2 * v_i), v_i = 1/N⁺_i + 1/N⁻_i
2139 double sum_delta2_over_v_BKLM_Scint = 0.0;
2140 double sum_delta2_over_v_BKLM_RPC_Phi = 0.0;
2141 double sum_delta2_over_v_BKLM_RPC_Z = 0.0;
2142 double sum_delta2_over_v_EKLM_Scint = 0.0;
2143
2144 int nDimuon_BKLM_Scint = 0;
2145 int nDimuon_BKLM_RPC_Phi = 0;
2146 int nDimuon_BKLM_RPC_Z = 0;
2147 int nDimuon_EKLM_Scint = 0;
2148
2149 for (const auto& eventPair : eventTrackMap) {
2150 const auto& trackMap = eventPair.second;
2151
2152 // Check if exactly 2 tracks (di-muon candidate)
2153 if (trackMap.size() != 2)
2154 continue;
2155
2156 auto it1 = trackMap.begin();
2157 auto it2 = trackMap.begin();
2158 ++it2;
2159 const TrackT0Info& track1 = it1->second;
2160 const TrackT0Info& track2 = it2->second;
2161
2162 // Check opposite charges
2163 if (track1.charge * track2.charge >= 0)
2164 continue;
2165
2166 // Identify mu+ and mu-
2167 const TrackT0Info& muPlus = (track1.charge > 0) ? track1 : track2;
2168 const TrackT0Info& muMinus = (track1.charge > 0) ? track2 : track1;
2169
2170 // === BKLM Scintillator ===
2171 if (muPlus.nHits_BKLM_Scint > 0 && muMinus.nHits_BKLM_Scint > 0) {
2172 double t0_plus = muPlus.sumT0_BKLM_Scint / muPlus.nHits_BKLM_Scint;
2173 double t0_minus = muMinus.sumT0_BKLM_Scint / muMinus.nHits_BKLM_Scint;
2174 double deltaT0 = t0_plus - t0_minus;
2175
2176 // v = 1/N⁺ + 1/N⁻ for this event
2177 double v = 1.0 / muPlus.nHits_BKLM_Scint + 1.0 / muMinus.nHits_BKLM_Scint;
2178 int nTotal = muPlus.nHits_BKLM_Scint + muMinus.nHits_BKLM_Scint;
2179
2180 if (v <= 0.0)
2181 continue;
2182
2183 // Accumulate for σ_hit^2 = (1/N_evt) Σ [ ΔT0^2 / v ]
2184 sum_delta2_over_v_BKLM_Scint += (deltaT0 * deltaT0) / v;
2185 nDimuon_BKLM_Scint++;
2186
2187 // Histograms for monitoring
2188 h_eventT0_scint->Fill(t0_plus);
2189 h_eventT0_scint->Fill(t0_minus);
2190 hc_eventT0_scint->Fill(deltaT0);
2191
2192 // ==== NEW DIAGNOSTIC FILLS ====
2193 h_nHits_plus_scint->Fill(muPlus.nHits_BKLM_Scint);
2194 h_nHits_minus_scint->Fill(muMinus.nHits_BKLM_Scint);
2195 h2_deltaT0_vs_v_scint->Fill(v, deltaT0);
2196 prof_deltaT0_rms_vs_v_scint->Fill(v, deltaT0);
2197 h2_deltaT0_vs_nhits_scint->Fill(nTotal, deltaT0);
2198
2199 // Fill multiplicity-binned histograms
2200 if (nTotal < 5) {
2201 hc_eventT0_scint_lowN->Fill(deltaT0);
2202 } else if (nTotal < 15) {
2203 hc_eventT0_scint_midN->Fill(deltaT0);
2204 } else {
2205 hc_eventT0_scint_highN->Fill(deltaT0);
2206 }
2207 }
2208
2209 // === BKLM RPC Phi ===
2210 if (muPlus.nHits_BKLM_RPC_Phi > 0 && muMinus.nHits_BKLM_RPC_Phi > 0) {
2211 double t0_plus = muPlus.sumT0_BKLM_RPC_Phi / muPlus.nHits_BKLM_RPC_Phi;
2212 double t0_minus = muMinus.sumT0_BKLM_RPC_Phi / muMinus.nHits_BKLM_RPC_Phi;
2213 double deltaT0 = t0_plus - t0_minus;
2214
2215 double v = 1.0 / muPlus.nHits_BKLM_RPC_Phi + 1.0 / muMinus.nHits_BKLM_RPC_Phi;
2216 int nTotal = muPlus.nHits_BKLM_RPC_Phi + muMinus.nHits_BKLM_RPC_Phi;
2217
2218 if (v <= 0.0)
2219 continue;
2220
2221 sum_delta2_over_v_BKLM_RPC_Phi += (deltaT0 * deltaT0) / v;
2222 nDimuon_BKLM_RPC_Phi++;
2223
2224 h_eventT0_rpc->Fill(t0_plus);
2225 h_eventT0_rpc->Fill(t0_minus);
2226 hc_eventT0_rpc->Fill(deltaT0);
2227
2228 // Diagnostic fills
2229 h_nHits_plus_rpc->Fill(muPlus.nHits_BKLM_RPC_Phi);
2230 h_nHits_minus_rpc->Fill(muMinus.nHits_BKLM_RPC_Phi);
2231 h2_deltaT0_vs_v_rpc->Fill(v, deltaT0);
2232 prof_deltaT0_rms_vs_v_rpc->Fill(v, deltaT0);
2233 h2_deltaT0_vs_nhits_rpc->Fill(nTotal, deltaT0);
2234
2235 if (nTotal < 10) {
2236 hc_eventT0_rpc_lowN->Fill(deltaT0);
2237 } else if (nTotal < 30) {
2238 hc_eventT0_rpc_midN->Fill(deltaT0);
2239 } else {
2240 hc_eventT0_rpc_highN->Fill(deltaT0);
2241 }
2242 }
2243
2244 // === BKLM RPC Z ===
2245 if (muPlus.nHits_BKLM_RPC_Z > 0 && muMinus.nHits_BKLM_RPC_Z > 0) {
2246 double t0_plus = muPlus.sumT0_BKLM_RPC_Z / muPlus.nHits_BKLM_RPC_Z;
2247 double t0_minus = muMinus.sumT0_BKLM_RPC_Z / muMinus.nHits_BKLM_RPC_Z;
2248 double deltaT0 = t0_plus - t0_minus;
2249
2250 double v = 1.0 / muPlus.nHits_BKLM_RPC_Z + 1.0 / muMinus.nHits_BKLM_RPC_Z;
2251 int nTotal = muPlus.nHits_BKLM_RPC_Z + muMinus.nHits_BKLM_RPC_Z;
2252
2253 if (v <= 0.0)
2254 continue;
2255
2256 sum_delta2_over_v_BKLM_RPC_Z += (deltaT0 * deltaT0) / v;
2257 nDimuon_BKLM_RPC_Z++;
2258
2259 h_eventT0_rpc->Fill(t0_plus);
2260 h_eventT0_rpc->Fill(t0_minus);
2261 hc_eventT0_rpc->Fill(deltaT0);
2262
2263 // Diagnostic fills
2264 h_nHits_plus_rpc->Fill(muPlus.nHits_BKLM_RPC_Z);
2265 h_nHits_minus_rpc->Fill(muMinus.nHits_BKLM_RPC_Z);
2266 h2_deltaT0_vs_v_rpc->Fill(v, deltaT0);
2267 prof_deltaT0_rms_vs_v_rpc->Fill(v, deltaT0);
2268 h2_deltaT0_vs_nhits_rpc->Fill(nTotal, deltaT0);
2269
2270 if (nTotal < 10) {
2271 hc_eventT0_rpc_lowN->Fill(deltaT0);
2272 } else if (nTotal < 30) {
2273 hc_eventT0_rpc_midN->Fill(deltaT0);
2274 } else {
2275 hc_eventT0_rpc_highN->Fill(deltaT0);
2276 }
2277 }
2278
2279 // === EKLM Scintillator ===
2280 if (muPlus.nHits_EKLM_Scint > 0 && muMinus.nHits_EKLM_Scint > 0) {
2281 double t0_plus = muPlus.sumT0_EKLM_Scint / muPlus.nHits_EKLM_Scint;
2282 double t0_minus = muMinus.sumT0_EKLM_Scint / muMinus.nHits_EKLM_Scint;
2283 double deltaT0 = t0_plus - t0_minus;
2284
2285 double v = 1.0 / muPlus.nHits_EKLM_Scint + 1.0 / muMinus.nHits_EKLM_Scint;
2286 int nTotal = muPlus.nHits_EKLM_Scint + muMinus.nHits_EKLM_Scint;
2287
2288 if (v <= 0.0)
2289 continue;
2290
2291 sum_delta2_over_v_EKLM_Scint += (deltaT0 * deltaT0) / v;
2292 nDimuon_EKLM_Scint++;
2293
2294 h_eventT0_scint_end->Fill(t0_plus);
2295 h_eventT0_scint_end->Fill(t0_minus);
2296 hc_eventT0_scint_end->Fill(deltaT0);
2297
2298 // ==== NEW DIAGNOSTIC FILLS ====
2299 h_nHits_plus_scint_end->Fill(muPlus.nHits_EKLM_Scint);
2300 h_nHits_minus_scint_end->Fill(muMinus.nHits_EKLM_Scint);
2301 h2_deltaT0_vs_v_scint_end->Fill(v, deltaT0);
2302 prof_deltaT0_rms_vs_v_scint_end->Fill(v, deltaT0);
2303 h2_deltaT0_vs_nhits_scint_end->Fill(nTotal, deltaT0);
2304
2305 // Fill multiplicity-binned histograms
2306 if (nTotal < 5) {
2307 hc_eventT0_scint_end_lowN->Fill(deltaT0);
2308 } else if (nTotal < 15) {
2309 hc_eventT0_scint_end_midN->Fill(deltaT0);
2310 } else {
2311 hc_eventT0_scint_end_highN->Fill(deltaT0);
2312 }
2313 }
2314 }
2315
2316 B2INFO("Di-muon ΔT0 data collected."
2317 << LogVar("BKLM Scint di-muon events", nDimuon_BKLM_Scint)
2318 << LogVar("BKLM RPC Phi di-muon events", nDimuon_BKLM_RPC_Phi)
2319 << LogVar("BKLM RPC Z di-muon events", nDimuon_BKLM_RPC_Z)
2320 << LogVar("EKLM Scint di-muon events", nDimuon_EKLM_Scint));
2321
2322 // === Extract σ_hit using the Gaussian MLE ===
2323 // σ_hit^2 = (1 / N_evt) Σ_i [ ΔT0_i^2 / (1/N⁺_i + 1/N⁻_i) ]
2324
2325 float sigma_BKLM_Scint = 10.0f; // Default fallback
2326 float sigma_BKLM_Scint_err = 1.0f;
2327 if (nDimuon_BKLM_Scint > 0) {
2328 double sigma2 = sum_delta2_over_v_BKLM_Scint / static_cast<double>(nDimuon_BKLM_Scint);
2329 sigma_BKLM_Scint = static_cast<float>(std::sqrt(sigma2));
2330 // Uncertainty approximation (Gaussian statistics)
2331 sigma_BKLM_Scint_err = sigma_BKLM_Scint / std::sqrt(2.0 * nDimuon_BKLM_Scint);
2332 }
2333
2334 float sigma_RPC_Phi = 10.0f;
2335 float sigma_RPC_Phi_err = 1.0f;
2336 if (nDimuon_BKLM_RPC_Phi > 0) {
2337 double sigma2 = sum_delta2_over_v_BKLM_RPC_Phi / static_cast<double>(nDimuon_BKLM_RPC_Phi);
2338 sigma_RPC_Phi = static_cast<float>(std::sqrt(sigma2));
2339 sigma_RPC_Phi_err = sigma_RPC_Phi / std::sqrt(2.0 * nDimuon_BKLM_RPC_Phi);
2340 }
2341
2342 float sigma_RPC_Z = 10.0f;
2343 float sigma_RPC_Z_err = 1.0f;
2344 if (nDimuon_BKLM_RPC_Z > 0) {
2345 double sigma2 = sum_delta2_over_v_BKLM_RPC_Z / static_cast<double>(nDimuon_BKLM_RPC_Z);
2346 sigma_RPC_Z = static_cast<float>(std::sqrt(sigma2));
2347 sigma_RPC_Z_err = sigma_RPC_Z / std::sqrt(2.0 * nDimuon_BKLM_RPC_Z);
2348 }
2349
2350 // Compute combined RPC sigma (weighted average by number of events)
2351 float sigma_RPC = 10.0f;
2352 float sigma_RPC_err = 1.0f;
2353 int nDimuon_RPC_total = nDimuon_BKLM_RPC_Phi + nDimuon_BKLM_RPC_Z;
2354 if (nDimuon_RPC_total > 0) {
2355 // Weighted average by statistics
2356 double w_phi = static_cast<double>(nDimuon_BKLM_RPC_Phi) / nDimuon_RPC_total;
2357 double w_z = static_cast<double>(nDimuon_BKLM_RPC_Z) / nDimuon_RPC_total;
2358 sigma_RPC = w_phi * sigma_RPC_Phi + w_z * sigma_RPC_Z;
2359 // Propagate uncertainties
2360 sigma_RPC_err = std::sqrt(w_phi * w_phi * sigma_RPC_Phi_err * sigma_RPC_Phi_err +
2361 w_z * w_z * sigma_RPC_Z_err * sigma_RPC_Z_err);
2362 }
2363
2364 float sigma_EKLM_Scint = 10.0f;
2365 float sigma_EKLM_Scint_err = 1.0f;
2366 if (nDimuon_EKLM_Scint > 0) {
2367 double sigma2 = sum_delta2_over_v_EKLM_Scint / static_cast<double>(nDimuon_EKLM_Scint);
2368 sigma_EKLM_Scint = static_cast<float>(std::sqrt(sigma2));
2369 sigma_EKLM_Scint_err = sigma_EKLM_Scint / std::sqrt(2.0 * nDimuon_EKLM_Scint);
2370 }
2371
2372 B2INFO("Extracted per-hit resolutions using event-by-event weighting:"
2373 << LogVar("σ_BKLM_Scint [ns]", sigma_BKLM_Scint) << LogVar("±", sigma_BKLM_Scint_err)
2374 << LogVar("σ_RPC_Phi [ns]", sigma_RPC_Phi) << LogVar("±", sigma_RPC_Phi_err)
2375 << LogVar("σ_RPC_Z [ns]", sigma_RPC_Z) << LogVar("±", sigma_RPC_Z_err)
2376 << LogVar("σ_RPC_combined [ns]", sigma_RPC) << LogVar("±", sigma_RPC_err)
2377 << LogVar("σ_EKLM_Scint [ns]", sigma_EKLM_Scint) << LogVar("±", sigma_EKLM_Scint_err));
2378
2379 // === Store in payload ===
2380 m_eventT0HitResolution->setSigmaBKLMScint(sigma_BKLM_Scint, sigma_BKLM_Scint_err);
2381 m_eventT0HitResolution->setSigmaRPC(sigma_RPC, sigma_RPC_err); // Combined for backward compatibility
2382 m_eventT0HitResolution->setSigmaRPCPhi(sigma_RPC_Phi, sigma_RPC_Phi_err); // Direction-specific
2383 m_eventT0HitResolution->setSigmaRPCZ(sigma_RPC_Z, sigma_RPC_Z_err); // Direction-specific
2384 m_eventT0HitResolution->setSigmaEKLMScint(sigma_EKLM_Scint, sigma_EKLM_Scint_err);
2385
2386 B2INFO("EventT0 hit resolution calibration complete and stored in payload.");
2387
2388 // Clear temporary data
2389 eventTrackMap.clear();
2390
2391 delete fcn_const;
2392 m_evts.clear();
2393 m_timeShift.clear();
2394 m_timeRes.clear();
2395 m_cFlag.clear();
2396
2397 saveHist();
2398
2399 saveCalibration(m_timeCableDelay, "KLMTimeCableDelay");
2400 saveCalibration(m_timeConstants, "KLMTimeConstants");
2401 saveCalibration(m_timeResolution, "KLMTimeResolution");
2402 saveCalibration(m_eventT0HitResolution, "KLMEventT0HitResolution");
2403
2405}
2406
2408{
2409 m_outFile->cd();
2410 B2INFO("Save Histograms into Files.");
2411
2412 /* Save vital plots. */
2413 TDirectory* dir_monitor = m_outFile->mkdir("monitor_Hists", "", true);
2414 dir_monitor->cd();
2415 h_calibrated->SetDirectory(dir_monitor);
2416 hc_calibrated->SetDirectory(dir_monitor);
2417 h_diff->SetDirectory(dir_monitor);
2418
2419 m_outFile->cd();
2420 TDirectory* dir_eventT0 = m_outFile->mkdir("EventT0", "", true);
2421 dir_eventT0->cd();
2422
2423 // Original histograms
2424 h_eventT0_rpc->SetDirectory(dir_eventT0);
2425 h_eventT0_scint->SetDirectory(dir_eventT0);
2426 h_eventT0_scint_end->SetDirectory(dir_eventT0);
2427
2428 hc_eventT0_rpc->SetDirectory(dir_eventT0);
2429 hc_eventT0_scint->SetDirectory(dir_eventT0);
2430 hc_eventT0_scint_end->SetDirectory(dir_eventT0);
2431
2432 // ==== NEW DIAGNOSTIC PLOTS ====
2433
2434 // Hit multiplicity
2435 h_nHits_plus_rpc->SetDirectory(dir_eventT0);
2436 h_nHits_minus_rpc->SetDirectory(dir_eventT0);
2437 h_nHits_plus_scint->SetDirectory(dir_eventT0);
2438 h_nHits_minus_scint->SetDirectory(dir_eventT0);
2439 h_nHits_plus_scint_end->SetDirectory(dir_eventT0);
2440 h_nHits_minus_scint_end->SetDirectory(dir_eventT0);
2441
2442 // ΔT0 vs v
2443 h2_deltaT0_vs_v_rpc->SetDirectory(dir_eventT0);
2444 h2_deltaT0_vs_v_scint->SetDirectory(dir_eventT0);
2445 h2_deltaT0_vs_v_scint_end->SetDirectory(dir_eventT0);
2446
2447 // Profile plots
2448 prof_deltaT0_rms_vs_v_rpc->SetDirectory(dir_eventT0);
2449 prof_deltaT0_rms_vs_v_scint->SetDirectory(dir_eventT0);
2450 prof_deltaT0_rms_vs_v_scint_end->SetDirectory(dir_eventT0);
2451
2452 // ΔT0 vs total hits
2453 h2_deltaT0_vs_nhits_rpc->SetDirectory(dir_eventT0);
2454 h2_deltaT0_vs_nhits_scint->SetDirectory(dir_eventT0);
2455 h2_deltaT0_vs_nhits_scint_end->SetDirectory(dir_eventT0);
2456
2457 // Multiplicity-binned ΔT0
2458 hc_eventT0_rpc_lowN->SetDirectory(dir_eventT0);
2459 hc_eventT0_rpc_midN->SetDirectory(dir_eventT0);
2460 hc_eventT0_rpc_highN->SetDirectory(dir_eventT0);
2461 hc_eventT0_scint_lowN->SetDirectory(dir_eventT0);
2462 hc_eventT0_scint_midN->SetDirectory(dir_eventT0);
2463 hc_eventT0_scint_highN->SetDirectory(dir_eventT0);
2464 hc_eventT0_scint_end_lowN->SetDirectory(dir_eventT0);
2465 hc_eventT0_scint_end_midN->SetDirectory(dir_eventT0);
2466 hc_eventT0_scint_end_highN->SetDirectory(dir_eventT0);
2467
2468 m_outFile->cd();
2469 TDirectory* dir_effC = m_outFile->mkdir("effC_Hists", "", true);
2470 dir_effC->cd();
2471 m_ProfileRpcPhi->SetDirectory(dir_effC);
2472 m_ProfileRpcZ->SetDirectory(dir_effC);
2473 m_ProfileBKLMScintillatorPhi->SetDirectory(dir_effC);
2474 m_ProfileBKLMScintillatorZ->SetDirectory(dir_effC);
2475 m_ProfileEKLMScintillatorPlane1->SetDirectory(dir_effC);
2476 m_ProfileEKLMScintillatorPlane2->SetDirectory(dir_effC);
2477 m_Profile2RpcPhi->SetDirectory(dir_effC);
2478 m_Profile2RpcZ->SetDirectory(dir_effC);
2479 m_Profile2BKLMScintillatorPhi->SetDirectory(dir_effC);
2480 m_Profile2BKLMScintillatorZ->SetDirectory(dir_effC);
2481 m_Profile2EKLMScintillatorPlane1->SetDirectory(dir_effC);
2482 m_Profile2EKLMScintillatorPlane2->SetDirectory(dir_effC);
2483
2484 m_outFile->cd();
2485 TDirectory* dir_time = m_outFile->mkdir("time", "", true);
2486 dir_time->cd();
2487
2488 h_time_scint->SetDirectory(dir_time);
2489 hc_time_scint->SetDirectory(dir_time);
2490
2491 h_time_scint_end->SetDirectory(dir_time);
2492 hc_time_scint_end->SetDirectory(dir_time);
2493
2494 h_time_rpc->SetDirectory(dir_time);
2495 hc_time_rpc->SetDirectory(dir_time);
2496
2497 gre_time_channel_rpc->Write("gre_time_channel_rpc");
2498 gre_time_channel_scint->Write("gre_time_channel_scint");
2499 gre_time_channel_scint_end->Write("gre_time_channel_scint_end");
2500 gr_timeShift_channel_rpc->Write("gr_timeShift_channel_rpc");
2501 gr_timeShift_channel_scint->Write("gr_timeShift_channel_scint");
2502 gr_timeShift_channel_scint_end->Write("gr_timeShift_channel_scint_end");
2503 gre_ctime_channel_rpc->Write("gre_ctime_channel_rpc");
2504 gre_ctime_channel_scint->Write("gre_ctime_channel_scint");
2505 gre_ctime_channel_scint_end->Write("gre_ctime_channel_scint_end");
2506 gr_timeRes_channel_rpc->Write("gr_timeRes_channel_rpc");
2507 gr_timeRes_channel_scint->Write("gr_timeRes_channel_scint");
2508 gr_timeRes_channel_scint_end->Write("gr_timeRes_channel_scint_end");
2509
2510 B2INFO("Top file setup Done.");
2511
2512 /* Save debug plots (only if m_saveAllPlots is true). */
2513 if (!m_saveAllPlots) {
2514 B2INFO("Skipping debug histogram directory creation (m_saveAllPlots = false)");
2515 m_outFile->cd();
2516 m_outFile->Write();
2517 m_outFile->Close();
2518 B2INFO("File Write and Close. Done.");
2519 return;
2520 }
2521
2522 TDirectory* dir_time_F[2];
2523 TDirectory* dir_time_FS[2][8];
2524 TDirectory* dir_time_FSL[2][8][15];
2525 TDirectory* dir_time_FSLP[2][8][15][2];
2526 TDirectory* dir_time_F_end[2];
2527 TDirectory* dir_time_FS_end[2][4];
2528 TDirectory* dir_time_FSL_end[2][4][14];
2529 TDirectory* dir_time_FSLP_end[2][4][14][2];
2530 char dirname[50];
2531 B2INFO("Sub files declare Done.");
2532 for (int iF = 0; iF < 2; ++iF) {
2533 h_timeF_rpc[iF]->SetDirectory(dir_time);
2534 hc_timeF_rpc[iF]->SetDirectory(dir_time);
2535
2536 h2_timeF_rpc[iF]->SetDirectory(dir_time);
2537 h2c_timeF_rpc[iF]->SetDirectory(dir_time);
2538
2539 h_timeF_scint[iF]->SetDirectory(dir_time);
2540 hc_timeF_scint[iF]->SetDirectory(dir_time);
2541
2542 h2_timeF_scint[iF]->SetDirectory(dir_time);
2543 h2c_timeF_scint[iF]->SetDirectory(dir_time);
2544
2545 h_timeF_scint_end[iF]->SetDirectory(dir_time);
2546 hc_timeF_scint_end[iF]->SetDirectory(dir_time);
2547
2548 h2_timeF_scint_end[iF]->SetDirectory(dir_time);
2549 h2c_timeF_scint_end[iF]->SetDirectory(dir_time);
2550
2551 sprintf(dirname, "isForward_%d", iF);
2552 dir_time_F[iF] = dir_time->mkdir(dirname, "", true);
2553 dir_time_F[iF]->cd();
2554
2555 for (int iS = 0; iS < 8; ++iS) {
2556 h_timeFS_rpc[iF][iS]->SetDirectory(dir_time_F[iF]);
2557 hc_timeFS_rpc[iF][iS]->SetDirectory(dir_time_F[iF]);
2558
2559 h_timeFS_scint[iF][iS]->SetDirectory(dir_time_F[iF]);
2560 hc_timeFS_scint[iF][iS]->SetDirectory(dir_time_F[iF]);
2561
2562 h2_timeFS[iF][iS]->SetDirectory(dir_time_F[iF]);
2563 h2c_timeFS[iF][iS]->SetDirectory(dir_time_F[iF]);
2564
2565 sprintf(dirname, "Sector_%d", iS + 1);
2566 dir_time_FS[iF][iS] = dir_time_F[iF]->mkdir(dirname, "", true);
2567 dir_time_FS[iF][iS]->cd();
2568
2569 for (int iL = 0; iL < 15; ++iL) {
2570 h_timeFSL[iF][iS][iL]->SetDirectory(dir_time_FS[iF][iS]);
2571 hc_timeFSL[iF][iS][iL]->SetDirectory(dir_time_FS[iF][iS]);
2572
2573 sprintf(dirname, "Layer_%d", iL + 1);
2574 dir_time_FSL[iF][iS][iL] = dir_time_FS[iF][iS]->mkdir(dirname, "", true);
2575 dir_time_FSL[iF][iS][iL]->cd();
2576 for (int iP = 0; iP < 2; ++iP) {
2577 h_timeFSLP[iF][iS][iL][iP]->SetDirectory(dir_time_FSL[iF][iS][iL]);
2578 hc_timeFSLP[iF][iS][iL][iP]->SetDirectory(dir_time_FSL[iF][iS][iL]);
2579 h2_timeFSLP[iF][iS][iL][iP]->SetDirectory(dir_time_FSL[iF][iS][iL]);
2580 h2c_timeFSLP[iF][iS][iL][iP]->SetDirectory(dir_time_FSL[iF][iS][iL]);
2581
2582 sprintf(dirname, "Plane_%d", iP);
2583 dir_time_FSLP[iF][iS][iL][iP] = dir_time_FSL[iF][iS][iL]->mkdir(dirname, "", true);
2584 dir_time_FSLP[iF][iS][iL][iP]->cd();
2585
2586 }
2587 }
2588 }
2589
2590 sprintf(dirname, "isForward_%d_end", iF + 1);
2591 dir_time_F_end[iF] = dir_time->mkdir(dirname, "", true);
2592 dir_time_F_end[iF]->cd();
2593 int maxLayer = 12 + 2 * iF;
2594 for (int iS = 0; iS < 4; ++iS) {
2595 h_timeFS_scint_end[iF][iS]->SetDirectory(dir_time_F_end[iF]);
2596 hc_timeFS_scint_end[iF][iS]->SetDirectory(dir_time_F_end[iF]);
2597
2598 h2_timeFS_end[iF][iS]->SetDirectory(dir_time_F_end[iF]);
2599 h2c_timeFS_end[iF][iS]->SetDirectory(dir_time_F_end[iF]);
2600
2601 sprintf(dirname, "Sector_%d_end", iS + 1);
2602 dir_time_FS_end[iF][iS] = dir_time_F_end[iF]->mkdir(dirname, "", true);
2603 dir_time_FS_end[iF][iS]->cd();
2604 for (int iL = 0; iL < maxLayer; ++iL) {
2605 h_timeFSL_end[iF][iS][iL]->SetDirectory(dir_time_FS_end[iF][iS]);
2606 hc_timeFSL_end[iF][iS][iL]->SetDirectory(dir_time_FS_end[iF][iS]);
2607
2608 sprintf(dirname, "Layer_%d_end", iL + 1);
2609 dir_time_FSL_end[iF][iS][iL] = dir_time_FS_end[iF][iS]->mkdir(dirname, "", true);
2610 dir_time_FSL_end[iF][iS][iL]->cd();
2611 for (int iP = 0; iP < 2; ++iP) {
2612 h_timeFSLP_end[iF][iS][iL][iP]->SetDirectory(dir_time_FSL_end[iF][iS][iL]);
2613 hc_timeFSLP_end[iF][iS][iL][iP]->SetDirectory(dir_time_FSL_end[iF][iS][iL]);
2614 h2_timeFSLP_end[iF][iS][iL][iP]->SetDirectory(dir_time_FSL_end[iF][iS][iL]);
2615 h2c_timeFSLP_end[iF][iS][iL][iP]->SetDirectory(dir_time_FSL_end[iF][iS][iL]);
2616
2617 sprintf(dirname, "plane_%d_end", iP);
2618 dir_time_FSLP_end[iF][iS][iL][iP] = dir_time_FSL_end[iF][iS][iL]->mkdir(dirname, "", true);
2619 dir_time_FSLP_end[iF][iS][iL][iP]->cd();
2620
2621 }
2622 }
2623 }
2624 }
2625 m_outFile->cd();
2626 m_outFile->Write();
2627 m_outFile->Close();
2628 B2INFO("File Write and Close. Done.");
2629}
2630
2632{
2633 double tS = 0.0;
2634 int iSub = klmChannel.getSubdetector();
2635 int iF = klmChannel.getSection();
2636 int iS = klmChannel.getSector();
2637 int iL = klmChannel.getLayer();
2638 int iP = klmChannel.getPlane();
2639 int iC = klmChannel.getStrip();
2641 if (iSub == KLMElementNumbers::c_BKLM)
2642 totNStrips = BKLMElementNumbers::getNStrips(iF, iS, iL, iP);
2643 if (iC == 1) {
2644 KLMChannelIndex kCIndex_upper(iSub, iF, iS, iL, iP, iC + 1);
2645 tS = tS_upperStrip(kCIndex_upper).second;
2646 } else if (iC == totNStrips) {
2647 KLMChannelIndex kCIndex_lower(iSub, iF, iS, iL, iP, iC - 1);
2648 tS = tS_lowerStrip(kCIndex_lower).second;
2649 } else {
2650 KLMChannelIndex kCIndex_upper(iSub, iF, iS, iL, iP, iC + 1);
2651 KLMChannelIndex kCIndex_lower(iSub, iF, iS, iL, iP, iC - 1);
2652 std::pair<int, double> tS_upper = tS_upperStrip(kCIndex_upper);
2653 std::pair<int, double> tS_lower = tS_lowerStrip(kCIndex_lower);
2654 unsigned int td_upper = tS_upper.first - iC;
2655 unsigned int td_lower = iC - tS_lower.first;
2656 unsigned int td = tS_upper.first - tS_lower.first;
2657 tS = (double(td_upper) * tS_lower.second + double(td_lower) * tS_upper.second) / double(td);
2658 }
2659 return tS;
2660}
2661
2662std::pair<int, double> KLMTimeAlgorithm::tS_upperStrip(const KLMChannelIndex& klmChannel)
2663{
2664 std::pair<int, double> tS;
2665 int cId = klmChannel.getKLMChannelNumber();
2666 int iSub = klmChannel.getSubdetector();
2667 int iF = klmChannel.getSection();
2668 int iS = klmChannel.getSector();
2669 int iL = klmChannel.getLayer();
2670 int iP = klmChannel.getPlane();
2671 int iC = klmChannel.getStrip();
2673 if (iSub == KLMElementNumbers::c_BKLM)
2674 totNStrips = BKLMElementNumbers::getNStrips(iF, iS, iL, iP);
2675 if (m_timeShift.find(cId) != m_timeShift.end()) {
2676 tS.first = iC;
2677 tS.second = m_timeShift[cId];
2678 } else if (iC == totNStrips) {
2679 tS.first = iC;
2680 tS.second = 0.0;
2681 } else {
2682 KLMChannelIndex kCIndex(iSub, iF, iS, iL, iP, iC + 1);
2683 tS = tS_upperStrip(kCIndex);
2684 }
2685 return tS;
2686}
2687
2688std::pair<int, double> KLMTimeAlgorithm::tS_lowerStrip(const KLMChannelIndex& klmChannel)
2689{
2690 std::pair<int, double> tS;
2691 int cId = klmChannel.getKLMChannelNumber();
2692 int iSub = klmChannel.getSubdetector();
2693 int iF = klmChannel.getSection();
2694 int iS = klmChannel.getSector();
2695 int iL = klmChannel.getLayer();
2696 int iP = klmChannel.getPlane();
2697 int iC = klmChannel.getStrip();
2698 if (m_timeShift.find(cId) != m_timeShift.end()) {
2699 tS.first = iC;
2700 tS.second = m_timeShift[cId];
2701 } else if (iC == 1) {
2702 tS.first = iC;
2703 tS.second = 0.0;
2704 } else {
2705 KLMChannelIndex kCIndex(iSub, iF, iS, iL, iP, iC - 1);
2706 tS = tS_lowerStrip(kCIndex);
2707 }
2708 return tS;
2709}
2710
2712{
2713 double tR = 0.0;
2714 int iSub = klmChannel.getSubdetector();
2715 int iF = klmChannel.getSection();
2716 int iS = klmChannel.getSector();
2717 int iL = klmChannel.getLayer();
2718 int iP = klmChannel.getPlane();
2719 int iC = klmChannel.getStrip();
2721 if (iSub == KLMElementNumbers::c_BKLM)
2722 totNStrips = BKLMElementNumbers::getNStrips(iF, iS, iL, iP);
2723 if (iC == 1) {
2724 KLMChannelIndex kCIndex_upper(iSub, iF, iS, iL, iP, iC + 1);
2725 tR = tR_upperStrip(kCIndex_upper).second;
2726 } else if (iC == totNStrips) {
2727 KLMChannelIndex kCIndex_lower(iSub, iF, iS, iL, iP, iC - 1);
2728 tR = tR_lowerStrip(kCIndex_lower).second;
2729 } else {
2730 KLMChannelIndex kCIndex_upper(iSub, iF, iS, iL, iP, iC + 1);
2731 KLMChannelIndex kCIndex_lower(iSub, iF, iS, iL, iP, iC - 1);
2732 std::pair<int, double> tR_upper = tR_upperStrip(kCIndex_upper);
2733 std::pair<int, double> tR_lower = tR_lowerStrip(kCIndex_lower);
2734 unsigned int tr_upper = tR_upper.first - iC;
2735 unsigned int tr_lower = iC - tR_lower.first;
2736 unsigned int tr = tR_upper.first - tR_lower.first;
2737 tR = (double(tr_upper) * tR_lower.second + double(tr_lower) * tR_upper.second) / double(tr);
2738 }
2739 return tR;
2740}
2741
2742std::pair<int, double> KLMTimeAlgorithm::tR_upperStrip(const KLMChannelIndex& klmChannel)
2743{
2744 std::pair<int, double> tR;
2745 int cId = klmChannel.getKLMChannelNumber();
2746 int iSub = klmChannel.getSubdetector();
2747 int iF = klmChannel.getSection();
2748 int iS = klmChannel.getSector();
2749 int iL = klmChannel.getLayer();
2750 int iP = klmChannel.getPlane();
2751 int iC = klmChannel.getStrip();
2753 if (iSub == KLMElementNumbers::c_BKLM)
2754 totNStrips = BKLMElementNumbers::getNStrips(iF, iS, iL, iP);
2755 if (m_timeRes.find(cId) != m_timeRes.end()) {
2756 tR.first = iC;
2757 tR.second = m_timeRes[cId];
2758 } else if (iC == totNStrips) {
2759 tR.first = iC;
2760 tR.second = 0.0;
2761 } else {
2762 KLMChannelIndex kCIndex(iSub, iF, iS, iL, iP, iC + 1);
2763 tR = tR_upperStrip(kCIndex);
2764 }
2765 return tR;
2766}
2767
2768std::pair<int, double> KLMTimeAlgorithm::tR_lowerStrip(const KLMChannelIndex& klmChannel)
2769{
2770 std::pair<int, double> tR;
2771 int cId = klmChannel.getKLMChannelNumber();
2772 int iSub = klmChannel.getSubdetector();
2773 int iF = klmChannel.getSection();
2774 int iS = klmChannel.getSector();
2775 int iL = klmChannel.getLayer();
2776 int iP = klmChannel.getPlane();
2777 int iC = klmChannel.getStrip();
2778 if (m_timeRes.find(cId) != m_timeRes.end()) {
2779 tR.first = iC;
2780 tR.second = m_timeRes[cId];
2781 } else if (iC == 1) {
2782 tR.first = iC;
2783 tR.second = 0.0;
2784 } else {
2785 KLMChannelIndex kCIndex(iSub, iF, iS, iL, iP, iC - 1);
2786 tR = tR_lowerStrip(kCIndex);
2787 }
2788 return tR;
2789}
@ c_FirstRPCLayer
First RPC layer.
static int getNStrips(int section, int sector, int layer, int plane)
Get number of strips.
void saveCalibration(TClonesArray *data, const std::string &name)
Store DBArray payload with given name with default IOV.
const std::vector< Calibration::ExpRun > & getRunList() const
Get the list of runs for which calibration is called.
EResult
The result of calibration.
@ c_OK
Finished successfully =0 in Python.
@ c_NotEnoughData
Needs more data =2 in Python.
CalibrationAlgorithm(const std::string &collectorModuleName)
Constructor - sets the prefix for collected objects (won't be accesses until execute(....
Class for accessing objects in the database.
Definition DBObjPtr.h:21
Singleton class to cache database objects.
Definition DBStore.h:31
static DataStore & Instance()
Instance of singleton Store.
Definition DataStore.cc:53
void setInitializeActive(bool active)
Setter for m_initializeActive.
Definition DataStore.cc:93
static constexpr int getMaximalStripNumber()
Get maximal strip number.
static const GeometryData & Instance(enum DataSource dataSource=c_Database, const GearDir *gearDir=nullptr)
Instantiation.
KLM channel index.
int getSubdetector() const
Get subdetector.
int getLayer() const
Get layer.
int getSection() const
Get section.
int getPlane() const
Get plane.
int getStrip() const
Get strip.
int getSector() const
Get sector.
KLMChannelNumber getKLMChannelNumber() const
Get KLM channel number.
static const KLMElementNumbers & Instance()
Instantiation.
Class to store per-hit time resolution (sigma) for KLM EventT0, separated by detector type.
TProfile * m_Profile2EKLMScintillatorPlane2
For EKLM scintillator plane2.
TDirectory * m_channelHistDir_BKLM[2][8][15][2]
Directory structure for per-channel histograms (BKLM).
double mc_etime_channelAvg_rpc
Calibrated central value error of the global time distribution (BKLM RPC part).
bool m_applyChargeRestriction
Whether to apply ADC/charge restriction cuts for scintillators.
TH2F * h2c_timeF_scint_end[2]
EKLM part.
KLMTimeResolution * m_timeResolution
DBObject of time resolution.
TH1F * hc_eventT0_scint_end_highN
Corrected EventT0 for EKLM scintillator with high hit count.
TProfile * prof_deltaT0_rms_vs_v_scint_end
DeltaT0 RMS vs inverse hit count profile for EKLM scintillator.
TH1F * h_time_scint_tc_end
EKLM part.
TH1F * hc_eventT0_scint_midN
Corrected EventT0 for BKLM scintillator with medium hit count.
void createHistograms()
Create histograms.
TGraphErrors * gre_time_channel_scint
BKLM Scintillator.
TH1F * h_timeFSL[2][8][15]
BKLM part.
TH1F * hc_timeFSL_end[2][4][14]
EKLM part.
TH1F * h_timeFSLP_end[2][4][14][2]
EKLM part.
TGraph * gr_timeRes_channel_rpc
BKLM RPC.
TH1F * hc_timeFSLP_end[2][4][14][2]
EKLM part.
TH1F * h_timeFSLP[2][8][15][2]
BKLM part.
TH1F * hc_timeF_scint_end[2]
EKLM part.
std::map< KLMChannelNumber, double > m_timeShift
Shift values of each channel.
TH1F * h_time_scint
BKLM scintillator part.
double m_time_channelAvg_scint
Central value of the global time distribution (BKLM scintillator part).
TH1F * hc_timeFS_scint_end[2][4]
EKLM part.
double esti_timeRes(const KLMChannelIndex &klmChannel)
Estimate value of calibration constant for calibrated channels.
double m_UpperTimeBoundaryCalibratedRPC
Upper time boundary for RPC (calibrated data).
double m_ctime_channelAvg_rpc
Calibrated central value of the global time distribution (BKLM RPC part).
KLMTimeConstants * m_timeConstants
DBObject of time cost on some parts of the detector.
void setupDatabase()
Setup the database.
TProfile * prof_deltaT0_rms_vs_v_rpc
DeltaT0 RMS vs inverse hit count profile for RPC.
TH1F * h_eventT0_scint
EventT0 seen by BKLM scintillator hits.
TH1F * hc_timeFS_scint[2][8]
BKLM scintillator part.
std::map< KLMChannelNumber, double > m_time_channel
Time distribution central value of each channel.
TH1F * hc_eventT0_rpc_lowN
Corrected EventT0 for RPC with low hit count.
double m_ctime_channelAvg_scint_end
Calibrated central value of the global time distribution (EKLM scintillator part).
CalibrationAlgorithm::EResult readCalibrationData()
Read calibration data.
TH1F * hc_eventT0_scint_lowN
Corrected EventT0 for BKLM scintillator with low hit count.
TGraph * gr_timeShift_channel_scint_end
EKLM.
TGraph * gr_timeRes_channel_scint
BKLM scintillator.
TH2F * h2_deltaT0_vs_v_scint
DeltaT0 vs inverse hit count for BKLM scintillator.
TH1F * hc_timeF_scint[2]
BKLM scintillator part.
TH1F * h_timeFS_scint[2][8]
BKLM scintillator part.
bool m_saveChannelHists
Write per-channel temporary histograms (tc/raw/hc) in minimal mode.
double m_UpperTimeBoundaryScintillatorsBKLM
Upper time boundary for BKLM scintillators.
const KLMElementNumbers * m_ElementNumbers
Element numbers.
void writeThenDelete_(TH1 *h, bool write, TDirectory *dir=nullptr)
Optionally write a histogram, then delete it to free memory.
std::pair< int, double > tR_upperStrip(const KLMChannelIndex &klmChannel)
Tracing available channels with increasing strip number.
TH1F * hc_timeF_rpc[2]
BKLM RPC part.
TH2F * h2c_timeFS_end[2][4]
EKLM part.
const EKLM::GeometryData * m_EKLMGeometry
EKLM geometry data.
TGraphErrors * gre_ctime_channel_scint_end
EKLM.
TH1F * h_nHits_plus_scint_end
Number of EKLM scintillator hits per mu+ track.
TProfile * m_Profile2BKLMScintillatorPhi
For BKLM scintillator phi plane.
TH1F * hc_time_scint_end
EKLM part.
TH1F * hc_eventT0_scint
Corrected EventT0 for BKLM scintillator hits.
TGraphErrors * gre_time_channel_scint_end
EKLM.
TH2F * h2_timeFSLP[2][8][15][2]
BKLM part.
double m_UpperTimeBoundaryCalibratedScintillatorsEKLM
Upper time boundary for BKLM scintillators (calibrated data).
TGraph * gr_timeShift_channel_scint
BKLM scintillator.
double m_time_channelAvg_scint_end
Central value of the global time distribution (EKLM scintillator part).
TProfile * m_Profile2EKLMScintillatorPlane1
For EKLM scintillator plane1.
TH1F * hc_timeFS_rpc[2][8]
BKLM RPC part.
double m_UpperTimeBoundaryScintillatorsEKLM
Upper time boundary for BKLM scintillators.
void fillTimeDistanceProfiles(TProfile *profileRpcPhi, TProfile *profileRpcZ, TProfile *profileBKLMScintillatorPhi, TProfile *profileBKLMScintillatorZ, TProfile *profileEKLMScintillatorPlane1, TProfile *profileEKLMScintillatorPlane2, bool fill2dHistograms)
Fill profiles of time versus distance.
TFile * m_outFile
Output file.
double esti_timeShift(const KLMChannelIndex &klmChannel)
Estimate value of calibration constant for uncalibrated channels.
std::pair< int, double > tS_upperStrip(const KLMChannelIndex &klmChannel)
Tracing available channels with increasing strip number.
double m_LowerTimeBoundaryCalibratedScintillatorsEKLM
Lower time boundary for EKLM scintillators (calibrated data).
TH1F * hc_timeFSLP[2][8][15][2]
BKLM part.
TGraphErrors * gre_ctime_channel_rpc
BKLM RPC.
void saveHist()
Save histograms to file.
const bklm::GeometryPar * m_BKLMGeometry
BKLM geometry data.
bool m_saveAllPlots
Default minimal unless you set true in your header script.
TH2F * h2c_timeFSLP[2][8][15][2]
BKLM part.
double m_ctime_channelAvg_scint
Calibrated central value of the global time distribution (BKLM scintillator part).
TF1 * fcn_const
Const function.
double m_UpperTimeBoundaryCalibratedScintillatorsBKLM
Upper time boundary for BKLM scintillators (calibrated data).
TProfile * m_Profile2RpcZ
For BKLM RPC z plane.
TH1F * h_nHits_minus_rpc
Number of RPC hits per mu- track.
TH2F * h2_timeFSLP_end[2][4][14][2]
EKLM part.
TH1I * hc_calibrated
Calibration statistics for each channel.
TH1F * h_eventT0_rpc
EventT0 seen by RPC hits.
void timeDistance2dFit(const std::vector< std::pair< KLMChannelNumber, unsigned int > > &channels, double &delay, double &delayError)
Two-dimensional fit for individual channels.
TGraph * gr_timeRes_channel_scint_end
EKLM.
TH1I * h_calibrated
Calibration statistics for each channel.
TProfile * m_ProfileBKLMScintillatorZ
For BKLM scintillator z plane.
double m_LowerTimeBoundaryScintillatorsBKLM
Lower time boundary for BKLM scintillators.
TH1F * hc_eventT0_scint_end_midN
Corrected EventT0 for EKLM scintillator with medium hit count.
TH1F * h_time_rpc_tc
BKLM RPC part.
TH1F * h_time_scint_end
EKLM part.
TH2F * h2c_timeF_scint[2]
BKLM scintillator part.
TF1 * fcn_pol1
Pol1 function.
double m_etime_channelAvg_scint_end
Central value error of the global time distribution (EKLM scintillator part).
TH1F * hc_time_rpc
BKLM RPC part.
double mc_etime_channelAvg_scint
Calibrated central value error of the global time distribution (BKLM scintillator part).
TH2F * h2c_timeFSLP_end[2][4][14][2]
EKLM part.
double mc_etime_channelAvg_scint_end
Calibrated central value error of the global time distribution (EKLM scintillator part).
bool passesADCCut(const Event &event, int subdetector, int layer) const
Check if event passes ADC count cuts for quality selection.
TH2F * h2_timeF_scint_end[2]
EKLM part.
KLMEventT0HitResolution * m_eventT0HitResolution
DBObject of per-hit time resolution for EventT0.
TF1 * fcn_gaus
Gaussian function.
double m_LowerTimeBoundaryCalibratedScintillatorsBKLM
Lower time boundary for BKLM scintillators (calibrated data).
TH1F * h_timeF_rpc[2]
BKLM RPC part.
TProfile * m_ProfileBKLMScintillatorPhi
For BKLM scintillator phi plane.
TH1F * hc_time_scint
BKLM scintillator part.
TH2F * h2_timeFS[2][8]
BKLM part.
double m_fixedRPCDelay
Fixed propagation delay for RPCs (ns/cm).
double m_etime_channelAvg_scint
Central value error of the global time distribution (BKLM scintillator part).
TH1F * h_timeF_scint_end[2]
EKLM part.
TH1F * hc_eventT0_scint_end
Corrected EventT0 for EKLM scintillator hits.
TProfile * m_ProfileRpcPhi
For BKLM RPC phi plane.
TGraphErrors * gre_ctime_channel_scint
BKLM Scintillator.
TH2F * h2_deltaT0_vs_nhits_scint
DeltaT0 vs total hit count for BKLM scintillator.
TH2F * h2_deltaT0_vs_nhits_rpc
DeltaT0 vs total hit count for RPC.
TProfile * m_ProfileEKLMScintillatorPlane2
For EKLM scintillator plane2.
TH1F * h_time_rpc
BKLM RPC part.
TH1F * h_timeFSL_end[2][4][14]
EKLM part.
TProfile * m_Profile2RpcPhi
For BKLM RPC phi plane.
TH1F * h_timeF_scint[2]
BKLM scintillator part.
TH1F * hc_timeFSL[2][8][15]
BKLM part.
TProfile * m_Profile2BKLMScintillatorZ
For BKLM scintillator z plane.
TH2F * h2_deltaT0_vs_v_scint_end
DeltaT0 vs inverse hit count for EKLM scintillator.
TH1F * h_timeFS_rpc[2][8]
BKLM RPC part.
KLMChannelIndex m_klmChannels
KLM ChannelIndex object.
TGraph * gr_timeShift_channel_rpc
BKLM RPC.
std::map< KLMChannelNumber, double > m_timeRes
Resolution values of each channel.
TH1F * h_eventT0_scint_end
EventT0 seen by EKLM scintillator hits.
TH1F * h_diff
Distance between global and local position.
TH1F * hc_eventT0_rpc_highN
Corrected EventT0 for RPC with high hit count.
TH1F * h_nHits_plus_scint
Number of BKLM scintillator hits per mu+ track.
TH2F * h2_timeF_scint[2]
BKLM scintillator part.
TH1F * h_time_scint_tc
BKLM scintillator part.
double m_LowerTimeBoundaryRPC
Lower time boundary for RPC.
virtual EResult calibrate() override
Run algorithm on data.
std::map< KLMChannelNumber, double > m_ctime_channel
Calibrated time distribution central value of each channel.
double m_LowerTimeBoundaryCalibratedRPC
Lower time boundary for RPC (calibrated data).
TH2F * h2_deltaT0_vs_nhits_scint_end
DeltaT0 vs total hit count for EKLM scintillator.
bool m_useEventT0
Whether to use event T0 from CDC.
int m_MinimalDigitNumber
Minimal digit number (total).
TProfile * m_ProfileEKLMScintillatorPlane1
For EKLM scintillator plane1.
double m_UpperTimeBoundaryRPC
Upper time boundary for RPC.
TH2F * h2c_timeF_rpc[2]
BKLM RPC part.
TH1F * hc_eventT0_scint_highN
Corrected EventT0 for BKLM scintillator with high hit count.
TF1 * fcn_land
Landau function.
KLMTimeCableDelay * m_timeCableDelay
DBObject of the calibration constant of each channel due to cable decay.
TH2F * h2_deltaT0_vs_v_rpc
DeltaT0 vs inverse hit count for RPC.
std::pair< int, double > tS_lowerStrip(const KLMChannelIndex &klmChannel)
Tracing available channels with decreasing strip number.
TProfile * m_ProfileRpcZ
For BKLM RPC z plane.
bool m_useFixedRPCDelay
Whether to use fixed propagation delay for RPCs.
std::map< KLMChannelNumber, double > mc_etime_channel
Calibrated time distribution central value Error of each channel.
std::pair< int, double > tR_lowerStrip(const KLMChannelIndex &klmChannel)
Tracing available channels with decreasing strip number.
void readCalibrationDataBatch(std::function< bool(const KLMChannelIndex &)> channelFilter)
Load calibration data for a specific batch of channels.
TH1F * hc_eventT0_rpc_midN
Corrected EventT0 for RPC with medium hit count.
TH1F * h_nHits_minus_scint
Number of BKLM scintillator hits per mu- track.
ROOT::Math::MinimizerOptions m_minimizerOptions
Minimization options.
TProfile * prof_deltaT0_rms_vs_v_scint
DeltaT0 RMS vs inverse hit count profile for BKLM scintillator.
std::map< KLMChannelNumber, int > m_cFlag
Calibration flag if the channel has enough hits collected and fitted OK.
TGraphErrors * gre_time_channel_rpc
BKLM RPC.
TH2F * h2_timeFS_end[2][4]
EKLM part.
TH2F * h2_timeF_rpc[2]
BKLM RPC part.
TH1F * h_nHits_plus_rpc
Number of RPC hits per mu+ track.
std::map< KLMChannelNumber, std::vector< struct Event > > m_evts
Container of hit information.
TDirectory * m_channelHistDir_EKLM[2][4][14][2]
Directory structure for per-channel histograms (EKLM).
TH1F * hc_eventT0_rpc
Corrected EventT0 for RPC hits.
TH1F * h_nHits_minus_scint_end
Number of EKLM scintillator hits per mu- track.
TH1F * h_timeFS_scint_end[2][4]
EKLM part.
double m_time_channelAvg_rpc
Central value of the global time distribution (BKLM RPC part).
TH2F * h2c_timeFS[2][8]
BKLM part.
double m_etime_channelAvg_rpc
Central value error of the global time distribution (BKLM RPC part).
double m_LowerTimeBoundaryScintillatorsEKLM
Lower time boundary for EKLM scintillators.
void readCalibrationDataCounts(std::map< KLMChannelNumber, unsigned int > &eventCounts)
Count events per channel (lightweight scan without loading full data).
std::map< KLMChannelNumber, double > m_etime_channel
Time distribution central value Error of each channel.
int m_lower_limit_counts
Lower limit of hits collected for on single channel.
TH1F * hc_eventT0_scint_end_lowN
Corrected EventT0 for EKLM scintillator with low hit count.
void readCalibrationDataFor2DFit(const std::vector< std::pair< KLMChannelNumber, unsigned int > > &channelsBKLM, const std::vector< std::pair< KLMChannelNumber, unsigned int > > &channelsEKLM)
Load calibration data only for channels needed for 2D fit.
Class to store BKLM delay time coused by cable in the database.
Class to store KLM constants related to time.
@ c_BKLM
BKLM scintillator.
@ c_EKLM
EKLM scintillator.
Class to store KLM time resolution in the database.
bool registerInDataStore(DataStore::EStoreFlags storeFlags=DataStore::c_WriteOut)
Register the object/array in the DataStore.
Type-safe access to single objects in the data store.
Definition StoreObjPtr.h:96
bool isValid() const
Check whether the object was created.
bool construct(Args &&... params)
Construct an object of type T in this StoreObjPtr, using the provided constructor arguments.
static const double cm
Standard units with the value = 1.
Definition Unit.h:47
static GeometryPar * instance(void)
Static method to get a reference to the singleton GeometryPar instance.
Define the geometry of a BKLM module Each sector [octant] contains Modules.
Definition Module.h:76
Class to store variables with their name which were sent to the logging service.
std::shared_ptr< T > getObjectPtr(const std::string &name, const std::vector< Calibration::ExpRun > &requestedRuns)
Get calibration data object by name and list of runs, the Merge function will be called to generate t...
static DBStore & Instance()
Instance of a singleton DBStore.
Definition DBStore.cc:26
void updateEvent()
Updates all intra-run dependent objects.
Definition DBStore.cc:140
void update()
Updates all objects that are outside their interval of validity.
Definition DBStore.cc:77
double sqrt(double a)
sqrt for double
Definition beamHelpers.h:28
uint16_t KLMChannelNumber
Channel number.
Abstract base class for different kinds of events.