Win32 API doesn't provide pthreads-like destructors, so we need to emulate them to free per-thread allocated memory. git-svn-id: https://svn.wxwidgets.org/svn/wx/wxWidgets/trunk@63653 c3d73ce0-8a6f-49c7-b76d-6d57e0e08775
		
			
				
	
	
		
			118 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			118 lines
		
	
	
		
			3.3 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| ///////////////////////////////////////////////////////////////////////////////
 | |
| // Name:        wx/unix/tls.h
 | |
| // Purpose:     Win32 implementation of wxTlsValue<>
 | |
| // Author:      Vadim Zeitlin
 | |
| // Created:     2008-08-08
 | |
| // RCS-ID:      $Id$
 | |
| // Copyright:   (c) 2008 Vadim Zeitlin <vadim@wxwidgets.org>
 | |
| // Licence:     wxWindows licence
 | |
| ///////////////////////////////////////////////////////////////////////////////
 | |
| 
 | |
| #ifndef _WX_MSW_TLS_H_
 | |
| #define _WX_MSW_TLS_H_
 | |
| 
 | |
| #include "wx/msw/wrapwin.h"
 | |
| #include "wx/thread.h"
 | |
| #include "wx/vector.h"
 | |
| 
 | |
| // ----------------------------------------------------------------------------
 | |
| // wxTlsKey is a helper class encapsulating a TLS slot
 | |
| // ----------------------------------------------------------------------------
 | |
| 
 | |
| class wxTlsKey
 | |
| {
 | |
| public:
 | |
|     // ctor allocates a new key
 | |
|     wxTlsKey(wxTlsDestructorFunction destructor)
 | |
|     {
 | |
|         m_destructor = destructor;
 | |
|         m_slot = ::TlsAlloc();
 | |
|     }
 | |
| 
 | |
|     // return true if the key was successfully allocated
 | |
|     bool IsOk() const { return m_slot != TLS_OUT_OF_INDEXES; }
 | |
| 
 | |
|     // get the key value, there is no error return
 | |
|     void *Get() const
 | |
|     {
 | |
|         return ::TlsGetValue(m_slot);
 | |
|     }
 | |
| 
 | |
|     // change the key value, return true if ok
 | |
|     bool Set(void *value)
 | |
|     {
 | |
|         void *old = Get();
 | |
| 
 | |
|         if ( ::TlsSetValue(m_slot, value) == 0 )
 | |
|             return false;
 | |
| 
 | |
|         if ( old )
 | |
|             m_destructor(old);
 | |
| 
 | |
|         // update m_allValues list of all values - remove old, add new
 | |
|         wxCriticalSectionLocker lock(m_csAllValues);
 | |
|         if ( old )
 | |
|         {
 | |
|             for ( wxVector<void*>::iterator i = m_allValues.begin();
 | |
|                   i != m_allValues.end();
 | |
|                   ++i )
 | |
|             {
 | |
|                 if ( *i == old )
 | |
|                 {
 | |
|                     if ( value )
 | |
|                         *i = value;
 | |
|                     else
 | |
|                         m_allValues.erase(i);
 | |
|                     return true;
 | |
|                 }
 | |
|             }
 | |
|             wxFAIL_MSG( "previous wxTlsKey value not recorded in m_allValues" );
 | |
|         }
 | |
| 
 | |
|         if ( value )
 | |
|             m_allValues.push_back(value);
 | |
| 
 | |
|         return true;
 | |
|     }
 | |
| 
 | |
|     // free the key
 | |
|     ~wxTlsKey()
 | |
|     {
 | |
|         if ( !IsOk() )
 | |
|             return;
 | |
| 
 | |
|         // Win32 API doesn't have the equivalent of pthread's destructor, so we
 | |
|         // have to keep track of all allocated values and destroy them manually;
 | |
|         // ideally we'd do that at thread exit time, but since we could only
 | |
|         // do that with wxThread and not otherwise created threads, we do it
 | |
|         // here.
 | |
|         //
 | |
|         // TODO: We should still call destructors for wxTlsKey used in the
 | |
|         //       thread from wxThread's thread shutdown code, *in addition*
 | |
|         //       to doing it in ~wxTlsKey.
 | |
|         //
 | |
|         // NB: No need to lock m_csAllValues, by the time this code is called,
 | |
|         //     no other thread can be using this key.
 | |
|         for ( wxVector<void*>::iterator i = m_allValues.begin();
 | |
|               i != m_allValues.end();
 | |
|               ++i )
 | |
|         {
 | |
|             m_destructor(*i);
 | |
|         }
 | |
| 
 | |
|         ::TlsFree(m_slot);
 | |
|     }
 | |
| 
 | |
| private:
 | |
|     wxTlsDestructorFunction m_destructor;
 | |
|     DWORD m_slot;
 | |
| 
 | |
|     wxVector<void*> m_allValues;
 | |
|     wxCriticalSection m_csAllValues;
 | |
| 
 | |
|     wxDECLARE_NO_COPY_CLASS(wxTlsKey);
 | |
| };
 | |
| 
 | |
| #endif // _WX_MSW_TLS_H_
 | |
| 
 |