#ifndef CTHREADREP_H #define CTHREADREP_H #include "common.h" #if HAVE_PTHREAD #include <pthread.h> #elif WINDOWS_LIKE #define WIN32_LEAN_AND_MEAN #include <windows.h> #endif class CMutex; class IJob; //! Internal thread class; do not use directly class CThreadRep { public: CThreadRep(IJob*, void* userData); // manipulators // initialize the thread library static void initThreads(); // change ref count void ref(); void unref(); // the calling thread sleeps for t seconds. if t == 0.0 then // the thread yields the CPU. void sleep(double timeout); // cancel the thread void cancel(); // set cancellation state bool enableCancel(bool enable); // permanently disable further cancellation and start cancel cleanup // if cancel has been called and cancellation hasn't been started yet. void testCancel(); // wait for thread to exit or for current thread to cancel bool wait(CThreadRep*, double timeout); #if WINDOWS_LIKE // wait for a message on the queue bool waitForEvent(double timeout); #endif // set the priority void setPriority(int n); // accessors // get the exit result for this thread. thread must be terminated. void* getResult() const; // get the user data passed to the constructor void* getUserData() const; // get the current cancellable state bool isCancellable() const; #if HAVE_PTHREAD bool isExited() const; #elif WINDOWS_LIKE HANDLE getExitEvent() const; HANDLE getCancelEvent() const; #endif // return the thread rep for the calling thread. the returned // rep has been ref()'d. static CThreadRep* getCurrentThreadRep(); protected: virtual ~CThreadRep(); private: // internal constructor CThreadRep(); // initialization/cleanup void init(); void fini(); // thread rep lookup static CThreadRep* find(); // thread functions #if HAVE_PTHREAD static void* threadFunc(void* arg); static void threadCancel(int); static void* threadSignalHandler(void*); #elif WINDOWS_LIKE static unsigned int __stdcall threadFunc(void* arg); #endif void doThreadFunc(); // not implemented CThreadRep(const CThreadRep&); CThreadRep& operator=(const CThreadRep&); private: static CMutex* s_mutex; static CThreadRep* s_head; CThreadRep* m_prev; CThreadRep* m_next; int m_refCount; IJob* m_job; void* m_userData; void* m_result; bool m_cancellable; bool m_cancelling; #if HAVE_PTHREAD pthread_t m_thread; bool m_exit; bool m_cancel; static pthread_t s_signalThread; #endif #if WINDOWS_LIKE HANDLE m_thread; DWORD m_id; HANDLE m_exit; HANDLE m_cancel; #endif }; // // CThreadPtr -- auto unref'ing pointer to thread rep // class CThreadPtr { public: CThreadPtr(CThreadRep* rep) : m_rep(rep) { } ~CThreadPtr() { m_rep->unref(); } CThreadRep* operator->() const { return m_rep; } private: // not implemented CThreadPtr(const CThreadPtr&); CThreadPtr& operator=(const CThreadPtr&); private: CThreadRep* m_rep; }; #endif