//%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_Threads_h #define Pegasus_Threads_h #include #include #include #include #include // ATTN: can we consolidate these someplace? #if defined(PEGASUS_HAVE_PTHREADS) # include # include # include #elif defined(PEGASUS_HAVE_WINDOWS_THREADS) # include # include #else # error ": not implemented" #endif PEGASUS_NAMESPACE_BEGIN //============================================================================== // // PEGASUS_THREAD_CDECL // //============================================================================== #if defined(PEGASUS_OS_TYPE_WINDOWS) # define PEGASUS_THREAD_CDECL __stdcall #else # define PEGASUS_THREAD_CDECL /* empty */ #endif //============================================================================== // // ThreadId // //============================================================================== struct ThreadId { // The character representation of a uint64 requires 22 bytes including the // null terminator. char buffer[22]; }; //============================================================================== // // ThreadType // //============================================================================== #if defined(PEGASUS_HAVE_PTHREADS) struct ThreadType { ThreadType() { memset(&thread, 0, sizeof(thread)); } ThreadType(pthread_t thread_) : thread(thread_) { } pthread_t thread; }; #endif /* PEGASUS_HAVE_PTHREADS */ #if defined(PEGASUS_HAVE_WINDOWS_THREADS) struct ThreadType { ThreadType() : handle(NULL) { } HANDLE handle; }; #endif /* PEGASUS_HAVE_WINDOWS_THREADS */ //============================================================================== // // ThreadReturnType // //============================================================================== #if defined(PEGASUS_HAVE_PTHREADS) typedef void* ThreadReturnType; #endif #if defined(PEGASUS_HAVE_WINDOWS_THREADS) typedef unsigned ThreadReturnType; #endif //============================================================================== // // ThreadHandle // //============================================================================== #if defined(PEGASUS_HAVE_PTHREADS) struct ThreadHandle { ThreadType thid; }; #elif defined(PEGASUS_HAVE_WINDOWS_THREADS) struct ThreadHandle { ThreadType thid; }; #endif //============================================================================== // // Threads // //============================================================================== class PEGASUS_COMMON_LINKAGE Threads { public: enum Type { DETACHED, JOINABLE }; static int create( ThreadType& thread, Type type, void* (*start)(void*), void* arg); static ThreadType self(); static bool equal(ThreadType x, ThreadType y); static void exit(ThreadReturnType rc); static void cancel(ThreadType th, ThreadReturnType rc); static void yield(); static void sleep(int msec); static void cleanup_push(void (*start)(void*), void* arg); static void cleanup_pop(int execute); static ThreadId id(const ThreadType& x = Threads::self()); static bool null(const ThreadType& x = Threads::self()); static void clear(ThreadType& x); }; //============================================================================== // // POSIX Threads Implementation // //============================================================================== #if defined(PEGASUS_HAVE_PTHREADS) inline bool Threads::equal(ThreadType x, ThreadType y) { return pthread_equal(x.thread, y.thread); } inline void Threads::exit(ThreadReturnType rc) { // NOTE: pthread_exit exhibits unusual behavior on RHEL 3 U2, as // documented in Bugzilla 3836. Where feasible, it may be advantageous // to avoid using this function. pthread_exit(rc); } inline void Threads::cancel(ThreadType th, ThreadReturnType rc) { pthread_cancel(th.thread); } inline void Threads::yield() { #if defined(PEGASUS_PLATFORM_AIX_RS_IBMCXX) || \ defined(PEGASUS_PLATFORM_PASE_ISERIES_IBMCXX) || \ defined(PEGASUS_PLATFORM_HPUX_ACC) || \ defined(PEGASUS_PLATFORM_TRU64_ALPHA_DECCXX) || \ defined(PEGASUS_OS_VMS) || \ defined(PEGASUS_OS_ZOS) || \ defined(PEGASUS_OS_VXWORKS) || \ defined(PEGASUS_OS_DARWIN) || \ defined(PEGASUS_OS_SOLARIS) sched_yield(); #else pthread_yield(); #endif } inline void Threads::cleanup_push(void (*func)(void*), void* arg) { // ATTN: it is doubtful whether cleanup handlers ever really worked. // They are only used in two places and not used in many other // places where mutexes are obtained. Further, they are only // implemented correctly on one or two platforms. For now, we // will defer their implementation until we can find a way to // implement them on all platforms (using thread local storage). } inline void Threads::cleanup_pop(int execute) { // ATTN: not implemented. } inline ThreadId Threads::id(const ThreadType& x) { ThreadId tid; #if defined(PEGASUS_PLATFORM_ZOS_ZSERIES_IBM) const char* s = x.thread.__; sprintf(tid.buffer, "%X%X%X%X%X%X%X%X", s[0], s[1], s[2], s[3], s[4], s[5], s[6], s[7]); #else sprintf(tid.buffer, "%" PEGASUS_64BIT_CONVERSION_WIDTH "u", Uint64(x.thread)); #endif return tid; } inline bool Threads::null(const ThreadType& x) { #if defined(PEGASUS_PLATFORM_ZOS_ZSERIES_IBM) Uint64 tmp; memcpy(&tmp, x.thread.__, sizeof(Uint64)); return tmp == 0; #else return x.thread == 0; #endif } inline void Threads::clear(ThreadType& x) { memset(&x, 0, sizeof(x)); } #endif /* defined(PEGASUS_HAVE_PTHREADS) */ //============================================================================== // // Windows Threads Implementation // //============================================================================== #if defined(PEGASUS_HAVE_WINDOWS_THREADS) inline ThreadType Threads::self() { ThreadType tt; tt.handle = GetCurrentThread(); return tt; } inline bool Threads::equal(ThreadType x, ThreadType y) { return x.handle == y.handle; } inline void Threads::exit(ThreadReturnType rc) { _endthreadex(rc); } inline void Threads::cancel(ThreadType th, ThreadReturnType rc) { TerminateThread(th.handle, rc); } inline void Threads::yield() { Sleep(0); } inline void Threads::cleanup_push(void (*func)(void*), void* arg) { // ATTN: Not implemented on Windows. } inline void Threads::cleanup_pop(int execute) { // ATTN: Not implemented on Windows. } inline ThreadId Threads::id(const ThreadType& x) { ThreadId tmp; sprintf(tmp.buffer, "%" PEGASUS_64BIT_CONVERSION_WIDTH "u", Uint64(x.handle)); return tmp; } inline bool Threads::null(const ThreadType& x) { return x.handle == NULL; } inline void Threads::clear(ThreadType& x) { x.handle = NULL; } #endif /* defined(PEGASUS_HAVE_WINDOWS_THREADS) */ PEGASUS_NAMESPACE_END #endif /* Pegasus_Threads_h */