chanegd wxTlsValue to be pointer-like instead of value-like which doesn't work for UDTs; use __thread keyword with mingw32 >= 4.3 too; use library-based thread-specific variables support in wxString cache now that it is fixed to work there; finally added a unit test for TLS stuff
git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@55361 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
This commit is contained in:
@@ -11,9 +11,6 @@
|
|||||||
#ifndef _WX_MSW_TLS_H_
|
#ifndef _WX_MSW_TLS_H_
|
||||||
#define _WX_MSW_TLS_H_
|
#define _WX_MSW_TLS_H_
|
||||||
|
|
||||||
#include "wx/log.h"
|
|
||||||
#include "wx/intl.h"
|
|
||||||
|
|
||||||
#include "wx/msw/wrapwin.h"
|
#include "wx/msw/wrapwin.h"
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
@@ -27,8 +24,6 @@ public:
|
|||||||
wxTlsKey()
|
wxTlsKey()
|
||||||
{
|
{
|
||||||
m_slot = ::TlsAlloc();
|
m_slot = ::TlsAlloc();
|
||||||
if ( m_slot == TLS_OUT_OF_INDEXES )
|
|
||||||
wxLogError("Creating TLS key failed");
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// return true if the key was successfully allocated
|
// return true if the key was successfully allocated
|
||||||
@@ -43,25 +38,14 @@ public:
|
|||||||
// change the key value, return true if ok
|
// change the key value, return true if ok
|
||||||
bool Set(void *value)
|
bool Set(void *value)
|
||||||
{
|
{
|
||||||
if ( !::TlsSetValue(m_slot, value) )
|
return ::TlsSetValue(m_slot, value) != 0;
|
||||||
{
|
|
||||||
wxLogSysError(_("Failed to set TLS value"));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// free the key
|
// free the key
|
||||||
~wxTlsKey()
|
~wxTlsKey()
|
||||||
{
|
{
|
||||||
if ( IsOk() )
|
if ( IsOk() )
|
||||||
{
|
::TlsFree(m_slot);
|
||||||
if ( !::TlsFree(m_slot) )
|
|
||||||
{
|
|
||||||
wxLogDebug("TlsFree() failed: %08x", ::GetLastError());
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
|||||||
@@ -75,12 +75,6 @@
|
|||||||
#define wxUSE_STRING_POS_CACHE 0
|
#define wxUSE_STRING_POS_CACHE 0
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
#ifndef wxHAS_COMPILER_TLS
|
|
||||||
// FIXME: currently the code only works with compiler TLS support
|
|
||||||
#undef wxUSE_STRING_POS_CACHE
|
|
||||||
#define wxUSE_STRING_POS_CACHE 0
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#if wxUSE_STRING_POS_CACHE
|
#if wxUSE_STRING_POS_CACHE
|
||||||
#include "wx/tls.h"
|
#include "wx/tls.h"
|
||||||
|
|
||||||
@@ -600,7 +594,26 @@ private:
|
|||||||
unsigned lastUsed;
|
unsigned lastUsed;
|
||||||
};
|
};
|
||||||
|
|
||||||
static wxTLS_TYPE(Cache) ms_cache;
|
// notice that we must use an accessor function and not a static variable
|
||||||
|
// because when the TLS variables support is implemented in the library (and
|
||||||
|
// not by the compiler), the global s_cache variable could be not yet
|
||||||
|
// initialized when a ctor of another global object is executed and if that
|
||||||
|
// ctor uses any wxString methods, bad things happen
|
||||||
|
//
|
||||||
|
// also note that for the same reason this function _is_ MT-safe: we know
|
||||||
|
// it's going to be called during the program startup (currently during
|
||||||
|
// globals initialization but even if they ever stop using wxString, it would
|
||||||
|
// still be called by wxInitialize()), i.e. before any threads are created
|
||||||
|
static Cache& GetCache()
|
||||||
|
{
|
||||||
|
static wxTLS_TYPE(Cache) s_cache;
|
||||||
|
|
||||||
|
return wxTLS_VALUE(s_cache);
|
||||||
|
}
|
||||||
|
|
||||||
|
static Cache::Element *GetCacheBegin() { return GetCache().cached; }
|
||||||
|
static Cache::Element *GetCacheEnd() { return GetCacheBegin() + Cache::SIZE; }
|
||||||
|
static unsigned& LastUsedCacheElement() { return GetCache().lastUsed; }
|
||||||
|
|
||||||
friend struct wxStrCacheDumper;
|
friend struct wxStrCacheDumper;
|
||||||
|
|
||||||
@@ -644,9 +657,7 @@ private:
|
|||||||
// profiling seems to show a small but consistent gain if we use this
|
// profiling seems to show a small but consistent gain if we use this
|
||||||
// simple loop instead of starting from the last used element (there are
|
// simple loop instead of starting from the last used element (there are
|
||||||
// a lot of misses in this function...)
|
// a lot of misses in this function...)
|
||||||
for ( Cache::Element *c = ms_cache.cached;
|
for ( Cache::Element *c = GetCacheBegin(); c != GetCacheEnd(); c++ )
|
||||||
c != ms_cache.cached + Cache::SIZE;
|
|
||||||
c++ )
|
|
||||||
{
|
{
|
||||||
if ( c->str == this )
|
if ( c->str == this )
|
||||||
return c;
|
return c;
|
||||||
@@ -660,9 +671,9 @@ private:
|
|||||||
// its corresponding index in the byte string or not
|
// its corresponding index in the byte string or not
|
||||||
Cache::Element *GetCacheElement() const
|
Cache::Element *GetCacheElement() const
|
||||||
{
|
{
|
||||||
Cache::Element * const cacheBegin = ms_cache.cached;
|
Cache::Element * const cacheBegin = GetCacheBegin();
|
||||||
Cache::Element * const cacheEnd = ms_cache.cached + Cache::SIZE;
|
Cache::Element * const cacheEnd = GetCacheEnd();
|
||||||
Cache::Element * const cacheStart = cacheBegin + ms_cache.lastUsed;
|
Cache::Element * const cacheStart = cacheBegin + LastUsedCacheElement();
|
||||||
|
|
||||||
// check the last used first, this does no (measurable) harm for a miss
|
// check the last used first, this does no (measurable) harm for a miss
|
||||||
// but does help for simple loops addressing the same string all the time
|
// but does help for simple loops addressing the same string all the time
|
||||||
@@ -685,7 +696,7 @@ private:
|
|||||||
c->Reset();
|
c->Reset();
|
||||||
|
|
||||||
// and remember the last used element
|
// and remember the last used element
|
||||||
ms_cache.lastUsed = c - cacheBegin;
|
LastUsedCacheElement() = c - cacheBegin;
|
||||||
}
|
}
|
||||||
|
|
||||||
return c;
|
return c;
|
||||||
|
|||||||
@@ -17,9 +17,19 @@
|
|||||||
// check for compiler support of thread-specific variables
|
// 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 wxHAS_COMPILER_TLS
|
||||||
#define wxTHREAD_SPECIFIC_DECL __thread
|
#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)
|
#elif wxCHECK_VISUALC_VERSION(7)
|
||||||
#define wxHAS_COMPILER_TLS
|
#define wxHAS_COMPILER_TLS
|
||||||
#define wxTHREAD_SPECIFIC_DECL __declspec(thread)
|
#define wxTHREAD_SPECIFIC_DECL __declspec(thread)
|
||||||
@@ -31,6 +41,8 @@
|
|||||||
|
|
||||||
#ifdef wxHAS_COMPILER_TLS
|
#ifdef wxHAS_COMPILER_TLS
|
||||||
#define wxTLS_TYPE(T) wxTHREAD_SPECIFIC_DECL T
|
#define wxTLS_TYPE(T) wxTHREAD_SPECIFIC_DECL T
|
||||||
|
#define wxTLS_PTR(var) (&(var))
|
||||||
|
#define wxTLS_VALUE(var) (var)
|
||||||
#else // !wxHAS_COMPILER_TLS
|
#else // !wxHAS_COMPILER_TLS
|
||||||
#ifdef __WXMSW__
|
#ifdef __WXMSW__
|
||||||
#include "wx/msw/tls.h"
|
#include "wx/msw/tls.h"
|
||||||
@@ -41,23 +53,76 @@
|
|||||||
#error Neither compiler nor OS support thread-specific variables.
|
#error Neither compiler nor OS support thread-specific variables.
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// wxTlsValue<T> represents a thread-specific value of type T
|
#include <stdlib.h> // for calloc()
|
||||||
|
|
||||||
|
// wxTlsValue<T> 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 <typename T>
|
template <typename T>
|
||||||
class wxTlsValue
|
class wxTlsValue
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
typedef T ValueType;
|
typedef T ValueType;
|
||||||
|
|
||||||
wxTlsValue() { *this = static_cast<T>(0); }
|
// ctor doesn't do anything, the object is created on first access
|
||||||
|
//
|
||||||
wxTlsValue& operator=(T value)
|
// 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()
|
||||||
|
#ifdef __UNIX__
|
||||||
|
: m_key(free)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
m_key.Set(wxUIntToPtr(value));
|
|
||||||
|
|
||||||
return *this;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
operator T() const { return wxPtrToUInt(m_key.Get()); }
|
// 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()
|
||||||
|
{
|
||||||
|
void * const value = m_key.Get();
|
||||||
|
if ( value)
|
||||||
|
{
|
||||||
|
free(value);
|
||||||
|
m_key.Set(NULL);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 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<ValueType *>(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
// pointer-like accessors
|
||||||
|
ValueType *operator->() { return Get(); }
|
||||||
|
ValueType& operator*() { return *Get(); }
|
||||||
|
|
||||||
private:
|
private:
|
||||||
wxTlsKey m_key;
|
wxTlsKey m_key;
|
||||||
@@ -66,6 +131,8 @@
|
|||||||
};
|
};
|
||||||
|
|
||||||
#define wxTLS_TYPE(T) wxTlsValue<T>
|
#define wxTLS_TYPE(T) wxTlsValue<T>
|
||||||
|
#define wxTLS_PTR(var) (var)
|
||||||
|
#define wxTLS_VALUE(var) (*(var))
|
||||||
#endif // wxHAS_COMPILER_TLS/!wxHAS_COMPILER_TLS
|
#endif // wxHAS_COMPILER_TLS/!wxHAS_COMPILER_TLS
|
||||||
|
|
||||||
#endif // _WX_TLS_H_
|
#endif // _WX_TLS_H_
|
||||||
|
|||||||
@@ -11,9 +11,6 @@
|
|||||||
#ifndef _WX_UNIX_TLS_H_
|
#ifndef _WX_UNIX_TLS_H_
|
||||||
#define _WX_UNIX_TLS_H_
|
#define _WX_UNIX_TLS_H_
|
||||||
|
|
||||||
#include "wx/intl.h"
|
|
||||||
#include "wx/log.h"
|
|
||||||
|
|
||||||
#include <pthread.h>
|
#include <pthread.h>
|
||||||
|
|
||||||
// ----------------------------------------------------------------------------
|
// ----------------------------------------------------------------------------
|
||||||
@@ -23,12 +20,13 @@
|
|||||||
class wxTlsKey
|
class wxTlsKey
|
||||||
{
|
{
|
||||||
public:
|
public:
|
||||||
// ctor allocates a new key
|
// ctor allocates a new key and possibly registering a destructor function
|
||||||
wxTlsKey()
|
// for it (notice that using destructor function is Pthreads-specific and
|
||||||
|
// not supported in Win32 implementation)
|
||||||
|
wxTlsKey(void (*destructor)(void *) = NULL)
|
||||||
{
|
{
|
||||||
int rc = pthread_key_create(&m_key, NULL);
|
if ( pthread_key_create(&m_key, destructor) != 0 )
|
||||||
if ( rc )
|
m_key = 0;
|
||||||
wxLogSysError(_("Creating TLS key failed"), rc);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// return true if the key was successfully allocated
|
// return true if the key was successfully allocated
|
||||||
@@ -43,14 +41,7 @@ public:
|
|||||||
// change the key value, return true if ok
|
// change the key value, return true if ok
|
||||||
bool Set(void *value)
|
bool Set(void *value)
|
||||||
{
|
{
|
||||||
int rc = pthread_setspecific(m_key, value);
|
return pthread_setspecific(m_key, value) == 0;
|
||||||
if ( rc )
|
|
||||||
{
|
|
||||||
wxLogSysError(_("Failed to set TLS value"));
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// free the key
|
// free the key
|
||||||
|
|||||||
@@ -10,18 +10,59 @@
|
|||||||
Macro to be used for thread-specific variables declarations.
|
Macro to be used for thread-specific variables declarations.
|
||||||
|
|
||||||
This macro can be used to define thread-specific variables of the specified
|
This macro can be used to define thread-specific variables of the specified
|
||||||
@a type. Such variables must be global or static. Example of use:
|
@a type. Such variables must be global or static and must be POD, i.e.
|
||||||
|
not have any constructors or destructor (even implicitly generated by the
|
||||||
|
compiler due to use of base classes or members which are not POD in a
|
||||||
|
struct).
|
||||||
|
|
||||||
|
Example of use:
|
||||||
@code
|
@code
|
||||||
struct PerThreadData
|
struct PerThreadData
|
||||||
{
|
{
|
||||||
... data which will be different for every thread ...
|
... data which will be different for every thread ...
|
||||||
};
|
};
|
||||||
|
|
||||||
static wxTLS_TYPE(PerThreadData *) s_threadPtr;
|
static wxTLS_TYPE(PerThreadData) s_threadDataVar;
|
||||||
|
#define s_threadData (wxTLS_VALUE(s_threadDataVar))
|
||||||
|
|
||||||
|
... use s_threadData as a variable of type PerThreadData ...
|
||||||
@endcode
|
@endcode
|
||||||
|
|
||||||
Currently only types of size less than size of a pointer are supported.
|
Notice that the use of the ugly wxTLS_VALUE() macro is unfortunately
|
||||||
This limitation will be lifted in the future.
|
required if you need to support platforms without native compiler support
|
||||||
|
for thread-specific variables. If you compile your code only on platforms
|
||||||
|
which do have such support (recent versions of GNU C++ compiler, Microsoft
|
||||||
|
Visual C++ and Sun C++ compiler are known to have it), you can avoid it and
|
||||||
|
use the variable directly.
|
||||||
*/
|
*/
|
||||||
#define wxTLS_TYPE(type)
|
#define wxTLS_TYPE(type) compiler-dependent-implementation
|
||||||
|
|
||||||
|
/**
|
||||||
|
Macro to access thread-specific variables.
|
||||||
|
|
||||||
|
This macro is used to hide the difference in implementation of
|
||||||
|
thread-specific variables under different platforms: they can be of type T
|
||||||
|
used in wxTLS_TYPE() if they are directly supported by the compiler or of
|
||||||
|
type emulating @c T @c *, i.e. a pointer to this type otherwise. This macro
|
||||||
|
always returns an expression of type @c T itself.
|
||||||
|
|
||||||
|
As shown in wxTLS_TYPE() example, you may want to @c #define a symbol
|
||||||
|
wrapping a thread-specific variable with this macro. And, as also explained
|
||||||
|
in wxTLS_TYPE() documentation, you may avoid using it entirely if you
|
||||||
|
target only recent compilers.
|
||||||
|
|
||||||
|
@see wxTLS_PTR()
|
||||||
|
*/
|
||||||
|
#define wxTLS_VALUE(var)
|
||||||
|
|
||||||
|
/**
|
||||||
|
Macro to return address of a thread-specific variables.
|
||||||
|
|
||||||
|
This macro is similar to wxTLS_VALUE() except that it always returns a
|
||||||
|
pointer to the type of thread-specific variable.
|
||||||
|
|
||||||
|
Notice that this is not a constant expression even if the macro is defined
|
||||||
|
simply as @c &var -- the value returned is still different for every
|
||||||
|
thread.
|
||||||
|
*/
|
||||||
|
#define wxTLS_PTR(var)
|
||||||
|
|||||||
@@ -59,7 +59,6 @@
|
|||||||
const size_t wxString::npos = (size_t) -1;
|
const size_t wxString::npos = (size_t) -1;
|
||||||
|
|
||||||
#if wxUSE_STRING_POS_CACHE
|
#if wxUSE_STRING_POS_CACHE
|
||||||
wxTLS_TYPE(wxString::Cache) wxString::ms_cache;
|
|
||||||
|
|
||||||
// gdb seems to be unable to display thread-local variables correctly, at least
|
// gdb seems to be unable to display thread-local variables correctly, at least
|
||||||
// not my 6.4.98 version under amd64, so provide this debugging helper to do it
|
// not my 6.4.98 version under amd64, so provide this debugging helper to do it
|
||||||
@@ -73,11 +72,11 @@ struct wxStrCacheDumper
|
|||||||
for ( unsigned n = 0; n < wxString::Cache::SIZE; n++ )
|
for ( unsigned n = 0; n < wxString::Cache::SIZE; n++ )
|
||||||
{
|
{
|
||||||
const wxString::Cache::Element&
|
const wxString::Cache::Element&
|
||||||
c = wxString::ms_cache.cached[n];
|
c = wxString::GetCacheBegin()[n];
|
||||||
|
|
||||||
printf("\t%u%s\t%p: pos=(%lu, %lu), len=%ld\n",
|
printf("\t%u%s\t%p: pos=(%lu, %lu), len=%ld\n",
|
||||||
n,
|
n,
|
||||||
n == wxString::ms_cache.lastUsed ? " [*]" : "",
|
n == wxString::LastUsedCacheElement() ? " [*]" : "",
|
||||||
c.str,
|
c.str,
|
||||||
(unsigned long)c.pos,
|
(unsigned long)c.pos,
|
||||||
(unsigned long)c.impl,
|
(unsigned long)c.impl,
|
||||||
|
|||||||
@@ -13,7 +13,6 @@ datarootdir = @datarootdir@
|
|||||||
INSTALL = @INSTALL@
|
INSTALL = @INSTALL@
|
||||||
EXEEXT = @EXEEXT@
|
EXEEXT = @EXEEXT@
|
||||||
WINDRES = @WINDRES@
|
WINDRES = @WINDRES@
|
||||||
REZ = @REZ@
|
|
||||||
SETFILE = @SETFILE@
|
SETFILE = @SETFILE@
|
||||||
BK_DEPS = @BK_DEPS@
|
BK_DEPS = @BK_DEPS@
|
||||||
BK_MAKE_PCH = @BK_MAKE_PCH@
|
BK_MAKE_PCH = @BK_MAKE_PCH@
|
||||||
@@ -101,6 +100,7 @@ TEST_OBJECTS = \
|
|||||||
test_textfiletest.o \
|
test_textfiletest.o \
|
||||||
test_atomic.o \
|
test_atomic.o \
|
||||||
test_queue.o \
|
test_queue.o \
|
||||||
|
test_tls.o \
|
||||||
test_uris.o \
|
test_uris.o \
|
||||||
test_vectors.o \
|
test_vectors.o \
|
||||||
test_evtconnection.o \
|
test_evtconnection.o \
|
||||||
@@ -165,20 +165,12 @@ COND_MONOLITHIC_0___WXLIB_XML_p = \
|
|||||||
@COND_USE_GUI_1@__test_gui___depname = test_gui$(EXEEXT)
|
@COND_USE_GUI_1@__test_gui___depname = test_gui$(EXEEXT)
|
||||||
@COND_PLATFORM_MAC_0@__test_gui___mac_setfilecmd = @true
|
@COND_PLATFORM_MAC_0@__test_gui___mac_setfilecmd = @true
|
||||||
@COND_PLATFORM_MAC_1@__test_gui___mac_setfilecmd = \
|
@COND_PLATFORM_MAC_1@__test_gui___mac_setfilecmd = \
|
||||||
@COND_PLATFORM_MAC_1@ $(SETFILE) -a C test_gui$(EXEEXT)
|
@COND_PLATFORM_MAC_1@ $(SETFILE) -t APPL test_gui$(EXEEXT)
|
||||||
@COND_PLATFORM_MAC_1@__test_gui___mac_rezcmd = $(__MACOSX_RESOURCES_p_1)
|
|
||||||
@COND_WXUNIV_1@__WXUNIV_DEFINE_p_4 = -d __WXUNIVERSAL__
|
|
||||||
@COND_WXUNIV_1@__WXUNIV_DEFINE_p_5 = --define __WXUNIVERSAL__
|
@COND_WXUNIV_1@__WXUNIV_DEFINE_p_5 = --define __WXUNIVERSAL__
|
||||||
@COND_USE_EXCEPTIONS_0@__EXCEPTIONS_DEFINE_p_4 = -d wxNO_EXCEPTIONS
|
|
||||||
@COND_USE_EXCEPTIONS_0@__EXCEPTIONS_DEFINE_p_5 = --define wxNO_EXCEPTIONS
|
@COND_USE_EXCEPTIONS_0@__EXCEPTIONS_DEFINE_p_5 = --define wxNO_EXCEPTIONS
|
||||||
@COND_USE_RTTI_0@__RTTI_DEFINE_p_4 = -d wxNO_RTTI
|
|
||||||
@COND_USE_RTTI_0@__RTTI_DEFINE_p_5 = --define wxNO_RTTI
|
@COND_USE_RTTI_0@__RTTI_DEFINE_p_5 = --define wxNO_RTTI
|
||||||
@COND_USE_THREADS_0@__THREAD_DEFINE_p_4 = -d wxNO_THREADS
|
|
||||||
@COND_USE_THREADS_0@__THREAD_DEFINE_p_5 = --define wxNO_THREADS
|
@COND_USE_THREADS_0@__THREAD_DEFINE_p_5 = --define wxNO_THREADS
|
||||||
@COND_SHARED_1@__DLLFLAG_p_4 = -d WXUSINGDLL
|
|
||||||
@COND_SHARED_1@__DLLFLAG_p_5 = --define WXUSINGDLL
|
@COND_SHARED_1@__DLLFLAG_p_5 = --define WXUSINGDLL
|
||||||
@COND_TOOLKIT_MSW@__RCDEFDIR_p = -i \
|
|
||||||
@COND_TOOLKIT_MSW@ $(LIBDIRNAME)/wx/include/$(TOOLCHAIN_FULLNAME)
|
|
||||||
@COND_TOOLKIT_MSW@__RCDEFDIR_p_1 = --include-dir \
|
@COND_TOOLKIT_MSW@__RCDEFDIR_p_1 = --include-dir \
|
||||||
@COND_TOOLKIT_MSW@ $(LIBDIRNAME)/wx/include/$(TOOLCHAIN_FULLNAME)
|
@COND_TOOLKIT_MSW@ $(LIBDIRNAME)/wx/include/$(TOOLCHAIN_FULLNAME)
|
||||||
@COND_PLATFORM_WIN32_1@__test_gui___win32rc = test_gui_sample_rc.o
|
@COND_PLATFORM_WIN32_1@__test_gui___win32rc = test_gui_sample_rc.o
|
||||||
@@ -217,12 +209,6 @@ COND_MONOLITHIC_0___WXLIB_CORE_p = \
|
|||||||
@COND_ICC_PCH_1@ .pch/testprec_printfbench/testprec.h.gch
|
@COND_ICC_PCH_1@ .pch/testprec_printfbench/testprec.h.gch
|
||||||
@COND_USE_PCH_1@___pch_testprec_printfbench_testprec_h_gch___depname \
|
@COND_USE_PCH_1@___pch_testprec_printfbench_testprec_h_gch___depname \
|
||||||
@COND_USE_PCH_1@ = .pch/testprec_printfbench/testprec.h.gch
|
@COND_USE_PCH_1@ = .pch/testprec_printfbench/testprec.h.gch
|
||||||
COND_TOOLKIT_MAC___MACOSX_RESOURCES_p_1 = $(REZ) -d __DARWIN__ -t APPL -d \
|
|
||||||
__WX$(TOOLKIT)__ $(__WXUNIV_DEFINE_p_4) $(__EXCEPTIONS_DEFINE_p_4) \
|
|
||||||
$(__RTTI_DEFINE_p_4) $(__THREAD_DEFINE_p_4) -i $(srcdir) $(__DLLFLAG_p_4) -i \
|
|
||||||
$(srcdir)/../samples $(__RCDEFDIR_p) -i $(top_srcdir)/include -o \
|
|
||||||
test_gui$(EXEEXT) Carbon.r sample.r
|
|
||||||
@COND_TOOLKIT_MAC@__MACOSX_RESOURCES_p_1 = $(COND_TOOLKIT_MAC___MACOSX_RESOURCES_p_1)
|
|
||||||
@COND_WXUNIV_1@__WXUNIV_DEFINE_p = -D__WXUNIVERSAL__
|
@COND_WXUNIV_1@__WXUNIV_DEFINE_p = -D__WXUNIVERSAL__
|
||||||
@COND_USE_EXCEPTIONS_0@__EXCEPTIONS_DEFINE_p = -DwxNO_EXCEPTIONS
|
@COND_USE_EXCEPTIONS_0@__EXCEPTIONS_DEFINE_p = -DwxNO_EXCEPTIONS
|
||||||
@COND_USE_RTTI_0@__RTTI_DEFINE_p = -DwxNO_RTTI
|
@COND_USE_RTTI_0@__RTTI_DEFINE_p = -DwxNO_RTTI
|
||||||
@@ -274,7 +260,7 @@ test$(EXEEXT): $(TEST_OBJECTS)
|
|||||||
|
|
||||||
@COND_USE_GUI_1@test_gui$(EXEEXT): $(TEST_GUI_OBJECTS) $(__test_gui___win32rc)
|
@COND_USE_GUI_1@test_gui$(EXEEXT): $(TEST_GUI_OBJECTS) $(__test_gui___win32rc)
|
||||||
@COND_USE_GUI_1@ $(CXX) -o $@ $(TEST_GUI_OBJECTS) $(LDFLAGS) -L$(LIBDIRNAME) $(SAMPLES_RPATH_FLAG) $(CPPUNIT_LIBS) $(LIBS) $(__WXLIB_CORE_p) $(__WXLIB_BASE_p) $(__WXLIB_MONO_p) $(__LIB_TIFF_p) $(__LIB_JPEG_p) $(__LIB_PNG_p) $(EXTRALIBS_FOR_GUI) $(__LIB_ZLIB_p) $(__LIB_REGEX_p) $(__LIB_EXPAT_p) $(EXTRALIBS_FOR_BASE)
|
@COND_USE_GUI_1@ $(CXX) -o $@ $(TEST_GUI_OBJECTS) $(LDFLAGS) -L$(LIBDIRNAME) $(SAMPLES_RPATH_FLAG) $(CPPUNIT_LIBS) $(LIBS) $(__WXLIB_CORE_p) $(__WXLIB_BASE_p) $(__WXLIB_MONO_p) $(__LIB_TIFF_p) $(__LIB_JPEG_p) $(__LIB_PNG_p) $(EXTRALIBS_FOR_GUI) $(__LIB_ZLIB_p) $(__LIB_REGEX_p) $(__LIB_EXPAT_p) $(EXTRALIBS_FOR_BASE)
|
||||||
@COND_USE_GUI_1@ $(__test_gui___mac_rezcmd)
|
@COND_USE_GUI_1@
|
||||||
@COND_USE_GUI_1@ $(__test_gui___mac_setfilecmd)
|
@COND_USE_GUI_1@ $(__test_gui___mac_setfilecmd)
|
||||||
@COND_USE_GUI_1@ $(SAMPLES_RPATH_POSTLINK)
|
@COND_USE_GUI_1@ $(SAMPLES_RPATH_POSTLINK)
|
||||||
|
|
||||||
@@ -486,6 +472,9 @@ test_atomic.o: $(srcdir)/thread/atomic.cpp $(TEST_ODEP)
|
|||||||
test_queue.o: $(srcdir)/thread/queue.cpp $(TEST_ODEP)
|
test_queue.o: $(srcdir)/thread/queue.cpp $(TEST_ODEP)
|
||||||
$(CXXC) -c -o $@ $(TEST_CXXFLAGS) $(srcdir)/thread/queue.cpp
|
$(CXXC) -c -o $@ $(TEST_CXXFLAGS) $(srcdir)/thread/queue.cpp
|
||||||
|
|
||||||
|
test_tls.o: $(srcdir)/thread/tls.cpp $(TEST_ODEP)
|
||||||
|
$(CXXC) -c -o $@ $(TEST_CXXFLAGS) $(srcdir)/thread/tls.cpp
|
||||||
|
|
||||||
test_uris.o: $(srcdir)/uris/uris.cpp $(TEST_ODEP)
|
test_uris.o: $(srcdir)/uris/uris.cpp $(TEST_ODEP)
|
||||||
$(CXXC) -c -o $@ $(TEST_CXXFLAGS) $(srcdir)/uris/uris.cpp
|
$(CXXC) -c -o $@ $(TEST_CXXFLAGS) $(srcdir)/uris/uris.cpp
|
||||||
|
|
||||||
|
|||||||
@@ -176,7 +176,8 @@ BENCHMARK_FUNC(BoostTLS)
|
|||||||
|
|
||||||
BENCHMARK_FUNC(wxTLS)
|
BENCHMARK_FUNC(wxTLS)
|
||||||
{
|
{
|
||||||
static wxTLS_TYPE(int) s_global;
|
static wxTLS_TYPE(int) s_globalVar;
|
||||||
|
#define s_global wxTLS_VALUE(s_globalVar)
|
||||||
|
|
||||||
for ( int n = 0; n < NUM_ITER; n++ )
|
for ( int n = 0; n < NUM_ITER; n++ )
|
||||||
{
|
{
|
||||||
|
|||||||
@@ -87,6 +87,7 @@ TEST_OBJECTS = \
|
|||||||
$(OBJS)\test_textfiletest.obj \
|
$(OBJS)\test_textfiletest.obj \
|
||||||
$(OBJS)\test_atomic.obj \
|
$(OBJS)\test_atomic.obj \
|
||||||
$(OBJS)\test_queue.obj \
|
$(OBJS)\test_queue.obj \
|
||||||
|
$(OBJS)\test_tls.obj \
|
||||||
$(OBJS)\test_uris.obj \
|
$(OBJS)\test_uris.obj \
|
||||||
$(OBJS)\test_vectors.obj \
|
$(OBJS)\test_vectors.obj \
|
||||||
$(OBJS)\test_evtconnection.obj \
|
$(OBJS)\test_evtconnection.obj \
|
||||||
@@ -515,6 +516,9 @@ $(OBJS)\test_atomic.obj: .\thread\atomic.cpp
|
|||||||
$(OBJS)\test_queue.obj: .\thread\queue.cpp
|
$(OBJS)\test_queue.obj: .\thread\queue.cpp
|
||||||
$(CXX) -q -c -P -o$@ $(TEST_CXXFLAGS) .\thread\queue.cpp
|
$(CXX) -q -c -P -o$@ $(TEST_CXXFLAGS) .\thread\queue.cpp
|
||||||
|
|
||||||
|
$(OBJS)\test_tls.obj: .\thread\tls.cpp
|
||||||
|
$(CXX) -q -c -P -o$@ $(TEST_CXXFLAGS) .\thread\tls.cpp
|
||||||
|
|
||||||
$(OBJS)\test_uris.obj: .\uris\uris.cpp
|
$(OBJS)\test_uris.obj: .\uris\uris.cpp
|
||||||
$(CXX) -q -c -P -o$@ $(TEST_CXXFLAGS) .\uris\uris.cpp
|
$(CXX) -q -c -P -o$@ $(TEST_CXXFLAGS) .\uris\uris.cpp
|
||||||
|
|
||||||
|
|||||||
@@ -79,6 +79,7 @@ TEST_OBJECTS = \
|
|||||||
$(OBJS)\test_textfiletest.o \
|
$(OBJS)\test_textfiletest.o \
|
||||||
$(OBJS)\test_atomic.o \
|
$(OBJS)\test_atomic.o \
|
||||||
$(OBJS)\test_queue.o \
|
$(OBJS)\test_queue.o \
|
||||||
|
$(OBJS)\test_tls.o \
|
||||||
$(OBJS)\test_uris.o \
|
$(OBJS)\test_uris.o \
|
||||||
$(OBJS)\test_vectors.o \
|
$(OBJS)\test_vectors.o \
|
||||||
$(OBJS)\test_evtconnection.o \
|
$(OBJS)\test_evtconnection.o \
|
||||||
@@ -493,6 +494,9 @@ $(OBJS)\test_atomic.o: ./thread/atomic.cpp
|
|||||||
$(OBJS)\test_queue.o: ./thread/queue.cpp
|
$(OBJS)\test_queue.o: ./thread/queue.cpp
|
||||||
$(CXX) -c -o $@ $(TEST_CXXFLAGS) $(CPPDEPS) $<
|
$(CXX) -c -o $@ $(TEST_CXXFLAGS) $(CPPDEPS) $<
|
||||||
|
|
||||||
|
$(OBJS)\test_tls.o: ./thread/tls.cpp
|
||||||
|
$(CXX) -c -o $@ $(TEST_CXXFLAGS) $(CPPDEPS) $<
|
||||||
|
|
||||||
$(OBJS)\test_uris.o: ./uris/uris.cpp
|
$(OBJS)\test_uris.o: ./uris/uris.cpp
|
||||||
$(CXX) -c -o $@ $(TEST_CXXFLAGS) $(CPPDEPS) $<
|
$(CXX) -c -o $@ $(TEST_CXXFLAGS) $(CPPDEPS) $<
|
||||||
|
|
||||||
|
|||||||
@@ -80,6 +80,7 @@ TEST_OBJECTS = \
|
|||||||
$(OBJS)\test_textfiletest.obj \
|
$(OBJS)\test_textfiletest.obj \
|
||||||
$(OBJS)\test_atomic.obj \
|
$(OBJS)\test_atomic.obj \
|
||||||
$(OBJS)\test_queue.obj \
|
$(OBJS)\test_queue.obj \
|
||||||
|
$(OBJS)\test_tls.obj \
|
||||||
$(OBJS)\test_uris.obj \
|
$(OBJS)\test_uris.obj \
|
||||||
$(OBJS)\test_vectors.obj \
|
$(OBJS)\test_vectors.obj \
|
||||||
$(OBJS)\test_evtconnection.obj \
|
$(OBJS)\test_evtconnection.obj \
|
||||||
@@ -600,6 +601,9 @@ $(OBJS)\test_atomic.obj: .\thread\atomic.cpp
|
|||||||
$(OBJS)\test_queue.obj: .\thread\queue.cpp
|
$(OBJS)\test_queue.obj: .\thread\queue.cpp
|
||||||
$(CXX) /c /nologo /TP /Fo$@ $(TEST_CXXFLAGS) .\thread\queue.cpp
|
$(CXX) /c /nologo /TP /Fo$@ $(TEST_CXXFLAGS) .\thread\queue.cpp
|
||||||
|
|
||||||
|
$(OBJS)\test_tls.obj: .\thread\tls.cpp
|
||||||
|
$(CXX) /c /nologo /TP /Fo$@ $(TEST_CXXFLAGS) .\thread\tls.cpp
|
||||||
|
|
||||||
$(OBJS)\test_uris.obj: .\uris\uris.cpp
|
$(OBJS)\test_uris.obj: .\uris\uris.cpp
|
||||||
$(CXX) /c /nologo /TP /Fo$@ $(TEST_CXXFLAGS) .\uris\uris.cpp
|
$(CXX) /c /nologo /TP /Fo$@ $(TEST_CXXFLAGS) .\uris\uris.cpp
|
||||||
|
|
||||||
|
|||||||
@@ -292,6 +292,7 @@ TEST_OBJECTS = &
|
|||||||
$(OBJS)\test_textfiletest.obj &
|
$(OBJS)\test_textfiletest.obj &
|
||||||
$(OBJS)\test_atomic.obj &
|
$(OBJS)\test_atomic.obj &
|
||||||
$(OBJS)\test_queue.obj &
|
$(OBJS)\test_queue.obj &
|
||||||
|
$(OBJS)\test_tls.obj &
|
||||||
$(OBJS)\test_uris.obj &
|
$(OBJS)\test_uris.obj &
|
||||||
$(OBJS)\test_vectors.obj &
|
$(OBJS)\test_vectors.obj &
|
||||||
$(OBJS)\test_evtconnection.obj &
|
$(OBJS)\test_evtconnection.obj &
|
||||||
@@ -546,6 +547,9 @@ $(OBJS)\test_atomic.obj : .AUTODEPEND .\thread\atomic.cpp
|
|||||||
$(OBJS)\test_queue.obj : .AUTODEPEND .\thread\queue.cpp
|
$(OBJS)\test_queue.obj : .AUTODEPEND .\thread\queue.cpp
|
||||||
$(CXX) -bt=nt -zq -fo=$^@ $(TEST_CXXFLAGS) $<
|
$(CXX) -bt=nt -zq -fo=$^@ $(TEST_CXXFLAGS) $<
|
||||||
|
|
||||||
|
$(OBJS)\test_tls.obj : .AUTODEPEND .\thread\tls.cpp
|
||||||
|
$(CXX) -bt=nt -zq -fo=$^@ $(TEST_CXXFLAGS) $<
|
||||||
|
|
||||||
$(OBJS)\test_uris.obj : .AUTODEPEND .\uris\uris.cpp
|
$(OBJS)\test_uris.obj : .AUTODEPEND .\uris\uris.cpp
|
||||||
$(CXX) -bt=nt -zq -fo=$^@ $(TEST_CXXFLAGS) $<
|
$(CXX) -bt=nt -zq -fo=$^@ $(TEST_CXXFLAGS) $<
|
||||||
|
|
||||||
|
|||||||
@@ -73,6 +73,7 @@
|
|||||||
textfile/textfiletest.cpp
|
textfile/textfiletest.cpp
|
||||||
thread/atomic.cpp
|
thread/atomic.cpp
|
||||||
thread/queue.cpp
|
thread/queue.cpp
|
||||||
|
thread/tls.cpp
|
||||||
uris/uris.cpp
|
uris/uris.cpp
|
||||||
vectors/vectors.cpp
|
vectors/vectors.cpp
|
||||||
weakref/evtconnection.cpp
|
weakref/evtconnection.cpp
|
||||||
|
|||||||
@@ -413,6 +413,10 @@ SOURCE=.\streams\textstreamtest.cpp
|
|||||||
# End Source File
|
# End Source File
|
||||||
# Begin Source File
|
# Begin Source File
|
||||||
|
|
||||||
|
SOURCE=.\thread\tls.cpp
|
||||||
|
# End Source File
|
||||||
|
# Begin Source File
|
||||||
|
|
||||||
SOURCE=.\strings\tokenizer.cpp
|
SOURCE=.\strings\tokenizer.cpp
|
||||||
# End Source File
|
# End Source File
|
||||||
# Begin Source File
|
# Begin Source File
|
||||||
|
|||||||
@@ -786,6 +786,8 @@
|
|||||||
RelativePath=".\textfile\textfiletest.cpp"/>
|
RelativePath=".\textfile\textfiletest.cpp"/>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\streams\textstreamtest.cpp"/>
|
RelativePath=".\streams\textstreamtest.cpp"/>
|
||||||
|
<File
|
||||||
|
RelativePath=".\thread\tls.cpp"/>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\strings\tokenizer.cpp"/>
|
RelativePath=".\strings\tokenizer.cpp"/>
|
||||||
<File
|
<File
|
||||||
|
|||||||
@@ -1004,6 +1004,9 @@
|
|||||||
<File
|
<File
|
||||||
RelativePath=".\streams\textstreamtest.cpp"
|
RelativePath=".\streams\textstreamtest.cpp"
|
||||||
/>
|
/>
|
||||||
|
<File
|
||||||
|
RelativePath=".\thread\tls.cpp"
|
||||||
|
/>
|
||||||
<File
|
<File
|
||||||
RelativePath=".\strings\tokenizer.cpp"
|
RelativePath=".\strings\tokenizer.cpp"
|
||||||
/>
|
/>
|
||||||
|
|||||||
128
tests/thread/tls.cpp
Normal file
128
tests/thread/tls.cpp
Normal file
@@ -0,0 +1,128 @@
|
|||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
// Name: tests/thread/tls.cpp
|
||||||
|
// Purpose: wxTlsValue unit test
|
||||||
|
// Author: Vadim Zeitlin
|
||||||
|
// Created: 2008-08-28
|
||||||
|
// RCS-ID: $Id$
|
||||||
|
// Copyright: (c) 2008 Vadim Zeitlin
|
||||||
|
///////////////////////////////////////////////////////////////////////////////
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// headers
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
#include "testprec.h"
|
||||||
|
|
||||||
|
#ifdef __BORLANDC__
|
||||||
|
#pragma hdrstop
|
||||||
|
#endif
|
||||||
|
|
||||||
|
#ifndef WX_PRECOMP
|
||||||
|
#endif // WX_PRECOMP
|
||||||
|
|
||||||
|
#include "wx/thread.h"
|
||||||
|
#include "wx/tls.h"
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// globals
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
namespace
|
||||||
|
{
|
||||||
|
|
||||||
|
// NB: this struct must be a POD, so don't use wxString as its member
|
||||||
|
struct PerThreadData
|
||||||
|
{
|
||||||
|
const char *name;
|
||||||
|
int number;
|
||||||
|
};
|
||||||
|
|
||||||
|
wxTLS_TYPE(PerThreadData) gs_threadDataVar;
|
||||||
|
#define gs_threadData wxTLS_VALUE(gs_threadDataVar)
|
||||||
|
|
||||||
|
wxTLS_TYPE(int) gs_threadIntVar;
|
||||||
|
#define gs_threadInt wxTLS_VALUE(gs_threadIntVar)
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// test thread
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
// this thread arbitrarily modifies all global thread-specific variables to
|
||||||
|
// make sure that the changes in it are not visible from the main thread
|
||||||
|
class TLSTestThread : public wxThread
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
// ctor both creates and starts the thread
|
||||||
|
TLSTestThread() : wxThread(wxTHREAD_JOINABLE) { Create(); Run(); }
|
||||||
|
|
||||||
|
virtual void *Entry()
|
||||||
|
{
|
||||||
|
gs_threadInt = 17;
|
||||||
|
|
||||||
|
gs_threadData.name = "worker";
|
||||||
|
gs_threadData.number = 2;
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT_EQUAL( "worker", gs_threadData.name );
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 2, gs_threadData.number );
|
||||||
|
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
} // anonymous namespace
|
||||||
|
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
// test class
|
||||||
|
// ----------------------------------------------------------------------------
|
||||||
|
|
||||||
|
class TLSTestCase : public CppUnit::TestCase
|
||||||
|
{
|
||||||
|
public:
|
||||||
|
TLSTestCase() { }
|
||||||
|
|
||||||
|
private:
|
||||||
|
CPPUNIT_TEST_SUITE( TLSTestCase );
|
||||||
|
CPPUNIT_TEST( TestInt );
|
||||||
|
CPPUNIT_TEST( TestStruct );
|
||||||
|
CPPUNIT_TEST_SUITE_END();
|
||||||
|
|
||||||
|
void TestInt();
|
||||||
|
void TestStruct();
|
||||||
|
|
||||||
|
DECLARE_NO_COPY_CLASS(TLSTestCase)
|
||||||
|
};
|
||||||
|
|
||||||
|
// register in the unnamed registry so that these tests are run by default
|
||||||
|
CPPUNIT_TEST_SUITE_REGISTRATION( TLSTestCase );
|
||||||
|
|
||||||
|
// also include in it's own registry so that these tests can be run alone
|
||||||
|
CPPUNIT_TEST_SUITE_NAMED_REGISTRATION( TLSTestCase, "TLSTestCase" );
|
||||||
|
|
||||||
|
void TLSTestCase::TestInt()
|
||||||
|
{
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 0, gs_threadInt );
|
||||||
|
|
||||||
|
gs_threadInt++;
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 1, gs_threadInt );
|
||||||
|
|
||||||
|
TLSTestThread().Wait();
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 1, gs_threadInt );
|
||||||
|
}
|
||||||
|
|
||||||
|
void TLSTestCase::TestStruct()
|
||||||
|
{
|
||||||
|
CPPUNIT_ASSERT_EQUAL( "", gs_threadData.name );
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 0, gs_threadData.number );
|
||||||
|
|
||||||
|
gs_threadData.name = "main";
|
||||||
|
gs_threadData.number = 1;
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 1, gs_threadData.number );
|
||||||
|
|
||||||
|
TLSTestThread().Wait();
|
||||||
|
|
||||||
|
CPPUNIT_ASSERT_EQUAL( "main", gs_threadData.name );
|
||||||
|
CPPUNIT_ASSERT_EQUAL( 1, gs_threadData.number );
|
||||||
|
}
|
||||||
|
|
||||||
Reference in New Issue
Block a user