From 71f01fb8f37708b54e32d99aa9c2200744a7ba78 Mon Sep 17 00:00:00 2001 From: Andrii Grynenko Date: Fri, 15 Apr 2016 19:50:46 -0700 Subject: [PATCH] Non-intrusive AtomicLinkedList Summary:Renamed AtomicLinkedList to AtomicIntrusiveLinkedList. AtomicLinkedList is a simple AtomicIntrusiveLinkedList wrapper, which handles intrusive list hook. Reviewed By: yfeldblum Differential Revision: D3188171 fb-gh-sync-id: 0823b04a48336d65e0a6a8cd412c75f52afe02b9 fbshipit-source-id: 0823b04a48336d65e0a6a8cd412c75f52afe02b9 --- folly/AtomicIntrusiveLinkedList.h | 137 +++++++++++++ folly/AtomicLinkedList.h | 99 +++------ folly/Makefile.am | 1 + folly/experimental/fibers/Fiber.h | 10 +- folly/experimental/fibers/FiberManager.h | 13 +- folly/experimental/fibers/FiberManagerMap.cpp | 1 - folly/test/AtomicLinkedListTest.cpp | 189 ++++++++++++++++++ 7 files changed, 364 insertions(+), 86 deletions(-) create mode 100644 folly/AtomicIntrusiveLinkedList.h create mode 100644 folly/test/AtomicLinkedListTest.cpp diff --git a/folly/AtomicIntrusiveLinkedList.h b/folly/AtomicIntrusiveLinkedList.h new file mode 100644 index 00000000..a78b685f --- /dev/null +++ b/folly/AtomicIntrusiveLinkedList.h @@ -0,0 +1,137 @@ +/* + * Copyright 2016 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 { + +/** + * A very simple atomic single-linked list primitive. + * + * Usage: + * + * class MyClass { + * AtomicIntrusiveLinkedListHook hook_; + * } + * + * AtomicIntrusiveLinkedList list; + * list.insert(&a); + * list.sweep([] (MyClass* c) { doSomething(c); } + */ +template +struct AtomicIntrusiveLinkedListHook { + T* next{nullptr}; +}; + +template T::*HookMember> +class AtomicIntrusiveLinkedList { + public: + AtomicIntrusiveLinkedList() {} + AtomicIntrusiveLinkedList(const AtomicIntrusiveLinkedList&) = delete; + AtomicIntrusiveLinkedList& operator=(const AtomicIntrusiveLinkedList&) = + delete; + AtomicIntrusiveLinkedList(AtomicIntrusiveLinkedList&& other) noexcept { + auto tmp = other.head_.load(); + other.head_ = head_.load(); + head_ = tmp; + } + AtomicIntrusiveLinkedList& operator=( + AtomicIntrusiveLinkedList&& other) noexcept { + auto tmp = other.head_.load(); + other.head_ = head_.load(); + head_ = tmp; + + return *this; + } + + /** + * Note: list must be empty on destruction. + */ + ~AtomicIntrusiveLinkedList() { + assert(empty()); + } + + bool empty() const { + return head_ == nullptr; + } + + /** + * Atomically insert t at the head of the list. + * @return True if the inserted element is the only one in the list + * after the call. + */ + bool insertHead(T* t) { + assert(next(t) == nullptr); + + auto oldHead = head_.load(std::memory_order_relaxed); + do { + next(t) = oldHead; + /* oldHead is updated by the call below. + + NOTE: we don't use next(t) instead of oldHead directly due to + compiler bugs (GCC prior to 4.8.3 (bug 60272), clang (bug 18899), + MSVC (bug 819819); source: + http://en.cppreference.com/w/cpp/atomic/atomic/compare_exchange */ + } while (!head_.compare_exchange_weak(oldHead, t, + std::memory_order_release, + std::memory_order_relaxed)); + + return oldHead == nullptr; + } + + /** + * Repeatedly replaces the head with nullptr, + * and calls func() on the removed elements in the order from tail to head. + * Stops when the list is empty. + */ + template + void sweep(F&& func) { + while (auto head = head_.exchange(nullptr)) { + auto rhead = reverse(head); + while (rhead != nullptr) { + auto t = rhead; + rhead = next(t); + next(t) = nullptr; + func(t); + } + } + } + + private: + std::atomic head_{nullptr}; + + static T*& next(T* t) { + return (t->*HookMember).next; + } + + /* Reverses a linked list, returning the pointer to the new head + (old tail) */ + static T* reverse(T* head) { + T* rhead = nullptr; + while (head != nullptr) { + auto t = head; + head = next(t); + next(t) = rhead; + rhead = t; + } + return rhead; + } +}; + +} // namespace folly diff --git a/folly/AtomicLinkedList.h b/folly/AtomicLinkedList.h index 459036ef..71f56bfc 100644 --- a/folly/AtomicLinkedList.h +++ b/folly/AtomicLinkedList.h @@ -16,8 +16,8 @@ #pragma once -#include -#include +#include +#include namespace folly { @@ -26,47 +26,26 @@ namespace folly { * * Usage: * - * class MyClass { - * AtomicLinkedListHook hook_; - * } - * - * AtomicLinkedList list; - * list.insert(&a); - * list.sweep([] (MyClass* c) { doSomething(c); } + * AtomicLinkedList list; + * list.insert(a); + * list.sweep([] (MyClass& c) { doSomething(c); } */ -template -struct AtomicLinkedListHook { - T* next{nullptr}; -}; -template T::* HookMember> +template class AtomicLinkedList { public: AtomicLinkedList() {} AtomicLinkedList(const AtomicLinkedList&) = delete; AtomicLinkedList& operator=(const AtomicLinkedList&) = delete; - AtomicLinkedList(AtomicLinkedList&& other) noexcept { - auto tmp = other.head_.load(); - other.head_ = head_.load(); - head_ = tmp; - } - AtomicLinkedList& operator=(AtomicLinkedList&& other) noexcept { - auto tmp = other.head_.load(); - other.head_ = head_.load(); - head_ = tmp; - - return *this; - } + AtomicLinkedList(AtomicLinkedList&& other) noexcept = default; + AtomicLinkedList& operator=(AtomicLinkedList&& other) = default; - /** - * Note: list must be empty on destruction. - */ ~AtomicLinkedList() { - assert(empty()); + sweep([](T&&) {}); } bool empty() const { - return head_ == nullptr; + return list_.empty(); } /** @@ -74,62 +53,34 @@ class AtomicLinkedList { * @return True if the inserted element is the only one in the list * after the call. */ - bool insertHead(T* t) { - assert(next(t) == nullptr); - - auto oldHead = head_.load(std::memory_order_relaxed); - do { - next(t) = oldHead; - /* oldHead is updated by the call below. + bool insertHead(T t) { + auto wrapper = folly::make_unique(std::move(t)); - NOTE: we don't use next(t) instead of oldHead directly due to - compiler bugs (GCC prior to 4.8.3 (bug 60272), clang (bug 18899), - MSVC (bug 819819); source: - http://en.cppreference.com/w/cpp/atomic/atomic/compare_exchange */ - } while (!head_.compare_exchange_weak(oldHead, t, - std::memory_order_release, - std::memory_order_relaxed)); - - return oldHead == nullptr; + return list_.insertHead(wrapper.release()); } /** - * Repeatedly replaces the head with nullptr, + * Repeatedly pops element from head, * and calls func() on the removed elements in the order from tail to head. * Stops when the list is empty. */ template void sweep(F&& func) { - while (auto head = head_.exchange(nullptr)) { - auto rhead = reverse(head); - while (rhead != nullptr) { - auto t = rhead; - rhead = next(t); - next(t) = nullptr; - func(t); - } - } + list_.sweep([&](Wrapper* wrapperPtr) mutable { + std::unique_ptr wrapper(wrapperPtr); + + func(std::move(wrapper->data)); + }); } private: - std::atomic head_{nullptr}; + struct Wrapper { + explicit Wrapper(T&& t) : data(std::move(t)) {} - static T*& next(T* t) { - return (t->*HookMember).next; - } - - /* Reverses a linked list, returning the pointer to the new head - (old tail) */ - static T* reverse(T* head) { - T* rhead = nullptr; - while (head != nullptr) { - auto t = head; - head = next(t); - next(t) = rhead; - rhead = t; - } - return rhead; - } + AtomicIntrusiveLinkedListHook hook; + T data; + }; + AtomicIntrusiveLinkedList list_; }; } // namespace folly diff --git a/folly/Makefile.am b/folly/Makefile.am index 2c199540..d9e8e771 100644 --- a/folly/Makefile.am +++ b/folly/Makefile.am @@ -31,6 +31,7 @@ nobase_follyinclude_HEADERS = \ AtomicHashArray-inl.h \ AtomicHashMap.h \ AtomicHashMap-inl.h \ + AtomicIntrusiveLinkedList.h \ AtomicLinkedList.h \ AtomicStruct.h \ AtomicUnorderedMap.h \ diff --git a/folly/experimental/fibers/Fiber.h b/folly/experimental/fibers/Fiber.h index a0e397f5..5025231b 100644 --- a/folly/experimental/fibers/Fiber.h +++ b/folly/experimental/fibers/Fiber.h @@ -18,15 +18,15 @@ #include #include -#include -#include -#include +#include #include #include #include +#include #include #include -#include +#include +#include namespace folly { namespace fibers { @@ -127,7 +127,7 @@ class Fiber { /** * Points to next fiber in remote ready list */ - folly::AtomicLinkedListHook nextRemoteReady_; + folly::AtomicIntrusiveLinkedListHook nextRemoteReady_; static constexpr size_t kUserBufferSize = 256; std::aligned_storage::type userBuffer_; diff --git a/folly/experimental/fibers/FiberManager.h b/folly/experimental/fibers/FiberManager.h index 5c893e85..11a631b5 100644 --- a/folly/experimental/fibers/FiberManager.h +++ b/folly/experimental/fibers/FiberManager.h @@ -24,12 +24,12 @@ #include #include -#include +#include #include -#include #include -#include +#include #include +#include #include #include @@ -334,7 +334,7 @@ class FiberManager : public ::folly::Executor { folly::Function func; std::unique_ptr localData; std::shared_ptr rcontext; - AtomicLinkedListHook nextRemoteTask; + AtomicIntrusiveLinkedListHook nextRemoteTask; }; intptr_t activateFiber(Fiber* fiber); @@ -430,9 +430,10 @@ class FiberManager : public ::folly::Executor { ExceptionCallback exceptionCallback_; /**< task exception callback */ - folly::AtomicLinkedList remoteReadyQueue_; + folly::AtomicIntrusiveLinkedList + remoteReadyQueue_; - folly::AtomicLinkedList + folly::AtomicIntrusiveLinkedList remoteTaskQueue_; std::shared_ptr timeoutManager_; diff --git a/folly/experimental/fibers/FiberManagerMap.cpp b/folly/experimental/fibers/FiberManagerMap.cpp index 93eb48ac..8f88cd22 100644 --- a/folly/experimental/fibers/FiberManagerMap.cpp +++ b/folly/experimental/fibers/FiberManagerMap.cpp @@ -18,7 +18,6 @@ #include #include -#include #include #include diff --git a/folly/test/AtomicLinkedListTest.cpp b/folly/test/AtomicLinkedListTest.cpp new file mode 100644 index 00000000..62f3dc60 --- /dev/null +++ b/folly/test/AtomicLinkedListTest.cpp @@ -0,0 +1,189 @@ +/* + * Copyright 2016 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 + +#include + +#include + +class TestIntrusiveObject { + public: + explicit TestIntrusiveObject(size_t id__) : id_(id__) {} + size_t id() { + return id_; + } + + private: + folly::AtomicIntrusiveLinkedListHook hook_; + size_t id_; + + public: + using List = folly::AtomicIntrusiveLinkedList< + TestIntrusiveObject, + &TestIntrusiveObject::hook_>; +}; + +TEST(AtomicIntrusiveLinkedList, Basic) { + TestIntrusiveObject a(1), b(2), c(3); + + TestIntrusiveObject::List list; + + EXPECT_TRUE(list.empty()); + + { + EXPECT_TRUE(list.insertHead(&a)); + EXPECT_FALSE(list.insertHead(&b)); + + EXPECT_FALSE(list.empty()); + + size_t id = 0; + list.sweep( + [&](TestIntrusiveObject* obj) mutable { EXPECT_EQ(++id, obj->id()); }); + + EXPECT_TRUE(list.empty()); + } + + // Try re-inserting the same item (b) and a new item (c) + { + EXPECT_TRUE(list.insertHead(&b)); + EXPECT_FALSE(list.insertHead(&c)); + + EXPECT_FALSE(list.empty()); + + size_t id = 1; + list.sweep( + [&](TestIntrusiveObject* obj) mutable { EXPECT_EQ(++id, obj->id()); }); + + EXPECT_TRUE(list.empty()); + } + + TestIntrusiveObject::List movedList = std::move(list); +} + +TEST(AtomicIntrusiveLinkedList, Move) { + TestIntrusiveObject a(1), b(2); + + TestIntrusiveObject::List list1; + + EXPECT_TRUE(list1.insertHead(&a)); + EXPECT_FALSE(list1.insertHead(&b)); + + EXPECT_FALSE(list1.empty()); + + TestIntrusiveObject::List list2(std::move(list1)); + + EXPECT_TRUE(list1.empty()); + EXPECT_FALSE(list2.empty()); + + TestIntrusiveObject::List list3; + + EXPECT_TRUE(list3.empty()); + + list3 = std::move(list2); + + EXPECT_TRUE(list2.empty()); + EXPECT_FALSE(list3.empty()); + + size_t id = 0; + list3.sweep( + [&](TestIntrusiveObject* obj) mutable { EXPECT_EQ(++id, obj->id()); }); +} + +TEST(AtomicIntrusiveLinkedList, Stress) { + constexpr size_t kNumThreads = 32; + constexpr size_t kNumElements = 100000; + + std::vector elements; + for (size_t i = 0; i < kNumThreads * kNumElements; ++i) { + elements.emplace_back(i); + } + + TestIntrusiveObject::List list; + + std::vector threads; + for (size_t threadId = 0; threadId < kNumThreads; ++threadId) { + threads.emplace_back( + [threadId, kNumThreads, kNumElements, &list, &elements]() { + for (size_t id = 0; id < kNumElements; ++id) { + list.insertHead(&elements[threadId + kNumThreads * id]); + } + }); + } + + std::vector ids; + TestIntrusiveObject* prev{nullptr}; + + while (ids.size() < kNumThreads * kNumElements) { + list.sweep([&](TestIntrusiveObject* current) { + ids.push_back(current->id()); + + if (prev && prev->id() % kNumThreads == current->id() % kNumThreads) { + EXPECT_EQ(prev->id() + kNumThreads, current->id()); + } + + prev = current; + }); + } + + std::sort(ids.begin(), ids.end()); + + for (size_t i = 0; i < kNumThreads * kNumElements; ++i) { + EXPECT_EQ(i, ids[i]); + } + + for (auto& thread : threads) { + thread.join(); + } +} + +class TestObject { + public: + TestObject(size_t id__, std::shared_ptr ptr) : id_(id__), ptr_(ptr) {} + + size_t id() { + return id_; + } + + private: + size_t id_; + std::shared_ptr ptr_; +}; + +TEST(AtomicLinkedList, Basic) { + constexpr size_t kNumElements = 10; + + using List = folly::AtomicLinkedList; + List list; + + std::shared_ptr ptr = std::make_shared(42); + + for (size_t id = 0; id < kNumElements; ++id) { + list.insertHead({id, ptr}); + } + + size_t counter = 0; + + list.sweep([&](TestObject object) { + EXPECT_EQ(counter, object.id()); + + EXPECT_EQ(1 + kNumElements - counter, ptr.use_count()); + + ++counter; + }); + + EXPECT_TRUE(ptr.unique()); +} -- 2.34.1