*/
#include <folly/io/async/EventBaseLocal.h>
+#include <folly/MapUtil.h>
#include <atomic>
#include <thread>
namespace folly { namespace detail {
EventBaseLocalBase::~EventBaseLocalBase() {
- // There's a race condition if an EventBase and an EventBaseLocal destruct
- // at the same time (each will lock eventBases_ and localStorageMutex_
- // in the opposite order), so we dance around it with a loop and try_lock.
- while (true) {
- SYNCHRONIZED(eventBases_) {
- auto it = eventBases_.begin();
- while (it != eventBases_.end()) {
- auto evb = *it;
- if (evb->localStorageMutex_.try_lock()) {
- evb->localStorage_.erase(key_);
- evb->localStorageToDtor_.erase(this);
- it = eventBases_.erase(it);
- evb->localStorageMutex_.unlock();
- } else {
- ++it;
- }
- }
-
- if (eventBases_.empty()) {
- return;
- }
- }
- std::this_thread::yield(); // let the other thread take the eventBases_ lock
+ for (auto* evb : *eventBases_.rlock()) {
+ evb->runInEventBaseThread([ this, evb, key = key_ ] {
+ evb->localStorage_.erase(key);
+ evb->localStorageToDtor_.erase(this);
+ });
}
}
void* EventBaseLocalBase::getVoid(EventBase& evb) {
- std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
- auto it2 = evb.localStorage_.find(key_);
- if (UNLIKELY(it2 != evb.localStorage_.end())) {
- return it2->second.get();
- }
+ DCHECK(evb.isInEventBaseThread());
- return nullptr;
+ return folly::get_default(evb.localStorage_, key_, {}).get();
}
void EventBaseLocalBase::erase(EventBase& evb) {
- std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
+ DCHECK(evb.isInEventBaseThread());
+
evb.localStorage_.erase(key_);
evb.localStorageToDtor_.erase(this);
}
void EventBaseLocalBase::onEventBaseDestruction(EventBase& evb) {
+ DCHECK(evb.isInEventBaseThread());
+
SYNCHRONIZED(eventBases_) {
eventBases_.erase(&evb);
}
}
void EventBaseLocalBase::setVoid(EventBase& evb, std::shared_ptr<void>&& ptr) {
- std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
- setVoidUnlocked(evb, std::move(ptr));
-}
-
-void EventBaseLocalBase::setVoidUnlocked(
- EventBase& evb, std::shared_ptr<void>&& ptr) {
+ DCHECK(evb.isInEventBaseThread());
auto alreadyExists =
evb.localStorage_.find(key_) != evb.localStorage_.end();
evb.localStorage_.emplace(key_, std::move(ptr));
if (!alreadyExists) {
- SYNCHRONIZED(eventBases_) {
- eventBases_.insert(&evb);
- }
+ eventBases_.wlock()->insert(&evb);
evb.localStorageToDtor_.insert(this);
}
}
protected:
void setVoid(EventBase& evb, std::shared_ptr<void>&& ptr);
- void setVoidUnlocked(EventBase& evb, std::shared_ptr<void>&& ptr);
void* getVoid(EventBase& evb);
folly::Synchronized<std::unordered_set<EventBase*>> eventBases_;
template <typename... Args>
T& getOrCreate(EventBase& evb, Args&&... args) {
- std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
-
- auto it2 = evb.localStorage_.find(key_);
- if (LIKELY(it2 != evb.localStorage_.end())) {
- return *static_cast<T*>(it2->second.get());
- } else {
- auto smartPtr = std::make_shared<T>(std::forward<Args>(args)...);
- auto ptr = smartPtr.get();
- setVoidUnlocked(evb, std::move(smartPtr));
- return *ptr;
+ if (auto ptr = getVoid(evb)) {
+ return *static_cast<T*>(ptr);
}
+ auto smartPtr = std::make_shared<T>(std::forward<Args>(args)...);
+ auto& ref = *smartPtr;
+ setVoid(evb, std::move(smartPtr));
+ return ref;
}
template <typename Func>
// If this looks like it's copy/pasted from above, that's because it is.
// gcc has a bug (fixed in 4.9) that doesn't allow capturing variadic
// params in a lambda.
- std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
-
- auto it2 = evb.localStorage_.find(key_);
- if (LIKELY(it2 != evb.localStorage_.end())) {
- return *static_cast<T*>(it2->second.get());
- } else {
- std::shared_ptr<T> smartPtr(fn());
- auto ptr = smartPtr.get();
- setVoidUnlocked(evb, std::move(smartPtr));
- return *ptr;
+ if (auto ptr = getVoid(evb)) {
+ return *static_cast<T*>(ptr);
}
+ std::shared_ptr<T> smartPtr(fn());
+ auto& ref = *smartPtr;
+ setVoid(evb, std::move(smartPtr));
+ return ref;
}
};