StatusCodeSvc.cpp
Go to the documentation of this file.
1 #include "StatusCodeSvc.h"
3 
4 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
5 
6 using namespace std;
7 //
9 //
10 inline void toupper(std::string &s)
11 {
12  std::transform(s.begin(), s.end(), s.begin(),
13  (int(*)(int)) toupper);
14 }
15 
17  : base_class( name, svc )
18 {
19 
20  declareProperty("Filter",m_pFilter);
21  declareProperty("AbortOnError",m_abort=false);
22  declareProperty("SuppressCheck", m_suppress=false);
23  declareProperty("IgnoreDicts",m_dict=true);
24 
25 }
26 
27 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
28 
29 
32 
34  if (!sc.isSuccess()) return sc;
35 
36  info() << "initialize" << endmsg;
37 
38  for (const auto& itr : m_pFilter.value()) {
39  // we need to do this if someone has gotten to regFnc before initialize
40 
41  string fnc,lib;
42  parseFilter(itr,fnc,lib);
43 
44  if (!fnc.empty()) {
45  filterFnc(fnc);
46  m_filterfnc.insert(fnc);
47  }
48 
49  if (!lib.empty()) {
50  filterLib(lib);
51  m_filterlib.insert(lib);
52  }
53 
54  }
55 
56  return StatusCode::SUCCESS;
57 
58 }
59 
60 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
61 
64 
65  info() << "reinitialize" << endmsg;
66 
67  return StatusCode::SUCCESS;
68 
69 }
70 
71 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
74 
75  if (!m_dat.empty()) {
76 
77  info() << "listing all unchecked return codes:" << endmsg;
78 
79  list();
80 
81  } else {
82 
83  if (msgLevel(MSG::DEBUG))
84  debug() << "all StatusCode instances where checked" << endmsg;
85 
86  }
87 
88  return StatusCode::SUCCESS;
89 
90 }
91 
92 
93 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
94 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
95 
96 void
98 
101  return;
102  }
103 
104  if (m_dict &&
105  (lib.compare(lib.length()-7, 7, "Dict.so") == 0 ||
106  lib.compare(lib.length()-8, 8, "Cling.so") == 0)) {
107  return;
108  }
109  // this appears only with gcc 4.9...
110  if (fnc == "_PyObject_GC_Malloc") return;
111  // GAUDI-1036
112  if (fnc == "PyThread_get_thread_ident") return;
113  if (fnc == "local") return;
114 
115  {
116  const string rlib = lib.substr(lib.rfind("/") + 1);
117 
118  if (m_filterfnc.find(fnc) != m_filterfnc.end() ||
119  m_filterlib.find(rlib) != m_filterlib.end() ) {
120  return;
121  }
122  }
123 
124  if (m_abort) {
125  fatal() << "Unchecked StatusCode in " << fnc << " from lib "
126  << lib << endmsg;
127  abort();
128  }
129 
130  string key = fnc + lib;
131 
132  auto itr = m_dat.find(key);
133 
134  if (itr != m_dat.end()) {
135  itr->second.count += 1;
136  } else {
137 
138  const string rlib = lib.substr(lib.rfind("/") + 1);
139 
140  StatCodeDat dat;
141  dat.fnc = fnc;
142  dat.lib = rlib;
143  dat.count = 1;
144 
145  m_dat[key] = dat;
146  }
147 
148 }
149 
150 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
151 
152 void
154 
155 
156 
158  os << "Num | Function | Source Library" << endl;
159  os << "----+--------------------------------+-------------------"
160  << "-----------------------" << endl;
161 
162  for(const auto& itr : m_dat) {
163  const auto& dat = itr.second;
164 
165  os.width(3);
166  os.setf(ios_base::right,ios_base::adjustfield);
167  os << dat.count;
168 
169  os << " | ";
170  os.width(30);
171  os.setf(ios_base::left,ios_base::adjustfield);
172  os << dat.fnc;
173 
174  os << " | ";
175  os.setf(ios_base::left,ios_base::adjustfield);
176  os << dat.lib;
177 
178  os << endl;
179 
180  }
181 
182  info() << endl << os.str() << endmsg;
183 
184 }
185 
186 
187 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
188 void
190 
191  auto itr = std::find_if(m_dat.begin(),m_dat.end(),
192  [&](const std::pair<std::string,StatCodeDat>& d) {
193  return d.second.fnc == str;
194  });
195  if (itr!=std::end(m_dat)) m_dat.erase(itr);
196 
197 }
198 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
199 
200 void
202 
203  auto itr = std::find_if(m_dat.begin(),m_dat.end(),
204  [&](const std::pair<std::string,StatCodeDat>& d) {
205  return d.second.lib == str;
206  });
207  if (itr!=std::end(m_dat)) m_dat.erase(itr);
208 }
209 
210 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
211 
212 void
213 StatusCodeSvc::parseFilter(const string& str, string& fnc, string& lib) {
214 
215 
216  auto loc = str.find("=");
217  if (loc == std::string::npos) {
218  fnc = str;
219  lib = "";
220  } else {
221  string key = str.substr(0,loc);
222  string val = str.substr(loc+1);
223 
224  toupper(key);
225 
226  if (key == "FCN" || key == "FNC") {
227  fnc = val;
228  lib.clear();
229  } else if (key == "LIB") {
230  fnc.clear();
231  lib = val;
232  } else {
233  fnc.clear();
234  lib.clear();
235 
236  warning() << "ignoring unknown token in Filter: " << str
237  << endmsg;
238  }
239  }
240 
241 }
242 
243 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
244 
BooleanProperty m_abort
Definition: StatusCodeSvc.h:43
T setf(T...args)
StatusCode initialize() override
Definition: Service.cpp:68
T empty(T...args)
The ISvcLocator is the interface implemented by the Service Factory in the Application Manager to loc...
Definition: ISvcLocator.h:25
Gaudi::StateMachine::State m_state
Service state.
Definition: Service.h:311
BooleanProperty m_dict
Definition: StatusCodeSvc.h:43
MsgStream & info() const
shortcut for the method msgStream(MSG::INFO)
StatusCode reinitialize() override
bool isSuccess() const
Test for a status code of SUCCESS.
Definition: StatusCode.h:76
T rfind(T...args)
T endl(T...args)
STL namespace.
void list() const override
T end(T...args)
std::map< std::string, StatCodeDat > m_dat
Definition: StatusCodeSvc.h:45
BooleanProperty m_suppress
Definition: StatusCodeSvc.h:43
#define DECLARE_COMPONENT(type)
Definition: PluginService.h:36
STL class.
void parseFilter(const std::string &str, std::string &fnc, std::string &lib)
StatusCode initialize() override
MsgStream & warning() const
shortcut for the method msgStream(MSG::WARNING)
T toupper(T...args)
This class is used for returning status codes from appropriate routines.
Definition: StatusCode.h:26
T width(T...args)
T clear(T...args)
std::set< std::string > m_filterfnc
Definition: StatusCodeSvc.h:46
const TYPE & value() const
explicit conversion
Definition: Property.h:341
void regFnc(const std::string &func, const std::string &lib) override
T insert(T...args)
StringArrayProperty m_pFilter
Definition: StatusCodeSvc.h:42
T find(T...args)
T length(T...args)
MsgStream & debug() const
shortcut for the method msgStream(MSG::DEBUG)
T begin(T...args)
StatusCodeSvc(const std::string &name, ISvcLocator *svc)
void filterLib(const std::string &)
string s
Definition: gaudirun.py:245
T substr(T...args)
T abort(T...args)
T transform(T...args)
MsgStream & fatal() const
shortcut for the method msgStream(MSG::FATAL)
Property * declareProperty(const std::string &name, T &property, const std::string &doc="none") const
Declare the named property.
Definition: Service.h:215
MSG::Level msgLevel() const
get the output level from the embedded MsgStream
void toupper(std::string &s)
MsgStream & endmsg(MsgStream &s)
MsgStream Modifier: endmsg. Calls the output method of the MsgStream.
Definition: MsgStream.h:244
T compare(T...args)
void filterFnc(const std::string &)
StatusCode finalize() override
std::set< std::string > m_filterlib
Definition: StatusCodeSvc.h:46