-
Notifications
You must be signed in to change notification settings - Fork 10
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
A simple doubly-linked intrusive list. Notably it only provides forward iteration, to avoid complexity in handling reverse iterators. However it does allow pushing and popping at both ends.
- Loading branch information
1 parent
aac48be
commit 2e444a5
Showing
4 changed files
with
496 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,175 @@ | ||
#pragma once | ||
|
||
#include <stdx/concepts.hpp> | ||
|
||
#include <cstddef> | ||
#include <iterator> | ||
#include <type_traits> | ||
|
||
namespace stdx { | ||
inline namespace v1 { | ||
|
||
#if __cpp_concepts >= 201907L | ||
namespace detail { | ||
template <typename T> | ||
concept base_single_linkable = requires(T node) { | ||
{ node->next } -> same_as<T &>; | ||
}; | ||
} // namespace detail | ||
|
||
template <typename T> | ||
concept double_linkable = requires(T *node) { | ||
requires detail::base_single_linkable< | ||
std::remove_cvref_t<decltype(node->next)>>; | ||
requires detail::base_single_linkable< | ||
std::remove_cvref_t<decltype(node->prev)>>; | ||
}; | ||
|
||
#define STDX_DOUBLE_LINKABLE double_linkable | ||
#else | ||
#define STDX_DOUBLE_LINKABLE typename | ||
#endif | ||
|
||
template <STDX_DOUBLE_LINKABLE NodeType> class intrusive_list { | ||
public: | ||
struct iterator { | ||
using difference_type = std::ptrdiff_t; | ||
using value_type = NodeType; | ||
using pointer = value_type *; | ||
using reference = value_type &; | ||
using iterator_category = std::forward_iterator_tag; | ||
|
||
constexpr iterator() = default; | ||
constexpr explicit iterator(pointer n) : node{n} {} | ||
|
||
constexpr auto operator*() -> reference { return *node; } | ||
constexpr auto operator*() const -> reference { return *node; } | ||
constexpr auto operator->() -> pointer { return node; } | ||
constexpr auto operator->() const -> pointer { return node; } | ||
|
||
constexpr auto operator++() -> iterator & { | ||
node = node->next; | ||
return *this; | ||
} | ||
constexpr auto operator++(int) -> iterator { | ||
auto tmp = *this; | ||
++(*this); | ||
return tmp; | ||
} | ||
|
||
private: | ||
pointer node{}; | ||
|
||
#if __cpp_impl_three_way_comparison < 201907L | ||
friend constexpr auto operator==(iterator lhs, iterator rhs) -> bool { | ||
return lhs.node == rhs.node; | ||
} | ||
friend constexpr auto operator!=(iterator lhs, iterator rhs) -> bool { | ||
return not(lhs == rhs); | ||
} | ||
#else | ||
friend constexpr auto operator==(iterator, iterator) -> bool = default; | ||
#endif | ||
}; | ||
|
||
using value_type = NodeType; | ||
using size_type = std::size_t; | ||
using difference_type = std::ptrdiff_t; | ||
using reference = value_type &; | ||
using const_reference = value_type const &; | ||
using pointer = value_type *; | ||
using const_pointer = value_type const *; | ||
using const_iterator = iterator const; | ||
|
||
private: | ||
pointer head{}; | ||
pointer tail{}; | ||
|
||
public: | ||
constexpr auto begin() -> iterator { return iterator{head}; } | ||
constexpr auto begin() const -> const_iterator { return iterator{head}; } | ||
constexpr auto cbegin() const -> const_iterator { return iterator{head}; } | ||
constexpr auto end() -> iterator { return {}; } | ||
constexpr auto end() const -> const_iterator { return {}; } | ||
constexpr auto cend() -> const_iterator { return {}; } | ||
|
||
constexpr auto push_front(pointer n) -> void { | ||
if (head != nullptr) { | ||
head->prev = n; | ||
} | ||
n->next = head; | ||
head = n; | ||
n->prev = nullptr; | ||
if (tail == nullptr) { | ||
tail = n; | ||
} | ||
} | ||
|
||
constexpr auto push_back(pointer n) -> void { | ||
if (tail != nullptr) { | ||
tail->next = n; | ||
} | ||
n->prev = tail; | ||
tail = n; | ||
n->next = nullptr; | ||
if (head == nullptr) { | ||
head = n; | ||
} | ||
} | ||
|
||
constexpr auto pop_front() -> pointer { | ||
pointer const poppedNode = head; | ||
head = head->next; | ||
|
||
if (head == nullptr) { | ||
tail = nullptr; | ||
} else { | ||
head->prev = nullptr; | ||
} | ||
|
||
return poppedNode; | ||
} | ||
|
||
constexpr auto pop_back() -> pointer { | ||
pointer const poppedNode = tail; | ||
tail = tail->prev; | ||
|
||
if (tail == nullptr) { | ||
head = nullptr; | ||
} else { | ||
tail->next = nullptr; | ||
} | ||
|
||
return poppedNode; | ||
} | ||
|
||
[[nodiscard]] constexpr auto empty() const -> bool { | ||
return head == nullptr; | ||
} | ||
|
||
constexpr auto clear() -> void { | ||
head = nullptr; | ||
tail = nullptr; | ||
} | ||
|
||
constexpr auto remove(pointer n) -> void { | ||
pointer const nextNode = n->next; | ||
pointer const prevNode = n->prev; | ||
|
||
if (prevNode == nullptr) { | ||
head = nextNode; | ||
} else { | ||
prevNode->next = nextNode; | ||
} | ||
|
||
if (nextNode == nullptr) { | ||
tail = prevNode; | ||
} else { | ||
nextNode->prev = prevNode; | ||
} | ||
} | ||
}; | ||
|
||
#undef STDX_DOUBLE_LINKABLE | ||
} // namespace v1 | ||
} // namespace stdx |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Oops, something went wrong.