12#ifndef DETECTOR_CALIB_TIMESLOTCALIB_H_
13#define DETECTOR_CALIB_TIMESLOTCALIB_H_
34class ProcessingContext;
39template <
typename Container>
98 LOGP(info,
"Start offset is not supported in the INFINITE_TF slot length or UpdateAtTheEndOfRunOnly or FinalizeWhenReady modes");
104 }
else if (
f > 0.95) {
110 LOGP(info,
"Imposing offset of {:4.2} x nominal slot length",
mStartOffsetFrac);
130 template <
typename... DATA>
167 template <
typename,
typename T>
170 std::integral_constant<T, false>::value,
171 "Second template parameter needs to be of function type.");
176 template <
typename C,
typename Ret,
typename... Args>
179 template <
typename T>
180 static constexpr auto check(T*)
181 ->
typename std::is_same<decltype(std::declval<T>().fill(std::declval<Args>()...)), Ret>
::type;
183 static constexpr std::false_type
check(...);
184 typedef decltype(check<C>(
nullptr)) type;
187 static constexpr bool value = type::value;
203 LOG(fatal) <<
"This method must be implemented by derived class to write content of the slot to save";
210 LOG(fatal) <<
"This method must be implemented by derived class to adopt content of the saved slot";
224 static bool suppressRunStartWarning =
false;
225 if (!suppressRunStartWarning) {
229 suppressRunStartWarning =
true;
232 suppressRunStartWarning =
true;
241 return uint32_t(orb);
280template <
typename Container>
281template <
typename... DATA>
284 static bool firstCall =
true;
291 TFType tf = mCurrentTFInfo.tfCounter;
292 uint64_t maxDelay64 = uint64_t(mSlotLength * mMaxSlotsDelay);
294 if (!mUpdateAtTheEndOfRunOnly) {
295 if (
tf < mLastClosedTF || (!mSlots.empty() && getLastSlot().getTFStart() >
tf + maxDelay64)) {
299 LOG(info) <<
"Ignoring TF " <<
tf <<
", mLastClosedTF = " << mLastClosedTF;
303 auto& slotTF = getSlotForTF(
tf);
304 using Cont_t =
typename std::remove_pointer<
decltype(slotTF.getContainer())>
::type;
306 slotTF.getContainer()->fill(mCurrentTFInfo,
data...);
308 slotTF.getContainer()->fill(
data...);
310 if (
tf > mMaxSeenTF) {
313 if (!mUpdateAtTheEndOfRunOnly) {
315 checkSlotsToFinalize(
tf, maxDelay);
322template <
typename Container>
328 if (mSlots.size() == 1 && mFinalizeWhenReady) {
329 TFType checkInterval = mCheckIntervalInfiniteSlot + mLastCheckedTFInfiniteSlot;
330 if (mWasCheckedInfiniteSlot) {
331 checkInterval = mCheckDeltaIntervalInfiniteSlot + mLastCheckedTFInfiniteSlot;
334 LOG(
debug) <<
"mMaxSeenTF = " << mMaxSeenTF <<
", mLastCheckedTFInfiniteSlot = " << mLastCheckedTFInfiniteSlot <<
", checkInterval = " << checkInterval <<
", mSlots[0].getTFStart() = " << mSlots[0].getTFStart();
336 LOG(info) <<
"End of run reached, trying to calibrate what we have, if we have enough statistics";
338 LOG(info) <<
"Calibrating as soon as we have enough statistics:";
339 LOG(info) <<
"Update interval passed (" << checkInterval <<
"), checking slot for " << mSlots[0].getTFStart() <<
" <= TF <= " <<
INFINITE_TF;
341 mLastCheckedTFInfiniteSlot =
tf;
342 if (hasEnoughData(mSlots[0])) {
343 mWasCheckedInfiniteSlot =
false;
344 mSlots[0].setTFStart(mLastClosedTF);
345 mSlots[0].setTFEnd(mMaxSeenTF);
346 LOG(info) <<
"Finalizing slot for " << mSlots[0].getTFStart() <<
" <= TF <= " << mSlots[0].getTFEnd();
347 finalizeSlot(mSlots[0]);
348 mLastClosedTF = mSlots[0].getTFEnd() <
INFINITE_TF ? (mSlots[0].getTFEnd() + 1) : mSlots[0].getTFEnd() <
INFINITE_TF;
349 mSlots.erase(mSlots.begin());
352 LOG(info) <<
"Creating new slot for " << mLastClosedTF <<
" <= TF <= " <<
INFINITE_TF;
353 auto& sl = emplaceNewSlot(
true, mLastClosedTF,
INFINITE_TF);
354 sl.setRunStartOrbit(getRunStartOrbit());
357 LOG(info) <<
"Not enough data to calibrate";
358 mWasCheckedInfiniteSlot =
true;
361 LOG(
debug) <<
"Not trying to calibrate: either not at EoS, or update interval not passed";
365 for (
auto slot = mSlots.begin(); slot != mSlots.end();) {
366 uint64_t lim64 = uint64_t(maxDelay) + slot->getTFEnd();
369 if (hasEnoughData(*slot)) {
370 LOG(
debug) <<
"Finalizing slot for " << slot->getTFStart() <<
" <= TF <= " << slot->getTFEnd();
372 }
else if ((slot + 1) != mSlots.end()) {
373 LOG(info) <<
"Merging underpopulated slot " << slot->getTFStart() <<
" <= TF <= " << slot->getTFEnd()
374 <<
" to slot " << (slot + 1)->getTFStart() <<
" <= TF <= " << (slot + 1)->getTFEnd();
375 (slot + 1)->mergeToPrevious(*slot);
377 LOG(info) <<
"Discard underpopulated slot " << slot->getTFStart() <<
" <= TF <= " << slot->getTFEnd();
380 mLastClosedTF = slot->getTFEnd() + 1;
381 LOG(info) <<
"closing slot " << slot->getTFStart() <<
" <= TF <= " << slot->getTFEnd();
382 slot = mSlots.erase(slot);
391template <
typename Container>
395 if (mSlots.empty()) {
396 LOG(warning) <<
"There are no slots defined";
399 finalizeSlot(mSlots.front());
400 mLastClosedTF = mSlots.front().getTFEnd() + 1;
401 mSlots.erase(mSlots.begin());
405template <
typename Container>
410 throw std::runtime_error(
"invalid TF");
412 if (mUpdateAtTheEndOfRunOnly) {
416 tft = int64_t(((
tf - mFirstTF + mStartOffsetTFs) / mSlotLength) * mSlotLength) + mFirstTF;
417 if (tft > mStartOffsetTFs) {
418 tft -= mStartOffsetTFs;
426template <
typename Container>
431 if (mUpdateAtTheEndOfRunOnly) {
432 if (!mSlots.empty() && mSlots.back().getTFEnd() <
tf) {
434 }
else if (mSlots.empty()) {
435 auto& sl = emplaceNewSlot(
true, mFirstTF,
tf);
436 sl.setRunStartOrbit(getRunStartOrbit());
437 sl.setStaticStartTimeMS(sl.getStartTimeMS());
439 return mSlots.back();
442 if (!mSlots.empty() && mSlots.front().getTFStart() >
tf) {
443 auto tfmn = tf2SlotMin(mSlots.front().getTFStart() - 1);
444 auto tftgt = tf2SlotMin(
tf);
445 while (tfmn >= tftgt) {
446 uint64_t tft = mSlots.front().getTFStart() - 1;
448 LOG(info) <<
"Adding new slot for " << tfmn <<
" <= TF <= " << tfmx;
449 auto& sl = emplaceNewSlot(
true, tfmn, tfmx);
450 sl.setRunStartOrbit(getRunStartOrbit());
451 sl.setStaticStartTimeMS(sl.getStartTimeMS());
455 tfmn = tf2SlotMin(mSlots.front().getTFStart() - 1);
459 for (
auto it = mSlots.begin(); it != mSlots.end(); it++) {
460 auto rel = (*it).relateToTF(
tf);
466 auto tfmn = mSlots.empty() ? tf2SlotMin(
tf) : tf2SlotMin(mSlots.back().getTFEnd() + 1);
468 uint64_t tft = uint64_t(tfmn) + mSlotLength - 1;
469 if (mSlots.empty() && mStartOffsetTFs &&
tf < mStartOffsetTFs) {
470 tft -= mStartOffsetTFs;
473 LOG(info) <<
"Adding new slot for " << tfmn <<
" <= TF <= " << tfmx;
474 auto& sl = emplaceNewSlot(
false, tfmn, tfmx);
475 sl.setRunStartOrbit(getRunStartOrbit());
476 sl.setStaticStartTimeMS(sl.getStartTimeMS());
478 }
while (
tf > mSlots.back().getTFEnd());
480 return mSlots.back();
484template <
typename Container>
487 for (
int i = 0;
i < getNSlots();
i++) {
488 LOG(info) <<
"Slot #" <<
i + 1 <<
" of " << getNSlots();
494template <
typename Container>
497 if (mSlots.empty()) {
498 LOG(warn) <<
"Nothing to save, no TimeSlots defined";
501 if (mSaveMetaData.startRun < 0) {
502 mSaveMetaData.startRun = mCurrentTFInfo.runNumber;
504 mSaveMetaData.endRun = mCurrentTFInfo.runNumber;
505 if (mSaveMetaData.startTime < 0) {
506 mSaveMetaData.startTime = mSlots.back().getStartTimeMS();
508 mSaveMetaData.endTime = mSlots.back().getEndTimeMS();
513template <
typename Container>
516 if (!getSavedSlotAllowed()) {
517 LOG(info) <<
"Slot saving is disabled";
520 if (!updateSaveMetaData()) {
524 if (!mSaveDirectory.empty() && !std::filesystem::exists(mSaveDirectory)) {
525 std::filesystem::create_directories(mSaveDirectory);
526 if (!std::filesystem::exists(mSaveDirectory)) {
527 LOGP(fatal,
"could not create output directory {}", mSaveDirectory);
529 LOGP(info,
"created calibration directory {}", mSaveDirectory);
533 auto pth = getSaveFilePath();
534 auto pthTmp = pth +
".part";
535 TFile flout(pthTmp.c_str(),
"recreate");
536 if (flout.IsZombie()) {
537 LOGP(error,
"failed to open save file {}", pth);
538 unlink(pthTmp.c_str());
541 if (!saveLastSlotData(flout)) {
543 unlink(pthTmp.c_str());
546 flout.WriteObjectAny(&mSaveMetaData,
"o2::calibration::TimeSlotMetaData",
"metadata");
548 std::filesystem::rename(pthTmp, pth);
549 LOGP(info,
"Saved data of the last slot to {}", pth);
554template <
typename Container>
557 if (!getSavedSlotAllowed()) {
558 LOG(info) <<
"Saved slot usage is disabled";
561 auto pth = getSaveFilePath();
562 if (!std::filesystem::exists(pth)) {
563 LOGP(info,
"No save file {} is found", pth);
566 TFile flin(pth.c_str());
567 if (flin.IsZombie()) {
568 LOGP(error,
"failed to open save file {}", pth);
573 LOGP(error,
"Failed to read metadata from {}", pth);
576 auto res = adoptSavedData(*meta, flin);
578 mSaveMetaData.startRun = meta->startRun;
579 mSaveMetaData.startTime = meta->startTime;
580 updateSaveMetaData();
588template <
typename Container>
591 if (mSaveFileName.empty()) {
592 LOGP(fatal,
"Save file name was not set");
594 return fmt::format(
"{}{}{}", mSaveDirectory, ((!mSaveDirectory.empty() && mSaveDirectory.back() !=
'/') ?
"/" :
""), mSaveFileName);
Helper for geometry and GRP related CCDB requests.
static int getNHBFPerTF()
static GRPGeomHelper & instance()
virtual bool adoptSavedData(const TimeSlotMetaData &metadata, TFile &fl)
static constexpr TFType INFINITE_TF
virtual bool saveLastSlotData(TFile &fl)
TFType getCheckDeltaIntervalInfiniteSlot() const
o2::calibration::TFType TFType
virtual void initOutput()=0
float getMaxSlotsDelay() const
TFType mCheckDeltaIntervalInfiniteSlot
void setFirstTF(TFType v)
virtual void checkSlotsToFinalize(TFType tf=INFINITE_TF, int maxDelay=0)
void setSaveFileName(const std::string &n)
TFType getFirstTF() const
ClassDef(TimeSlotCalibration, 1)
TFType mLastCheckedTFInfiniteSlot
const Slot & getLastSlot() const
void setCheckDeltaIntervalInfiniteSlot(TFType v)
std::string mSaveDirectory
virtual bool updateSaveMetaData()
void setStartOffsetFrac(float f)
virtual bool saveLastSlot()
const o2::dataformats::TFIDInfo & getCurrentTFInfo() const
virtual void print() const
Slot & getSlotForTF(TFType tf)
void setSlotLengthInOrbits(int n)
std::deque< Slot > mSlots
bool process(const DATA &... data)
TFType getSlotLength() const
TimeSlotCalibration()=default
o2::dataformats::TFIDInfo & getCurrentTFInfo()
virtual void finalizeOldestSlot()
const Slot & getSlot(int i) const
virtual ~TimeSlotCalibration()=default
virtual void finalizeSlot(Slot &slot)=0
uint32_t getRunStartOrbit() const
void setSaveDirectory(const std::string &n)
void setSavedSlotAllowed(bool v)
void setSlotLengthInSeconds(int s)
void setFinalizeWhenReady()
o2::dataformats::TFIDInfo mCurrentTFInfo
void setSlotLength(TFType v)
virtual bool hasEnoughData(const Slot &slot) const =0
std::string mSaveFileName
void setUpdateAtTheEndOfRunOnly()
const std::string & getSaveFileName() const
const Slot & getFirstSlot() const
TFType getCheckIntervalInfiniteSlot() const
TFType mCheckIntervalInfiniteSlot
TimeSlotMetaData mSaveMetaData
bool getSavedSlotAllowed() const
bool mUpdateAtTheEndOfRunOnly
bool mWasCheckedInfiniteSlot
TFType tf2SlotMin(TFType tf) const
virtual bool loadSavedSlot()
std::string getSaveFilePath() const
virtual Slot & emplaceNewSlot(bool front, TFType tstart, TFType tend)=0
void setCheckIntervalInfiniteSlot(TFType v)
void setMaxSlotsDelay(float v)
GLsizei const GLfloat * value
GLint GLint GLsizei GLint GLenum GLenum type
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
constexpr TFType INFINITE_TF
constexpr double LHCOrbitMUS
void check(const std::vector< std::string > &arguments, const std::vector< ConfigParamSpec > &workflowOptions, const std::vector< DeviceSpec > &deviceSpecs, CheckMatrix &matrix)
a couple of static helper functions to create timestamp values for CCDB queries or override obsolete ...
std::unique_ptr< GPUReconstructionTimeframe > tf
LOG(info)<< "Compressed in "<< sw.CpuTime()<< " s"