diff options
author | Chris Robinson <[email protected]> | 2014-04-17 00:56:02 -0700 |
---|---|---|
committer | Chris Robinson <[email protected]> | 2014-04-17 00:56:02 -0700 |
commit | e5d39a5f4c871183c7e409abff3c3a6d6c40bf17 (patch) | |
tree | 3b33b86e7ea81de8990b3bec78f7668a6fba7324 /Alc/threads.c | |
parent | c3b1c31d9b530e64e84805435784ec53eb5ea745 (diff) |
Don't use DllMain to call altss destructors
Diffstat (limited to 'Alc/threads.c')
-rw-r--r-- | Alc/threads.c | 40 |
1 files changed, 38 insertions, 2 deletions
diff --git a/Alc/threads.c b/Alc/threads.c index e5df3086..ca004903 100644 --- a/Alc/threads.c +++ b/Alc/threads.c @@ -198,6 +198,42 @@ int almtx_timedlock(almtx_t *mtx, const struct timespec *ts) } +/* An associative map of uint:void* pairs. The key is the TLS index (given by + * TlsAlloc), and the value is the altss_dtor_t callback. When a thread exits, + * it iterates over the thread-local value for each TLS key and calls the + * destructor function if they're both not NULL. Placing a PIMAGE_TLS_CALLBACK + * function pointer in a ".CRT$XLx" section (where x is a character A to Z) + * ensures the CRT will call it similar to DllMain. + */ +UIntMap TlsDestructors; + +static void NTAPI altss_callback(void* UNUSED(handle), DWORD reason, void* UNUSED(reserved)) +{ + ALsizei i; + + if(reason != DLL_THREAD_DETACH) + return; + + LockUIntMapRead(&TlsDestructors); + for(i = 0;i < TlsDestructors.size;i++) + { + void *ptr = altss_get(TlsDestructors.array[i].key); + altss_dtor_t callback = (altss_dtor_t)TlsDestructors.array[i].value; + if(ptr && callback) + callback(ptr); + } + UnlockUIntMapRead(&TlsDestructors); +} +#ifdef _MSC_VER +#pragma section(".CRT$XLB",read) +__declspec(allocate(".CRT$XLB")) PIMAGE_TLS_CALLBACK altss_callback_ = altss_callback; +#elif defined(__GNUC__) +PIMAGE_TLS_CALLBACK altss_callback_ __attribute__((section(".CRT$XLB"))) = altss_callback; +#else +#warning "No TLS callback support, thread-local contexts may leak references on poorly written applications." +PIMAGE_TLS_CALLBACK altss_callback_ = altss_callback; +#endif + int altss_create(altss_t *tss_id, altss_dtor_t callback) { DWORD key = TlsAlloc(); @@ -206,13 +242,13 @@ int altss_create(altss_t *tss_id, altss_dtor_t callback) *tss_id = key; if(callback != NULL) - InsertUIntMapEntry(&TlsDestructor, key, callback); + InsertUIntMapEntry(&TlsDestructors, key, callback); return althrd_success; } void altss_delete(altss_t tss_id) { - InsertUIntMapEntry(&TlsDestructor, tss_id, NULL); + RemoveUIntMapKey(&TlsDestructors, tss_id); TlsFree(tss_id); } |