Project
Loading...
Searching...
No Matches
TimeSlotCalibration.h
Go to the documentation of this file.
1// Copyright 2019-2020 CERN and copyright holders of ALICE O2.
2// See https://alice-o2.web.cern.ch/copyright for details of the copyright holders.
3// All rights not expressly granted are reserved.
4//
5// This software is distributed under the terms of the GNU General Public
6// License v3 (GPL Version 3), copied verbatim in the file "COPYING".
7//
8// In applying this license CERN does not waive the privileges and immunities
9// granted to it by virtue of its status as an Intergovernmental Organization
10// or submit itself to any jurisdiction.
11
12#ifndef DETECTOR_CALIB_TIMESLOTCALIB_H_
13#define DETECTOR_CALIB_TIMESLOTCALIB_H_
14
16
22#include <TFile.h>
23#include <filesystem>
24#include <deque>
25#include <gsl/gsl>
26#include <limits>
27#include <type_traits>
28#include <unistd.h>
29
30namespace o2
31{
32namespace framework
33{
34class ProcessingContext;
35}
36namespace calibration
37{
38
39template <typename Container>
41{
42 public:
45
47
49 virtual ~TimeSlotCalibration() = default;
50 float getMaxSlotsDelay() const { return mMaxSlotsDelay; }
51 void setMaxSlotsDelay(float v) { mMaxSlotsDelay = v > 0. ? v : 0.; }
52
53 TFType getSlotLength() const { return mSlotLength; }
55 {
56 if (v == 0) {
58 } else {
59 mSlotLength = v;
60 }
61 }
62
65
68
69 TFType getFirstTF() const { return mFirstTF; }
71
72 void setSlotLengthInSeconds(int s) { mSlotLengthInSeconds = s > 0 ? s : 1; }
73 void setSlotLengthInOrbits(int n) { mSlotLengthInOrbits = n > 0 ? n : 1; }
75 {
76 if (mSlotLengthInSeconds > 0) {
78 LOGP(info, "Redefining slot duration from {} s. to {} TFs", mSlotLengthInSeconds, ntf);
79 setSlotLength(ntf);
81 } else if (mSlotLengthInOrbits > 0) {
83 if (ntf < 1) {
84 ntf = 1;
85 }
86 LOGP(info, "Redefining slot duration from {} orbits to {} TFs", mSlotLengthInOrbits, ntf);
87 setSlotLength(ntf);
89 }
90 setStartOffsetFrac(mStartOffsetFrac); // set once more to account for eventual dependencies
92 }
93
95 {
96 if (mUpdateAtTheEndOfRunOnly || mFinalizeWhenReady || mSlotLength == INFINITE_TF) { // offset makes no sense for run-wide objects
97 if (f) {
98 LOGP(info, "Start offset is not supported in the INFINITE_TF slot length or UpdateAtTheEndOfRunOnly or FinalizeWhenReady modes");
99 }
100 return;
101 }
102 if (f < 0.) {
103 mStartOffsetFrac = 0.;
104 } else if (f > 0.95) {
105 mStartOffsetFrac = 0.95;
106 } else {
108 }
110 LOGP(info, "Imposing offset of {:4.2} x nominal slot length", mStartOffsetFrac);
111 }
112 }
113
120
122
123 int getNSlots() const { return mSlots.size(); }
125 Slot& getSlot(int i) { return (Slot&)mSlots.at(i); }
126 const Slot& getSlot(int i) const { return (Slot&)mSlots.at(i); }
127 const Slot& getLastSlot() const { return (Slot&)mSlots.back(); }
128 const Slot& getFirstSlot() const { return (Slot&)mSlots.front(); }
129
130 template <typename... DATA>
131 bool process(const DATA&... data);
132 virtual void checkSlotsToFinalize(TFType tf = INFINITE_TF, int maxDelay = 0);
133 virtual void finalizeOldestSlot();
134
135 virtual void reset()
136 { // reset to virgin state (need for start - stop - start)
137 mSlots.clear();
138 mLastClosedTF = 0;
139 mFirstTF = 0;
140 mMaxSeenTF = 0;
143 initOutput();
144 }
145
146 // Methods to be implemented by the derived user class
147
148 // implement and call this method te reset the output slots once they are not needed
149 virtual void initOutput() = 0;
150 // process the time slot container and add results to the output
151 virtual void finalizeSlot(Slot& slot) = 0;
152 // create new time slot in the beginning or the end of the slots pool
153 virtual Slot& emplaceNewSlot(bool front, TFType tstart, TFType tend) = 0;
154 // check if the slot has enough data to be finalized
155 virtual bool hasEnoughData(const Slot& slot) const = 0;
156
157 virtual void print() const;
158
161
162 // from https://stackoverflow.com/questions/87372/check-if-a-class-has-a-member-function-of-a-given-signature
163 // Primary template with a static assertion
164 // for a meaningful error message
165 // if it ever gets instantiated.
166 // We could leave it undefined if we didn't care.
167 template <typename, typename T>
169 static_assert(
170 std::integral_constant<T, false>::value,
171 "Second template parameter needs to be of function type.");
172 };
173
174 // specialization that does the checking
175
176 template <typename C, typename Ret, typename... Args>
177 struct has_fill_method<C, Ret(Args...)> {
178 private:
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; // attempt to call it and see if the return type is correct
182 template <typename>
183 static constexpr std::false_type check(...);
184 typedef decltype(check<C>(nullptr)) type;
185
186 public:
187 static constexpr bool value = type::value;
188 };
189
190 // methods for saving/reading data in the of the run in case of insufficient statistics
191 bool getSavedSlotAllowed() const { return mSavedSlotAllowed; }
193 std::string getSaveFilePath() const;
194 const std::string& getSaveFileName() const { return mSaveFileName; }
195 void setSaveFileName(const std::string& n) { mSaveFileName = n; }
196 void setSaveDirectory(const std::string& n) { mSaveDirectory = n; }
197 virtual bool updateSaveMetaData();
198
199 // derived class using slot saving functionality must implement this method to write the
200 // content of the slot, returning true on success
201 virtual bool saveLastSlotData(TFile& fl)
202 {
203 LOG(fatal) << "This method must be implemented by derived class to write content of the slot to save";
204 return false;
205 }
206 // derived class using slot saving functionality must implement this method to adopt the content of the
207 // saved slot, returning true on success. Provided metadata should be used to judge if the saved data is useful.
208 virtual bool adoptSavedData(const TimeSlotMetaData& metadata, TFile& fl)
209 {
210 LOG(fatal) << "This method must be implemented by derived class to adopt content of the saved slot";
211 return false;
212 }
213 virtual bool loadSavedSlot();
214 virtual bool saveLastSlot();
215
216 protected:
217 auto& getSlots() { return mSlots; }
218 uint32_t getRunStartOrbit() const
219 {
221 static unsigned int threshold = 512 * o2::base::GRPGeomHelper::getNHBFPerTF();
222 if (orb < 0) {
223 // If we have a firstTForbit between 1 and 512 * tf len, we disable the warning for negative runStartOrbit permanently, since this is a SYNTHETIC run.
224 static bool suppressRunStartWarning = false;
225 if (!suppressRunStartWarning) {
226 const auto* grpecs = o2::base::GRPGeomHelper::instance().getGRPECS();
227 if (grpecs) {
228 if (grpecs->getRunType() == o2::parameters::GRPECS::SYNTHETIC) {
229 suppressRunStartWarning = true;
230 }
231 } else if (mCurrentTFInfo.firstTForbit < threshold && mCurrentTFInfo.firstTForbit > 0) {
232 suppressRunStartWarning = true;
233 }
234 }
235
236 if (!suppressRunStartWarning && mCurrentTFInfo.firstTForbit >= threshold) {
237 LOGP(alarm, "Negative runStartOrbit = {} deduced from tfCounter={} and firstTForbit={}, enforcing runStartOrbit to 0", orb, mCurrentTFInfo.tfCounter, mCurrentTFInfo.firstTForbit);
238 }
239 orb = 0;
240 }
241 return uint32_t(orb);
242 }
243
245 std::deque<Slot> mSlots;
246
248 int mSlotLengthInSeconds = -1; // optionally provided slot length in seconds
249 int mSlotLengthInOrbits = -1; // optionally provided slot length in orbits
252 TFType mMaxSeenTF = 0; // largest TF processed
253 TFType mSlotLength = 1; // slot length in TFs
254 TFType mStartOffsetTFs = 0; // shift start of all TFs backwards by this amount (to make 1st slot effectively shorter: run_1st_tf to run_1st_tf - offset + mSlotLength), derived from mStartOffsetFrac
255 float mStartOffsetFrac = 0.; // shift start of all TFs backwards mSlotLength*mStartOffsetFrac TFs.
256 TFType mCheckIntervalInfiniteSlot = 1; // will be used if the TF length is INFINITE_TF_int64 to decide
257 // when to check if to call the finalize; otherwise it is called
258 // at every new TF; note that this is an approximation,
259 // since TFs come in async order
260 TFType mLastCheckedTFInfiniteSlot = 0; // will be used if the TF length is INFINITE_TF_int64 to book-keep
261 // the last TF at which we tried to calibrate
262 TFType mCheckDeltaIntervalInfiniteSlot = 1; // will be used if the TF length is INFINITE_TF_int64 when
263 // the check on the statistics returned false, to determine
264 // after how many TF to check again.
265 float mMaxSlotsDelay = 3.0; // difference in slot units between the current TF and oldest slot (end TF) to account for the TF
266
267 bool mWasCheckedInfiniteSlot = false; // flag to know whether the statistics of the infinite slot was already checked
269 bool mFinalizeWhenReady = false; // if true: single bin is filled until ready, then closed and new one is added
270
271 std::string mSaveDirectory = ""; // directory where the file is saved
272 std::string mSaveFileName = ""; // filename for data saves in the end of the run
274 bool mSavedSlotAllowed = false;
275
277};
278
279//_________________________________________________
280template <typename Container>
281template <typename... DATA>
283{
284 static bool firstCall = true;
285 if (firstCall) {
286 firstCall = false;
287 checkSlotLength();
288 }
289
290 // process current TF
291 TFType tf = mCurrentTFInfo.tfCounter;
292 uint64_t maxDelay64 = uint64_t(mSlotLength * mMaxSlotsDelay);
293 TFType maxDelay = maxDelay64 > o2::calibration::INFINITE_TF ? o2::calibration::INFINITE_TF : TFType(maxDelay64);
294 if (!mUpdateAtTheEndOfRunOnly) { // if you update at the end of run only, then you accept everything
295 if (tf < mLastClosedTF || (!mSlots.empty() && getLastSlot().getTFStart() > tf + maxDelay64)) { // ignore TF; note that if you have only 1 timeslot
296 // which is INFINITE_TF wide, then maxDelay
297 // does not matter: you won't accept TFs from the past,
298 // so the first condition will be used
299 LOG(info) << "Ignoring TF " << tf << ", mLastClosedTF = " << mLastClosedTF;
300 return false;
301 }
302 }
303 auto& slotTF = getSlotForTF(tf);
304 using Cont_t = typename std::remove_pointer<decltype(slotTF.getContainer())>::type;
305 if constexpr (has_fill_method<Cont_t, void(const o2::dataformats::TFIDInfo&, const DATA&...)>::value) {
306 slotTF.getContainer()->fill(mCurrentTFInfo, data...);
307 } else {
308 slotTF.getContainer()->fill(data...);
309 }
310 if (tf > mMaxSeenTF) {
311 mMaxSeenTF = tf; // keep track of the most recent TF processed
312 }
313 if (!mUpdateAtTheEndOfRunOnly) { // if you update at the end of run only, you don't check at every TF which slots can be closed
314 // check if some slots are done
315 checkSlotsToFinalize(tf, maxDelay);
316 }
317
318 return true;
319}
320
321//_________________________________________________
322template <typename Container>
324{
325 // Check which slots can be finalized, provided the newly arrived TF is tf
326
327 // if slot finalization is asked as soon as the slot is ready, we need to check if we got enough statistics, and if so, redefine the slot
328 if (mSlots.size() == 1 && mFinalizeWhenReady) {
329 TFType checkInterval = mCheckIntervalInfiniteSlot + mLastCheckedTFInfiniteSlot;
330 if (mWasCheckedInfiniteSlot) {
331 checkInterval = mCheckDeltaIntervalInfiniteSlot + mLastCheckedTFInfiniteSlot;
332 }
333 if (tf >= checkInterval || tf == INFINITE_TF) {
334 LOG(debug) << "mMaxSeenTF = " << mMaxSeenTF << ", mLastCheckedTFInfiniteSlot = " << mLastCheckedTFInfiniteSlot << ", checkInterval = " << checkInterval << ", mSlots[0].getTFStart() = " << mSlots[0].getTFStart();
335 if (tf == INFINITE_TF) {
336 LOG(info) << "End of run reached, trying to calibrate what we have, if we have enough statistics";
337 } else {
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;
340 }
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]); // will be removed after finalization
348 mLastClosedTF = mSlots[0].getTFEnd() < INFINITE_TF ? (mSlots[0].getTFEnd() + 1) : mSlots[0].getTFEnd() < INFINITE_TF; // will not accept any TF below this
349 mSlots.erase(mSlots.begin());
350 // creating a new slot if we are not at the end of run
351 if (tf != INFINITE_TF) {
352 LOG(info) << "Creating new slot for " << mLastClosedTF << " <= TF <= " << INFINITE_TF;
353 auto& sl = emplaceNewSlot(true, mLastClosedTF, INFINITE_TF);
354 sl.setRunStartOrbit(getRunStartOrbit());
355 }
356 } else {
357 LOG(info) << "Not enough data to calibrate";
358 mWasCheckedInfiniteSlot = true;
359 }
360 } else {
361 LOG(debug) << "Not trying to calibrate: either not at EoS, or update interval not passed";
362 }
363 } else {
364 // check if some slots are done
365 for (auto slot = mSlots.begin(); slot != mSlots.end();) {
366 uint64_t lim64 = uint64_t(maxDelay) + slot->getTFEnd();
367 TFType tfLim = lim64 < INFINITE_TF ? TFType(lim64) : INFINITE_TF;
368 if (tfLim < tf) {
369 if (hasEnoughData(*slot)) {
370 LOG(debug) << "Finalizing slot for " << slot->getTFStart() << " <= TF <= " << slot->getTFEnd();
371 finalizeSlot(*slot); // will be removed after finalization
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);
376 } else {
377 LOG(info) << "Discard underpopulated slot " << slot->getTFStart() << " <= TF <= " << slot->getTFEnd();
378 break; // slot has no enough stat. and there is no other slot to merge it to
379 }
380 mLastClosedTF = slot->getTFEnd() + 1; // will not accept any TF below this
381 LOG(info) << "closing slot " << slot->getTFStart() << " <= TF <= " << slot->getTFEnd();
382 slot = mSlots.erase(slot);
383 } else {
384 break; // all following slots will be even closer to the new TF
385 }
386 }
387 }
388}
389
390//_________________________________________________
391template <typename Container>
393{
394 // Enforce finalization and removal of the oldest slot
395 if (mSlots.empty()) {
396 LOG(warning) << "There are no slots defined";
397 return;
398 }
399 finalizeSlot(mSlots.front());
400 mLastClosedTF = mSlots.front().getTFEnd() + 1; // do not accept any TF below this
401 mSlots.erase(mSlots.begin());
402}
403
404//________________________________________
405template <typename Container>
407{
408 // returns the min TF of the slot to which "tf" belongs
409 if (tf < mFirstTF) {
410 throw std::runtime_error("invalid TF");
411 }
412 if (mUpdateAtTheEndOfRunOnly) {
413 return mFirstTF;
414 }
415 int64_t tft = 0;
416 tft = int64_t(((tf - mFirstTF + mStartOffsetTFs) / mSlotLength) * mSlotLength) + mFirstTF;
417 if (tft > mStartOffsetTFs) {
418 tft -= mStartOffsetTFs;
419 } else {
420 tft = 0;
421 }
422 return tft < o2::calibration::INFINITE_TF ? TFType(tft) : INFINITE_TF;
423}
424
425//_________________________________________________
426template <typename Container>
428{
429
430 LOG(debug) << "Getting slot for TF " << tf;
431 if (mUpdateAtTheEndOfRunOnly) {
432 if (!mSlots.empty() && mSlots.back().getTFEnd() < tf) {
433 mSlots.back().setTFEnd(tf);
434 } else if (mSlots.empty()) {
435 auto& sl = emplaceNewSlot(true, mFirstTF, tf);
436 sl.setRunStartOrbit(getRunStartOrbit());
437 sl.setStaticStartTimeMS(sl.getStartTimeMS());
438 }
439 return mSlots.back();
440 }
441
442 if (!mSlots.empty() && mSlots.front().getTFStart() > tf) { // we need to add a slot to the beginning
443 auto tfmn = tf2SlotMin(mSlots.front().getTFStart() - 1); // min TF of the slot corresponding to a TF smaller than the first seen
444 auto tftgt = tf2SlotMin(tf); // min TF of the slot to which the TF "tf" would belong
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());
452 if (!tfmn) {
453 break;
454 }
455 tfmn = tf2SlotMin(mSlots.front().getTFStart() - 1);
456 }
457 return mSlots[0];
458 }
459 for (auto it = mSlots.begin(); it != mSlots.end(); it++) {
460 auto rel = (*it).relateToTF(tf);
461 if (rel == 0) {
462 return (*it);
463 }
464 }
465 // need to add in the end
466 auto tfmn = mSlots.empty() ? tf2SlotMin(tf) : tf2SlotMin(mSlots.back().getTFEnd() + 1);
467 do {
468 uint64_t tft = uint64_t(tfmn) + mSlotLength - 1;
469 if (mSlots.empty() && mStartOffsetTFs && tf < mStartOffsetTFs) { // if this was lowest possible TF, its length might be smaller than mSlotLength
470 tft -= mStartOffsetTFs;
471 }
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());
477 tfmn = tft < o2::calibration::INFINITE_TF ? mSlots.back().getTFEnd() + 1 : tft;
478 } while (tf > mSlots.back().getTFEnd());
479
480 return mSlots.back();
481}
482
483//_________________________________________________
484template <typename Container>
486{
487 for (int i = 0; i < getNSlots(); i++) {
488 LOG(info) << "Slot #" << i + 1 << " of " << getNSlots();
489 getSlot(i).print();
490 }
491}
492
493//_________________________________________________
494template <typename Container>
496{
497 if (mSlots.empty()) {
498 LOG(warn) << "Nothing to save, no TimeSlots defined";
499 return false;
500 }
501 if (mSaveMetaData.startRun < 0) {
502 mSaveMetaData.startRun = mCurrentTFInfo.runNumber;
503 }
504 mSaveMetaData.endRun = mCurrentTFInfo.runNumber;
505 if (mSaveMetaData.startTime < 0) {
506 mSaveMetaData.startTime = mSlots.back().getStartTimeMS();
507 }
508 mSaveMetaData.endTime = mSlots.back().getEndTimeMS();
509 return true;
510}
511
512//_________________________________________________
513template <typename Container>
515{
516 if (!getSavedSlotAllowed()) {
517 LOG(info) << "Slot saving is disabled";
518 return false;
519 }
520 if (!updateSaveMetaData()) {
521 return false;
522 }
523
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);
528 } else {
529 LOGP(info, "created calibration directory {}", mSaveDirectory);
530 }
531 }
532
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());
539 return false;
540 }
541 if (!saveLastSlotData(flout)) { // call used method to store data
542 flout.Close();
543 unlink(pthTmp.c_str());
544 return false;
545 }
546 flout.WriteObjectAny(&mSaveMetaData, "o2::calibration::TimeSlotMetaData", "metadata");
547 flout.Close();
548 std::filesystem::rename(pthTmp, pth);
549 LOGP(info, "Saved data of the last slot to {}", pth);
550 return true;
551}
552
553//_________________________________________________
554template <typename Container>
556{
557 if (!getSavedSlotAllowed()) {
558 LOG(info) << "Saved slot usage is disabled";
559 return false;
560 }
561 auto pth = getSaveFilePath();
562 if (!std::filesystem::exists(pth)) {
563 LOGP(info, "No save file {} is found", pth);
564 return false;
565 }
566 TFile flin(pth.c_str());
567 if (flin.IsZombie()) {
568 LOGP(error, "failed to open save file {}", pth);
569 return false;
570 }
571 auto meta = (o2::calibration::TimeSlotMetaData*)flin.GetObjectChecked("metadata", "o2::calibration::TimeSlotMetaData");
572 if (!meta) {
573 LOGP(error, "Failed to read metadata from {}", pth);
574 return false;
575 }
576 auto res = adoptSavedData(*meta, flin); // up to the detector to decide if data should be accepted
577 if (res) {
578 mSaveMetaData.startRun = meta->startRun;
579 mSaveMetaData.startTime = meta->startTime;
580 updateSaveMetaData();
581 }
582 flin.Close();
583 unlink(pth.c_str()); // cleanup used file
584 return true;
585}
586
587//_________________________________________________
588template <typename Container>
590{
591 if (mSaveFileName.empty()) {
592 LOGP(fatal, "Save file name was not set");
593 }
594 return fmt::format("{}{}{}", mSaveDirectory, ((!mSaveDirectory.empty() && mSaveDirectory.back() != '/') ? "/" : ""), mSaveFileName);
595}
596
597} // namespace calibration
598} // namespace o2
599
600#endif
int32_t i
Helper for geometry and GRP related CCDB requests.
uint32_t res
Definition RawData.h:0
std::ostringstream debug
static GRPGeomHelper & instance()
virtual bool adoptSavedData(const TimeSlotMetaData &metadata, TFile &fl)
virtual void checkSlotsToFinalize(TFType tf=INFINITE_TF, int maxDelay=0)
void setSaveFileName(const std::string &n)
ClassDef(TimeSlotCalibration, 1)
const o2::dataformats::TFIDInfo & getCurrentTFInfo() const
o2::dataformats::TFIDInfo & getCurrentTFInfo()
virtual void finalizeSlot(Slot &slot)=0
void setSaveDirectory(const std::string &n)
virtual bool hasEnoughData(const Slot &slot) const =0
const std::string & getSaveFileName() const
virtual Slot & emplaceNewSlot(bool front, TFType tstart, TFType tend)=0
void setTFEnd(TFType v)
Definition TimeSlot.h:58
GLdouble n
Definition glcorearb.h:1982
const GLdouble * v
Definition glcorearb.h:832
GLdouble f
Definition glcorearb.h:310
GLsizei const GLfloat * value
Definition glcorearb.h:819
GLint GLint GLsizei GLint GLenum GLenum type
Definition glcorearb.h:275
GLboolean * data
Definition glcorearb.h:298
typedef void(APIENTRYP PFNGLCULLFACEPROC)(GLenum mode)
uint32_t TFType
Definition TimeSlot.h:29
constexpr TFType INFINITE_TF
Definition TimeSlot.h:30
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"