From 763b84fbc36895d3b6f1da10979fb050a399133f Mon Sep 17 00:00:00 2001 From: Dave Watson Date: Wed, 26 Jul 2017 08:07:56 -0700 Subject: [PATCH] Allow stealing pointer bits Summary: Currently hazard pointers doesn't support stealing any of the pointer bits. You can *almost* roll it yourself using try_protect, but this prevents implementations from choosing their type of barrier. This adds a new get_protected interface that you can use to steal bits, or otherwise manipulate pointers as you would like. This also adds a MWMR list based set example that uses it, that is wait-free for readers (unlike the SWMR example, that is only lock-free). Reviewed By: magedm Differential Revision: D5455615 fbshipit-source-id: 53d282eda433e00b6b53cd804d4e1c32c74c2fb8 --- folly/experimental/hazptr/example/MWMRSet.h | 251 ++++++++++++++++++ folly/experimental/hazptr/hazptr-impl.h | 20 +- folly/experimental/hazptr/hazptr.h | 10 + folly/experimental/hazptr/test/HazptrTest.cpp | 31 +++ 4 files changed, 310 insertions(+), 2 deletions(-) create mode 100644 folly/experimental/hazptr/example/MWMRSet.h diff --git a/folly/experimental/hazptr/example/MWMRSet.h b/folly/experimental/hazptr/example/MWMRSet.h new file mode 100644 index 00000000..49d561ad --- /dev/null +++ b/folly/experimental/hazptr/example/MWMRSet.h @@ -0,0 +1,251 @@ +/* + * Copyright 2017 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. + */ +#pragma once + +#include +#include + +namespace folly { +namespace hazptr { + +/** Set implemented as an ordered singly-linked list. + * + * Multiple writers may add or remove elements. Multiple reader + * threads may search the set concurrently with each other and with + * the writers' operations. + */ +template +class MWMRListSet { + class Node : public hazptr_obj_base { + friend MWMRListSet; + T elem_; + std::atomic refcount_{1}; + std::atomic next_{nullptr}; + + // Node must be refcounted for wait-free access: A deleted node + // may have hazptrs pointing at it, so the rest of the list (or at + // least, what existed at the time of the hazptr load) must still + // be accessible. + void release() { + if (refcount_.fetch_sub(1) == 1) { + this->retire(); + } + } + + // Optimization in the case that we know there are no hazptrs pointing + // at the list. + void releaseFast() { + if (refcount_.load(std::memory_order_relaxed) == 1) { + auto next = getPtr(next_.load(std::memory_order_relaxed)); + if (next) { + next->releaseFast(); + next_.store(nullptr, std::memory_order_relaxed); + } + delete this; + } + } + + void acquire() { + DCHECK(refcount_.load() != 0); + refcount_.fetch_add(1); + } + + public: + explicit Node(T e) : elem_(e) { + DEBUG_PRINT(this << " " << e); + } + + ~Node() { + DEBUG_PRINT(this); + auto next = getPtr(next_.load(std::memory_order_relaxed)); + if (next) { + next->release(); + } + } + }; + + static bool getDeleted(Node* ptr) { + return uintptr_t(ptr) & 1; + } + + static Node* getPtr(Node* ptr) { + return (Node*)(uintptr_t(ptr) & ~1UL); + } + + mutable std::atomic head_ = {nullptr}; + + // Remove a single deleted item. + // Although it doesn't have to be our item. + // + // Note that standard lock-free Michael linked lists put this in the + // contains() path, while this implementation leaves it only in + // remove(), such that contains() is wait-free. + void fixlist( + hazptr_holder& hptr_prev, + hazptr_holder& hptr_curr, + std::atomic*& prev, + Node*& curr) const { + while (true) { + prev = &head_; + curr = hptr_curr.get_protected(*prev, getPtr); + while (getPtr(curr)) { + auto next = getPtr(curr)->next_.load(std::memory_order_acquire); + if (getDeleted(next)) { + auto nextp = getPtr(next); + if (nextp) { + nextp->acquire(); + } + // Try to fix + auto curr_no_mark = getPtr(curr); + if (prev->compare_exchange_weak(curr_no_mark, nextp)) { + // Physically delete + curr_no_mark->release(); + return; + } else { + if (nextp) { + nextp->release(); + } + break; + } + } + prev = &(getPtr(curr)->next_); + curr = hptr_prev.get_protected(getPtr(curr)->next_, getPtr); + + swap(hptr_curr, hptr_prev); + } + DCHECK(getPtr(curr)); + } + } + + /* wait-free set search */ + bool find( + const T& val, + hazptr_holder& hptr_prev, + hazptr_holder& hptr_curr, + std::atomic*& prev, + Node*& curr) const { + prev = &head_; + curr = hptr_curr.get_protected(*prev, getPtr); + while (getPtr(curr)) { + auto next = getPtr(curr)->next_.load(std::memory_order_acquire); + if (!getDeleted(next)) { + if (getPtr(curr)->elem_ == val) { + return true; + } else if (!(getPtr(curr)->elem_ < val)) { + break; // Because the list is sorted. + } + } + prev = &(getPtr(curr)->next_); + curr = hptr_prev.get_protected(getPtr(curr)->next_, getPtr); + /* Swap does not change the values of the owned hazard + * pointers themselves. After the swap, The hazard pointer + * owned by hptr_prev continues to protect the node that + * contains the pointer *prev. The hazard pointer owned by + * hptr_curr will continue to protect the node that contains + * the old *prev (unless the old prev was &head), which no + * longer needs protection, so hptr_curr's hazard pointer is + * now free to protect *curr in the next iteration (if curr != + * null). + */ + swap(hptr_curr, hptr_prev); + } + + return false; + } + + public: + explicit MWMRListSet() {} + + ~MWMRListSet() { + Node* next = head_.load(); + if (next) { + next->releaseFast(); + } + } + + bool add(T v) { + hazptr_holder hptr_prev; + hazptr_holder hptr_curr; + std::atomic* prev; + Node* cur; + + auto newnode = folly::make_unique(v); + + while (true) { + if (find(v, hptr_prev, hptr_curr, prev, cur)) { + return false; + } + newnode->next_.store(cur, std::memory_order_relaxed); + auto cur_no_mark = getPtr(cur); + if (prev->compare_exchange_weak(cur_no_mark, newnode.get())) { + newnode.release(); + return true; + } + // Ensure ~Node() destructor doesn't destroy next_ + newnode->next_.store(nullptr, std::memory_order_relaxed); + } + } + + bool remove(const T& v) { + hazptr_holder hptr_prev; + hazptr_holder hptr_curr; + std::atomic* prev; + Node* curr; + + while (true) { + if (!find(v, hptr_prev, hptr_curr, prev, curr)) { + return false; + } + auto next = getPtr(curr)->next_.load(std::memory_order_acquire); + auto next_no_mark = getPtr(next); // Ensure only one deleter wins + // Logically delete + if (!getPtr(curr)->next_.compare_exchange_weak( + next_no_mark, (Node*)(uintptr_t(next_no_mark) | 1))) { + continue; + } + if (next) { + next->acquire(); + } + + // Swing prev around + auto curr_no_mark = getPtr(curr); /* ensure not deleted */ + if (prev->compare_exchange_weak(curr_no_mark, next)) { + // Physically delete + curr->release(); + return true; + } + if (next) { + next->release(); + } + + // Someone else modified prev. Call fixlist + // to unlink deleted element by re-walking list. + fixlist(hptr_prev, hptr_curr, prev, curr); + } + } + + bool contains(const T& v) const { + hazptr_holder hptr_prev; + hazptr_holder hptr_curr; + std::atomic* prev; + Node* curr; + + return find(v, hptr_prev, hptr_curr, prev, curr); + } +}; + +} // namespace folly { +} // namespace hazptr { diff --git a/folly/experimental/hazptr/hazptr-impl.h b/folly/experimental/hazptr/hazptr-impl.h index 70123659..60cf5ecd 100644 --- a/folly/experimental/hazptr/hazptr-impl.h +++ b/folly/experimental/hazptr/hazptr-impl.h @@ -231,8 +231,16 @@ template FOLLY_ALWAYS_INLINE bool hazptr_holder::try_protect( T*& ptr, const std::atomic& src) noexcept { + return try_protect(ptr, src, [](T* t) { return t; }); +} + +template +FOLLY_ALWAYS_INLINE bool hazptr_holder::try_protect( + T*& ptr, + const std::atomic& src, + Func f) noexcept { DEBUG_PRINT(this << " " << ptr << " " << &src); - reset(ptr); + reset(f(ptr)); /*** Full fence ***/ hazptr_mb::light(); T* p = src.load(std::memory_order_acquire); if (p != ptr) { @@ -246,8 +254,16 @@ FOLLY_ALWAYS_INLINE bool hazptr_holder::try_protect( template FOLLY_ALWAYS_INLINE T* hazptr_holder::get_protected( const std::atomic& src) noexcept { + return get_protected(src, [](T* t) { return t; }); +} + +template +FOLLY_ALWAYS_INLINE T* hazptr_holder::get_protected( + const std::atomic& src, + Func f) noexcept { T* p = src.load(std::memory_order_relaxed); - while (!try_protect(p, src)) {} + while (!try_protect(p, src, f)) { + } DEBUG_PRINT(this << " " << p << " " << &src); return p; } diff --git a/folly/experimental/hazptr/hazptr.h b/folly/experimental/hazptr/hazptr.h index d5944adf..301fdfab 100644 --- a/folly/experimental/hazptr/hazptr.h +++ b/folly/experimental/hazptr/hazptr.h @@ -119,10 +119,20 @@ class hazptr_holder { /* Returns a protected pointer from the source */ template T* get_protected(const std::atomic& src) noexcept; + /* Returns a protected pointer from the source, filtering + the protected pointer through function Func. Useful for + stealing bits of the pointer word */ + template + T* get_protected(const std::atomic& src, Func f) noexcept; /* Return true if successful in protecting ptr if src == ptr after * setting the hazard pointer. Otherwise sets ptr to src. */ template bool try_protect(T*& ptr, const std::atomic& src) noexcept; + /* Return true if successful in protecting ptr if src == ptr after + * setting the hazard pointer, filtering the pointer through Func. + * Otherwise sets ptr to src. */ + template + bool try_protect(T*& ptr, const std::atomic& src, Func f) noexcept; /* Set the hazard pointer to ptr */ template void reset(const T* ptr) noexcept; diff --git a/folly/experimental/hazptr/test/HazptrTest.cpp b/folly/experimental/hazptr/test/HazptrTest.cpp index d8ca18da..301aaa80 100644 --- a/folly/experimental/hazptr/test/HazptrTest.cpp +++ b/folly/experimental/hazptr/test/HazptrTest.cpp @@ -19,6 +19,7 @@ #include #include +#include #include #include #include @@ -224,6 +225,36 @@ TEST_F(HazptrTest, SWMRLIST) { } } +TEST_F(HazptrTest, MWMRSet) { + using T = uint64_t; + + CHECK_GT(FLAGS_num_threads, 0); + for (int i = 0; i < FLAGS_num_reps; ++i) { + DEBUG_PRINT("========== start of rep scope"); + MWMRListSet s; + std::vector threads(FLAGS_num_threads); + for (int tid = 0; tid < FLAGS_num_threads; ++tid) { + threads[tid] = std::thread([&s, tid]() { + for (int j = tid; j < FLAGS_num_ops; j += FLAGS_num_threads) { + s.contains(j); + s.add(j); + s.remove(j); + } + }); + } + for (int j = 0; j < 10; ++j) { + s.add(j); + } + for (int j = 0; j < 10; ++j) { + s.remove(j); + } + for (auto& t : threads) { + t.join(); + } + DEBUG_PRINT("========== end of rep scope"); + } +} + TEST_F(HazptrTest, WIDECAS) { WideCAS s; std::string u = ""; -- 2.34.1