// // Copyright 2018 Pixar // // Licensed under the Apache License, Version 2.0 (the "Apache License") // with the following modification; you may not use this file except in // compliance with the Apache License and the following modification to it: // Section 6. Trademarks. is deleted and replaced with: // // 6. Trademarks. This License does not grant permission to use the trade // names, trademarks, service marks, or product names of the Licensor // and its affiliates, except as required to comply with Section 4(c) of // the License and to reproduce the content of the NOTICE file. // // You may obtain a copy of the Apache License at // // http://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the Apache License with the above modification is // distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY // KIND, either express or implied. See the Apache License for the specific // language governing permissions and limitations under the Apache License. // #ifndef PXR_BASE_TRACE_CONCURRENT_LIST_H #define PXR_BASE_TRACE_CONCURRENT_LIST_H #include "pxr/pxr.h" #include "pxr/base/arch/align.h" #include #include #include PXR_NAMESPACE_OPEN_SCOPE //////////////////////////////////////////////////////////////////////////////// /// \class TraceConcurrentList /// /// This class supports thread safe insertion and iteration over a list of items. /// template class TraceConcurrentList { // Linked list node that is cache line aligned to prevent false sharing. struct alignas(ARCH_CACHE_LINE_SIZE*2) Node { T value; Node* next; }; public: //////////////////////////////////////////////////////////////////////////// /// \class iterator /// /// This class provides forward iterator support to iterate over all the /// items. /// class iterator { public: // iterator types using iterator_category = std::forward_iterator_tag; using value = T; using pointer = T*; using reference = T&; using difference_type = ptrdiff_t; iterator() : _node(nullptr) {} pointer operator->() { return _node ? &_node->value : nullptr; } reference operator*() { return _node->value; } iterator& operator++() { _node = _node->next; return *this; } iterator operator++(int) { iterator result(*this); _node = _node->next; return result; } bool operator !=(const iterator& other) const { return _node != other._node; } bool operator ==(const iterator& other) const { return _node == other._node; } private: explicit iterator(Node* node) : _node(node) {} Node* _node; friend class TraceConcurrentList; }; /// Constructor. TraceConcurrentList() : _head(nullptr) {} /// Destructor. ~TraceConcurrentList() { // Delete all nodes in the list. Node* curNode = _head.load(std::memory_order_acquire); while (curNode) { Node* nodeToDelete = curNode; curNode = curNode->next; _alloc.destroy(nodeToDelete); _alloc.deallocate(nodeToDelete, 1); } } // No copies TraceConcurrentList(const TraceConcurrentList&) = delete; TraceConcurrentList& operator=(const TraceConcurrentList&) = delete; /// \name Iterator support. /// @{ iterator begin() { return iterator(_head.load(std::memory_order_acquire)); } iterator end() { return iterator(); } /// @} /// Inserts an item at the beginning of the list and returns an iterator to /// the newly created item. iterator Insert() { Node* newNode = _alloc.allocate(1); _alloc.construct(newNode); // Add the node to the linked list in an atomic manner. do { newNode->next = _head.load(std::memory_order_relaxed); } while (!_head.compare_exchange_weak(newNode->next, newNode)); return iterator(newNode); } private: std::atomic _head; tbb::cache_aligned_allocator _alloc; }; PXR_NAMESPACE_CLOSE_SCOPE #endif // PXR_BASE_TRACE_CONCURRENT_LIST_H