// Copyright 2022 The Forgotten Server Authors. All rights reserved. // Use of this source code is governed by the GPL-2.0 License that can be found in the LICENSE file. #ifndef FS_LOCKFREE_H_8C707AEB7C7235A2FBC5D4EDDF03B008 #define FS_LOCKFREE_H_8C707AEB7C7235A2FBC5D4EDDF03B008 #include #if _MSC_FULL_VER >= 190023918 // Workaround for VS2015 Update 2. Boost.Lockfree is a header-only library, so this should be safe to do. #define _ENABLE_ATOMIC_ALIGNMENT_FIX #endif /* * we use this to avoid instantiating multiple free lists for objects of the * same size and it can be replaced by a variable template in C++14 * * template * boost::lockfree::stack lockfreeFreeList; */ template struct LockfreeFreeList { using FreeList = boost::lockfree::stack>; static FreeList& get() { static FreeList freeList; return freeList; } }; template class LockfreePoolingAllocator { public: template struct rebind { using other = LockfreePoolingAllocator; }; LockfreePoolingAllocator() = default; template explicit constexpr LockfreePoolingAllocator(const LockfreePoolingAllocator&) {} using value_type = T; T* allocate(size_t) const { auto& inst = LockfreeFreeList::get(); void* p; // NOTE: p doesn't have to be initialized if (!inst.pop(p)) { //Acquire memory without calling the constructor of T p = operator new (sizeof(T)); } return static_cast(p); } void deallocate(T* p, size_t) const { auto& inst = LockfreeFreeList::get(); if (!inst.bounded_push(p)) { //Release memory without calling the destructor of T //(it has already been called at this point) operator delete(p); } } }; #endif