Ensure the destruction order of PosixEnv and ThreadLocalPtr
Summary: By default, RocksDB initializes the singletons of ThreadLocalPtr first, then initializes PosixEnv via static initializer. Destructor terminates objects in reverse order, so terminating PosixEnv (calling pthread_mutex_lock), then ThreadLocal (calling pthread_mutex_destroy). However, in certain case, application might initialize PosixEnv first, then ThreadLocalPtr. This will cause core dump at the end of the program (eg. https://github.com/facebook/mysql-5.6/issues/122) This patch fix this issue by ensuring the destruction order by moving the global static singletons to function static singletons. Since function static singletons are initialized when the function is first called, this property allows us invoke to enforce the construction of the static PosixEnv and the singletons of ThreadLocalPtr by calling the function where the ThreadLocalPtr singletons belongs right before we initialize the static PosixEnv. Test Plan: Verified in the MyRocks. Reviewers: yoshinorim, IslamAbdelRahman, rven, kradhakrishnan, anthony, sdong, MarkCallaghan Reviewed By: anthony Subscribers: dhruba, leveldb Differential Revision: https://reviews.facebook.net/D51789
This commit is contained in:
parent
64fa43843b
commit
00d6edf6a0
@ -48,6 +48,7 @@
|
|||||||
#include "util/posix_logger.h"
|
#include "util/posix_logger.h"
|
||||||
#include "util/string_util.h"
|
#include "util/string_util.h"
|
||||||
#include "util/sync_point.h"
|
#include "util/sync_point.h"
|
||||||
|
#include "util/thread_local.h"
|
||||||
#include "util/thread_status_updater.h"
|
#include "util/thread_status_updater.h"
|
||||||
|
|
||||||
#if !defined(TMPFS_MAGIC)
|
#if !defined(TMPFS_MAGIC)
|
||||||
@ -761,6 +762,17 @@ std::string Env::GenerateUniqueId() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
Env* Env::Default() {
|
Env* Env::Default() {
|
||||||
|
// The following function call initializes the singletons of ThreadLocalPtr
|
||||||
|
// right before the static default_env. This guarantees default_env will
|
||||||
|
// always being destructed before the ThreadLocalPtr singletons get
|
||||||
|
// destructed as C++ guarantees that the destructions of static variables
|
||||||
|
// is in the reverse order of their constructions.
|
||||||
|
//
|
||||||
|
// Since static members are destructed in the reverse order
|
||||||
|
// of their construction, having this call here guarantees that
|
||||||
|
// the destructor of static PosixEnv will go first, then the
|
||||||
|
// the singletons of ThreadLocalPtr.
|
||||||
|
ThreadLocalPtr::InitSingletons();
|
||||||
static PosixEnv default_env;
|
static PosixEnv default_env;
|
||||||
return &default_env;
|
return &default_env;
|
||||||
}
|
}
|
||||||
|
@ -14,7 +14,6 @@
|
|||||||
|
|
||||||
namespace rocksdb {
|
namespace rocksdb {
|
||||||
|
|
||||||
port::Mutex ThreadLocalPtr::StaticMeta::mutex_;
|
|
||||||
#if ROCKSDB_SUPPORT_THREAD_LOCAL
|
#if ROCKSDB_SUPPORT_THREAD_LOCAL
|
||||||
__thread ThreadLocalPtr::ThreadData* ThreadLocalPtr::StaticMeta::tls_ = nullptr;
|
__thread ThreadLocalPtr::ThreadData* ThreadLocalPtr::StaticMeta::tls_ = nullptr;
|
||||||
#endif
|
#endif
|
||||||
@ -103,11 +102,33 @@ PIMAGE_TLS_CALLBACK p_thread_callback_on_exit = wintlscleanup::WinOnThreadExit;
|
|||||||
|
|
||||||
#endif // OS_WIN
|
#endif // OS_WIN
|
||||||
|
|
||||||
|
void ThreadLocalPtr::InitSingletons() {
|
||||||
|
ThreadLocalPtr::StaticMeta::InitSingletons();
|
||||||
|
ThreadLocalPtr::Instance();
|
||||||
|
}
|
||||||
|
|
||||||
ThreadLocalPtr::StaticMeta* ThreadLocalPtr::Instance() {
|
ThreadLocalPtr::StaticMeta* ThreadLocalPtr::Instance() {
|
||||||
|
// Here we prefer function static variable instead of global
|
||||||
|
// static variable as function static variable is initialized
|
||||||
|
// when the function is first call. As a result, we can properly
|
||||||
|
// control their construction order by properly preparing their
|
||||||
|
// first function call.
|
||||||
static ThreadLocalPtr::StaticMeta inst;
|
static ThreadLocalPtr::StaticMeta inst;
|
||||||
return &inst;
|
return &inst;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void ThreadLocalPtr::StaticMeta::InitSingletons() { Mutex(); }
|
||||||
|
|
||||||
|
port::Mutex* ThreadLocalPtr::StaticMeta::Mutex() {
|
||||||
|
// Here we prefer function static variable instead of global
|
||||||
|
// static variable as function static variable is initialized
|
||||||
|
// when the function is first call. As a result, we can properly
|
||||||
|
// control their construction order by properly preparing their
|
||||||
|
// first function call.
|
||||||
|
static port::Mutex mutex;
|
||||||
|
return &mutex;
|
||||||
|
}
|
||||||
|
|
||||||
void ThreadLocalPtr::StaticMeta::OnThreadExit(void* ptr) {
|
void ThreadLocalPtr::StaticMeta::OnThreadExit(void* ptr) {
|
||||||
auto* tls = static_cast<ThreadData*>(ptr);
|
auto* tls = static_cast<ThreadData*>(ptr);
|
||||||
assert(tls != nullptr);
|
assert(tls != nullptr);
|
||||||
@ -115,7 +136,7 @@ void ThreadLocalPtr::StaticMeta::OnThreadExit(void* ptr) {
|
|||||||
auto* inst = Instance();
|
auto* inst = Instance();
|
||||||
pthread_setspecific(inst->pthread_key_, nullptr);
|
pthread_setspecific(inst->pthread_key_, nullptr);
|
||||||
|
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
inst->RemoveThreadData(tls);
|
inst->RemoveThreadData(tls);
|
||||||
// Unref stored pointers of current thread from all instances
|
// Unref stored pointers of current thread from all instances
|
||||||
uint32_t id = 0;
|
uint32_t id = 0;
|
||||||
@ -175,7 +196,7 @@ ThreadLocalPtr::StaticMeta::StaticMeta() : next_instance_id_(0) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ThreadLocalPtr::StaticMeta::AddThreadData(ThreadLocalPtr::ThreadData* d) {
|
void ThreadLocalPtr::StaticMeta::AddThreadData(ThreadLocalPtr::ThreadData* d) {
|
||||||
mutex_.AssertHeld();
|
Mutex()->AssertHeld();
|
||||||
d->next = &head_;
|
d->next = &head_;
|
||||||
d->prev = head_.prev;
|
d->prev = head_.prev;
|
||||||
head_.prev->next = d;
|
head_.prev->next = d;
|
||||||
@ -184,7 +205,7 @@ void ThreadLocalPtr::StaticMeta::AddThreadData(ThreadLocalPtr::ThreadData* d) {
|
|||||||
|
|
||||||
void ThreadLocalPtr::StaticMeta::RemoveThreadData(
|
void ThreadLocalPtr::StaticMeta::RemoveThreadData(
|
||||||
ThreadLocalPtr::ThreadData* d) {
|
ThreadLocalPtr::ThreadData* d) {
|
||||||
mutex_.AssertHeld();
|
Mutex()->AssertHeld();
|
||||||
d->next->prev = d->prev;
|
d->next->prev = d->prev;
|
||||||
d->prev->next = d->next;
|
d->prev->next = d->next;
|
||||||
d->next = d->prev = d;
|
d->next = d->prev = d;
|
||||||
@ -204,14 +225,14 @@ ThreadLocalPtr::ThreadData* ThreadLocalPtr::StaticMeta::GetThreadLocal() {
|
|||||||
{
|
{
|
||||||
// Register it in the global chain, needs to be done before thread exit
|
// Register it in the global chain, needs to be done before thread exit
|
||||||
// handler registration
|
// handler registration
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
inst->AddThreadData(tls_);
|
inst->AddThreadData(tls_);
|
||||||
}
|
}
|
||||||
// Even it is not OS_MACOSX, need to register value for pthread_key_ so that
|
// Even it is not OS_MACOSX, need to register value for pthread_key_ so that
|
||||||
// its exit handler will be triggered.
|
// its exit handler will be triggered.
|
||||||
if (pthread_setspecific(inst->pthread_key_, tls_) != 0) {
|
if (pthread_setspecific(inst->pthread_key_, tls_) != 0) {
|
||||||
{
|
{
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
inst->RemoveThreadData(tls_);
|
inst->RemoveThreadData(tls_);
|
||||||
}
|
}
|
||||||
delete tls_;
|
delete tls_;
|
||||||
@ -233,7 +254,7 @@ void ThreadLocalPtr::StaticMeta::Reset(uint32_t id, void* ptr) {
|
|||||||
auto* tls = GetThreadLocal();
|
auto* tls = GetThreadLocal();
|
||||||
if (UNLIKELY(id >= tls->entries.size())) {
|
if (UNLIKELY(id >= tls->entries.size())) {
|
||||||
// Need mutex to protect entries access within ReclaimId
|
// Need mutex to protect entries access within ReclaimId
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
tls->entries.resize(id + 1);
|
tls->entries.resize(id + 1);
|
||||||
}
|
}
|
||||||
tls->entries[id].ptr.store(ptr, std::memory_order_release);
|
tls->entries[id].ptr.store(ptr, std::memory_order_release);
|
||||||
@ -243,7 +264,7 @@ void* ThreadLocalPtr::StaticMeta::Swap(uint32_t id, void* ptr) {
|
|||||||
auto* tls = GetThreadLocal();
|
auto* tls = GetThreadLocal();
|
||||||
if (UNLIKELY(id >= tls->entries.size())) {
|
if (UNLIKELY(id >= tls->entries.size())) {
|
||||||
// Need mutex to protect entries access within ReclaimId
|
// Need mutex to protect entries access within ReclaimId
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
tls->entries.resize(id + 1);
|
tls->entries.resize(id + 1);
|
||||||
}
|
}
|
||||||
return tls->entries[id].ptr.exchange(ptr, std::memory_order_acquire);
|
return tls->entries[id].ptr.exchange(ptr, std::memory_order_acquire);
|
||||||
@ -254,7 +275,7 @@ bool ThreadLocalPtr::StaticMeta::CompareAndSwap(uint32_t id, void* ptr,
|
|||||||
auto* tls = GetThreadLocal();
|
auto* tls = GetThreadLocal();
|
||||||
if (UNLIKELY(id >= tls->entries.size())) {
|
if (UNLIKELY(id >= tls->entries.size())) {
|
||||||
// Need mutex to protect entries access within ReclaimId
|
// Need mutex to protect entries access within ReclaimId
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
tls->entries.resize(id + 1);
|
tls->entries.resize(id + 1);
|
||||||
}
|
}
|
||||||
return tls->entries[id].ptr.compare_exchange_strong(
|
return tls->entries[id].ptr.compare_exchange_strong(
|
||||||
@ -263,7 +284,7 @@ bool ThreadLocalPtr::StaticMeta::CompareAndSwap(uint32_t id, void* ptr,
|
|||||||
|
|
||||||
void ThreadLocalPtr::StaticMeta::Scrape(uint32_t id, autovector<void*>* ptrs,
|
void ThreadLocalPtr::StaticMeta::Scrape(uint32_t id, autovector<void*>* ptrs,
|
||||||
void* const replacement) {
|
void* const replacement) {
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
for (ThreadData* t = head_.next; t != &head_; t = t->next) {
|
for (ThreadData* t = head_.next; t != &head_; t = t->next) {
|
||||||
if (id < t->entries.size()) {
|
if (id < t->entries.size()) {
|
||||||
void* ptr =
|
void* ptr =
|
||||||
@ -276,12 +297,12 @@ void ThreadLocalPtr::StaticMeta::Scrape(uint32_t id, autovector<void*>* ptrs,
|
|||||||
}
|
}
|
||||||
|
|
||||||
void ThreadLocalPtr::StaticMeta::SetHandler(uint32_t id, UnrefHandler handler) {
|
void ThreadLocalPtr::StaticMeta::SetHandler(uint32_t id, UnrefHandler handler) {
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
handler_map_[id] = handler;
|
handler_map_[id] = handler;
|
||||||
}
|
}
|
||||||
|
|
||||||
UnrefHandler ThreadLocalPtr::StaticMeta::GetHandler(uint32_t id) {
|
UnrefHandler ThreadLocalPtr::StaticMeta::GetHandler(uint32_t id) {
|
||||||
mutex_.AssertHeld();
|
Mutex()->AssertHeld();
|
||||||
auto iter = handler_map_.find(id);
|
auto iter = handler_map_.find(id);
|
||||||
if (iter == handler_map_.end()) {
|
if (iter == handler_map_.end()) {
|
||||||
return nullptr;
|
return nullptr;
|
||||||
@ -290,7 +311,7 @@ UnrefHandler ThreadLocalPtr::StaticMeta::GetHandler(uint32_t id) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint32_t ThreadLocalPtr::StaticMeta::GetId() {
|
uint32_t ThreadLocalPtr::StaticMeta::GetId() {
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
if (free_instance_ids_.empty()) {
|
if (free_instance_ids_.empty()) {
|
||||||
return next_instance_id_++;
|
return next_instance_id_++;
|
||||||
}
|
}
|
||||||
@ -301,7 +322,7 @@ uint32_t ThreadLocalPtr::StaticMeta::GetId() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
uint32_t ThreadLocalPtr::StaticMeta::PeekId() const {
|
uint32_t ThreadLocalPtr::StaticMeta::PeekId() const {
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
if (!free_instance_ids_.empty()) {
|
if (!free_instance_ids_.empty()) {
|
||||||
return free_instance_ids_.back();
|
return free_instance_ids_.back();
|
||||||
}
|
}
|
||||||
@ -311,7 +332,7 @@ uint32_t ThreadLocalPtr::StaticMeta::PeekId() const {
|
|||||||
void ThreadLocalPtr::StaticMeta::ReclaimId(uint32_t id) {
|
void ThreadLocalPtr::StaticMeta::ReclaimId(uint32_t id) {
|
||||||
// This id is not used, go through all thread local data and release
|
// This id is not used, go through all thread local data and release
|
||||||
// corresponding value
|
// corresponding value
|
||||||
MutexLock l(&mutex_);
|
MutexLock l(Mutex());
|
||||||
auto unref = GetHandler(id);
|
auto unref = GetHandler(id);
|
||||||
for (ThreadData* t = head_.next; t != &head_; t = t->next) {
|
for (ThreadData* t = head_.next; t != &head_; t = t->next) {
|
||||||
if (id < t->entries.size()) {
|
if (id < t->entries.size()) {
|
||||||
|
@ -63,6 +63,15 @@ class ThreadLocalPtr {
|
|||||||
// data for all existing threads
|
// data for all existing threads
|
||||||
void Scrape(autovector<void*>* ptrs, void* const replacement);
|
void Scrape(autovector<void*>* ptrs, void* const replacement);
|
||||||
|
|
||||||
|
// Initialize the static singletons of the ThreadLocalPtr.
|
||||||
|
//
|
||||||
|
// If this function is not called, then the singletons will be
|
||||||
|
// automatically initialized when they are used.
|
||||||
|
//
|
||||||
|
// Calling this function twice or after the singletons have been
|
||||||
|
// initialized will be no-op.
|
||||||
|
static void InitSingletons();
|
||||||
|
|
||||||
protected:
|
protected:
|
||||||
struct Entry {
|
struct Entry {
|
||||||
Entry() : ptr(nullptr) {}
|
Entry() : ptr(nullptr) {}
|
||||||
@ -121,6 +130,15 @@ class ThreadLocalPtr {
|
|||||||
// Register the UnrefHandler for id
|
// Register the UnrefHandler for id
|
||||||
void SetHandler(uint32_t id, UnrefHandler handler);
|
void SetHandler(uint32_t id, UnrefHandler handler);
|
||||||
|
|
||||||
|
// Initialize all the singletons associated with StaticMeta.
|
||||||
|
//
|
||||||
|
// If this function is not called, then the singletons will be
|
||||||
|
// automatically initialized when they are used.
|
||||||
|
//
|
||||||
|
// Calling this function twice or after the singletons have been
|
||||||
|
// initialized will be no-op.
|
||||||
|
static void InitSingletons();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
// Get UnrefHandler for id with acquiring mutex
|
// Get UnrefHandler for id with acquiring mutex
|
||||||
// REQUIRES: mutex locked
|
// REQUIRES: mutex locked
|
||||||
@ -153,7 +171,22 @@ class ThreadLocalPtr {
|
|||||||
|
|
||||||
// protect inst, next_instance_id_, free_instance_ids_, head_,
|
// protect inst, next_instance_id_, free_instance_ids_, head_,
|
||||||
// ThreadData.entries
|
// ThreadData.entries
|
||||||
static port::Mutex mutex_;
|
//
|
||||||
|
// Note that here we prefer function static variable instead of the usual
|
||||||
|
// global static variable. The reason is that c++ destruction order of
|
||||||
|
// static variables in the reverse order of their construction order.
|
||||||
|
// However, C++ does not guarantee any construction order when global
|
||||||
|
// static variables are defined in different files, while the function
|
||||||
|
// static variables are initialized when their function are first called.
|
||||||
|
// As a result, the construction order of the function static variables
|
||||||
|
// can be controlled by properly invoke their first function calls in
|
||||||
|
// the right order.
|
||||||
|
//
|
||||||
|
// For instance, the following function contains a function static
|
||||||
|
// variable. We place a dummy function call of this inside
|
||||||
|
// Env::Default() to ensure the construction order of the construction
|
||||||
|
// order.
|
||||||
|
static port::Mutex* Mutex();
|
||||||
#if ROCKSDB_SUPPORT_THREAD_LOCAL
|
#if ROCKSDB_SUPPORT_THREAD_LOCAL
|
||||||
// Thread local storage
|
// Thread local storage
|
||||||
static __thread ThreadData* tls_;
|
static __thread ThreadData* tls_;
|
||||||
|
Loading…
Reference in New Issue
Block a user