The Gaudi Framework  v33r0 (d5ea422b)
ToolSvc.cpp
Go to the documentation of this file.
1 /***********************************************************************************\
2 * (c) Copyright 1998-2019 CERN for the benefit of the LHCb and ATLAS collaborations *
3 * *
4 * This software is distributed under the terms of the Apache version 2 licence, *
5 * copied verbatim in the file "LICENSE". *
6 * *
7 * In applying this licence, CERN does not waive the privileges and immunities *
8 * granted to it by virtue of its status as an Intergovernmental Organization *
9 * or submit itself to any jurisdiction. *
10 \***********************************************************************************/
11 // Include Files
12 #include "ToolSvc.h"
13 #include "GaudiKernel/AlgTool.h"
15 #include "GaudiKernel/IAlgorithm.h"
18 #include "GaudiKernel/MsgStream.h"
19 #include "GaudiKernel/Service.h"
20 #include "boost/algorithm/string/erase.hpp"
21 #include "boost/algorithm/string/predicate.hpp"
22 #include "boost/circular_buffer.hpp"
23 #include <algorithm>
24 #include <cassert>
25 #include <functional>
26 #include <map>
27 #include <numeric>
28 #include <string>
29 namespace ba = boost::algorithm;
30 
31 #define ON_DEBUG if ( msgLevel( MSG::DEBUG ) )
32 #define ON_VERBOSE if ( msgLevel( MSG::VERBOSE ) )
33 
34 namespace {
35  //------------------------------------------------------------------------------
36  template <typename C>
37  unsigned long totalRefCount( const C& toolList )
38  //------------------------------------------------------------------------------
39  {
40  return std::accumulate( std::begin( toolList ), std::end( toolList ), 0ul,
41  [&]( unsigned long count, const IAlgTool* tool ) { return count + tool->refCount(); } );
42  }
43 
45  template <typename C>
46  void remove( C& c, typename C::const_reference i ) {
47  c.erase( std::remove( std::begin( c ), std::end( c ), i ), std::end( c ) );
48  }
49 } // namespace
50 
51 // Instantiation of a static factory class used by clients to create
52 // instances of this service
54 
55 //------------------------------------------------------------------------------
57  // tell the remaining observers that we're gone, and forget about unregistering..
58  std::for_each( std::begin( m_observers ), std::end( m_observers ),
59  [&]( IToolSvc::Observer* obs ) { obs->setUnregister( {} ); } );
60 }
61 //------------------------------------------------------------------------------
63 //------------------------------------------------------------------------------
64 {
65 
66  // initialize the Service Base class
68  if ( UNLIKELY( status.isFailure() ) ) {
69  error() << "Unable to initialize the Service" << endmsg;
70  return status;
71  }
72 
73  // set my own (ToolSvc) properties via the jobOptionService
74  if ( UNLIKELY( setProperties().isFailure() ) ) {
75  error() << "Unable to set base properties" << endmsg;
76  return StatusCode::FAILURE;
77  }
78 
79  return status;
80 }
81 
82 //------------------------------------------------------------------------------
84 //------------------------------------------------------------------------------
85 {
86  // Finalize and delete all left-over tools. Normally all tools created with
87  // ToolSvc are left over, since ToolSvc holds a refCount (via AlgTool ctor).
88  // Several cases need to be covered:
89  // 1) Simple dependencies: no circular dependencies between tools,
90  // and tools not using other tools
91  // 2) Tools-using-tools (but no circular dependencies)
92  // a) release() is called in the tool::finalize() (e.g. via GaudiAlgorithm)
93  // b) release() is called in the tool destructor (e.g. via ToolHandle)
94  // 3) Circular dependencies between tools
95  // a) release() is called in the tool::finalize() (e.g. via GaudiAlgorithm)
96  // b) release() is called in the tool destructor (e.g. via ToolHandle)
97  // 4) In addition to each of the above cases, refCounting of a particular tool
98  // might not have been done correctly in the code. Typically release()
99  // is not called, so we are left with a too high refCount.
100  // What to do with those, and how to avoid a crash while handling them...
101 
110  info() << "Removing all tools created by ToolSvc" << endmsg;
111 
112  // Print out list of tools
113  ON_DEBUG {
114  auto& log = debug();
115  log << " Tool List : ";
116  for ( const auto& iTool : m_instancesTools ) { log << iTool->name() << ":" << iTool->refCount() << " "; }
117  log << endmsg;
118  }
119 
120  //
121  // first pass: Finalize all tools (but don't delete them)
122  //
135  boost::circular_buffer<IAlgTool*> finalizedTools( m_instancesTools.size() ); // list of tools that have been finalized
136  bool fail( false );
137  size_t toolCount = m_instancesTools.size();
138  unsigned long startRefCount = 0;
139  unsigned long endRefCount = totalToolRefCount();
140  unsigned long startMinRefCount = 0;
141  unsigned long endMinRefCount = minimumToolRefCount();
142  while ( toolCount > 0 && endRefCount > 0 && ( endRefCount != startRefCount || endMinRefCount != startMinRefCount ) ) {
143  ON_DEBUG if ( endMinRefCount != startMinRefCount ) {
144  debug() << toolCount << " tools left to finalize. Summed refCounts: " << endRefCount << endmsg;
145  debug() << "Will finalize tools with refCount <= " << endMinRefCount << endmsg;
146  }
147  startMinRefCount = endMinRefCount;
148  startRefCount = endRefCount;
149  unsigned long maxLoop = toolCount + 1;
150  while ( --maxLoop > 0 && !m_instancesTools.empty() ) {
151  IAlgTool* pTool = m_instancesTools.back();
152  // removing tool from list makes ToolSvc::releaseTool( IAlgTool* ) a noop
154  unsigned long count = pTool->refCount();
155  // cache tool name
156  const std::string& toolName = pTool->name();
157  if ( count <= startMinRefCount ) {
158  ON_DEBUG debug() << " Performing finalization of " << toolName << " (refCount " << count << ")" << endmsg;
159  // finalize of one tool may trigger a release of another tool
160  // pTool->sysFinalize().ignore();
161  if ( !finalizeTool( pTool ).isSuccess() ) {
162  warning() << " FAILURE finalizing " << toolName << endmsg;
163  fail = true;
164  }
165  // postpone deletion
166  finalizedTools.push_back( pTool );
167  } else {
168  // Place back at the front of the list to try again later
169  // ToolSvc::releaseTool( IAlgTool* ) remains active for this tool
170  ON_DEBUG debug() << " Delaying finalization of " << toolName << " (refCount " << count << ")" << endmsg;
172  }
173  } // end of inner loop
174  toolCount = m_instancesTools.size();
175  endRefCount = totalToolRefCount();
176  endMinRefCount = minimumToolRefCount();
177  }; // end of outer loop
178 
179  //
180  // Second pass: Delete all finalized tools
181  //
182  // Delete in the order of increasing number of refCounts.
183  // Loop over tools in the same order as the order in which they were finalized.
184  // All tools in the list of finalized tools are no longer in the instancesTools list.
185  // If a tool destructor calls releaseTool() on another tool, this will have no
186  // effect on the 'other tool' if this 'other tool' is in the list of finalized tools.
187  // If this 'other tool' is still in the instancesTools list, it may trigger finalization
188  // (in releaseTool()), deletion and removal from the instancesTools list.
189  // Therefore, the check on non-finalised tools should happen *after* the deletion
190  // of the finalized tools.
191  ON_DEBUG debug() << "Deleting " << finalizedTools.size() << " finalized tools" << endmsg;
192  auto maxLoop = totalRefCount( finalizedTools ) + 1;
193  while ( --maxLoop > 0 && !finalizedTools.empty() ) {
194  IAlgTool* pTool = finalizedTools.front();
195  finalizedTools.pop_front();
196  auto count = pTool->refCount();
197  if ( count == 1 ) {
198  ON_DEBUG debug() << " Performing deletion of " << pTool->name() << endmsg;
199  } else {
200  ON_VERBOSE verbose() << " Delaying deletion of " << pTool->name() << " (refCount " << count << ")" << endmsg;
201  // Move to the end when refCount still not zero
202  finalizedTools.push_back( pTool );
203  }
204  // do a forced release
205  pTool->release();
206  }
207 
208  // Error if by now not all tools are properly finalised
209  if ( !m_instancesTools.empty() ) {
210  error() << "Unable to finalize and delete the following tools : ";
211  for ( const auto& iTool : m_instancesTools ) { error() << iTool->name() << ": " << iTool->refCount() << " "; }
212  error() << endmsg;
213  }
214 
215  // by now, all tools should be deleted and removed.
216  if ( !finalizedTools.empty() ) {
217  error() << "Failed to delete the following " << finalizedTools.size()
218  << " finalized tools. Bug in ToolSvc::finalize()?: ";
219  for ( const auto& iTool : finalizedTools ) { error() << iTool->name() << ": " << iTool->refCount() << " "; }
220  error() << endmsg;
221  }
222 
223  if ( m_pHistorySvc ) m_pHistorySvc->release();
224 
225  // Finalize this specific service
226  return ( Service::finalize().isSuccess() && !fail ) ? StatusCode::SUCCESS : StatusCode::FAILURE;
227 }
228 
229 // ===================================================================================
233 // ===================================================================================
234 namespace {
235  static const std::string s_PUBLIC = ":PUBLIC";
236 }
237 
238 //------------------------------------------------------------------------------
239 StatusCode ToolSvc::retrieve( const std::string& tooltype, const InterfaceID& iid, IAlgTool*& tool,
240  const IInterface* parent, bool createIf )
241 //------------------------------------------------------------------------------
242 {
243  // check for tools, which by name are required to be public:
244  if ( ba::ends_with( tooltype, s_PUBLIC ) ) {
245  // parent for PUBLIC tool is 'this', i.e. ToolSvc
246  return retrieve( ba::erase_tail_copy( tooltype, s_PUBLIC.size() ), iid, tool, this, createIf );
247  }
248 
249  // protect against empty type
250  if ( tooltype.empty() ) {
251  error() << "retrieve(): No Tool Type/Name given" << endmsg;
252  return StatusCode::FAILURE;
253  }
254  auto pos = tooltype.find( '/' );
255  if ( std::string::npos == pos ) { return retrieve( tooltype, tooltype, iid, tool, parent, createIf ); }
256  return retrieve( tooltype.substr( 0, pos ), tooltype.substr( pos + 1 ), iid, tool, parent, createIf );
257 }
258 
259 // ===================================================================================
260 
261 //------------------------------------------------------------------------------
263  IAlgTool*& tool, const IInterface* parent, bool createIf )
264 //------------------------------------------------------------------------------
265 {
266  // check the applicability of another method:
267  // ignore the provided name if it is empty or the type contains a name
268  if ( toolname.empty() || ( std::string::npos != tooltype.find( '/' ) ) ) {
269  return retrieve( tooltype, iid, tool, parent, createIf );
270  }
271 
272  // check for tools, which by name are required to be public:
273  if ( ba::ends_with( toolname, s_PUBLIC ) ) {
274  // parent for PUBLIC tool is this, i.e. ToolSvc
275  return retrieve( tooltype, ba::erase_tail_copy( toolname, s_PUBLIC.size() ), iid, tool, this, createIf );
276  }
277 
279 
280  IAlgTool* itool = nullptr;
282 
283  tool = nullptr;
284 
285  // If parent is not specified it means it is the ToolSvc itself
286  if ( !parent ) parent = this;
287  const std::string fullname = nameTool( toolname, parent );
288 
289  // Find tool in list of those already existing, and tell its
290  // interface that it has been used one more time
292  [&]( const IAlgTool* i ) { return i->name() == fullname && i->parent() == parent; } );
293  if ( it != std::end( m_instancesTools ) ) {
294  ON_DEBUG debug() << "Retrieved tool " << toolname << " with parent " << parent << endmsg;
295  itool = *it;
296  }
297 
298  if ( !itool ) {
299  // Instances of this tool do not exist, create an instance if desired
300  // otherwise return failure
301  if ( UNLIKELY( !createIf ) ) {
302  warning() << "Tool " << toolname << " not found and creation not requested" << endmsg;
303  return sc;
304  }
305  sc = create( tooltype, toolname, parent, itool );
306  if ( sc.isFailure() ) { return sc; }
307  }
308 
309  // Get the right interface of it
310  sc = itool->queryInterface( iid, pp_cast<void>( &tool ) );
311  if ( UNLIKELY( sc.isFailure() ) ) {
312  error() << "Tool " << toolname << " either does not implement the correct interface, or its version is incompatible"
313  << endmsg;
314  return sc;
315  }
316 
321  [&]( IToolSvc::Observer* obs ) { obs->onRetrieve( itool ); } );
322  return sc;
323 }
324 //------------------------------------------------------------------------------
326 //------------------------------------------------------------------------------
327 {
328 
331  for ( const auto& tool : m_instancesTools ) {
332  if ( tool->type() == toolType ) tools.push_back( tool->name() );
333  }
334  return tools;
335 }
336 //------------------------------------------------------------------------------
338 //------------------------------------------------------------------------------
339 {
343  []( const IAlgTool* t ) { return t->name(); } );
344  return tools;
345 }
346 //------------------------------------------------------------------------------
348 //------------------------------------------------------------------------------
349 {
352 }
353 //------------------------------------------------------------------------------
355 //------------------------------------------------------------------------------
356 {
359  // test if tool is in known list (protect trying to access a previously deleted tool)
361  unsigned long count = tool->refCount();
362  if ( count == 1 ) {
363  // finalize the tool
364 
366  // We are being called during ToolSvc::finalize()
367  // message format matches the one in ToolSvc::finalize()
368  debug() << " Performing finalization of " << tool->name() << " (refCount " << count << ")" << endmsg;
369  // message format matches the one in ToolSvc::finalize()
370  debug() << " Performing deletion of " << tool->name() << endmsg;
371  } else {
372  debug() << "Performing finalization and deletion of " << tool->name() << endmsg;
373  }
374  sc = finalizeTool( tool );
375  // remove from known tools...
376  remove( m_instancesTools, tool );
377  }
378  tool->release();
379  }
380  return sc;
381 }
382 
383 //------------------------------------------------------------------------------
384 StatusCode ToolSvc::create( const std::string& tooltype, const IInterface* parent, IAlgTool*& tool )
385 //------------------------------------------------------------------------------
386 {
387  const std::string& toolname = tooltype;
388  return create( tooltype, toolname, parent, tool );
389 }
390 
391 namespace {
394  template <typename T>
395  class ToolCreateGuard {
397  T& m_tools;
400 
401  public:
402  explicit ToolCreateGuard( T& tools ) : m_tools( tools ) {}
403  // we don't get a move constructor by default because we
404  // have a non-trivial destructor... However, the default
405  // one is just fine...
406  ToolCreateGuard( ToolCreateGuard&& ) noexcept = default;
408  void create( const std::string& tooltype, const std::string& fullname, const IInterface* parent ) {
409  // remove previous content
410  if ( m_tool ) remove( m_tools, m_tool.get() );
411  m_tool = AlgTool::Factory::create( tooltype, tooltype, fullname, parent );
412  // set new content
413  if ( m_tool ) m_tools.push_back( m_tool.get() );
414  }
416  IAlgTool* get() { return m_tool.get(); }
417  IAlgTool* operator->() const {
418  assert( m_tool );
419  return m_tool.get();
420  }
422  IAlgTool* release() { return m_tool.release(); }
424  ~ToolCreateGuard() {
425  if ( m_tool ) remove( m_tools, m_tool.get() );
426  }
427  };
428 
429  template <typename C>
430  ToolCreateGuard<C> make_toolCreateGuard( C& c ) {
431  return ToolCreateGuard<C>{c};
432  }
433 } // namespace
434 
435 //------------------------------------------------------------------------------
443 StatusCode ToolSvc::create( const std::string& tooltype, const std::string& toolname, const IInterface* parent,
444  IAlgTool*& tool )
445 //------------------------------------------------------------------------------
446 {
447 
449  // protect against empty type
450  if ( UNLIKELY( tooltype.empty() ) ) {
451  error() << "create(): No Tool Type given" << endmsg;
452  return StatusCode::FAILURE;
453  }
454 
455  // If parent has not been specified, assume it is the ToolSvc
456  if ( !parent ) parent = this;
457 
458  tool = nullptr;
459  // Automatically deletes the tool if not explicitly kept (i.e. on success).
460  // The tool is removed from the list of known tools too.
461  auto toolguard = make_toolCreateGuard( m_instancesTools );
462 
463  // Check if the tool already exist : this could happen with clones
464  std::string fullname = nameTool( toolname, parent );
465  if ( UNLIKELY( existsTool( fullname ) ) ) {
466  // Now check if the parent is the same. This allows for clones
467  for ( IAlgTool* iAlgTool : m_instancesTools ) {
468  if ( iAlgTool->name() == toolname && iAlgTool->parent() == parent ) {
469  // The tool exist with this name, type and parent: this is bad!
470  // This excludes the possibility of cloning public tools intrinsecally
471  error() << "Tool " << fullname << " already exists with the same parent" << endmsg;
472  if ( parent == this )
473  error() << "... In addition, the parent is the ToolSvc: public tools cannot be cloned!" << endmsg;
474 
475  return StatusCode::FAILURE;
476  }
477  }
478  ON_DEBUG debug() << "Creating clone of " << fullname << endmsg;
479  }
480  // instantiate the tool using the factory
481  try {
482  toolguard.create( tooltype, fullname, parent );
483  if ( UNLIKELY( !toolguard.get() ) ) {
484  error() << "Cannot create tool " << tooltype << " (No factory found)" << endmsg;
485  return StatusCode::FAILURE;
486  }
487  } catch ( const GaudiException& Exception ) {
488  // (1) perform the printout of message
489  fatal() << "Exception with tag=" << Exception.tag() << " is caught whilst instantiating tool '" << tooltype << "'"
490  << endmsg;
491  // (2) print the exception itself
492  // (NB! - GaudiException is a linked list of all "previous exceptions")
493  fatal() << Exception << endmsg;
494  return StatusCode::FAILURE;
495  } catch ( const std::exception& Exception ) {
496  // (1) perform the printout of message
497  fatal() << "Standard std::exception is caught whilst instantiating tool '" << tooltype << "'" << endmsg;
498  // (2) print the exception itself
499  // (NB! - GaudiException is a linked list of all "previous exceptions")
500  fatal() << Exception.what() << endmsg;
501  return StatusCode::FAILURE;
502  } catch ( ... ) {
503  // (1) perform the printout
504  fatal() << "UNKNOWN Exception is caught whilst instantiating tool '" << tooltype << "'" << endmsg;
505  return StatusCode::FAILURE;
506  }
507  ON_VERBOSE verbose() << "Created tool " << tooltype << "/" << fullname << endmsg;
508 
509  // Since only AlgTool has the setProperties() method it is necessary to cast
510  // to downcast IAlgTool to AlgTool in order to set the properties via the JobOptions
511  // service
512  AlgTool* mytool = dynamic_cast<AlgTool*>( toolguard.get() );
513  if ( mytool ) {
514  StatusCode sc = mytool->setProperties();
515  if ( UNLIKELY( sc.isFailure() ) ) {
516  error() << "Error setting properties for tool '" << fullname << "'" << endmsg;
517  return sc;
518  }
519  }
520 
521  // Initialize the Tool
522  StatusCode sc( StatusCode::FAILURE, true );
523  try {
524  sc = toolguard->sysInitialize();
525  }
526  // Catch any exceptions
527  catch ( const GaudiException& Exception ) {
528  error() << "GaudiException with tag=" << Exception.tag() << " caught whilst initializing tool '" << fullname << "'"
529  << endmsg << Exception << endmsg;
530  return StatusCode::FAILURE;
531  } catch ( const std::exception& Exception ) {
532  error() << "Standard std::exception caught whilst initializing tool '" << fullname << "'" << endmsg
533  << Exception.what() << endmsg;
534  return StatusCode::FAILURE;
535  } catch ( ... ) {
536  error() << "UNKNOWN Exception caught whilst initializing tool '" << fullname << "'" << endmsg;
537  return StatusCode::FAILURE;
538  }
539 
540  // Status of tool initialization
541  if ( UNLIKELY( sc.isFailure() ) ) {
542  error() << "Error initializing tool '" << fullname << "'" << endmsg;
543  return sc;
544  }
545 
546  // Start the tool if we are running.
548  sc = toolguard->sysStart();
549 
550  if ( UNLIKELY( sc.isFailure() ) ) {
551  error() << "Error starting tool '" << fullname << "'" << endmsg;
552  return sc;
553  }
554  }
555 
556  // The tool has been successfully created and initialized,
557  // so we inform the guard that it can release it
558  tool = toolguard.release();
559 
563  std::for_each( m_observers.begin(), m_observers.end(), [&]( IToolSvc::Observer* obs ) { obs->onCreate( tool ); } );
564  // TODO: replace by generic callback
565  // Register the tool with the HistorySvc
566  if ( m_pHistorySvc || service( "HistorySvc", m_pHistorySvc, false ).isSuccess() ) {
568  }
569  return StatusCode::SUCCESS;
570 }
571 
572 //------------------------------------------------------------------------------
574 //------------------------------------------------------------------------------
575 {
576 
577  if ( !parent ) { return this->name() + "." + toolname; } // RETURN
578 
579  // check that parent has a name!
580  auto named_parent = SmartIF<INamedInterface>( const_cast<IInterface*>( parent ) );
581  if ( named_parent ) {
582  auto fullname = named_parent->name() + "." + toolname;
583  return fullname; // RETURN
584  }
585 
586  error() << "Private Tools only allowed for components implementing INamedInterface" << endmsg;
587  //
588  return "." + toolname;
589 }
590 
591 //------------------------------------------------------------------------------
592 bool ToolSvc::existsTool( const std::string& fullname ) const
593 //------------------------------------------------------------------------------
594 {
595  std::lock_guard<CallMutex> lock( m_mut );
596  auto i = std::find_if( std::begin( m_instancesTools ), std::end( m_instancesTools ),
597  [&]( const IAlgTool* tool ) { return tool->name() == fullname; } );
598  return i != std::end( m_instancesTools );
599 }
600 
601 //------------------------------------------------------------------------------
603 //------------------------------------------------------------------------------
604 {
605 
606  // Cache tool name in case of errors
607  const std::string toolName = itool->name();
608  StatusCode sc;
609 
610  // Finalise the tool inside a try block
611  try {
612  sc = itool->sysFinalize();
613  }
614  // Catch any exceptions
615  catch ( const GaudiException& Exception ) {
616  error() << "GaudiException with tag=" << Exception.tag() << " caught whilst finalizing tool '" << toolName << "'"
617  << endmsg << Exception << endmsg;
618  sc = StatusCode::FAILURE;
619  } catch ( const std::exception& Exception ) {
620  error() << "Standard std::exception caught whilst finalizing tool '" << toolName << "'" << endmsg
621  << Exception.what() << endmsg;
622  sc = StatusCode::FAILURE;
623  } catch ( ... ) {
624  error() << "UNKNOWN Exception caught whilst finalizing tool '" << toolName << "'" << endmsg;
625  sc = StatusCode::FAILURE;
626  }
627 
628  return sc;
629 }
630 
631 //------------------------------------------------------------------------------
632 unsigned long ToolSvc::totalToolRefCount() const
633 //------------------------------------------------------------------------------
634 {
635  return totalRefCount( m_instancesTools );
636 }
637 //------------------------------------------------------------------------------
638 unsigned long ToolSvc::minimumToolRefCount() const
639 //------------------------------------------------------------------------------
640 {
641  auto i =
643  []( const IAlgTool* lhs, const IAlgTool* rhs ) { return lhs->refCount() < rhs->refCount(); } );
644  return i != std::end( m_instancesTools ) ? ( *i )->refCount() : 0;
645 }
646 
647 //------------------------------------------------------------------------------
649  //------------------------------------------------------------------------------
650  if ( !obs ) throw GaudiException( "Received NULL pointer", this->name() + "::registerObserver", StatusCode::FAILURE );
651 
653  obs->setUnregister( [this, obs]() {
655  auto i = std::find( m_observers.begin(), m_observers.end(), obs );
656  if ( i != m_observers.end() ) m_observers.erase( i );
657  } );
658  m_observers.push_back( obs );
659 }
660 
661 //------------------------------------------------------------------------------
663 //------------------------------------------------------------------------------
664 {
665 
666  ON_DEBUG debug() << "START transition for AlgTools" << endmsg;
667 
668  bool fail( false );
669  for ( auto& iTool : m_instancesTools ) {
670  ON_VERBOSE verbose() << iTool->name() << "::start()" << endmsg;
671 
672  if ( UNLIKELY( !iTool->sysStart().isSuccess() ) ) {
673  fail = true;
674  error() << iTool->name() << " failed to start()" << endmsg;
675  }
676  }
677 
678  if ( UNLIKELY( fail ) ) {
679  error() << "One or more AlgTools failed to start()" << endmsg;
680  return StatusCode::FAILURE;
681  }
682  return StatusCode::SUCCESS;
683 }
684 
685 //------------------------------------------------------------------------------
687 //------------------------------------------------------------------------------
688 {
689 
690  ON_DEBUG debug() << "STOP transition for AlgTools" << endmsg;
691 
692  bool fail( false );
693  for ( auto& iTool : m_instancesTools ) {
694  ON_VERBOSE verbose() << iTool->name() << "::stop()" << endmsg;
695 
696  if ( UNLIKELY( !iTool->sysStop().isSuccess() ) ) {
697  fail = true;
698  error() << iTool->name() << " failed to stop()" << endmsg;
699  }
700  }
701 
702  if ( UNLIKELY( fail ) ) {
703  error() << "One or more AlgTools failed to stop()" << endmsg;
704  return StatusCode::FAILURE;
705  }
706  return StatusCode::SUCCESS;
707 }
IHistorySvc * m_pHistorySvc
Pointer to HistorySvc.
Definition: ToolSvc.h:104
std::vector< std::string > getInstances() const override
Get names of all tool instances.
Definition: ToolSvc.cpp:337
#define UNLIKELY(x)
Definition: Kernel.h:106
Gaudi::StateMachine::State m_targetState
Service state.
Definition: Service.h:165
StatusCode initialize() override
Definition: Service.cpp:70
T empty(T... args)
Define general base for Gaudi exception.
StatusCode releaseTool(IAlgTool *tool) override
Release tool.
Definition: ToolSvc.cpp:354
Gaudi::StateMachine::State m_state
Service state.
Definition: Service.h:163
StatusCode finalize() override
Definition: Service.cpp:174
MsgStream & warning() const
shortcut for the method msgStream(MSG::WARNING)
std::string nameTool(const std::string &nameByUser, const IInterface *parent)
Get Tool full name by combining nameByUser and "parent" part.
Definition: ToolSvc.cpp:573
T rend(T... args)
allow call-backs when a tool is a created or retrieved
Definition: IToolSvc.h:231
StatusCode create(const std::string &type, const IInterface *parent, IAlgTool *&tool)
Create Tool standard way with automatically assigned name.
Definition: ToolSvc.cpp:384
virtual const std::string & tag() const
name tag for the exception, or exception type
constexpr static const auto SUCCESS
Definition: StatusCode.h:96
STL namespace.
void registerObserver(IToolSvc::Observer *obs) override
Definition: ToolSvc.cpp:648
unsigned long minimumToolRefCount() const
The minimum number of refCounts of all tools.
Definition: ToolSvc.cpp:638
T end(T... args)
MsgStream & info() const
shortcut for the method msgStream(MSG::INFO)
auto get(const Handle &handle, const Algo &, const EventContext &) -> decltype(details::deref(handle.get()))
T remove(T... args)
std::vector< IToolSvc::Observer * > m_observers
Definition: ToolSvc.h:106
StatusCode initialize() override
Initialize the service.
Definition: ToolSvc.cpp:62
This service manages tools.
Definition: ToolSvc.h:34
StatusCode setProperties()
Method for setting declared properties to the values specified in the jobOptions via the job option s...
Definition: AlgTool.cpp:117
virtual const std::string & name() const =0
Retrieve the name of the instance.
STL class.
#define DECLARE_COMPONENT(type)
StatusCode start() override
Definition: ToolSvc.cpp:662
virtual void onCreate(const IAlgTool *)
Definition: IToolSvc.h:238
const std::string & name() const override
Retrieve name of the service.
Definition: Service.cpp:284
T push_back(T... args)
Interface ID class.
Definition: IInterface.h:39
MsgStream & error() const
shortcut for the method msgStream(MSG::ERROR)
T what(T... args)
This class is used for returning status codes from appropriate routines.
Definition: StatusCode.h:61
virtual StatusCode registerAlgTool(const IAlgTool &)=0
Definition of the basic interface.
Definition: IInterface.h:254
virtual unsigned long refCount() const =0
Current reference count.
T erase(T... args)
T pop_back(T... args)
MsgStream & verbose() const
shortcut for the method msgStream(MSG::VERBOSE)
std::vector< IAlgTool * > getTools() const override
Get pointers to all tool instances.
Definition: ToolSvc.cpp:347
bool existsTool(const std::string &toolname) const
Check if the tool instance exists.
Definition: ToolSvc.cpp:592
MsgStream & debug() const
shortcut for the method msgStream(MSG::DEBUG)
bool isSuccess() const
Definition: StatusCode.h:361
STL class.
StatusCode setProperties()
Method for setting declared properties to the values specified for the job.
Definition: Service.cpp:290
virtual const IInterface * parent() const =0
The parent of the concrete AlgTool.
T get(T... args)
T insert(T... args)
unsigned long totalToolRefCount() const
The total number of refCounts on all tools in the instancesTools list.
Definition: ToolSvc.cpp:632
T find(T... args)
T size(T... args)
const StatusCode & ignore() const
Ignore/check StatusCode.
Definition: StatusCode.h:164
void setUnregister(std::function< void()> unregister)
Definition: IToolSvc.h:236
virtual unsigned long release()=0
Release Interface instance.
STL class.
std::vector< IAlgTool * > m_instancesTools
Common Tools.
Definition: ToolSvc.h:101
T begin(T... args)
virtual void onRetrieve(const IAlgTool *)
Definition: IToolSvc.h:239
Base class from which all the concrete tool classes should be derived.
Definition: AlgTool.h:57
The interface implemented by the AlgTool base class.
Definition: IAlgTool.h:33
#define ON_DEBUG
Definition: ToolSvc.cpp:31
StatusCode finalize() override
Finalize the service.
Definition: ToolSvc.cpp:83
T back(T... args)
constexpr static const auto FAILURE
Definition: StatusCode.h:97
T substr(T... args)
StatusCode service(const std::string &name, const T *&psvc, bool createIf=true) const
Access a service by name, creating it if it doesn't already exist.
Definition: Service.h:93
StatusCode finalizeTool(IAlgTool *itool) const
Finalize the given tool, with exception handling.
Definition: ToolSvc.cpp:602
T transform(T... args)
bool isFailure() const
Definition: StatusCode.h:141
T accumulate(T... args)
CallMutex m_mut
Definition: ToolSvc.h:109
T min_element(T... args)
T for_each(T... args)
StatusCode stop() override
Definition: ToolSvc.cpp:686
#define ON_VERBOSE
Definition: ToolSvc.cpp:32
MsgStream & fatal() const
shortcut for the method msgStream(MSG::FATAL)
MsgStream & endmsg(MsgStream &s)
MsgStream Modifier: endmsg. Calls the output method of the MsgStream.
Definition: MsgStream.h:202
virtual StatusCode queryInterface(const InterfaceID &ti, void **pp)=0
Set the void** to the pointer to the requested interface of the instance.
StatusCode retrieve(const std::string &type, const InterfaceID &iid, IAlgTool *&tool, const IInterface *parent, bool createIf) override
Retrieve tool, create it by default as common tool if it does not already exist.
Definition: ToolSvc.cpp:239
T rbegin(T... args)