⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 pasync.h

📁 PTypes (C++ Portable Types Library) is a simple alternative to the STL that includes multithreading
💻 H
字号:
/* * *  C++ Portable Types Library (PTypes) *  Version 1.7.5   Released 9-Mar-2003 * *  Copyright (c) 2001, 2002, 2003 Hovik Melikyan * *  http://www.melikyan.com/ptypes/ *  http://ptypes.sourceforge.net/ * */#ifndef __PASYNC_H__#define __PASYNC_H__#ifdef WIN32#  define _WINSOCKAPI_   // prevent inclusion of winsock.h in windows.h#  include <windows.h>#else#  include <pthread.h>#  include <semaphore.h>#endif#ifndef __PPORT_H__#include "pport.h"#endif#ifndef __PTYPES_H__#include "ptypes.h"#endifPTYPES_BEGIN////  Summary of implementation:////  atomic increment/decrement/exchange//    Win32: internal, asm//    GCC/i386: internal, asm//    Other: internal, mutex hash table////  mutex//    Win32: Critical section//    Other: POSIX mutex////  trigger//    Win32: Event//    Other: internal, POSIX cond/mutex////  rwlock://    Win32: internal, Event/mutex//    MacOS: internal, POSIX cond/mutex//    Other: POSIX rwlock////  semaphore://    Win32: = tsemaphore//    MacOS: = tsemaphore//    Other: POSIX semaphore////  tsemaphore (with timed waiting)://    Win32: Semaphore//    Other: internal, POSIX mutex/cond//#ifdef _MSC_VER#pragma pack(push, 4)#endif#ifdef WIN32   typedef int pthread_id_t;   typedef HANDLE pthread_t;#  define __PFASTCALL __fastcall#else   typedef pthread_t pthread_id_t;#  define __PFASTCALL#endifptpublic int   __PFASTCALL pincrement(int* target);ptpublic int   __PFASTCALL pdecrement(int* target);ptpublic int   __PFASTCALL pexchange(int* target, int value);ptpublic void* __PFASTCALL pexchange(void** target, void* value);template <class T> inline T* tpexchange(T** target, T* value)    { return (T*)pexchange((void**)target, (void*)value); }ptpublic void psleep(uint milliseconds);ptpublic bool pthrequal(pthread_id_t id);  // note: this is NOT the thread handle, use thread::get_id()ptpublic pthread_id_t pthrself();          // ... same// -------------------------------------------------------------------- //// --- mutex ---------------------------------------------------------- //// -------------------------------------------------------------------- //#ifdef WIN32struct ptpublic mutex{protected:    CRITICAL_SECTION critsec;public:    mutex()         { InitializeCriticalSection(&critsec); }    ~mutex()        { DeleteCriticalSection(&critsec); }    void enter()    { EnterCriticalSection(&critsec); }    void leave()    { LeaveCriticalSection(&critsec); }    void lock()     { enter(); }    void unlock()   { leave(); }};#elsestruct mutex{protected:    pthread_mutex_t mtx;public:    mutex()         { pthread_mutex_init(&mtx, 0); }    ~mutex()        { pthread_mutex_destroy(&mtx); }    void enter()    { pthread_mutex_lock(&mtx); }    void leave()    { pthread_mutex_unlock(&mtx); }    void lock()     { enter(); }    void unlock()   { leave(); }};#endif//// mutex table for hashed memory locking (undocumented)//#define _MUTEX_HASH_SIZE     17      // a prime number for hashing#ifdef WIN32#  define pmemlock        mutex#  define pmementer(m)    (m)->lock()#  define pmemleave(m)    (m)->unlock()#else#  define _MTX_INIT       PTHREAD_MUTEX_INITIALIZER#  define pmemlock        pthread_mutex_t#  define pmementer       pthread_mutex_lock#  define pmemleave       pthread_mutex_unlock#endifptpublic extern pmemlock _mtxtable[_MUTEX_HASH_SIZE];inline pmemlock* pgetmemlock(void* addr){    return _mtxtable + uint(addr) % _MUTEX_HASH_SIZE;}// -------------------------------------------------------------------- //// --- trigger -------------------------------------------------------- //// -------------------------------------------------------------------- //#ifdef WIN32class ptpublic trigger{protected:    HANDLE handle;      // Event objectpublic:    trigger(bool autoreset, bool state);    ~trigger()          { CloseHandle(handle); }    void wait()         { WaitForSingleObject(handle, INFINITE); }    void post()         { SetEvent(handle); }    void signal()       { post(); }    void reset()        { ResetEvent(handle); }};#elseclass trigger{protected:    pthread_mutex_t mtx;    pthread_cond_t cond;    int state;    bool autoreset;public:    trigger(bool autoreset, bool state);    ~trigger();    void wait();    void post();    void signal()  { post(); }    void reset();};#endif// -------------------------------------------------------------------- //// --- rwlock --------------------------------------------------------- //// -------------------------------------------------------------------- //#if defined(WIN32) || defined(__DARWIN__)#  define __PTYPES_RWLOCK__#elif defined(linux)   // on Linux rwlocks are included only with -D_GNU_SOURCE.   // programs that don't use rwlocks, do not need to define   // _GNU_SOURCE either.#  if defined(_GNU_SOURCE) || defined(__USE_UNIX98)#    define __POSIX_RWLOCK__#  endif#else#  define __POSIX_RWLOCK__#endif#ifdef __PTYPES_RWLOCK__struct ptpublic rwlock: protected mutex{protected:#ifdef WIN32    HANDLE  reading;    // Event object    HANDLE  finished;   // Event object    int     readcnt;    int     writecnt;#else    pthread_mutex_t mtx;    pthread_cond_t readcond;    pthread_cond_t writecond;    int locks;    int writers;    int readers;#endifpublic:    rwlock();    ~rwlock();    void rdlock();    void wrlock();    void unlock();    void lock()     { wrlock(); }};#elif defined(__POSIX_RWLOCK__)struct rwlock{protected:    pthread_rwlock_t rw;public:    rwlock();    ~rwlock()       { pthread_rwlock_destroy(&rw); }    void rdlock()   { pthread_rwlock_rdlock(&rw); }    void wrlock()   { pthread_rwlock_wrlock(&rw); }    void unlock()   { pthread_rwlock_unlock(&rw); }    void lock()     { wrlock(); }};#endif// -------------------------------------------------------------------- //// --- semaphore ------------------------------------------------------ //// -------------------------------------------------------------------- //#if defined(WIN32) || defined(__DARWIN__)#  define __SEM_TO_TIMEDSEM__#endif#ifdef __SEM_TO_TIMEDSEM__// map ordinary semaphore to timed semaphoreclass tsemaphore;typedef tsemaphore semaphore;#elseclass semaphore: public unknown{protected:    sem_t handle;public:    semaphore(int initvalue);    virtual ~semaphore();    void wait();    void post();    void signal()  { post(); }};#endifclass ptpublic tsemaphore: public unknown{protected:#ifdef WIN32    HANDLE handle;#else    int count;    pthread_mutex_t mtx;    pthread_cond_t cond;#endifpublic:    tsemaphore(int initvalue);    virtual ~tsemaphore();    bool wait(int msecs = -1);    void post();    void signal()  { post(); }};// -------------------------------------------------------------------- //// --- thread --------------------------------------------------------- //// -------------------------------------------------------------------- //class ptpublic thread: public unknown{protected:#ifdef WIN32    ulong id;#endif    pthread_t  handle;    bool autofree;    int  running;    int  signaled;    int  freed;    bool finished;    tsemaphore relaxsem;    virtual void execute() = 0;    virtual void cleanup();    bool relax(int msecs) { return relaxsem.wait(msecs); }    friend void _threadepilog(thread* thr);#ifdef WIN32    friend ulong __stdcall _threadproc(void* arg);#else    friend void* _threadproc(void* arg);#endifpublic:    thread(bool iautofree);    virtual ~thread();#ifdef WIN32    pthread_id_t get_id()   { return int(id); }#else    pthread_id_t get_id()   { return handle; }#endif    bool get_running()    { return running != 0; }    bool get_finished()   { return finished; }    bool get_signaled()   { return signaled != 0; }    void start();    void signal();    void waitfor();};// -------------------------------------------------------------------- //// --- msgqueue ------------------------------------------------------- //// -------------------------------------------------------------------- //const int MSG_USER = 0;const int MSG_USER_MAX = 0xBFFFF;const int MSG_LIB = 0xC0000;const int MSG_QUIT = MSG_LIB;class ptpublic message: public unknown{protected:    message* next;          // next in the message chain, used internally    semaphore* sync;        // used internally by msgqueue::send(), when called from a different thread    friend class msgqueue;  // my friend, message queue...public:    int id;    int result;    int param;    message(int iid, int iparam = 0);    virtual ~message();};class ptpublic msgqueue{private:    message*  head;         // queue head    message*  tail;         // queue tail    int       qcount;       // number of items in the queue    semaphore sem;          // queue semaphore    mutex     qlock;        // critical sections in enqueue and dequeue    mutex     thrlock;      // lock for the queue processing    pthread_id_t owner;     // thread ID of the queue processing thread    void enqueue(message* msg);    void push(message* msg);    message* dequeue(bool safe = true);    void purgequeue();    int  finishmsg(message* msg);    void handlemsg(message* msg);    void takeownership();protected:    bool quit;    void defhandler(message& msg);    virtual void msghandler(message& msg) = 0;public:    msgqueue();    virtual ~msgqueue();    // functions calling from the owner thread:    int  msgsavail()  { return qcount; }    void processone();  // process one message, may hang if no msgs in the queue    void processmsgs(); // process all available messages and return    void run();         // process messages until MSG_QUIT    // functions calling from any thread:    void post(message* msg);    void post(int id, int param = 0);    void posturgent(message* msg);    void posturgent(int id, int param = 0);    int  send(message* msg);    int  send(int id, int param = 0);};#ifdef _MSC_VER#pragma pack(pop)#endifPTYPES_END#endif // __PASYNC_H__

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -