13#include <dqm/core/DQMHistAnalysis.h>
14#include <boost/algorithm/string.hpp>
36std::vector <chid> DQMHistAnalysisModule::m_epicsChID;
62 if (dirname.size() > 0) {
63 fullname = dirname +
"/" + histname;
72 B2DEBUG(20,
"Found Delta" << fullname);
100 if (dirname.size() > 0) {
101 fullname = dirname +
"/" + histname;
110 std::string fullname;
111 if (dirname.size() > 0) {
112 fullname = dirname +
"/" + histname;
121 std::string fullname;
122 if (dirname.size() > 0) {
123 fullname = dirname +
"/" + histname;
127 return getDelta(fullname, n, onlyIfUpdated);
134 return it->second.getDelta(n, onlyIfUpdated);
136 B2WARNING(
"Delta hist " << fullname <<
" not found");
143 obj->SetName(objName.c_str());
149 TIter nextkey(gROOT->GetListOfCanvases());
152 while ((obj =
dynamic_cast<TObject*
>(nextkey()))) {
153 if (obj->IsA()->InheritsFrom(
"TCanvas")) {
154 if (obj->GetName() == canvas_name)
155 return dynamic_cast<TCanvas*
>(obj);
164 if (was_updated && !
s_histList[histname].isUpdated())
return nullptr;
168 B2ERROR(
"Histogram " << histname <<
" in histogram list but nullptr.");
171 B2INFO(
"Histogram " << histname <<
" not in list.");
177 if (dirname.size() > 0) {
178 return findHist(dirname +
"/" + histname, updated);
195 if (hist->Integral() != 0 and ref->Integral() != 0) {
196 ref->Scale(hist->Integral() / ref->Integral());
201 if (hist->GetMaximum() != 0 and ref->GetMaximum() != 0) {
202 ref->Scale(hist->GetMaximum() / ref->GetMaximum());
223 if (dirname.size() > 0) {
224 return findRefHist(dirname +
"/" + histname, scaling, hist);
231 TCanvas* cnv =
nullptr;
233 if (cobj) cnv = *cobj;
235 if (cnv ==
nullptr) {
239 B2ERROR(
"findHistInCanvas: histoname not valid (missing dir?), should be 'dirname/histname': " << histo_name);
242 auto dirname = s.at(0);
243 auto hname = s.at(1);
244 std::string canvas_name = dirname +
"/c_" + hname;
247 if (cnv && cobj) *cobj = cnv;
251 if (cnv !=
nullptr) {
252 TIter nextkey(cnv->GetListOfPrimitives());
254 while ((obj =
dynamic_cast<TObject*
>(nextkey()))) {
255 if (obj->IsA()->InheritsFrom(
"TH1")) {
256 if (obj->GetName() == histo_name)
257 return dynamic_cast<TH1*
>(obj);
268 if (file && file->IsOpen()) {
269 auto obj = file->Get(histname.data());
270 if (obj !=
nullptr) {
272 if (obj->IsA()->InheritsFrom(
"TH1")) {
273 B2DEBUG(20,
"Histogram " << histname <<
" found in file");
274 return dynamic_cast<TH1*
>(obj);
276 B2INFO(
"Found Object " << histname <<
" in file is not a histogram");
279 B2INFO(
"Histogram " << histname <<
" not found in file");
290 B2INFO(
"MonitoringObject " << objName <<
" not in memfile.");
296 double probs[2] = {0.16, 1 - 0.16};
297 double quant[2] = {0, 0};
298 h->GetQuantiles(2, quant, probs);
299 const double sigma68 = (-quant[0] + quant[1]) / 2;
305 std::vector <std::string> out;
306 boost::split(out, in, [delim](
char c) {
return c == delim;});
312 TIter nextckey(gROOT->GetListOfCanvases());
313 TObject* cobj =
nullptr;
315 while ((cobj =
dynamic_cast<TObject*
>(nextckey()))) {
316 if (cobj->IsA()->InheritsFrom(
"TCanvas")) {
317 TCanvas* cnv =
dynamic_cast<TCanvas*
>(cobj);
328 it.second.resetBeforeEvent();
332 it.second.setNotUpdated();
368 for (
size_t i = 0; i < hs.size(); i++) {
369 if (hs[i]->GetName() == std::string(
"DQMInfo/rtype")) {
374 B2ERROR(
"ExtractRunType: Histogram \"DQMInfo/rtype\" missing");
380 for (
size_t i = 0; i < hs.size(); i++) {
381 if (hs[i]->GetName() == std::string(
"DAQ/Nevent")) {
386 B2ERROR(
"ExtractEvent: Histogram \"DAQ/Nevent\" missing");
403 if (m_epicsNameToChID[pvname] !=
nullptr) {
404 B2ERROR(
"Epics PV " << pvname <<
" already registered!");
407 if (keyname !=
"" && m_epicsNameToChID[keyname] !=
nullptr) {
408 B2ERROR(
"Epics PV with key " << keyname <<
" already registered!");
412 m_epicsChID.emplace_back();
413 auto ptr = &m_epicsChID.back();
414 if (!ca_current_context()) SEVCHK(ca_context_create(ca_disable_preemptive_callback),
"ca_context_create");
416 CheckEpicsError(ca_create_channel((prefix + pvname).data(), NULL, NULL, 10, ptr),
"ca_create_channel failure", pvname);
418 m_epicsNameToChID[pvname] = *ptr;
419 if (keyname !=
"") m_epicsNameToChID[keyname] = *ptr;
420 return m_epicsChID.size() - 1;
430 if (m_epicsNameToChID[keyname] ==
nullptr) {
431 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
434 CheckEpicsError(ca_put(DBR_DOUBLE, m_epicsNameToChID[keyname], (
void*)&value),
"ca_set failure", keyname);
442 if (m_epicsNameToChID[keyname] ==
nullptr) {
443 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
446 CheckEpicsError(ca_put(DBR_SHORT, m_epicsNameToChID[keyname], (
void*)&value),
"ca_set failure", keyname);
454 if (m_epicsNameToChID[keyname] ==
nullptr) {
455 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
458 if (value.length() > 40) {
459 B2ERROR(
"Epics string PV " << keyname <<
" too long (>40 characters)!");
463 strcpy(text, value.c_str());
464 CheckEpicsError(ca_put(DBR_STRING, m_epicsNameToChID[keyname], text),
"ca_set failure", keyname);
472 if (index < 0 || index >= (
int)m_epicsChID.size()) {
473 B2ERROR(
"Epics PV with " << index <<
" not registered!");
476 CheckEpicsError(ca_put(DBR_DOUBLE, m_epicsChID[index], (
void*)&value),
"ca_set failure", m_epicsChID[index]);
484 if (index < 0 || index >= (
int)m_epicsChID.size()) {
485 B2ERROR(
"Epics PV with " << index <<
" not registered!");
488 CheckEpicsError(ca_put(DBR_SHORT, m_epicsChID[index], (
void*)&value),
"ca_set failure", m_epicsChID[index]);
496 if (index < 0 || index >= (
int)m_epicsChID.size()) {
497 B2ERROR(
"Epics PV with " << index <<
" not registered!");
501 strncpy(text, value.c_str(), 40);
503 CheckEpicsError(ca_put(DBR_STRING, m_epicsChID[index], text),
"ca_set failure", m_epicsChID[index]);
512 if (m_epicsNameToChID[keyname] ==
nullptr) {
513 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
519 auto r = ca_get(DBR_DOUBLE, m_epicsNameToChID[keyname], (
void*)&value);
520 if (r == ECA_NORMAL) r = ca_pend_io(5.0);
521 if (r == ECA_NORMAL) {
524 CheckEpicsError(r,
"Read PV failed in ca_get or ca_pend_io failure", keyname);
535 if (index < 0 || index >= (
int)m_epicsChID.size()) {
536 B2ERROR(
"Epics PV with " << index <<
" not registered!");
542 auto r = ca_get(DBR_DOUBLE, m_epicsChID[index], (
void*)&value);
543 if (r == ECA_NORMAL) r = ca_pend_io(5.0);
544 if (r == ECA_NORMAL) {
547 CheckEpicsError(r,
"Read PV failed in ca_get or ca_pend_io failure", m_epicsChID[index]);
559 if (m_epicsNameToChID[keyname] ==
nullptr) {
560 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
561 return std::string(value);
566 auto r = ca_get(DBR_STRING, m_epicsNameToChID[keyname], value);
567 if (r == ECA_NORMAL) r = ca_pend_io(5.0);
568 if (r == ECA_NORMAL) {
570 return std::string(value);
572 CheckEpicsError(r,
"Read PV (string) failed in ca_get or ca_pend_io failure", keyname);
575 return std::string(value);
584 if (index < 0 || index >= (
int)m_epicsChID.size()) {
585 B2ERROR(
"Epics PV with " << index <<
" not registered!");
586 return std::string(value);
591 auto r = ca_get(DBR_DOUBLE, m_epicsChID[index], value);
592 if (r == ECA_NORMAL) r = ca_pend_io(5.0);
593 if (r == ECA_NORMAL) {
595 return std::string(value);
597 CheckEpicsError(r,
"Read PV (string) failed in ca_get or ca_pend_io failure", m_epicsChID[index]);
600 return std::string(value);
607 if (m_epicsNameToChID[keyname] !=
nullptr) {
608 return m_epicsNameToChID[keyname];
610 B2ERROR(
"Epics PV " << keyname <<
" not registered!");
621 if (index >= 0 && index < (
int)m_epicsChID.size()) {
622 return m_epicsChID[index];
624 B2ERROR(
"Epics PV with " << index <<
" not registered!");
633 int state = ECA_NORMAL;
637 state = ca_pend_io(wait);
638 SEVCHK(state,
"ca_pend_io failure");
649 for (
auto& it : m_epicsChID)
CheckEpicsError(ca_clear_channel(it),
"ca_clear_channel failure", it);
653 m_epicsNameToChID.clear();
678 struct dbr_ctrl_double tPvData;
682 auto r = ca_get(DBR_CTRL_DOUBLE, pv, &tPvData);
683 if (r == ECA_NORMAL) r = ca_pend_io(5.0);
684 if (r == ECA_NORMAL) {
685 if (!std::isnan(tPvData.lower_alarm_limit)) {
686 lowerAlarm = tPvData.lower_alarm_limit;
688 if (!std::isnan(tPvData.lower_warning_limit)) {
689 lowerWarn = tPvData.lower_warning_limit;
691 if (!std::isnan(tPvData.upper_warning_limit)) {
692 upperWarn = tPvData.upper_warning_limit;
694 if (!std::isnan(tPvData.upper_alarm_limit)) {
695 upperAlarm = tPvData.upper_alarm_limit;
699 CheckEpicsError(r,
"Reading PV Limits failed in ca_get or ca_pend_io failure", pv);
713 }
else if (warn_flag) {
755 canvas->Pad()->SetFillColor(color);
757 canvas->Pad()->SetFrameFillColor(10);
758 canvas->Pad()->SetFrameFillStyle(1001);
759 canvas->Pad()->Modified();
760 canvas->Pad()->Update();
765 B2INFO(
"Check PV Connections");
767 for (
auto& it : m_epicsChID) {
770 B2INFO(
"Check PVs done");
776 B2WARNING(
"PV chid was nullptr");
779 auto state = ca_state(pv);
782 B2WARNING(
"Channel never connected " << ca_name(pv));
785 B2WARNING(
"Channel was connected, but now is not " << ca_name(pv));
788 B2WARNING(
"Channel deleted already " << ca_name(pv));
791 if (!onlyError) B2INFO(
"Channel connected and OK " << ca_name(pv));
794 B2WARNING(
"Undefined status for channel " << ca_name(pv));
801 if (state != ECA_NORMAL) {
802 B2WARNING(message <<
": " << name);
809 if (state != ECA_NORMAL) {
811 if (
id) name = ca_name(
id);
812 B2WARNING(message <<
": " << name);
static MonObjList s_monObjList
The list of MonitoringObjects.
TCanvas * findCanvas(TString cname)
Find canvas by name.
void printPVStatus(chid pv, bool onlyError=true)
check the status of a PVs and report if disconnected or not found
static TH1 * findRefHist(const std::string &histname, ERefScaling scaling=ERefScaling::c_RefScaleNone, const TH1 *hist=nullptr)
Get referencehistogram from list (no other search).
bool hasDeltaPar(const std::string &dirname, const std::string &histname)
Check if Delta histogram parameters exist for histogram.
std::map< std::string, HistObject > HistList
The type of list of histograms.
static MonitoringObject * getMonitoringObject(const std::string &name)
Get MonitoringObject with given name (new object is created if non-existing)
std::map< std::string, MonitoringObject > MonObjList
The type of list of MonitoringObjects.
static TH1 * scaleReference(ERefScaling scaling, const TH1 *hist, TH1 *ref)
Using the original and reference, create scaled version.
void addDeltaPar(const std::string &dirname, const std::string &histname, HistDelta::EDeltaType t, int p, unsigned int a=1)
Add Delta histogram parameters.
static TH1 * findHistInFile(TFile *file, const std::string &histname)
Find histogram in specific TFile (e.g.
EStatusColor getStatusColor(EStatus status)
Return color for canvas state.
void colorizeCanvas(TCanvas *canvas, EStatus status)
Helper function for Canvas colorization.
static TH1 * findHist(const std::string &histname, bool onlyIfUpdated=false)
Get histogram from list (no other search).
static MonitoringObject * findMonitoringObject(const std::string &objName)
Find MonitoringObject.
void clearlist(void)
Clear all static global lists.
double getSigma68(TH1 *h) const
Helper function to compute half of the central interval covering 68% of a distribution.
EStatusColor
Status colors of histogram/canvas (corresponding to status)
@ c_ColorWarning
Analysis result: Warning, there may be minor issues.
@ c_ColorError
Analysis result: Severe issue found.
@ c_ColorTooFew
Not enough entries/event to judge.
@ c_ColorGood
Analysis result: Good.
@ c_ColorDefault
default for non-coloring
double getEpicsPV(std::string keyname)
Read value from a EPICS PV.
static int s_eventProcessed
Number of Events processed to fill histograms.
std::map< std::string, bool > CanvasUpdatedList
The type of list of canvas updated status.
std::string getEpicsStringPV(std::string keyname, bool &status)
Read value from a EPICS PV.
int registerExternalEpicsPV(std::string pvname, std::string keyname="")
Register a PV with its name and a key name.
static HistList s_histList
The list of Histograms.
static RefList s_refList
The list of references.
static std::string s_runType
The Run type.
static void clearHistList(void)
Clears the list of histograms.
TH1 * getDelta(const std::string &fullname, int n=0, bool onlyIfUpdated=true)
Get Delta histogram.
std::vector< std::string > StringSplit(const std::string &s, const char delim)
Helper function for string token split.
static void clearRefList(void)
Clears the list of ref histograms.
std::map< std::string, HistDelta > DeltaList
The type of list of delta settings and histograms.
void setEpicsPV(std::string keyname, double value)
Write value to a EPICS PV.
static DeltaList s_deltaList
The list of Delta Histograms and settings.
DQMHistAnalysisModule()
Constructor / Destructor.
void checkPVStatus(void)
Check the status of all PVs and report if disconnected or not found.
std::map< std::string, RefHistObject > RefList
The type of list of references.
static bool m_epicsReadOnly
Flag if to use EPICS in ReadOnly mode (for reading limits) do not set by yourself,...
EStatus
Status flag of histogram/canvas.
@ c_StatusDefault
default for non-coloring
@ c_StatusTooFew
Not enough entries/event to judge.
@ c_StatusError
Analysis result: Severe issue found.
@ c_StatusWarning
Analysis result: Warning, there may be minor issues.
@ c_StatusGood
Analysis result: Good.
static bool addHist(const std::string &dirname, const std::string &histname, TH1 *h)
Add histogram.
bool getUseEpics(void)
Getter for EPICS usage.
void ExtractRunType(std::vector< TH1 * > &hs)
Extract Run Type from histogram title, called from input module.
void CheckEpicsError(int state, const std::string &message, const std::string &name)
check the return status and check PV in case of error
TH1 * findHistInCanvas(const std::string &hname, TCanvas **canvas=nullptr)
Find histogram in corresponding canvas.
static std::string m_PVPrefix
The Prefix for EPICS PVs.
void cleanupEpicsPVs(void)
Unsubscribe from EPICS PVs on terminate.
void clearCanvases(void)
Clear content of all Canvases.
ERefScaling
Reference plot scaling type.
@ c_RefScaleEntries
to number of entries (integral)
@ c_RefScaleMax
to maximum (bin entry)
@ c_RefScaleNone
no scaling
EStatus makeStatus(bool enough, bool warn_flag, bool error_flag)
Helper function to judge the status for coloring and EPICS.
static bool m_useEpics
Flag if to use EPICS do not set by yourself, use EpicsEnable module to set.
static void initHistListBeforeEvent(void)
Reset the list of histograms.
void ExtractEvent(std::vector< TH1 * > &hs)
Extract event processed from daq histogram, called from input module.
int registerEpicsPV(std::string pvname, std::string keyname="")
EPICS related Functions.
void UpdateCanvas(std::string name, bool updated=true)
Mark canvas as updated (or not)
void resetDeltaList(void)
Reset Delta.
chid getEpicsPVChID(std::string keyname)
Get EPICS PV Channel Id.
bool requestLimitsFromEpicsPVs(chid id, double &lowerAlarm, double &lowerWarn, double &upperWarn, double &upperAlarm)
Get Alarm Limits from EPICS PV.
int registerEpicsPVwithPrefix(std::string prefix, std::string pvname, std::string keyname="")
Register a PV with its name and a key name.
void setEpicsStringPV(std::string keyname, std::string value)
Write string to a EPICS PV.
int updateEpicsPVs(float timeout)
Update all EPICS PV (flush to network)
static CanvasUpdatedList s_canvasUpdatedList
The list of canvas updated status.
EDeltaType
enum definition for delta algo Disabled: nothing Entries: use nr histogram entries Underflow: use ent...
void setDescription(const std::string &description)
Sets the description of the module.
MonitoringObject is a basic object to hold data for the run-dependency monitoring Run summary TCanvas...
#define REG_MODULE(moduleName)
Register the given module (without 'Module' suffix) with the framework.
Abstract base class for different kinds of events.