// -*- C++ -*- #ifndef TAO_SINGLETON_CPP #define TAO_SINGLETON_CPP #include "tao/TAO_Singleton.h" #if !defined (ACE_LACKS_PRAGMA_ONCE) # pragma once #endif /* ACE_LACKS_PRAGMA_ONCE */ #include "tao/TAO_Singleton_Manager.h" #include "ace/Guard_T.h" #include "ace/Object_Manager.h" #include "tao/debug.h" #include "ace/os_include/os_typeinfo.h" #if !defined (__ACE_INLINE__) #include "tao/TAO_Singleton.inl" #endif /* __ACE_INLINE__ */ TAO_BEGIN_VERSIONED_NAMESPACE_DECL template void TAO_Singleton::dump (void) { #if defined (ACE_HAS_DUMP) ACE_TRACE ("TAO_Singleton::dump"); #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) TAOLIB_DEBUG ((LM_DEBUG, ACE_TEXT ("instance_ = %@"), TAO_Singleton::instance_i ())); TAOLIB_DEBUG ((LM_DEBUG, ACE_END_DUMP)); #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ #endif /* ACE_HAS_DUMP */ } template TAO_Singleton *& TAO_Singleton::instance_i (void) { #if defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) // Pointer to the Singleton instance. This works around a bug with // G++ and it's (mis-)handling of templates and statics... static TAO_Singleton *singleton_ = 0; return singleton_; #else return TAO_Singleton::singleton_; #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ } template TYPE * TAO_Singleton::instance (void) { ACE_TRACE ("TAO_Singleton::instance"); TAO_Singleton *&singleton = TAO_Singleton::instance_i (); // Perform the Double-Check pattern... if (singleton == 0) { if (TAO_Singleton_Manager::starting_up () || TAO_Singleton_Manager::shutting_down ()) { // The program is still starting up, and therefore assumed // to be single threaded. There's no need to double-check. // Or, the TAO_Singleton_Manager instance has been destroyed, // so the preallocated lock is not available. Either way, // don't register for destruction with the // TAO_Singleton_Manager: we'll have to leak this instance. ACE_NEW_RETURN (singleton, (TAO_Singleton), 0); } else { #if defined (ACE_MT_SAFE) && (ACE_MT_SAFE != 0) // Obtain a lock from the ACE_Object_Manager. The pointer // is static, so we only obtain one per TAO_Singleton // instantiation. static ACE_LOCK *lock = 0; if (ACE_Object_Manager::get_singleton_lock (lock) != 0) // Failed to acquire the lock! return 0; ACE_GUARD_RETURN (ACE_LOCK, ace_mon, *lock, 0); if (singleton == 0) { #endif /* ACE_MT_SAFE */ ACE_NEW_RETURN (singleton, (TAO_Singleton), 0); // Register for destruction with TAO_Singleton_Manager. #if !defined (ACE_MT_SAFE) || (ACE_MT_SAFE == 0) TAO_Singleton_Manager::at_exit (singleton, 0, typeid (TYPE).name ()); #else TAO_Singleton_Manager::at_exit (singleton, &lock, typeid (TYPE).name()); } #endif /* ACE_MT_SAFE */ } } return &singleton->instance_; } template void TAO_Singleton::cleanup (void *param) { delete this; TAO_Singleton::instance_i () = 0; #if defined ACE_MT_SAFE && ACE_MT_SAFE != 0 if (param) { ACE_LOCK **lock = static_cast (param); *lock = 0; } #endif } #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) // Pointer to the Singleton instance. template TAO_Singleton * TAO_Singleton::singleton_ = 0; template TAO_TSS_Singleton * TAO_TSS_Singleton::singleton_ = 0; #endif /* !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) */ template void TAO_TSS_Singleton::dump (void) { #if defined (ACE_HAS_DUMP) ACE_TRACE ("TAO_TSS_Singleton::dump"); #if !defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) TAOLIB_DEBUG ((LM_DEBUG, ACE_TEXT ("instance_ = %@"), TAO_TSS_Singleton::instance_i ())); TAOLIB_DEBUG ((LM_DEBUG, ACE_END_DUMP)); #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ #endif /* ACE_HAS_DUMP */ } template TAO_TSS_Singleton *& TAO_TSS_Singleton::instance_i (void) { #if defined (ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES) // Pointer to the Singleton instance. This works around a bug with // G++ and it's (mis-)handling of templates and statics... static TAO_TSS_Singleton *singleton_ = 0; return singleton_; #else return TAO_TSS_Singleton::singleton_; #endif /* ACE_LACKS_STATIC_DATA_MEMBER_TEMPLATES */ } template TYPE * TAO_TSS_Singleton::instance (void) { ACE_TRACE ("TAO_TSS_Singleton::instance"); TAO_TSS_Singleton *&singleton = TAO_TSS_Singleton::instance_i (); // Perform the Double-Check pattern... if (singleton == 0) { if (TAO_Singleton_Manager::starting_up () || TAO_Singleton_Manager::shutting_down ()) { // The program is still starting up, and therefore assumed // to be single threaded. There's no need to double-check. // Or, the TAO_Singleton_Manager instance has been destroyed, // so the preallocated lock is not available. Either way, // don't register for destruction with the // TAO_Singleton_Manager: we'll have to leak this instance. ACE_NEW_RETURN (singleton, (TAO_TSS_Singleton), 0); } else { #if defined (ACE_MT_SAFE) && (ACE_MT_SAFE != 0) // Obtain a lock from the ACE_Object_Manager. The pointer // is static, so we only obtain one per TAO_Singleton // instantiation. static ACE_LOCK *lock = 0; if (ACE_Object_Manager::get_singleton_lock (lock) != 0) // Failed to acquire the lock! return 0; ACE_GUARD_RETURN (ACE_LOCK, ace_mon, *lock, 0); if (singleton == 0) { #endif /* ACE_MT_SAFE */ ACE_NEW_RETURN (singleton, (TAO_TSS_Singleton), 0); // Register for destruction with TAO_Singleton_Manager. TAO_Singleton_Manager::at_exit (singleton, 0, typeid (TYPE).name ()); #if defined (ACE_MT_SAFE) && (ACE_MT_SAFE != 0) } #endif /* ACE_MT_SAFE */ } } return ACE_TSS_GET (&singleton->instance_, TYPE); } template void TAO_TSS_Singleton::cleanup (void *) { delete this; TAO_TSS_Singleton::instance_i () = 0; } TAO_END_VERSIONED_NAMESPACE_DECL #endif /* TAO_SINGLETON_CPP */