X-Git-Url: https://git.saurik.com/wxWidgets.git/blobdiff_plain/64a044d5a64dd92473b4cc666a6877db78bd37d3..c4021a7920ab9f41b1553e28f1177b6e39a6d901:/include/wx/tls.h diff --git a/include/wx/tls.h b/include/wx/tls.h index 9d1d6373b3..7846af1e3b 100644 --- a/include/wx/tls.h +++ b/include/wx/tls.h @@ -17,9 +17,19 @@ // check for compiler support of thread-specific variables // ---------------------------------------------------------------------------- -#ifdef HAVE___THREAD_KEYWORD +// when not using threads at all, there is no need for thread-specific +// values to be really thread-specific +#if !wxUSE_THREADS + #define wxHAS_COMPILER_TLS + #define wxTHREAD_SPECIFIC_DECL +// __thread keyword is supported if configure detected it or when using mingw32 +// >= 4.3 which is known to have it too +#elif defined(HAVE___THREAD_KEYWORD) || \ + (defined(__MINGW32__) && wxCHECK_GCC_VERSION(4, 3)) #define wxHAS_COMPILER_TLS #define wxTHREAD_SPECIFIC_DECL __thread +// MSVC has its own version which might be supported by some other Windows +// compilers, to be tested #elif wxCHECK_VISUALC_VERSION(7) #define wxHAS_COMPILER_TLS #define wxTHREAD_SPECIFIC_DECL __declspec(thread) @@ -31,9 +41,13 @@ #ifdef wxHAS_COMPILER_TLS #define wxTLS_TYPE(T) wxTHREAD_SPECIFIC_DECL T + #define wxTLS_PTR(var) (&(var)) + #define wxTLS_VALUE(var) (var) #else // !wxHAS_COMPILER_TLS #ifdef __WXMSW__ #include "wx/msw/tls.h" + #elif defined(__OS2__) + #include "wx/os2/tls.h" #elif defined(__UNIX__) #include "wx/unix/tls.h" #else @@ -41,23 +55,76 @@ #error Neither compiler nor OS support thread-specific variables. #endif - // wxTlsValue represents a thread-specific value of type T + #include // for calloc() + + // wxTlsValue represents a thread-specific value of type T but, unlike + // with native compiler thread-specific variables, it behaves like a + // (never NULL) pointer to T and so needs to be dereferenced before use template class wxTlsValue { public: typedef T ValueType; - wxTlsValue() { *this = static_cast(0); } + // ctor doesn't do anything, the object is created on first access + // + // FIXME: the thread-specific values are currently not freed under + // Windows, resulting in memory leaks, this must be implemented + // there somehow (probably by keeping a list of all TLS objects + // and cleaning them up in wxThread cleanup) + wxTlsValue() +#if !defined(__OS2__) && defined(__UNIX__) + : m_key(free) +#endif + { + } - wxTlsValue& operator=(T value) + // dtor is only called in the main thread context and so is not enough + // to free memory allocated by us for the other threads, we use + // destructor function when using Pthreads for this (which is not + // called for the main thread as it doesn't call pthread_exit() but + // just to be safe we also reset the key anyhow) and simply leak the + // memory under Windows (see the FIXME above) + ~wxTlsValue() { - m_key.Set(wxUIntToPtr(value)); + void * const value = m_key.Get(); + if ( value) + { + free(value); + m_key.Set(NULL); + } + } - return *this; + // access the object creating it on demand + ValueType *Get() + { + void *value = m_key.Get(); + if ( !value ) + { + // ValueType must be POD to be used in wxHAS_COMPILER_TLS case + // anyhow (at least gcc doesn't accept non-POD values being + // declared with __thread) so initialize it as a POD too + value = calloc(1, sizeof(ValueType)); + + if ( !m_key.Set(value) ) + { + free(value); + + // this will probably result in a crash in the caller but + // it's arguably better to crash immediately instead of + // slowly dying from out-of-memory errors which would + // happen as the next access to this object would allocate + // another ValueType instance and so on forever + value = NULL; + } + } + + return static_cast(value); } - operator T() const { return wxPtrToUInt(m_key.Get()); } + // pointer-like accessors + ValueType *operator->() { return Get(); } + ValueType& operator*() { return *Get(); } private: wxTlsKey m_key; @@ -66,6 +133,8 @@ }; #define wxTLS_TYPE(T) wxTlsValue + #define wxTLS_PTR(var) (var) + #define wxTLS_VALUE(var) (*(var)) #endif // wxHAS_COMPILER_TLS/!wxHAS_COMPILER_TLS #endif // _WX_TLS_H_