// // Thread_POSIX.h // // $Id: //poco/1.4/Foundation/include/Poco/Thread_POSIX.h#3 $ // // Library: Foundation // Package: Threading // Module: Thread // // Definition of the ThreadImpl class for POSIX Threads. // // Copyright (c) 2004-2007, Applied Informatics Software Engineering GmbH. // and Contributors. // // SPDX-License-Identifier: BSL-1.0 // #ifndef Foundation_Thread_POSIX_INCLUDED #define Foundation_Thread_POSIX_INCLUDED #include "Poco/Foundation.h" #include "Poco/Runnable.h" #include "Poco/SignalHandler.h" #include "Poco/Event.h" #include "Poco/RefCountedObject.h" #include "Poco/AutoPtr.h" #include "Poco/SharedPtr.h" #include // must be limits.h (not ) for PTHREAD_STACK_MIN on Solaris #include #if !defined(POCO_NO_SYS_SELECT_H) #include #endif #include #if defined(POCO_VXWORKS) #include #endif namespace Poco { class Foundation_API ThreadImpl { public: #if POCO_OS == POCO_OS_LINUX // OS kernel thread ID typedef pid_t TIDImpl; #elif POCO_OS == POCO_OS_MAC_OS_X // OS kernel thread ID typedef mach_port_t TIDImpl; #else // Default: pthread id typedef pthread_t TIDImpl; #endif typedef void (*Callable)(void*); enum Priority { PRIO_LOWEST_IMPL, PRIO_LOW_IMPL, PRIO_NORMAL_IMPL, PRIO_HIGH_IMPL, PRIO_HIGHEST_IMPL }; enum Policy { POLICY_DEFAULT_IMPL = SCHED_OTHER }; ThreadImpl(); ~ThreadImpl(); TIDImpl tidImpl() const; void setPriorityImpl(int prio); int getPriorityImpl() const; void setOSPriorityImpl(int prio, int policy = SCHED_OTHER); int getOSPriorityImpl() const; static int getMinOSPriorityImpl(int policy); static int getMaxOSPriorityImpl(int policy); void setStackSizeImpl(int size); int getStackSizeImpl() const; void setAffinityImpl(int cpu); int getAffinityImpl() const; void startImpl(SharedPtr pTarget); void joinImpl(); bool joinImpl(long milliseconds); bool isRunningImpl() const; static void sleepImpl(long milliseconds); static void yieldImpl(); static ThreadImpl* currentImpl(); static TIDImpl currentTidImpl(); protected: static void* runnableEntry(void* pThread); static int mapPrio(int prio, int policy = SCHED_OTHER); static int reverseMapPrio(int osPrio, int policy = SCHED_OTHER); private: class CurrentThreadHolder { public: CurrentThreadHolder() { if (pthread_key_create(&_key, NULL)) throw SystemException("cannot allocate thread context key"); } ~CurrentThreadHolder() { pthread_key_delete(_key); } ThreadImpl* get() const { return reinterpret_cast(pthread_getspecific(_key)); } void set(ThreadImpl* pThread) { pthread_setspecific(_key, pThread); } private: pthread_key_t _key; }; struct ThreadData: public RefCountedObject { ThreadData(): thread(0), prio(PRIO_NORMAL_IMPL), osPrio(), policy(SCHED_OTHER), done(Event::EVENT_MANUALRESET), stackSize(POCO_THREAD_STACK_SIZE), started(false), joined(false) { #if defined(POCO_VXWORKS) // This workaround is for VxWorks 5.x where // pthread_init() won't properly initialize the thread. std::memset(&thread, 0, sizeof(thread)); #endif } SharedPtr pRunnableTarget; pthread_t thread; TIDImpl tid; int prio; int osPrio; int policy; Event done; std::size_t stackSize; bool started; bool joined; }; AutoPtr _pData; static CurrentThreadHolder _currentThreadHolder; #if defined(POCO_OS_FAMILY_UNIX) && !defined(POCO_VXWORKS) SignalHandler::JumpBufferVec _jumpBufferVec; friend class SignalHandler; #endif }; // // inlines // inline int ThreadImpl::getPriorityImpl() const { return _pData->prio; } inline int ThreadImpl::getOSPriorityImpl() const { return _pData->osPrio; } inline bool ThreadImpl::isRunningImpl() const { return !_pData->pRunnableTarget.isNull(); } inline void ThreadImpl::yieldImpl() { sched_yield(); } inline int ThreadImpl::getStackSizeImpl() const { return static_cast(_pData->stackSize); } inline ThreadImpl::TIDImpl ThreadImpl::tidImpl() const { return _pData->tid; } } // namespace Poco #endif // Foundation_Thread_POSIX_INCLUDED