mdbx: more rework TLS cleanup on thread termination.

One more re-fix https://github.com/ReOpen/ReOpenLDAP/issues/48
Here is only part of the work for glibc >= 2.18

--

Unfortunately, the two bugs related to pthread_key_delete()
are present in all glibc versions:

1) The race condition between pthread_key_delete() and thread's
finalization path, from where a TSD-destructors are called.
Therefore some TSD-destructor(s) could be executed after that
the pthread_key_delete() was competed.

2) ld.so infrastructure does not tracks a TDS-destructors.
Therefore a lib.so could be unloaded while corresponding a
TSD-destructor(s) were even not completed or still were not called.

Change-Id: I47eba97df57fd4c7a5bf3a8f9f12d72ba898cae5
This commit is contained in:
Leo Yuriev 2017-01-06 02:09:08 +03:00
parent f42b6a982a
commit 02de457f3c

203
mdb.c
View File

@ -1002,12 +1002,13 @@ typedef struct MDB_pgstate {
/** Context for deferred cleanup of reader's threads. /** Context for deferred cleanup of reader's threads.
* to avoid https://github.com/ReOpen/ReOpenLDAP/issues/48 */ * to avoid https://github.com/ReOpen/ReOpenLDAP/issues/48 */
typedef struct MDBX_rthc { typedef struct MDBX_rthc {
struct MDBX_rthc *rc_next;
pthread_t rc_thread;
MDB_reader *rc_reader; MDB_reader *rc_reader;
} MDBX_rthc; } MDBX_rthc;
static MDBX_rthc* mdbx_rthc_alloc(void); static MDBX_rthc* mdbx_rthc_get(pthread_key_t key);
static void mdbx_rthc_cleanup(MDBX_rthc*); static void mdbx_rthc_cleanup(MDB_env *env);
static int mdbx_rthc_assigned(MDBX_rthc* rthc);
/** The database environment. */ /** The database environment. */
struct MDB_env { struct MDB_env {
@ -2762,18 +2763,10 @@ mdb_txn_renew0(MDB_txn *txn, unsigned flags)
txn->mt_flags = MDB_TXN_RDONLY; txn->mt_flags = MDB_TXN_RDONLY;
if (likely(env->me_flags & MDB_ENV_TXKEY)) { if (likely(env->me_flags & MDB_ENV_TXKEY)) {
mdb_assert(env, !(env->me_flags & MDB_NOTLS)); mdb_assert(env, !(env->me_flags & MDB_NOTLS));
rthc = pthread_getspecific(env->me_txkey); rthc = mdbx_rthc_get(env->me_txkey);
if (unlikely(! rthc)) {
rthc = mdbx_rthc_alloc();
if (unlikely(! rthc)) if (unlikely(! rthc))
return MDB_READERS_FULL; return ENOMEM;
rc = pthread_setspecific(env->me_txkey, rthc); if (likely(rthc->rc_reader)) {
if (unlikely(rc)) {
free(rthc);
return rc;
}
}
if (mdbx_rthc_assigned(rthc)) {
r = rthc->rc_reader; r = rthc->rc_reader;
mdb_assert(env, r->mr_pid == env->me_pid); mdb_assert(env, r->mr_pid == env->me_pid);
mdb_assert(env, r->mr_tid == pthread_self()); mdb_assert(env, r->mr_tid == pthread_self());
@ -4507,15 +4500,48 @@ mdb_env_open2(MDB_env *env, MDB_meta *meta)
/****************************************************************************/ /****************************************************************************/
static pthread_mutex_t mdbx_rthc_mutex = PTHREAD_MUTEX_INITIALIZER;
static MDBX_rthc *mdbx_rthc_list;
static pthread_key_t mdbx_pthread_crutch_key;
static pthread_cond_t mdbx_rthc_cond = PTHREAD_COND_INITIALIZER;
extern void *__dso_handle __attribute__ ((__weak__));
extern int __cxa_thread_atexit_impl(void (*dtor)(void*), void *obj, void *dso_symbol);
static __inline
void mdbx_rthc_lock(void) {
mdb_ensure(NULL, pthread_mutex_lock(&mdbx_rthc_mutex) == 0);
}
static __inline
void mdbx_rthc_unlock(void) {
mdb_ensure(NULL, pthread_mutex_unlock(&mdbx_rthc_mutex) == 0);
}
static __attribute__((constructor)) __cold
void mdbx_pthread_crutch_ctor(void) {
mdbx_pthread_crutch_key = -1;
mdb_ensure(NULL, pthread_key_create(&mdbx_pthread_crutch_key, NULL) == 0);
}
static __attribute__((destructor)) __cold
void mdbx_pthread_crutch_dtor(void)
{
mdbx_rthc_lock();
while (mdbx_rthc_list != NULL)
mdb_ensure(NULL, pthread_cond_wait(&mdbx_rthc_cond, &mdbx_rthc_mutex) == 0);
pthread_key_delete(mdbx_pthread_crutch_key);
mdbx_pthread_crutch_key = -1;
}
/** Release a reader thread's slot in the reader lock table. /** Release a reader thread's slot in the reader lock table.
* This function is called automatically when a thread exits. * This function is called automatically when a thread exits.
* @param[in] ptr This points to the MDB_rthc of a slot in the reader lock table. * @param[in] ptr This points to the MDB_rthc of a slot in the reader lock table.
*/ */
static __cold static __cold
void mdb_env_reader_destr(void *ptr) void mdbx_rthc_dtor(void *ptr)
{ {
MDBX_rthc* rthc = ptr;
/* LY: Основная задача этого деструктора была и есть в освобождении /* LY: Основная задача этого деструктора была и есть в освобождении
* слота таблицы читателей при завершении треда, но тут есть пара * слота таблицы читателей при завершении треда, но тут есть пара
* не очевидных сложностей: * не очевидных сложностей:
@ -4528,81 +4554,114 @@ void mdb_env_reader_destr(void *ptr)
* и завершающимися тредами. Грубо говоря, при старте mdb_env_close() * и завершающимися тредами. Грубо говоря, при старте mdb_env_close()
* деструктор уже может выполняться в некоторых тредах, и завершиться * деструктор уже может выполняться в некоторых тредах, и завершиться
* эти выполнения могут во время или после окончания mdb_env_close(). * эти выполнения могут во время или после окончания mdb_env_close().
* - БОЛЕЕ ТОГО, схожая проблема может возникнуть при выгрузке dso/dll, * - БОЛЕЕ ТОГО, схожая проблема возникает при выгрузке dso/dll,
* в случае если пользователь не вызвал mdb_env_close(), на что он * так как в текущей glibc (2.24) подсистема ld.so ничего не знает о
* имеет право. * TSD-деструкторах и поэтому может выгрузить lib.so до того как
* отработали все деструкторы.
* - Исходное проявление проблемы было зафиксировано * - Исходное проявление проблемы было зафиксировано
* в https://github.com/ReOpen/ReOpenLDAP/issues/48 * в https://github.com/ReOpen/ReOpenLDAP/issues/48
* *
* Решение посредством выделяемого динамически struct MDB_rthc было * Решение посредством выделяемого динамически struct MDB_rthc было
* не удачным, так как порождало либо утечку памяти, либо вероятностное * не удачным, так как порождало либо утечку памяти, либо вероятностное
* обращение к уже освобожденной памяти из этого деструктора. * обращение к уже освобожденной памяти из этого деструктора.
*
* Текущее решение достаточно "развесисто" и решает все описанные выше
* проблемы ценой переносимости, но без пенальти по производительности.
*/ */
mdbx_rthc_cleanup(rthc); MDBX_rthc* head = ptr;
} mdbx_rthc_lock();
mdb_ensure(NULL, head == pthread_getspecific(mdbx_pthread_crutch_key));
mdb_ensure(NULL, pthread_setspecific(mdbx_pthread_crutch_key, NULL) == 0);
static pthread_mutex_t mdbx_rthc_lock = PTHREAD_MUTEX_INITIALIZER; pid_t pid = getpid();
pthread_t thread = pthread_self();
#define MDBX_RTHC_MAX 512 for (MDBX_rthc** ref = &mdbx_rthc_list; *ref; ) {
static MDBX_rthc mdbx_rthc_tlb[MDBX_RTHC_MAX]; MDBX_rthc* rthc = *ref;
if (rthc->rc_thread == thread) {
static MDBX_INLINE if (rthc->rc_reader && rthc->rc_reader->mr_pid == pid) {
int mdbx_rthc_assigned(MDBX_rthc* rthc)
{
return rthc->rc_reader && rthc->rc_reader != (void*) rthc;
}
static __cold
MDBX_rthc* mdbx_rthc_alloc(void)
{
mdb_ensure(NULL, pthread_mutex_lock(&mdbx_rthc_lock) == 0);
for (size_t i = 0; i < MDBX_RTHC_MAX; ++i) {
if (mdbx_rthc_tlb[i].rc_reader == NULL) {
MDBX_rthc* rthc = mdbx_rthc_tlb + i;
rthc->rc_reader = (void *) rthc;
mdb_ensure(NULL, pthread_mutex_unlock(&mdbx_rthc_lock) == 0);
return rthc;
}
}
mdb_ensure(NULL, pthread_mutex_unlock(&mdbx_rthc_lock) == 0);
return NULL;
}
static __cold
void mdbx_rthc_cleanup_locked(MDBX_rthc* rthc)
{
if (mdbx_rthc_assigned(rthc)) {
if (rthc->rc_reader->mr_pid == getpid()) {
rthc->rc_reader->mr_pid = 0; rthc->rc_reader->mr_pid = 0;
mdbx_coherent_barrier(); mdbx_coherent_barrier();
} }
*ref = rthc->rc_next;
free(rthc);
} else {
ref = &(*ref)->rc_next;
}
}
if (mdbx_rthc_list == NULL)
pthread_cond_broadcast(&mdbx_rthc_cond);
mdbx_rthc_unlock();
}
static __cold
MDBX_rthc* mdbx_rthc_add(pthread_key_t key)
{
MDBX_rthc *rthc = malloc(sizeof(MDBX_rthc));
if (unlikely(rthc == NULL))
goto bailout;
rthc->rc_next = NULL;
rthc->rc_reader = NULL; rthc->rc_reader = NULL;
rthc->rc_thread = pthread_self();
if (unlikely(pthread_setspecific(key, rthc) != 0))
goto bailout_free;
mdbx_rthc_lock();
if (pthread_getspecific(mdbx_pthread_crutch_key) == NULL) {
void *dso_anchor = (&__dso_handle && __dso_handle)
? __dso_handle : (void *)mdb_version;
if (unlikely(__cxa_thread_atexit_impl(mdbx_rthc_dtor, rthc, dso_anchor) != 0))
goto bailout_unlock;
mdb_ensure(NULL, pthread_setspecific(mdbx_pthread_crutch_key, rthc) == 0);
} }
rthc->rc_next = mdbx_rthc_list;
mdbx_rthc_list = rthc;
mdbx_rthc_unlock();
return rthc;
bailout_unlock:
mdbx_rthc_unlock();
bailout_free:
free(rthc);
bailout:
return NULL;
}
static __inline
MDBX_rthc* mdbx_rthc_get(pthread_key_t key)
{
MDBX_rthc *rthc = pthread_getspecific(key);
if (likely(rthc != NULL))
return rthc;
return mdbx_rthc_add(key);
} }
static __cold static __cold
void mdbx_rthc_cleanup_env(MDB_env *env) void mdbx_rthc_cleanup(MDB_env *env)
{ {
mdb_ensure(env, pthread_mutex_lock(&mdbx_rthc_lock) == 0); mdbx_rthc_lock();
MDB_reader *begin = env->me_txns->mti_readers; MDB_reader *begin = env->me_txns->mti_readers;
MDB_reader* end = env->me_txns->mti_readers + env->me_close_readers; MDB_reader *end = begin + env->me_close_readers;
for (size_t i = 0; i < MDBX_RTHC_MAX; ++i) {
MDBX_rthc* rthc = mdbx_rthc_tlb + i; for (MDBX_rthc** ref = &mdbx_rthc_list; *ref; ) {
if (rthc->rc_reader >= begin && rthc->rc_reader < end) MDBX_rthc* rthc = *ref;
mdbx_rthc_cleanup_locked(rthc); if (rthc->rc_reader >= begin && rthc->rc_reader < end) {
if (rthc->rc_reader->mr_pid == env->me_pid) {
rthc->rc_reader->mr_pid = 0;
mdbx_coherent_barrier();
}
*ref = rthc->rc_next;
free(rthc);
} else {
ref = &(*ref)->rc_next;
} }
mdb_ensure(env, pthread_mutex_unlock(&mdbx_rthc_lock) == 0);
} }
static __cold if (mdbx_rthc_list == NULL)
void mdbx_rthc_cleanup(MDBX_rthc* rthc) pthread_cond_broadcast(&mdbx_rthc_cond);
{ mdbx_rthc_unlock();
if (rthc->rc_reader) {
mdb_ensure(NULL, pthread_mutex_lock(&mdbx_rthc_lock) == 0);
mdbx_rthc_cleanup_locked(rthc);
mdb_ensure(NULL, pthread_mutex_unlock(&mdbx_rthc_lock) == 0);
}
} }
/****************************************************************************/ /****************************************************************************/
@ -4776,7 +4835,7 @@ mdb_env_setup_locks(MDB_env *env, char *lpath, int mode, int *excl)
fcntl(env->me_lfd, F_SETFD, fdflags); fcntl(env->me_lfd, F_SETFD, fdflags);
if (!(env->me_flags & MDB_NOTLS)) { if (!(env->me_flags & MDB_NOTLS)) {
rc = pthread_key_create(&env->me_txkey, mdb_env_reader_destr); rc = pthread_key_create(&env->me_txkey, NULL);
if (rc) if (rc)
return rc; return rc;
env->me_flags |= MDB_ENV_TXKEY; env->me_flags |= MDB_ENV_TXKEY;
@ -5086,7 +5145,7 @@ mdb_env_close0(MDB_env *env)
* our readers), and clear each reader atomically. * our readers), and clear each reader atomically.
*/ */
if (env->me_pid == getpid()) if (env->me_pid == getpid())
mdbx_rthc_cleanup_env(env); mdbx_rthc_cleanup(env);
munmap((void *)env->me_txns, (env->me_maxreaders-1)*sizeof(MDB_reader)+sizeof(MDB_txninfo)); munmap((void *)env->me_txns, (env->me_maxreaders-1)*sizeof(MDB_reader)+sizeof(MDB_txninfo));
env->me_txns = NULL; env->me_txns = NULL;