Loading [MathJax]/extensions/tex2jax.js
The Gaudi Framework  v31r0 (aeb156f0)
All Classes Namespaces Files Functions Variables Typedefs Enumerations Enumerator Properties Friends Macros Modules Pages
JemallocProfileSvc.cpp
Go to the documentation of this file.
1 // Include files
4 #include "GaudiKernel/Service.h"
5 #include <climits>
6 #include <iostream>
7 
8 // local
9 #include "JemallocProfileSvc.h"
10 
11 // including jemmalloc.h is difficult as the malloc signature is not exactly identical
12 // to the system one (issue with throw).
13 // We therefore declare mallctl here.
14 extern "C" {
15 int mallctl( const char* name, void* oldp, size_t* oldlenp, void* newp, size_t newlen );
16 }
17 
18 //-----------------------------------------------------------------------------
19 // Implementation file for class : JemallocProfileSvc
20 //
21 // 2016-01-11 : Ben Couturier
22 //-----------------------------------------------------------------------------
23 
24 //=============================================================================
25 // Initializer
26 //=============================================================================
28  StatusCode sc = base_class::initialize();
29  if ( sc.isFailure() ) return sc;
30 
31  // register to the incident service
32  static const std::string serviceName = "IncidentSvc";
33  m_incidentSvc = serviceLocator()->service( serviceName );
34  if ( !m_incidentSvc ) {
35  error() << "Cannot retrieve " << serviceName << endmsg;
36  return StatusCode::FAILURE;
37  }
38 
39  debug() << "Register to the IncidentSvc" << endmsg;
40  m_incidentSvc->addListener( this, IncidentType::BeginEvent );
41  m_incidentSvc->addListener( this, IncidentType::EndEvent );
42  for ( std::string incident : m_startFromIncidents ) { m_incidentSvc->addListener( this, incident ); }
43  for ( std::string incident : m_stopAtIncidents ) { m_incidentSvc->addListener( this, incident ); }
44 
45  // Resetting the event counter
46  m_eventNumber = 0;
47  m_profiling = false;
48 
49  // Cache whether we have start/stop incidents
50  m_hasStartIncident = ( m_startFromIncidents.size() > 0 );
51  m_hasStopIncident = ( m_stopAtIncidents.size() > 0 );
52 
53  // Checking the consistency of the start/stop events
55  info() << "Stop profiling trigger was specified but no start. Defaulting to first events" << endmsg;
57  }
58 
59  return StatusCode::SUCCESS;
60 }
61 
62 // Finalization of the service.
64  if ( m_profiling ) { stopProfiling(); }
65  // unregistering from the IncidentSvc
66  m_incidentSvc->removeListener( this, IncidentType::BeginEvent );
67  m_incidentSvc->removeListener( this, IncidentType::EndEvent );
69  return base_class::finalize();
70 }
71 
72 //=============================================================================
73 // Event handling methods
74 
75 //=============================================================================
76 
77 // Handler for incidents
78 void JemallocProfileSvc::handle( const Incident& incident ) {
79  if ( IncidentType::BeginEvent == incident.type() ) {
80  handleBegin();
81  } else if ( IncidentType::EndEvent == incident.type() ) {
82  handleEnd();
83  }
84 
85  // If already processing we can ignore the incidents for start
86  if ( !m_profiling && m_hasStartIncident ) {
87  for ( std::string startincident : m_startFromIncidents ) {
88  if ( startincident == incident.type() ) {
89  info() << "Starting Jemalloc profile at incident " << incident.type() << endmsg;
91  break;
92  }
93  } // Loop on incidents
94  } // If checking incidents to start
95 
96  // If already processing we can ignore the incidents for start
97  if ( m_profiling && m_hasStopIncident ) {
98  for ( std::string stopincident : m_stopAtIncidents ) {
99  if ( stopincident == incident.type() ) {
100  info() << "Stopping Jemalloc profile at incident " << incident.type() << endmsg;
101  stopProfiling();
102  break;
103  }
104  } // Loop on incidents
105  } // If checking incidents to stop
106 }
107 
108 // Handler for incidents
109 // Called on at begin events
111  m_eventNumber += 1;
112 
114 }
115 
116 // Handler for incidents
117 // Called on at End events
120  ( ( m_eventNumber - m_nStartFromEvent ) % m_dumpPeriod == 0 ) ) {
121  dumpProfile();
122  }
123 
125 }
126 
127 //=============================================================================
128 // Utilities calling mallctl
129 
130 //=============================================================================
131 
136  m_profiling = true;
137  info() << "Starting Jemalloc profile at event " << m_eventNumber << endmsg;
138  mallctl( "prof.dump", NULL, NULL, NULL, 0 );
139 }
140 
145  m_profiling = false;
146  dumpProfile();
147 }
148 
153  info() << "Dumping Jemalloc profile at event " << m_eventNumber << endmsg;
154  mallctl( "prof.dump", NULL, NULL, NULL, 0 );
155 }
156 
157 //=============================================================================
158 // Declaration of the factory
const std::string & type() const
Access to the incident type.
Definition: Incident.h:38
void startProfiling()
Utility method to start profiling with jemalloc.
MsgStream & info() const
shortcut for the method msgStream(MSG::INFO)
StatusCode finalize() override
Finalizer.
Gaudi::Property< int > m_dumpPeriod
Service that enables the Jemalloc profiler on demand.
constexpr static const auto SUCCESS
Definition: StatusCode.h:85
void handle(const Incident &incident) override
bool isFailure() const
Definition: StatusCode.h:130
int mallctl(const char *name, void *oldp, size_t *oldlenp, void *newp, size_t newlen)
STL class.
#define DECLARE_COMPONENT(type)
int m_eventNumber
Current event number.
StatusCode initialize() override
Initializer.
StatusCode service(const Gaudi::Utils::TypeNameString &name, T *&svc, bool createIf=true)
Templated method to access a service by name.
Definition: ISvcLocator.h:76
MsgStream & error() const
shortcut for the method msgStream(MSG::ERROR)
This class is used for returning status codes from appropriate routines.
Definition: StatusCode.h:50
Gaudi::Property< int > m_nStopAtEvent
void stopProfiling()
Utility method to stop profiling with jemalloc.
Gaudi::Property< int > m_nStartFromEvent
Gaudi::Property< std::vector< std::string > > m_stopAtIncidents
MsgStream & debug() const
shortcut for the method msgStream(MSG::DEBUG)
Base class for all Incidents (computing events).
Definition: Incident.h:17
virtual void addListener(IIncidentListener *lis, const std::string &type="", long priority=0, bool rethrow=false, bool singleShot=false)=0
Add listener.
void dumpProfile()
Utility method to dump profile with jemalloc.
constexpr static const auto FAILURE
Definition: StatusCode.h:86
SmartIF< IIncidentSvc > m_incidentSvc
Pointer to the incident service.
void reset(TYPE *ptr=nullptr)
Set the internal pointer to the passed one disposing of the old one.
Definition: SmartIF.h:86
virtual void removeListener(IIncidentListener *lis, const std::string &type="")=0
Remove listener.
SmartIF< ISvcLocator > & serviceLocator() const override
Retrieve pointer to service locator.
Definition: Service.cpp:277
Gaudi::Property< std::vector< std::string > > m_startFromIncidents
MsgStream & endmsg(MsgStream &s)
MsgStream Modifier: endmsg. Calls the output method of the MsgStream.
Definition: MsgStream.h:192
bool m_profiling
Status of the profiling.