unioil-loyalty-rn-app/ios/Pods/Flipper-Folly/folly/synchronization/HazptrObj.h

543 lines
15 KiB
C++

/*
* Copyright (c) Facebook, Inc. and its affiliates.
*
* 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.
*/
#pragma once
#include <atomic>
#include <memory>
#include <glog/logging.h>
#include <folly/CPortability.h>
#include <folly/Portability.h>
#include <folly/concurrency/CacheLocality.h>
#include <folly/synchronization/Hazptr-fwd.h>
#include <folly/synchronization/detail/HazptrUtils.h>
///
/// Classes related to objects protectable by hazard pointers.
///
namespace folly {
/**
* hazptr_obj
*
* Private base class for objects protectable by hazard pointers.
*
* Data members:
* - next_: link to next object in private singly linked lists.
* - reclaim_: reclamation function for this object.
* - cohort_tag_: A pointer to a cohort (where the object is to be
* pushed when retired). It can also be used as a tag (see
* below).
*
* Cohorts, Tags, Tagged Objects, and Untagged Objects:
*
* - Cohorts: Cohorts (instances of hazptr_obj_cohort) are sets of
* retired hazptr_obj-s. Cohorts are used to keep related objects
* together instead of being spread across thread local structures
* and/or mixed with unrelated objects.
*
* - Tags: A tag is a unique identifier used for fast identification
* of related objects for synchronous reclamation. Tags are
* implemented as addresses of cohorts, with the lowest bit set (to
* save the space of separate cohort and tag data members and to
* differentiate from cohorts of untagged objects.
*
* - Tagged objects: Objects are tagged for fast identification. The
* primary use case is for synchronous reclamation ((e.g., the
* destruction of all Key and Value objects that were part of a
* Folly ConcurrentHashMap instance). Member function
* set_cohort_tag makes an object tagged.
*
* - Untagged objects: Objects that do not need to be identified
* separately from unrelated objects are not tagged (to keep tagged
* objects uncluttered). Untagged objects may or may not be
* associated with cohorts. An example of untagged objects
* associated with cohorts are Segment-s of Folly UnboundedQueue.
* Although such objects do not need synchronous reclamation,
* keeping them in cohorts helps avoid cases of a few missing
* objects delaying the reclamation of large numbers of
* link-counted objects. Objects are untagged either by default or
* after calling set_cohort_no_tag.
*
* - Thread Safety: Member functions set_cohort_tag and
* set_cohort_no_tag are not thread-safe. Thread safety must be
* ensured by the calling thread.
*/
template <template <typename> class Atom>
class hazptr_obj {
using ReclaimFnPtr = void (*)(hazptr_obj<Atom>*, hazptr_obj_list<Atom>&);
template <template <typename> class>
friend class hazptr_domain;
template <typename, template <typename> class, typename>
friend class hazptr_obj_base;
template <typename, template <typename> class, typename>
friend class hazptr_obj_base_linked;
template <template <typename> class>
friend class hazptr_obj_list;
template <template <typename> class>
friend class hazptr_priv;
friend class hazptr_detail::linked_list<hazptr_obj<Atom>>;
friend class hazptr_detail::shared_head_only_list<hazptr_obj<Atom>, Atom>;
friend class hazptr_detail::shared_head_tail_list<hazptr_obj<Atom>, Atom>;
static constexpr uintptr_t kTagBit = 1u;
ReclaimFnPtr reclaim_;
hazptr_obj<Atom>* next_;
uintptr_t cohort_tag_;
public:
/** Constructors */
/* All constructors set next_ to this in order to catch misuse bugs
such as double retire. By default, objects are untagged and not
associated with a cohort. */
hazptr_obj() noexcept : next_(this), cohort_tag_(0) {}
hazptr_obj(const hazptr_obj<Atom>& o) noexcept
: next_(this), cohort_tag_(o.cohort_tag_) {}
hazptr_obj(hazptr_obj<Atom>&& o) noexcept
: next_(this), cohort_tag_(o.cohort_tag_) {}
/** Copy operator */
hazptr_obj<Atom>& operator=(const hazptr_obj<Atom>&) noexcept {
return *this;
}
/** Move operator */
hazptr_obj<Atom>& operator=(hazptr_obj<Atom>&&) noexcept { return *this; }
/** cohort_tag */
uintptr_t cohort_tag() { return cohort_tag_; }
/** cohort */
hazptr_obj_cohort<Atom>* cohort() {
uintptr_t btag = cohort_tag_;
btag -= btag & kTagBit;
return reinterpret_cast<hazptr_obj_cohort<Atom>*>(btag);
}
/** tagged */
bool tagged() { return (cohort_tag_ & kTagBit) == kTagBit; }
/** set_cohort_tag: Set cohort and make object tagged. */
void set_cohort_tag(hazptr_obj_cohort<Atom>* cohort) {
cohort_tag_ = reinterpret_cast<uintptr_t>(cohort) + kTagBit;
}
/** set_cohort_no_tag: Set cohort and make object untagged. */
void set_cohort_no_tag(hazptr_obj_cohort<Atom>* cohort) {
cohort_tag_ = reinterpret_cast<uintptr_t>(cohort);
}
private:
friend class hazptr_domain<Atom>;
template <typename, template <typename> class, typename>
friend class hazptr_obj_base;
template <typename, template <typename> class, typename>
friend class hazptr_obj_base_refcounted;
friend class hazptr_obj_cohort<Atom>;
friend class hazptr_priv<Atom>;
hazptr_obj<Atom>* next() const noexcept { return next_; }
void set_next(hazptr_obj* obj) noexcept { next_ = obj; }
ReclaimFnPtr reclaim() noexcept { return reclaim_; }
const void* raw_ptr() const { return this; }
void pre_retire_check() noexcept {
// Only for catching misuse bugs like double retire
if (next_ != this) {
pre_retire_check_fail();
}
}
void push_obj(hazptr_domain<Atom>& domain) {
auto coh = cohort();
if (coh) {
DCHECK_EQ(&domain, &default_hazptr_domain<Atom>());
coh->push_obj(this);
} else {
push_to_retired(domain);
}
}
void push_to_retired(hazptr_domain<Atom>& domain) {
#if FOLLY_HAZPTR_THR_LOCAL
if (&domain == &default_hazptr_domain<Atom>() && !domain.shutdown_) {
hazptr_priv_tls<Atom>().push(this);
return;
}
#endif
hazptr_obj_list<Atom> l(this);
hazptr_domain_push_retired(l, true, domain);
}
FOLLY_NOINLINE void pre_retire_check_fail() noexcept {
CHECK_EQ(next_, this);
}
}; // hazptr_obj
/**
* hazptr_obj_list
*
* List of hazptr_obj-s.
*/
template <template <typename> class Atom>
class hazptr_obj_list {
using Obj = hazptr_obj<Atom>;
using List = hazptr_detail::linked_list<Obj>;
List l_;
int count_;
public:
hazptr_obj_list() noexcept : l_(nullptr, nullptr), count_(0) {}
explicit hazptr_obj_list(Obj* obj) noexcept : l_(obj, obj), count_(1) {
obj->set_next(nullptr);
}
explicit hazptr_obj_list(Obj* head, Obj* tail, int count) noexcept
: l_(head, tail), count_(count) {}
Obj* head() const noexcept { return l_.head(); }
Obj* tail() const noexcept { return l_.tail(); }
int count() const noexcept { return count_; }
void set_count(int val) { count_ = val; }
bool empty() const noexcept { return head() == nullptr; }
void push(Obj* obj) {
l_.push(obj);
++count_;
}
void splice(hazptr_obj_list<Atom>& l) {
if (l.count() == 0) {
return;
}
l_.splice(l.l_);
count_ += l.count();
l.clear();
}
void clear() {
l_.clear();
count_ = 0;
}
}; // hazptr_obj_list
/**
* hazptr_obj_cohort
*
* List of retired objects. For objects to be retred to a cohort,
* either of the hazptr_obj member functions set_cohort_tag or
* set_cohort_no_tag needs to be called before the object is retired.
*
* See description of hazptr_obj for notes on cohorts, tags, and
* tageed and untagged objects.
*
* [Note: For now supports only the default domain.]
*/
template <template <typename> class Atom>
class hazptr_obj_cohort {
using Obj = hazptr_obj<Atom>;
using List = hazptr_detail::linked_list<Obj>;
using SharedList = hazptr_detail::shared_head_tail_list<Obj, Atom>;
static constexpr int kThreshold = 20;
SharedList l_;
Atom<int> count_;
Atom<bool> active_;
Atom<bool> pushed_to_domain_tagged_;
Atom<Obj*> safe_list_top_;
public:
/** Constructor */
hazptr_obj_cohort() noexcept
: l_(),
count_(0),
active_(true),
pushed_to_domain_tagged_{false},
safe_list_top_{nullptr} {}
/** Not copyable or moveable */
hazptr_obj_cohort(const hazptr_obj_cohort& o) = delete;
hazptr_obj_cohort(hazptr_obj_cohort&& o) = delete;
hazptr_obj_cohort& operator=(const hazptr_obj_cohort&& o) = delete;
hazptr_obj_cohort& operator=(hazptr_obj_cohort&& o) = delete;
/** Destructor */
~hazptr_obj_cohort() {
if (active()) {
shutdown_and_reclaim();
}
DCHECK(!active());
DCHECK(l_.empty());
}
/** shutdown_and_reclaim */
void shutdown_and_reclaim() {
DCHECK(active());
clear_active();
if (pushed_to_domain_tagged_.load(std::memory_order_relaxed)) {
default_hazptr_domain<Atom>().cleanup_cohort_tag(this);
}
reclaim_safe_list();
if (!l_.empty()) {
List l = l_.pop_all();
clear_count();
Obj* obj = l.head();
reclaim_list(obj);
}
DCHECK(l_.empty());
}
private:
friend class hazptr_domain<Atom>;
friend class hazptr_obj<Atom>;
bool active() { return active_.load(std::memory_order_relaxed); }
void clear_active() { active_.store(false, std::memory_order_relaxed); }
int count() const noexcept { return count_.load(std::memory_order_acquire); }
void clear_count() noexcept { count_.store(0, std::memory_order_release); }
void inc_count() noexcept { count_.fetch_add(1, std::memory_order_release); }
bool cas_count(int& expected, int newval) noexcept {
return count_.compare_exchange_weak(
expected, newval, std::memory_order_acq_rel, std::memory_order_acquire);
}
Obj* safe_list_top() const noexcept {
return safe_list_top_.load(std::memory_order_acquire);
}
bool cas_safe_list_top(Obj*& expected, Obj* newval) noexcept {
return safe_list_top_.compare_exchange_weak(
expected, newval, std::memory_order_acq_rel, std::memory_order_relaxed);
}
/** push_obj */
void push_obj(Obj* obj) {
if (active()) {
l_.push(obj);
inc_count();
check_threshold_push();
if (safe_list_top())
reclaim_safe_list();
} else {
obj->set_next(nullptr);
reclaim_list(obj);
}
}
/** push_safe_obj */
void push_safe_obj(Obj* obj) noexcept {
while (true) {
Obj* top = safe_list_top();
obj->set_next(top);
if (cas_safe_list_top(top, obj))
return;
}
}
/** reclaim_list */
void reclaim_list(hazptr_obj<Atom>* obj) {
while (obj) {
hazptr_obj_list<Atom> children;
while (obj) {
Obj* next = obj->next();
(*(obj->reclaim()))(obj, children);
obj = next;
}
obj = children.head();
}
}
/** check_threshold_push */
void check_threshold_push() {
auto c = count();
while (c >= kThreshold) {
if (cas_count(c, 0)) {
List ll = l_.pop_all();
if (ll.head() && ll.head()->tagged()) {
pushed_to_domain_tagged_.store(true, std::memory_order_relaxed);
}
if (kIsDebug) {
Obj* p = ll.head();
for (int i = 1; p; ++i, p = p->next()) {
DCHECK_EQ(reinterpret_cast<uintptr_t>(p) & 7, uintptr_t{0})
<< p << " " << i;
}
}
hazptr_obj_list<Atom> l(ll.head(), ll.tail(), c);
hazptr_domain_push_list<Atom>(l);
return;
}
}
}
/** reclaim_safe_list */
void reclaim_safe_list() {
Obj* top = safe_list_top_.exchange(nullptr, std::memory_order_acq_rel);
reclaim_list(top);
}
}; // hazptr_obj_cohort
/**
* hazptr_obj_retired_list
*
* Used for maintaining lists of retired objects in domain
* structure. Locked operations are used for lists of tagged
* objects. Unlocked operations are used for the untagged list.
*/
/** hazptr_obj_retired_list */
template <template <typename> class Atom>
class hazptr_obj_retired_list {
using Obj = hazptr_obj<Atom>;
using List = hazptr_detail::linked_list<Obj>;
using RetiredList = hazptr_detail::shared_head_only_list<Obj, Atom>;
alignas(hardware_destructive_interference_size) RetiredList retired_;
Atom<int> count_;
public:
static constexpr bool kAlsoLock = RetiredList::kAlsoLock;
static constexpr bool kDontLock = RetiredList::kDontLock;
static constexpr bool kMayBeLocked = RetiredList::kMayBeLocked;
static constexpr bool kMayNotBeLocked = RetiredList::kMayNotBeLocked;
public:
hazptr_obj_retired_list() noexcept : count_(0) {}
void push(hazptr_obj_list<Atom>& l, bool may_be_locked) noexcept {
List ll(l.head(), l.tail());
retired_.push(ll, may_be_locked);
add_count(l.count());
}
void push_unlock(hazptr_obj_list<Atom>& l) noexcept {
List ll(l.head(), l.tail());
retired_.push_unlock(ll);
auto count = l.count();
if (count) {
add_count(count);
}
}
int count() const noexcept { return count_.load(std::memory_order_acquire); }
bool empty() { return retired_.empty(); }
bool check_threshold_try_zero_count(int thresh) {
auto oldval = count();
while (oldval >= thresh) {
if (cas_count(oldval, 0)) {
return true;
}
}
return false;
}
Obj* pop_all(bool lock) { return retired_.pop_all(lock); }
bool check_lock() { return retired_.check_lock(); }
private:
void add_count(int val) { count_.fetch_add(val, std::memory_order_release); }
bool cas_count(int& expected, int newval) {
return count_.compare_exchange_weak(
expected, newval, std::memory_order_acq_rel, std::memory_order_acquire);
}
}; // hazptr_obj_retired_list
/**
* hazptr_deleter
*
* For empty base optimization.
*/
template <typename T, typename D>
class hazptr_deleter {
D deleter_;
public:
void set_deleter(D d = {}) { deleter_ = std::move(d); }
void delete_obj(T* p) { deleter_(p); }
};
template <typename T>
class hazptr_deleter<T, std::default_delete<T>> {
public:
void set_deleter(std::default_delete<T> = {}) {}
void delete_obj(T* p) { delete p; }
};
/**
* hazptr_obj_base
*
* Base template for objects protected by hazard pointers.
*/
template <typename T, template <typename> class Atom, typename D>
class hazptr_obj_base : public hazptr_obj<Atom>, public hazptr_deleter<T, D> {
public:
/* Retire a removed object and pass the responsibility for
* reclaiming it to the hazptr library */
void retire(
D deleter = {},
hazptr_domain<Atom>& domain = default_hazptr_domain<Atom>()) {
pre_retire(std::move(deleter));
set_reclaim();
this->push_obj(domain); // defined in hazptr_obj
}
void retire(hazptr_domain<Atom>& domain) { retire({}, domain); }
private:
void pre_retire(D deleter) {
this->pre_retire_check(); // defined in hazptr_obj
this->set_deleter(std::move(deleter));
}
void set_reclaim() {
this->reclaim_ = [](hazptr_obj<Atom>* p, hazptr_obj_list<Atom>&) {
auto hobp = static_cast<hazptr_obj_base<T, Atom, D>*>(p);
auto obj = static_cast<T*>(hobp);
hobp->delete_obj(obj);
};
}
}; // hazptr_obj_base
} // namespace folly