//%2006//////////////////////////////////////////////////////////////////////// // // Copyright (c) 2000, 2001, 2002 BMC Software; Hewlett-Packard Development // Company, L.P.; IBM Corp.; The Open Group; Tivoli Systems. // Copyright (c) 2003 BMC Software; Hewlett-Packard Development Company, L.P.; // IBM Corp.; EMC Corporation, The Open Group. // Copyright (c) 2004 BMC Software; Hewlett-Packard Development Company, L.P.; // IBM Corp.; EMC Corporation; VERITAS Software Corporation; The Open Group. // Copyright (c) 2005 Hewlett-Packard Development Company, L.P.; IBM Corp.; // EMC Corporation; VERITAS Software Corporation; The Open Group. // Copyright (c) 2006 Hewlett-Packard Development Company, L.P.; IBM Corp.; // EMC Corporation; Symantec Corporation; The Open Group. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to // deal in the Software without restriction, including without limitation the // rights to use, copy, modify, merge, publish, distribute, sublicense, and/or // sell copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // THE ABOVE COPYRIGHT NOTICE AND THIS PERMISSION NOTICE SHALL BE INCLUDED IN // ALL COPIES OR SUBSTANTIAL PORTIONS OF THE SOFTWARE. THE SOFTWARE IS PROVIDED // "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT // LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR // PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT // HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN // ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION // WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. // //============================================================================== // //%///////////////////////////////////////////////////////////////////////////// #ifndef Pegasus_Mutex_h #define Pegasus_Mutex_h #include #include #include #include #include PEGASUS_NAMESPACE_BEGIN //============================================================================== // // MutexRep // //============================================================================== #if defined(PEGASUS_HAVE_PTHREADS) typedef pthread_mutex_t MutexType; inline void mutex_lock(MutexType* mutex) { pthread_mutex_lock(mutex); } inline void mutex_unlock(MutexType* mutex) { pthread_mutex_unlock(mutex); } struct MutexRep { pthread_mutex_t mutex; int count; }; # define PEGASUS_MUTEX_INITIALIZER PTHREAD_MUTEX_INITIALIZER #endif #if defined(PEGASUS_HAVE_WINDOWS_THREADS) typedef HANDLE MutexType; inline void mutex_lock(MutexType* m) { WaitForSingleObject(*m, INFINITE); } inline void mutex_unlock(MutexType* m) { ReleaseMutex(*m); } struct MutexRep { MutexType handle; size_t count; }; # define PEGASUS_MUTEX_INITIALIZER (CreateMutex(NULL, FALSE, NULL)) #endif //============================================================================== // // Mutex // //============================================================================== class PEGASUS_COMMON_LINKAGE Mutex { public: enum RecursiveTag { RECURSIVE }; enum NonRecursiveTag { NON_RECURSIVE }; /** Default constructor creates a recursive mutex. */ Mutex(); /** Call as Mutex(Mutex::RECURSIVE) to create a recursive mutex. */ Mutex(RecursiveTag); /** Call as Mutex(Mutex::NON_RECURSIVE) to create a non-recursive mutex. */ Mutex(NonRecursiveTag); ~Mutex(); void lock(); /** Attempts to lock the mutex without blocking. @return A Boolean indicating whether the lock was acquired. */ Boolean try_lock(); /** Attempts to lock the mutex within the specified time. @param milliseconds The maximum time to block while attempting to acquire the lock. @return A Boolean indicating whether the lock was acquired. */ Boolean timed_lock(Uint32 milliseconds); void unlock(); #if defined(PEGASUS_OS_LINUX) || \ (defined(PEGASUS_OS_ZOS) && !(__TARGET_LIB__ < 0x41090000)) /** This method must only be called after a fork() to reset the mutex lock status in the new process. Any other use of this method is unsafe. */ void reinitialize(); #endif private: Mutex(const Mutex&); Mutex& operator=(const Mutex&); MutexRep _rep; Magic<0x57D11485> _magic; friend class Condition; }; //============================================================================== // // AutoMutex // //============================================================================== class PEGASUS_COMMON_LINKAGE AutoMutex { public: AutoMutex(Mutex& mutex) : _mutex(mutex) { _mutex.lock(); } ~AutoMutex() { _mutex.unlock(); } private: AutoMutex(); // Unimplemented AutoMutex(const AutoMutex& x); // Unimplemented AutoMutex& operator=(const AutoMutex& x); // Unimplemented Mutex& _mutex; }; //============================================================================== // // PEGASUS_FORK_SAFE_MUTEX // //============================================================================== // Use of this macro ensures that a static Mutex is not locked across a fork(). #if !defined(PEGASUS_HAVE_PTHREADS) || \ (defined(PEGASUS_OS_ZOS) && (__TARGET_LIB__ < 0x41090000)) || \ defined(PEGASUS_OS_VMS) # define PEGASUS_FORK_SAFE_MUTEX(mutex) #elif defined(PEGASUS_OS_LINUX) || \ (defined(PEGASUS_OS_ZOS) && !(__TARGET_LIB__ < 0x41090000)) # define PEGASUS_FORK_SAFE_MUTEX(mutex) \ class ForkSafeMutex ## mutex \ { \ public: \ ForkSafeMutex ## mutex() \ { \ pthread_atfork( \ 0, \ 0, \ _reinitializeMutex); \ } \ \ private: \ static void _reinitializeMutex() \ { \ mutex.reinitialize(); \ } \ }; \ \ static ForkSafeMutex ## mutex __forkSafeMutex ## mutex; #else # define PEGASUS_FORK_SAFE_MUTEX(mutex) \ class ForkSafeMutex ## mutex \ { \ public: \ ForkSafeMutex ## mutex() \ { \ pthread_atfork( \ _lockMutex, \ _unlockMutex, \ _unlockMutex); \ } \ \ private: \ static void _lockMutex() \ { \ mutex.lock(); \ } \ \ static void _unlockMutex() \ { \ mutex.unlock(); \ } \ }; \ \ static ForkSafeMutex ## mutex __forkSafeMutex ## mutex; #endif PEGASUS_NAMESPACE_END #endif /* Pegasus_Mutex_h */