| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159 | 
							- /////////////////////////////////////////////////////////////////////////////
 
- // Name:        wx/atomic.h
 
- // Purpose:     functions to manipulate atomically integers and pointers
 
- // Author:      Armel Asselin
 
- // Created:     12/13/2006
 
- // Copyright:   (c) Armel Asselin
 
- // Licence:     wxWindows licence
 
- /////////////////////////////////////////////////////////////////////////////
 
- #ifndef _WX_ATOMIC_H_
 
- #define _WX_ATOMIC_H_
 
- // ----------------------------------------------------------------------------
 
- // headers
 
- // ----------------------------------------------------------------------------
 
- // get the value of wxUSE_THREADS configuration flag
 
- #include "wx/defs.h"
 
- // constraints on the various functions:
 
- //  - wxAtomicDec must return a zero value if the value is zero once
 
- //  decremented else it must return any non-zero value (the true value is OK
 
- //  but not necessary).
 
- #if wxUSE_THREADS
 
- #if defined(HAVE_GCC_ATOMIC_BUILTINS)
 
- // NB: we intentionally don't use Linux's asm/atomic.h header, because it's
 
- //     an internal kernel header that doesn't always work in userspace:
 
- //     http://bugs.mysql.com/bug.php?id=28456
 
- //     http://golubenco.org/blog/atomic-operations/
 
- inline void wxAtomicInc (wxUint32 &value)
 
- {
 
-     __sync_fetch_and_add(&value, 1);
 
- }
 
- inline wxUint32 wxAtomicDec (wxUint32 &value)
 
- {
 
-     return __sync_sub_and_fetch(&value, 1);
 
- }
 
- #elif defined(__WINDOWS__)
 
- // include standard Windows headers
 
- #include "wx/msw/wrapwin.h"
 
- inline void wxAtomicInc (wxUint32 &value)
 
- {
 
-     InterlockedIncrement ((LONG*)&value);
 
- }
 
- inline wxUint32 wxAtomicDec (wxUint32 &value)
 
- {
 
-     return InterlockedDecrement ((LONG*)&value);
 
- }
 
- #elif defined(__WXMAC__) || defined(__DARWIN__)
 
- #include "libkern/OSAtomic.h"
 
- inline void wxAtomicInc (wxUint32 &value)
 
- {
 
-     OSAtomicIncrement32 ((int32_t*)&value);
 
- }
 
- inline wxUint32 wxAtomicDec (wxUint32 &value)
 
- {
 
-     return OSAtomicDecrement32 ((int32_t*)&value);
 
- }
 
- #elif defined (__SOLARIS__)
 
- #include <atomic.h>
 
- inline void wxAtomicInc (wxUint32 &value)
 
- {
 
-     atomic_add_32 ((uint32_t*)&value, 1);
 
- }
 
- inline wxUint32 wxAtomicDec (wxUint32 &value)
 
- {
 
-     return atomic_add_32_nv ((uint32_t*)&value, (uint32_t)-1);
 
- }
 
- #else // unknown platform
 
- // it will result in inclusion if the generic implementation code a bit later in this page
 
- #define wxNEEDS_GENERIC_ATOMIC_OPS
 
- #endif // unknown platform
 
- #else // else of wxUSE_THREADS
 
- // if no threads are used we can safely use simple ++/--
 
- inline void wxAtomicInc (wxUint32 &value) { ++value; }
 
- inline wxUint32 wxAtomicDec (wxUint32 &value) { return --value; }
 
- #endif // !wxUSE_THREADS
 
- // ----------------------------------------------------------------------------
 
- // proxies to actual implementations, but for various other types with same
 
- //  behaviour
 
- // ----------------------------------------------------------------------------
 
- #ifdef wxNEEDS_GENERIC_ATOMIC_OPS
 
- #include "wx/thread.h" // for wxCriticalSection
 
- class wxAtomicInt32
 
- {
 
- public:
 
-     wxAtomicInt32() { } // non initialized for consistency with basic int type
 
-     wxAtomicInt32(wxInt32 v) : m_value(v) { }
 
-     wxAtomicInt32(const wxAtomicInt32& a) : m_value(a.m_value) {}
 
-     operator wxInt32() const { return m_value; }
 
-     operator volatile wxInt32&() { return m_value; }
 
-     wxAtomicInt32& operator=(wxInt32 v) { m_value = v; return *this; }
 
-     void Inc()
 
-     {
 
-         wxCriticalSectionLocker lock(m_locker);
 
-         ++m_value;
 
-     }
 
-     wxInt32 Dec()
 
-     {
 
-         wxCriticalSectionLocker lock(m_locker);
 
-         return --m_value;
 
-     }
 
- private:
 
-     volatile wxInt32  m_value;
 
-     wxCriticalSection m_locker;
 
- };
 
- inline void wxAtomicInc(wxAtomicInt32 &value) { value.Inc(); }
 
- inline wxInt32 wxAtomicDec(wxAtomicInt32 &value) { return value.Dec(); }
 
- #else // !wxNEEDS_GENERIC_ATOMIC_OPS
 
- #define wxHAS_ATOMIC_OPS
 
- inline void wxAtomicInc(wxInt32 &value) { wxAtomicInc((wxUint32&)value); }
 
- inline wxInt32 wxAtomicDec(wxInt32 &value) { return wxAtomicDec((wxUint32&)value); }
 
- typedef wxInt32 wxAtomicInt32;
 
- #endif // wxNEEDS_GENERIC_ATOMIC_OPS
 
- // all the native implementations use 32 bits currently
 
- // for a 64 bits implementation we could use (a future) wxAtomicInt64 as
 
- // default type
 
- typedef wxAtomicInt32 wxAtomicInt;
 
- #endif // _WX_ATOMIC_H_
 
 
  |