blob: 0f08b639e39f4a64d8c2d45acadce3d6923045a0 [file] [log] [blame]
/*
* 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/io/async/EventBaseLocal.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
}
}
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();
}
return nullptr;
}
void EventBaseLocalBase::erase(EventBase& evb) {
std::lock_guard<std::mutex> lg(evb.localStorageMutex_);
evb.localStorage_.erase(key_);
evb.localStorageToDtor_.erase(this);
SYNCHRONIZED(eventBases_) {
eventBases_.erase(&evb);
}
}
void EventBaseLocalBase::onEventBaseDestruction(EventBase& evb) {
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) {
auto alreadyExists =
evb.localStorage_.find(key_) != evb.localStorage_.end();
evb.localStorage_.emplace(key_, std::move(ptr));
if (!alreadyExists) {
SYNCHRONIZED(eventBases_) {
eventBases_.insert(&evb);
}
evb.localStorageToDtor_.insert(this);
}
}
std::atomic<uint64_t> EventBaseLocalBase::keyCounter_{0};
}}