futures/QueuedImmediateExecutor.cpp \
futures/ThreadWheelTimekeeper.cpp \
detail/Futex.cpp \
+ detail/ThreadLocalDetail.cpp \
GroupVarint.cpp \
GroupVarintTables.cpp \
IPAddress.cpp \
#endif
}
+#if defined(__APPLE__) || defined(_MSC_VER)
+#define MAX_STATIC_CONSTRUCTOR_PRIORITY
+#else
+// 101 is the highest priority allowed by the init_priority attribute.
+// This priority is already used by JEMalloc and other memory allocators so
+// we will take the next one.
+#define MAX_STATIC_CONSTRUCTOR_PRIORITY __attribute__ ((__init_priority__(102)))
+#endif
+
} // namespace folly
#endif // FOLLY_PORTABILITY_H_
--- /dev/null
+/*
+ * Copyright 2015 Facebook, Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+#include <folly/ThreadLocal.h>
+
+namespace folly { namespace threadlocal_detail {
+
+PthreadKeyUnregister
+MAX_STATIC_CONSTRUCTOR_PRIORITY PthreadKeyUnregister::instance_;
+
+}}
constexpr uint32_t kEntryIDInvalid = std::numeric_limits<uint32_t>::max();
+/**
+ * We want to disable onThreadExit call at the end of shutdown, we don't care
+ * about leaking memory at that point.
+ *
+ * Otherwise if ThreadLocal is used in a shared library, onThreadExit may be
+ * called after dlclose().
+ */
+class PthreadKeyUnregister {
+ public:
+ ~PthreadKeyUnregister() {
+ std::lock_guard<std::mutex> lg(mutex_);
+
+ for (const auto& key: keys_) {
+ pthread_key_delete(key);
+ }
+ }
+
+ static void registerKey(pthread_key_t key) {
+ instance_.registerKeyImpl(key);
+ }
+
+ private:
+ PthreadKeyUnregister() {}
+
+ void registerKeyImpl(pthread_key_t key) {
+ std::lock_guard<std::mutex> lg(mutex_);
+
+ keys_.push_back(key);
+ }
+
+ std::mutex mutex_;
+ std::vector<pthread_key_t> keys_;
+
+ static PthreadKeyUnregister instance_;
+};
+
+
// Held in a singleton to track our global instances.
// We have one of these per "Tag", by default one for the whole system
// (Tag=void).
head_.next = head_.prev = &head_;
int ret = pthread_key_create(&pthreadKey_, &onThreadExit);
checkPosixError(ret, "pthread_key_create failed");
+ PthreadKeyUnregister::registerKey(pthreadKey_);
#if FOLLY_HAVE_PTHREAD_ATFORK
ret = pthread_atfork(/*prepare*/ &StaticMeta::preFork,
#include <folly/ThreadLocal.h>
+#include <dlfcn.h>
#include <sys/types.h>
#include <sys/wait.h>
#include <unistd.h>
#include <gtest/gtest.h>
#include <folly/Benchmark.h>
+#include <folly/Baton.h>
using namespace folly;
}
}
+TEST(ThreadLocal, SharedLibrary)
+{
+ auto handle = dlopen("./_bin/folly/test/lib_thread_local_test.so",
+ RTLD_LAZY);
+ EXPECT_NE(nullptr, handle);
+
+ typedef void (*useA_t)();
+ dlerror();
+ useA_t useA = (useA_t) dlsym(handle, "useA");
+
+ const char *dlsym_error = dlerror();
+ EXPECT_EQ(nullptr, dlsym_error);
+
+ useA();
+
+ folly::Baton<> b11, b12, b21, b22;
+
+ std::thread t1([&]() {
+ useA();
+ b11.post();
+ b12.wait();
+ });
+
+ std::thread t2([&]() {
+ useA();
+ b21.post();
+ b22.wait();
+ });
+
+ b11.wait();
+ b21.wait();
+
+ dlclose(handle);
+
+ b12.post();
+ b22.post();
+
+ t1.join();
+ t2.join();
+}
+
// clang is unable to compile this code unless in c++14 mode.
#if __cplusplus >= 201402L
namespace {
--- /dev/null
+/*
+ * Copyright 2015 Facebook, Inc.
+ *
+ * Licensed under the Apache License, Version 2.0 (the "License");
+ * you may not use this file except in compliance with the License.
+ * You may obtain a copy of the License at
+ *
+ * http://www.apache.org/licenses/LICENSE-2.0
+ *
+ * Unless required by applicable law or agreed to in writing, software
+ * distributed under the License is distributed on an "AS IS" BASIS,
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
+ * See the License for the specific language governing permissions and
+ * limitations under the License.
+ */
+#include <iostream>
+#include <thread>
+
+#include <folly/ThreadLocal.h>
+
+class A {
+ public:
+ void use() const {
+ }
+};
+
+folly::ThreadLocal<A> a;
+
+extern "C" {
+
+void useA() {
+ a->use();
+}
+
+}