10#include <dqm/analysis/modules/DQMHistAnalysisECLSummary.h>
13#include <ecl/geometry/ECLGeometryPar.h>
20#include "boost/format.hpp"
31 B2DEBUG(20,
"DQMHistAnalysisECLSummary: Constructor done.");
33 std::string(
"ECL:channels_info:"));
35 "Mask Cell IDs based on information from ECL PVs",
38 "The higher this parameter, the larger differences in occupancy are allowed for adjacent channels", 0.28);
40 "The higher this parameter, the larger differences in the number of hits with bad chi2 are allowed for adjacent channels",
42 addParam(
"onlyIfUpdated",
m_onlyIfUpdated,
"If true (default), update EPICS PVs only if histograms were updated.",
57 {
"dead",
"#splitline{dead}{channels}", 1, 1, 3e4},
58 {
"cold",
"#splitline{cold}{channels}", 1, 2, 1e5},
59 {
"hot",
"#splitline{hot}{channels}", 25, 50, 1e5},
60 {
"bad_chi2",
"#splitline{bad #chi^{2}}{channels}", 5, 10, 1e6},
61 {
"bad_fit",
"#splitline{fit incon-}{sistencies}", 5, 10, 0 },
67 for (
int crate_id = 1; crate_id <= ECL::ECL_CRATES; crate_id++) {
68 std::string pv_name = (boost::format(
"crate%02d:%s") % crate_id % alarm.name).str();
72 for (
auto& ecl_part : {
"All",
"FWDEndcap",
"Barrel",
"BWDEndcap"}) {
73 std::string pv_name = (boost::format(
"%s:%s") % ecl_part % alarm.name).str();
77 std::string mask_pv_name = (boost::format(
"mask:%s") % alarm.name).str();
85 TString title =
"#splitline{ECL errors monitoring}";
86 title +=
"{E - Error, W - Warning, L - Low statistics}";
87 title +=
";ECLCollector ID (same as Crate ID)";
89 ECL::ECL_CRATES, 1, ECL::ECL_CRATES + 1,
99 for (
int i = 1; i <= ECL::ECL_CRATES; i++) {
119 c_occupancy =
new TCanvas(
"ECL/c_cid_Thr5MeV_overlaid_analysis");
120 c_bad_chi2 =
new TCanvas(
"ECL/c_bad_quality_overlaid_analysis");
123 1, ECL::ECL_TOTAL_CHANNELS + 1);
130 1, ECL::ECL_TOTAL_CHANNELS + 1);
137 1, ECL::ECL_TOTAL_CHANNELS + 1);
144 1, ECL::ECL_TOTAL_CHANNELS + 1);
150 B2DEBUG(20,
"DQMHistAnalysisECLSummary: initialized.");
155 B2DEBUG(20,
"DQMHistAnalysisECLSummary: beginRun called.");
174 if (!h_total_events)
return;
185 const double HISTCOLOR_RED = 0.9;
186 const double HISTCOLOR_GREEN = 0.45;
187 const double HISTCOLOR_ORANGE = 0.65;
188 const double HISTCOLOR_GRAY = 0.00;
204 bool warn_flag =
false;
205 bool error_flag =
false;
207 for (
size_t alarm_idx = 0; alarm_idx < alarm_counts.size(); alarm_idx++) {
208 for (
size_t crate = 0; crate < alarm_counts[alarm_idx].size(); crate++) {
210 char label_text[2] = {0};
212 int warning_limit =
m_ecl_alarms[alarm_idx].warning_limit;
214 int alarms = alarm_counts[alarm_idx][crate];
220 color = HISTCOLOR_GRAY;
222 }
else if (alarms >= alarm_limit) {
223 color = HISTCOLOR_RED;
227 }
else if (alarms >= warning_limit) {
228 color = HISTCOLOR_ORANGE;
233 color = HISTCOLOR_GREEN;
236 if (label_text[0] ==
'E' || label_text[0] ==
'W') {
237 B2DEBUG(100,
"Non-zero (" << alarm_counts[alarm_idx][crate]
238 <<
") for alarm_idx, crate = " << alarm_idx <<
", " << crate);
241 if (label_text[0] != 0) {
242 auto text =
new TText((crate + 1.5), (alarm_idx + 0.5), label_text);
243 if (label_text[0] !=
'L') text->SetTextColor(kWhite);
244 text->SetTextSize(0.03);
245 text->SetTextAlign(22);
254 auto gstyle_title_h = gStyle->GetTitleH();
255 auto gstyle_title_x = gStyle->GetTitleX();
256 auto gstyle_title_y = gStyle->GetTitleY();
257 gStyle->SetTitleH(0.04);
258 gStyle->SetTitleX(0.60);
259 gStyle->SetTitleY(1.00);
286 "gStyle->SetPalette(kRainBow);");
289 "gStyle->SetPalette(kBird);");
292 gStyle->SetTitleH(gstyle_title_h);
293 gStyle->SetTitleX(gstyle_title_x);
294 gStyle->SetTitleY(gstyle_title_y);
299 B2DEBUG(20,
"DQMHistAnalysisECLSummary: endRun called");
307 B2DEBUG(20,
"terminate called");
319 if (alarm_info.name == name)
return {index, alarm_info};
322 B2FATAL(
"Could not get ECL alarm " + name);
334 std::string pv_name = (boost::format(
"crate%02d:%s") % crate_id % alarm.name).str();
336 double upper_warning_limit = 0;
337 double upper_alarm_limit = 0;
340 if (!accessed || upper_alarm_limit <= 0 || upper_warning_limit <= 0) {
341 B2WARNING(
"Failed to get alarm limits");
345 alarm.alarm_limit = upper_alarm_limit;
346 alarm.warning_limit = upper_warning_limit;
352 static std::map<std::string, dbr_sts_long_array> mask_info;
354 B2WARNING(
"Failed to get arrays of masked channels");
360 m_mask[alarm.name].clear();
362 int cell_id = mask_info[alarm.name].value[i];
365 if (cell_id == 0)
break;
366 m_mask[alarm.name].insert(cell_id);
374 std::string mask_pv_name = (boost::format(
"mask:%s") % alarm.name).str();
377 if (mask_chid ==
nullptr)
return false;
381 if (r != ECA_NORMAL)
return false;
387 auto r = ca_pend_io(5.0);
388 if (r != ECA_NORMAL)
return false;
395 std::vector< std::vector<int> > alarm_counts(
396 m_ecl_alarms.size(), std::vector<int>(ECL::ECL_CRATES));
400 TH1* h_fail_crateid =
findHist(
"ECL/fail_crateid",
false);
403 for (
int crate_id = 1; crate_id <= ECL::ECL_CRATES; crate_id++) {
404 int errors_count = 0;
405 if (h_fail_crateid) {
406 errors_count = h_fail_crateid->GetBinContent(crate_id);
411 alarm_counts[fit_alarm_index][crate_id - 1] = errors_count;
415 std::map<int, int> error_bitmasks;
420 error_bitmasks[cell_id] |= error_bitmask;
425 error_bitmasks[cell_id] |= error_bitmask;
430 if (!update_mirabelle) {
437 static std::vector<std::string> indices = {
"dead",
"cold",
"hot",
"bad_chi2"};
438 for (
auto& index_name : indices) {
440 int alarm_bit = 1 << alarm_index;
442 for (
auto& [cid, error_bitmask] : error_bitmasks) {
443 if ((error_bitmask & alarm_bit) == 0)
continue;
445 bool masked = (
m_mask[index_name].find(cid) !=
m_mask[index_name].end());
448 int crate_id =
m_mapper.getCrateID(cid);
450 alarm_counts[alarm_index][crate_id - 1] += 1;
453 if (update_mirabelle)
continue;
458 if (index_name ==
"bad_chi2") {
461 }
else if (index_name ==
"dead" || index_name ==
"cold" || index_name ==
"hot") {
467 if (update_mirabelle)
continue;
470 if (index_name ==
"hot" || index_name ==
"bad_chi2") {
471 TCanvas* current_canvas;
474 TH1F* overlay_hist_green;
475 if (index_name ==
"hot") {
487 if (main_hist && main_hist->GetEntries() > 0) {
488 for (
auto& overlay : {overlay_hist, overlay_hist_green}) {
489 for (
int bin_id = 1; bin_id <= ECL::ECL_TOTAL_CHANNELS; bin_id++) {
490 if (overlay->GetBinContent(bin_id) == 0)
continue;
492 if (main_hist->GetBinContent(bin_id) < 1e-6)
continue;
493 overlay->SetBinContent(bin_id, main_hist->GetBinContent(bin_id));
497 current_canvas->Clear();
498 current_canvas->cd();
499 main_hist->Draw(
"hist");
500 overlay_hist->Draw(
"hist;same");
501 overlay_hist_green->Draw(
"hist;same");
502 current_canvas->Modified();
503 current_canvas->Update();
504 current_canvas->Draw();
511 for (
size_t alarm_idx = 0; alarm_idx < alarm_counts.size(); alarm_idx++) {
513 std::map<std::string, int> total;
515 for (
size_t crate = 0; crate < alarm_counts[alarm_idx].size(); crate++) {
516 int crate_id = crate + 1;
517 int value = alarm_counts[alarm_idx][crate];
519 if (!update_mirabelle) {
520 std::string pv_name = (boost::format(
"crate%02d:%s") % crate_id % alarm.name).str();
524 total[
"All"] += value;
525 if (crate_id <= ECL::ECL_BARREL_CRATES) {
526 total[
"Barrel"] += value;
527 }
else if (crate_id <= ECL::ECL_BARREL_CRATES + ECL::ECL_FWD_CRATES) {
528 total[
"FWDEndcap"] += value;
530 total[
"BWDEndcap"] += value;
534 for (
auto& ecl_part : {
"All",
"FWDEndcap",
"Barrel",
"BWDEndcap"}) {
535 std::string pv_name = (boost::format(
"%s:%s") % ecl_part % alarm.name).str();
536 if (update_mirabelle) {
537 std::string var_name = pv_name;
538 std::replace(var_name.begin(), var_name.end(),
':',
'_');
539 B2DEBUG(100, var_name <<
" = " << total[ecl_part]);
540 m_monObj->setVariable(var_name, total[ecl_part]);
552 static std::vector< std::vector<short> > neighbours(ECL::ECL_TOTAL_CHANNELS);
553 if (neighbours[0].size() == 0) {
554 for (
int cid0 = 0; cid0 < ECL::ECL_TOTAL_CHANNELS; cid0++) {
562 neighbours[cid0].erase(neighbours[cid0].begin());
569 max_deviation,
true);
576 static std::vector< std::vector<short> > neighbours(ECL::ECL_TOTAL_CHANNELS);
577 if (neighbours[0].size() == 0) {
578 for (
int cid_center = 1; cid_center <= ECL::ECL_TOTAL_CHANNELS; cid_center++) {
579 geom->Mapping(cid_center - 1);
580 const int theta_id_center = geom->GetThetaID();
581 int phi_id_center = geom->GetPhiID();
582 const int crystals_in_ring =
m_neighbours_obj.getCrystalsPerRing(theta_id_center);
583 phi_id_center = phi_id_center * 144 / crystals_in_ring;
584 for (
int cid0 = 0; cid0 < 8736; cid0++) {
585 if (cid0 == cid_center - 1)
continue;
587 int theta_id = geom->GetThetaID();
588 int phi_id = geom->GetPhiID();
590 if (std::abs(theta_id - theta_id_center) <= 2 &&
591 std::abs(phi_id - phi_id_center) <= 2) {
592 neighbours[cid_center - 1].push_back(cid0);
601 max_deviation,
false);
605 TH1* hist,
double total_events,
606 const std::vector< std::vector<short> >& neighbours,
607 double max_deviation,
bool occupancy_histogram)
609 std::map<int, int> retval;
611 if (!hist)
return retval;
614 if (hist->Integral() <= 0)
return retval;
624 const auto& [chi2_index, chi2_alarm] =
getAlarmByName(
"bad_chi2");
626 double min_required_events;
628 if (occupancy_histogram) {
629 min_required_events = std::min({
630 dead_alarm.required_statistics,
631 cold_alarm.required_statistics,
632 hot_alarm.required_statistics,
635 min_required_events = chi2_alarm.required_statistics;
638 if (total_events < min_required_events)
return retval;
640 int dead_bit = 1 << dead_index;
641 int cold_bit = 1 << cold_index;
642 int hot_bit = 1 << hot_index;
643 int chi2_bit = 1 << chi2_index;
647 if (occupancy_histogram) {
649 bool not_normalized = (
findCanvas(
"ECL/c_cid_Thr5MeV_analysis") ==
nullptr);
650 if (total_events >= dead_alarm.required_statistics) {
651 double min_occupancy;
653 if (run_type ==
"physics") {
655 min_occupancy = 1e-4;
658 min_occupancy = 1e-6;
660 if (not_normalized) {
662 min_occupancy *= total_events;
664 for (
int cid = 1; cid <= ECL::ECL_TOTAL_CHANNELS; cid++) {
665 if (hist->GetBinContent(cid) > min_occupancy)
continue;
666 retval[cid] |= dead_bit;
669 if (total_events >= hot_alarm.required_statistics) {
671 double max_occupancy = 0.7;
672 if (not_normalized) {
674 max_occupancy *= total_events;
676 for (
int cid = 1; cid <= ECL::ECL_TOTAL_CHANNELS; cid++) {
677 if (hist->GetBinContent(cid) < max_occupancy)
continue;
678 retval[cid] |= hot_bit;
685 for (
int cid = 1; cid <= ECL::ECL_TOTAL_CHANNELS; cid++) {
686 double actual_value = hist->GetBinContent(cid);
688 std::vector<short> neighb = neighbours[cid - 1];
689 std::multiset<double> values_sorted;
690 for (
auto& neighbour_cid : neighb) {
691 values_sorted.insert(hist->GetBinContent(neighbour_cid));
694 double expected_value;
695 if (occupancy_histogram) {
697 expected_value = *std::next(values_sorted.begin(), 1 * neighb.size() / 4);
700 expected_value = *std::next(values_sorted.begin(), 35 * neighb.size() / 48);
702 double deviation = std::abs((actual_value - expected_value) /
705 if (!occupancy_histogram) {
707 double min_occupancy = 1.41e-5;
708 if (
findCanvas(
"ECL/c_bad_quality_analysis") ==
nullptr) {
710 min_occupancy *= total_events;
713 if (actual_value < min_occupancy)
continue;
716 if (deviation < max_deviation)
continue;
718 if (occupancy_histogram) {
719 if (actual_value < expected_value) retval[cid] |= cold_bit;
720 if (actual_value > 2 * expected_value) retval[cid] |= hot_bit;
722 if (actual_value > expected_value) retval[cid] |= chi2_bit;
731 static std::map<std::string, std::vector<TLine*> > lines;
732 std::string name = hist->GetName();
733 if (lines[name].empty()) {
734 int x_min = hist->GetXaxis()->GetXmin();
735 int x_max = hist->GetXaxis()->GetXmax();
736 int y_min = hist->GetYaxis()->GetXmin();
737 int y_max = hist->GetYaxis()->GetXmax();
738 for (
int x = x_min + 1; x < x_max; x++) {
739 auto l =
new TLine(x, 0, x, 5);
740 l->SetLineStyle(kDashed);
741 lines[name].push_back(l);
743 for (
int y = y_min + 1; y < y_max; y++) {
744 auto l =
new TLine(1, y, ECL::ECL_CRATES + 1, y);
745 l->SetLineStyle(kDashed);
746 lines[name].push_back(l);
749 for (
auto line : lines[name]) {
std::map< int, int > getSuspiciousChannels(TH1 *hist, double total_events, const std::vector< std::vector< short > > &neighbours, double max_deviation, bool occupancy_histogram)
Get outlier channels, ones with values (occupancy, errors, etc) that are much higher than the values ...
double m_total_events
Number of events with ECL data in the current run.
double m_maxDeviationForChi2
The higher this parameter, the larger differences in the number of hits with bad chi2 are allowed for...
TH1F * h_bad_occ_overlay
Overlay to indicate bad channels on occupancy histogram.
void initialize() override final
Initialize the module.
void updateAlarmConfig()
Set alarm limits in DQM based on EPICS PV limits.
std::vector< std::vector< int > > updateAlarmCounts(bool update_mirabelle=false)
Calculate and return number of problems per crate for all alarm categories.
std::vector< TText * > m_labels
Labels on the ECL alarms histogram.
ECL::ECLChannelMapper m_mapper
Object to map ECL crystal ID to ECL crate ID.
void drawGrid(TH2 *hist)
Draw grid with TLine primitives for the specified histogram.
TCanvas * c_channels_summary
TCanvas for ECL alarms regarding suspicious channels.
std::map< int, int > getChannelsWithChi2Problems()
static const int c_max_masked_channels
Size of an array with masked channels.
std::pair< int, ECLAlarmType > getAlarmByName(std::string name)
Returns index of the specific alarm type and detailed information.
TH2F * h_channels_summary
Summarized information about ECL channels.
double m_maxDeviationForOccupancy
The higher this parameter, the larger differences in occupancy are allowed for adjacent channels.
ECL::ECLNeighbours m_neighbours_obj
Object to get ECL crystal neighbours.
TCanvas * c_occupancy
ECL occupancy histogram with highlighted suspicious channels.
TExec * m_ecl_style
Special object to specify style parameters.
std::string m_pvPrefix
Prefix to use for PVs registered by this module.
MonitoringObject * m_monObj
MiraBelle monitoring object.
void terminate() override final
Terminate.
void event() override final
Event processor.
TCanvas * c_bad_chi2
ECL bad chi2 histogram with highlighted suspicious channels.
TH1F * h_bad_chi2_overlay
Overlay to indicate bad channels on chi2 histogram.
void endRun() override final
Call when a run ends.
bool m_onlyIfUpdated
If true (default), update EPICS PVs only if there were changes in the histograms.
std::map< int, int > getChannelsWithOccupancyProblems()
TH1F * h_bad_chi2_overlay_green
Overlay to indicate masked bad channels on chi2 histogram.
void beginRun() override final
Call when a run begins.
DQMHistAnalysisECLSummaryModule()
< derived from DQMHistAnalysisModule class.
TExec * m_default_style
Special object to revert changes done by ecl_style.
TH1F * h_bad_occ_overlay_green
Overlay to indicate masked bad channels on occupancy histogram.
std::map< std::string, std::set< int > > m_mask
List of masked Cell IDs for each alarm type.
bool getMaskedChannels(std::map< std::string, dbr_sts_long_array > &mask_info)
Get the array of masked channels for each type of alarm case monitored by this module.
bool m_useChannelMask
If true, mask Cell IDs based on DQM:ECL:channels_info:{alarm_type} PV info.
std::vector< ECLAlarmType > m_ecl_alarms
Different alarms monitored in h_channels_summary.
static TCanvas * findCanvas(TString cname)
Find canvas by name.
chid getEpicsPVChID(const std::string &keyname)
Get EPICS PV Channel Id.
int registerEpicsPV(const std::string &pvname, const std::string &keyname="")
EPICS related Functions.
static MonitoringObject * getMonitoringObject(const std::string &name)
Get MonitoringObject with given name (new object is created if non-existing)
static 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).
@ c_ColorTooFew
Not enough entries/event to judge.
static const std::string & getRunType(void)
Get the list of the reference histograms.
DQMHistAnalysisModule()
Constructor / Destructor.
static EStatus makeStatus(bool enough, bool warn_flag, bool error_flag)
Helper function to judge the status for coloring and EPICS.
bool requestLimitsFromEpicsPVs(chid id, double &lowerAlarm, double &lowerWarn, double &upperWarn, double &upperAlarm)
Get Alarm Limits from EPICS PV.
void setEpicsPV(const std::string &keyname, double value)
Write value to a EPICS PV.
The Class for ECL Geometry Parameters.
static ECLGeometryPar * Instance()
Static method to get a reference to the ECLGeometryPar instance.
void addParam(const std::string &name, T ¶mVariable, const std::string &description, const T &defaultValue)
Adds a new parameter to the module.
#define REG_MODULE(moduleName)
Register the given module (without 'Module' suffix) with the framework.
Abstract base class for different kinds of events.