2020-10-18 19:21:11 +02:00
|
|
|
/*
|
2022-11-11 17:15:15 +01:00
|
|
|
* Copyright 2020-2022 Max Kellermann <max.kellermann@gmail.com>
|
2020-10-18 19:21:11 +02:00
|
|
|
*
|
|
|
|
* Redistribution and use in source and binary forms, with or without
|
|
|
|
* modification, are permitted provided that the following conditions
|
|
|
|
* are met:
|
|
|
|
*
|
|
|
|
* - Redistributions of source code must retain the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer.
|
|
|
|
*
|
|
|
|
* - Redistributions in binary form must reproduce the above copyright
|
|
|
|
* notice, this list of conditions and the following disclaimer in the
|
|
|
|
* documentation and/or other materials provided with the
|
|
|
|
* distribution.
|
|
|
|
*
|
|
|
|
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
|
|
|
|
* ``AS IS'' AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
|
|
|
|
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS
|
|
|
|
* FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE
|
|
|
|
* FOUNDATION OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT,
|
|
|
|
* INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
|
|
|
|
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
|
|
|
|
* SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION)
|
|
|
|
* HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT,
|
|
|
|
* STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE)
|
|
|
|
* ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED
|
|
|
|
* OF THE POSSIBILITY OF SUCH DAMAGE.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#pragma once
|
|
|
|
|
|
|
|
#include "Cast.hxx"
|
2022-06-08 18:01:18 +02:00
|
|
|
#include "MemberPointer.hxx"
|
2022-06-09 09:47:48 +02:00
|
|
|
#include "OptionalCounter.hxx"
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
#include <iterator>
|
|
|
|
#include <type_traits>
|
|
|
|
#include <utility>
|
|
|
|
|
|
|
|
struct IntrusiveListNode {
|
|
|
|
IntrusiveListNode *next, *prev;
|
2022-11-10 16:46:19 +01:00
|
|
|
|
|
|
|
static constexpr void Connect(IntrusiveListNode &a,
|
|
|
|
IntrusiveListNode &b) noexcept {
|
|
|
|
a.next = &b;
|
|
|
|
b.prev = &a;
|
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
class IntrusiveListHook {
|
2022-06-08 10:31:09 +02:00
|
|
|
template<typename T> friend struct IntrusiveListBaseHookTraits;
|
2022-06-08 18:01:18 +02:00
|
|
|
template<auto member> friend struct IntrusiveListMemberHookTraits;
|
2022-06-09 09:47:48 +02:00
|
|
|
template<typename T, typename HookTraits, bool> friend class IntrusiveList;
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
protected:
|
|
|
|
IntrusiveListNode siblings;
|
|
|
|
|
|
|
|
public:
|
2021-12-02 14:57:48 +01:00
|
|
|
IntrusiveListHook() noexcept = default;
|
|
|
|
|
|
|
|
IntrusiveListHook(const IntrusiveListHook &) = delete;
|
|
|
|
IntrusiveListHook &operator=(const IntrusiveListHook &) = delete;
|
|
|
|
|
2020-10-18 19:21:11 +02:00
|
|
|
void unlink() noexcept {
|
2022-11-10 16:46:19 +01:00
|
|
|
IntrusiveListNode::Connect(*siblings.prev, *siblings.next);
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
static constexpr auto &Cast(IntrusiveListNode &node) noexcept {
|
|
|
|
return ContainerCast(node, &IntrusiveListHook::siblings);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const auto &Cast(const IntrusiveListNode &node) noexcept {
|
|
|
|
return ContainerCast(node, &IntrusiveListHook::siblings);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2021-02-24 20:28:47 +01:00
|
|
|
* A variant of #IntrusiveListHook which keeps track of whether it is
|
|
|
|
* currently in a list.
|
2020-10-18 19:21:11 +02:00
|
|
|
*/
|
2021-02-24 20:28:47 +01:00
|
|
|
class SafeLinkIntrusiveListHook : public IntrusiveListHook {
|
2020-10-18 19:21:11 +02:00
|
|
|
public:
|
2021-02-24 20:28:47 +01:00
|
|
|
SafeLinkIntrusiveListHook() noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
siblings.next = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
void unlink() noexcept {
|
|
|
|
IntrusiveListHook::unlink();
|
|
|
|
siblings.next = nullptr;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool is_linked() const noexcept {
|
|
|
|
return siblings.next != nullptr;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2021-02-24 20:28:47 +01:00
|
|
|
/**
|
|
|
|
* A variant of #IntrusiveListHook which auto-unlinks itself from the
|
|
|
|
* list upon destruction. As a side effect, it has an is_linked()
|
|
|
|
* method.
|
|
|
|
*/
|
|
|
|
class AutoUnlinkIntrusiveListHook : public SafeLinkIntrusiveListHook {
|
|
|
|
public:
|
|
|
|
~AutoUnlinkIntrusiveListHook() noexcept {
|
|
|
|
if (is_linked())
|
|
|
|
unlink();
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-06-08 21:39:53 +02:00
|
|
|
/**
|
|
|
|
* Detect the hook type; this is important because
|
|
|
|
* SafeLinkIntrusiveListHook::unlink() needs to clear the "next"
|
|
|
|
* pointer. This is a template to postpone the type checks, to allow
|
|
|
|
* forward-declared types.
|
|
|
|
*/
|
|
|
|
template<typename U>
|
|
|
|
struct IntrusiveListHookDetection {
|
|
|
|
static_assert(std::is_base_of_v<IntrusiveListHook, U>);
|
2021-02-25 13:43:33 +01:00
|
|
|
|
2022-06-08 21:39:53 +02:00
|
|
|
using type = std::conditional_t<std::is_base_of_v<SafeLinkIntrusiveListHook, U>,
|
|
|
|
SafeLinkIntrusiveListHook,
|
|
|
|
IntrusiveListHook>;
|
|
|
|
};
|
2021-02-25 13:43:33 +01:00
|
|
|
|
2022-06-08 10:31:09 +02:00
|
|
|
/**
|
|
|
|
* For classes which embed #IntrusiveListHook as base class.
|
|
|
|
*/
|
2022-06-08 21:39:53 +02:00
|
|
|
template<typename T>
|
2022-06-08 10:31:09 +02:00
|
|
|
struct IntrusiveListBaseHookTraits {
|
2021-02-25 13:43:33 +01:00
|
|
|
template<typename U>
|
2022-06-08 21:39:53 +02:00
|
|
|
using Hook = typename IntrusiveListHookDetection<U>::type;
|
2021-02-25 13:43:33 +01:00
|
|
|
|
2022-06-09 09:47:48 +02:00
|
|
|
static constexpr bool IsAutoUnlink() noexcept {
|
|
|
|
return std::is_base_of_v<AutoUnlinkIntrusiveListHook, T>;
|
|
|
|
}
|
|
|
|
|
2020-10-18 19:21:11 +02:00
|
|
|
static constexpr T *Cast(IntrusiveListNode *node) noexcept {
|
2021-02-25 13:43:33 +01:00
|
|
|
auto *hook = &Hook<T>::Cast(*node);
|
2020-10-18 19:21:11 +02:00
|
|
|
return static_cast<T *>(hook);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const T *Cast(const IntrusiveListNode *node) noexcept {
|
2021-02-25 13:43:33 +01:00
|
|
|
const auto *hook = &Hook<T>::Cast(*node);
|
2020-10-18 19:21:11 +02:00
|
|
|
return static_cast<const T *>(hook);
|
|
|
|
}
|
|
|
|
|
2021-02-25 13:43:33 +01:00
|
|
|
static constexpr auto &ToHook(T &t) noexcept {
|
|
|
|
return static_cast<Hook<T> &>(t);
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
2021-02-25 13:43:33 +01:00
|
|
|
static constexpr const auto &ToHook(const T &t) noexcept {
|
|
|
|
return static_cast<const Hook<T> &>(t);
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
2022-06-08 10:31:09 +02:00
|
|
|
};
|
|
|
|
|
2022-06-08 18:01:18 +02:00
|
|
|
/**
|
|
|
|
* For classes which embed #IntrusiveListHook as member.
|
|
|
|
*/
|
|
|
|
template<auto member>
|
|
|
|
struct IntrusiveListMemberHookTraits {
|
|
|
|
using T = MemberPointerContainerType<decltype(member)>;
|
|
|
|
using _Hook = MemberPointerType<decltype(member)>;
|
|
|
|
using Hook = typename IntrusiveListHookDetection<_Hook>::type;
|
|
|
|
|
2022-06-09 09:47:48 +02:00
|
|
|
static constexpr bool IsAutoUnlink() noexcept {
|
|
|
|
return std::is_base_of_v<AutoUnlinkIntrusiveListHook, _Hook>;
|
|
|
|
}
|
|
|
|
|
2022-06-08 18:01:18 +02:00
|
|
|
static constexpr T *Cast(IntrusiveListNode *node) noexcept {
|
|
|
|
auto &hook = Hook::Cast(*node);
|
|
|
|
return &ContainerCast(hook, member);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const T *Cast(const IntrusiveListNode *node) noexcept {
|
|
|
|
const auto &hook = Hook::Cast(*node);
|
|
|
|
return &ContainerCast(hook, member);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr auto &ToHook(T &t) noexcept {
|
|
|
|
return t.*member;
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const auto &ToHook(const T &t) noexcept {
|
|
|
|
return t.*member;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-06-09 09:47:48 +02:00
|
|
|
/**
|
|
|
|
* @param constant_time_size make size() constant-time by caching the
|
|
|
|
* number of items in a field?
|
|
|
|
*/
|
|
|
|
template<typename T,
|
|
|
|
typename HookTraits=IntrusiveListBaseHookTraits<T>,
|
|
|
|
bool constant_time_size=false>
|
2022-06-08 10:31:09 +02:00
|
|
|
class IntrusiveList {
|
|
|
|
template<typename U>
|
|
|
|
using Hook = typename IntrusiveListHookDetection<U>::type;
|
|
|
|
|
|
|
|
IntrusiveListNode head{&head, &head};
|
|
|
|
|
2022-06-09 09:47:48 +02:00
|
|
|
[[no_unique_address]]
|
|
|
|
OptionalCounter<constant_time_size> counter;
|
|
|
|
|
2022-06-08 10:31:09 +02:00
|
|
|
static constexpr T *Cast(IntrusiveListNode *node) noexcept {
|
|
|
|
return HookTraits::Cast(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const T *Cast(const IntrusiveListNode *node) noexcept {
|
|
|
|
return HookTraits::Cast(node);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr auto &ToHook(T &t) noexcept {
|
|
|
|
return HookTraits::ToHook(t);
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const auto &ToHook(const T &t) noexcept {
|
|
|
|
return HookTraits::ToHook(t);
|
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
static constexpr IntrusiveListNode &ToNode(T &t) noexcept {
|
|
|
|
return ToHook(t).siblings;
|
|
|
|
}
|
|
|
|
|
|
|
|
static constexpr const IntrusiveListNode &ToNode(const T &t) noexcept {
|
|
|
|
return ToHook(t).siblings;
|
|
|
|
}
|
|
|
|
|
|
|
|
public:
|
2022-11-11 16:51:49 +01:00
|
|
|
using value_type = T;
|
|
|
|
using reference = T &;
|
|
|
|
using const_reference = const T &;
|
|
|
|
using pointer = T *;
|
|
|
|
using const_pointer = const T *;
|
2022-06-09 09:47:56 +02:00
|
|
|
using size_type = std::size_t;
|
|
|
|
|
2021-02-24 20:16:18 +01:00
|
|
|
constexpr IntrusiveList() noexcept = default;
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
IntrusiveList(IntrusiveList &&src) noexcept {
|
|
|
|
if (src.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
head = src.head;
|
|
|
|
head.next->prev = &head;
|
|
|
|
head.prev->next = &head;
|
|
|
|
|
|
|
|
src.head.next = &src.head;
|
|
|
|
src.head.prev = &src.head;
|
2022-06-09 09:47:48 +02:00
|
|
|
|
|
|
|
using std::swap;
|
|
|
|
swap(counter, src.counter);
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
~IntrusiveList() noexcept {
|
2022-06-13 20:21:29 +02:00
|
|
|
if constexpr (std::is_base_of_v<SafeLinkIntrusiveListHook, T>)
|
2020-10-18 19:21:11 +02:00
|
|
|
clear();
|
|
|
|
}
|
|
|
|
|
|
|
|
IntrusiveList &operator=(IntrusiveList &&) = delete;
|
|
|
|
|
2021-10-02 16:23:08 +02:00
|
|
|
friend void swap(IntrusiveList &a, IntrusiveList &b) noexcept {
|
|
|
|
using std::swap;
|
|
|
|
|
|
|
|
if (a.empty()) {
|
|
|
|
if (b.empty())
|
|
|
|
return;
|
|
|
|
|
|
|
|
a.head = b.head;
|
|
|
|
a.head.next->prev = &a.head;
|
|
|
|
a.head.prev->next = &a.head;
|
|
|
|
|
|
|
|
b.head = {&b.head, &b.head};
|
2022-11-11 18:29:44 +01:00
|
|
|
} else if (b.empty()) {
|
|
|
|
b.head = a.head;
|
|
|
|
b.head.next->prev = &b.head;
|
|
|
|
b.head.prev->next = &b.head;
|
|
|
|
|
|
|
|
a.head = {&a.head, &a.head};
|
2021-10-02 16:23:08 +02:00
|
|
|
} else {
|
|
|
|
swap(a.head, b.head);
|
|
|
|
|
|
|
|
a.head.next->prev = &a.head;
|
|
|
|
a.head.prev->next = &a.head;
|
|
|
|
|
|
|
|
b.head.next->prev = &b.head;
|
|
|
|
b.head.prev->next = &b.head;
|
|
|
|
}
|
2022-06-09 09:47:48 +02:00
|
|
|
|
|
|
|
swap(a.counter, b.counter);
|
2021-10-02 16:23:08 +02:00
|
|
|
}
|
|
|
|
|
2020-10-18 19:21:11 +02:00
|
|
|
constexpr bool empty() const noexcept {
|
|
|
|
return head.next == &head;
|
|
|
|
}
|
|
|
|
|
2022-06-09 09:47:56 +02:00
|
|
|
constexpr size_type size() const noexcept {
|
2022-06-09 09:47:48 +02:00
|
|
|
if constexpr (constant_time_size)
|
|
|
|
return counter;
|
|
|
|
else
|
|
|
|
return std::distance(begin(), end());
|
2022-06-09 09:47:56 +02:00
|
|
|
}
|
|
|
|
|
2020-10-18 19:21:11 +02:00
|
|
|
void clear() noexcept {
|
2022-06-13 20:21:29 +02:00
|
|
|
if constexpr (std::is_base_of_v<SafeLinkIntrusiveListHook, T>) {
|
2021-02-24 20:28:47 +01:00
|
|
|
/* for SafeLinkIntrusiveListHook, we need to
|
2020-10-18 19:21:11 +02:00
|
|
|
remove each item manually, or else its
|
|
|
|
is_linked() method will not work */
|
|
|
|
while (!empty())
|
|
|
|
pop_front();
|
2022-06-09 09:47:48 +02:00
|
|
|
} else {
|
2020-10-18 19:21:11 +02:00
|
|
|
head = {&head, &head};
|
2022-06-09 09:47:48 +02:00
|
|
|
counter.reset();
|
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename D>
|
|
|
|
void clear_and_dispose(D &&disposer) noexcept {
|
|
|
|
while (!empty()) {
|
|
|
|
auto *item = &front();
|
|
|
|
pop_front();
|
|
|
|
disposer(item);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
template<typename P, typename D>
|
|
|
|
void remove_and_dispose_if(P &&pred, D &&dispose) noexcept {
|
|
|
|
auto *n = head.next;
|
|
|
|
|
|
|
|
while (n != &head) {
|
|
|
|
auto *i = Cast(n);
|
|
|
|
n = n->next;
|
|
|
|
|
|
|
|
if (pred(*i)) {
|
2022-06-08 10:40:57 +02:00
|
|
|
ToHook(*i).unlink();
|
2022-06-09 09:47:48 +02:00
|
|
|
--counter;
|
2020-10-18 19:21:11 +02:00
|
|
|
dispose(i);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
const_reference front() const noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return *Cast(head.next);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
reference front() noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return *Cast(head.next);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop_front() noexcept {
|
2022-06-08 10:40:57 +02:00
|
|
|
ToHook(front()).unlink();
|
2022-06-09 09:47:48 +02:00
|
|
|
--counter;
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
template<typename D>
|
|
|
|
void pop_front_and_dispose(D &&disposer) noexcept {
|
|
|
|
auto &i = front();
|
2022-06-08 10:40:57 +02:00
|
|
|
ToHook(i).unlink();
|
2022-06-09 09:47:48 +02:00
|
|
|
--counter;
|
2020-10-18 19:21:11 +02:00
|
|
|
disposer(&i);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
reference back() noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return *Cast(head.prev);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop_back() noexcept {
|
2022-06-08 10:40:57 +02:00
|
|
|
ToHook(back()).unlink();
|
2022-06-09 09:47:48 +02:00
|
|
|
--counter;
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
class const_iterator;
|
|
|
|
|
2021-07-19 11:03:44 +02:00
|
|
|
class iterator final {
|
2020-10-18 19:21:11 +02:00
|
|
|
friend IntrusiveList;
|
|
|
|
friend const_iterator;
|
|
|
|
|
|
|
|
IntrusiveListNode *cursor;
|
|
|
|
|
|
|
|
constexpr iterator(IntrusiveListNode *_cursor) noexcept
|
|
|
|
:cursor(_cursor) {}
|
|
|
|
|
|
|
|
public:
|
2022-11-11 09:18:15 +01:00
|
|
|
using iterator_category = std::bidirectional_iterator_tag;
|
2021-07-19 11:03:44 +02:00
|
|
|
using value_type = T;
|
|
|
|
using difference_type = std::ptrdiff_t;
|
|
|
|
using pointer = value_type *;
|
|
|
|
using reference = value_type &;
|
|
|
|
|
2021-02-24 20:17:26 +01:00
|
|
|
iterator() noexcept = default;
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
constexpr bool operator==(const iterator &other) const noexcept {
|
|
|
|
return cursor == other.cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool operator!=(const iterator &other) const noexcept {
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
constexpr reference operator*() const noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return *Cast(cursor);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
constexpr pointer operator->() const noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return Cast(cursor);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto &operator++() noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
cursor = cursor->next;
|
|
|
|
return *this;
|
|
|
|
}
|
2022-11-11 09:18:15 +01:00
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto &operator--() noexcept {
|
2022-11-11 09:18:15 +01:00
|
|
|
cursor = cursor->prev;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto operator--(int) noexcept {
|
2022-11-11 09:18:15 +01:00
|
|
|
auto old = *this;
|
|
|
|
cursor = cursor->prev;
|
|
|
|
return old;
|
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
constexpr iterator begin() noexcept {
|
|
|
|
return {head.next};
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr iterator end() noexcept {
|
|
|
|
return {&head};
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
static constexpr iterator iterator_to(reference t) noexcept {
|
2021-02-25 14:10:51 +01:00
|
|
|
return {&ToNode(t)};
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
2021-07-19 11:03:44 +02:00
|
|
|
class const_iterator final {
|
2020-10-18 19:21:11 +02:00
|
|
|
friend IntrusiveList;
|
|
|
|
|
|
|
|
const IntrusiveListNode *cursor;
|
|
|
|
|
|
|
|
constexpr const_iterator(const IntrusiveListNode *_cursor) noexcept
|
|
|
|
:cursor(_cursor) {}
|
|
|
|
|
|
|
|
public:
|
2022-11-11 09:18:15 +01:00
|
|
|
using iterator_category = std::bidirectional_iterator_tag;
|
2021-07-19 11:03:44 +02:00
|
|
|
using value_type = const T;
|
|
|
|
using difference_type = std::ptrdiff_t;
|
|
|
|
using pointer = value_type *;
|
|
|
|
using reference = value_type &;
|
|
|
|
|
2021-02-24 20:17:26 +01:00
|
|
|
const_iterator() noexcept = default;
|
2020-10-18 19:21:11 +02:00
|
|
|
|
|
|
|
const_iterator(iterator src) noexcept
|
|
|
|
:cursor(src.cursor) {}
|
|
|
|
|
|
|
|
constexpr bool operator==(const const_iterator &other) const noexcept {
|
|
|
|
return cursor == other.cursor;
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr bool operator!=(const const_iterator &other) const noexcept {
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
constexpr reference operator*() const noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return *Cast(cursor);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
constexpr pointer operator->() const noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
return Cast(cursor);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto &operator++() noexcept {
|
2020-10-18 19:21:11 +02:00
|
|
|
cursor = cursor->next;
|
|
|
|
return *this;
|
|
|
|
}
|
2022-11-11 09:18:15 +01:00
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto &operator--() noexcept {
|
2022-11-11 09:18:15 +01:00
|
|
|
cursor = cursor->prev;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:32:54 +01:00
|
|
|
auto operator--(int) noexcept {
|
2022-11-11 09:18:15 +01:00
|
|
|
auto old = *this;
|
|
|
|
cursor = cursor->prev;
|
|
|
|
return old;
|
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
constexpr const_iterator begin() const noexcept {
|
|
|
|
return {head.next};
|
|
|
|
}
|
|
|
|
|
|
|
|
constexpr const_iterator end() const noexcept {
|
|
|
|
return {&head};
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
static constexpr const_iterator iterator_to(const_reference t) noexcept {
|
2021-10-13 11:04:17 +02:00
|
|
|
return {&ToNode(t)};
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
2021-02-25 14:00:26 +01:00
|
|
|
iterator erase(iterator i) noexcept {
|
|
|
|
auto result = std::next(i);
|
2021-02-25 14:12:09 +01:00
|
|
|
ToHook(*i).unlink();
|
2022-06-09 09:47:48 +02:00
|
|
|
--counter;
|
2021-02-25 14:00:26 +01:00
|
|
|
return result;
|
2021-02-17 20:19:19 +01:00
|
|
|
}
|
|
|
|
|
2021-02-25 13:38:39 +01:00
|
|
|
template<typename D>
|
|
|
|
iterator erase_and_dispose(iterator i, D &&disposer) noexcept {
|
|
|
|
auto result = erase(i);
|
|
|
|
disposer(&*i);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
void push_front(reference t) noexcept {
|
2021-02-17 20:14:29 +01:00
|
|
|
insert(begin(), t);
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
void push_back(reference t) noexcept {
|
2021-02-17 20:14:29 +01:00
|
|
|
insert(end(), t);
|
|
|
|
}
|
|
|
|
|
2022-11-11 16:51:49 +01:00
|
|
|
void insert(iterator p, reference t) noexcept {
|
2022-06-09 09:47:48 +02:00
|
|
|
static_assert(!constant_time_size ||
|
|
|
|
!HookTraits::IsAutoUnlink(),
|
|
|
|
"Can't use auto-unlink hooks with constant_time_size");
|
|
|
|
|
2021-02-17 20:14:29 +01:00
|
|
|
auto &existing_node = ToNode(*p);
|
2020-10-18 19:21:11 +02:00
|
|
|
auto &new_node = ToNode(t);
|
2021-02-17 20:14:29 +01:00
|
|
|
|
2022-11-10 16:46:19 +01:00
|
|
|
IntrusiveListNode::Connect(*existing_node.prev,
|
|
|
|
new_node);
|
|
|
|
IntrusiveListNode::Connect(new_node, existing_node);
|
2022-06-09 09:47:48 +02:00
|
|
|
|
|
|
|
++counter;
|
2020-10-18 19:21:11 +02:00
|
|
|
}
|
2022-11-10 15:57:12 +01:00
|
|
|
|
2022-11-11 17:37:31 +01:00
|
|
|
/**
|
|
|
|
* Move one item of the given list to this one before the
|
|
|
|
* given position.
|
|
|
|
*/
|
|
|
|
void splice(iterator position,
|
|
|
|
IntrusiveList &from, iterator i) noexcept {
|
|
|
|
auto &item = *i;
|
|
|
|
from.erase(i);
|
|
|
|
insert(position, item);
|
|
|
|
}
|
|
|
|
|
2022-11-10 15:57:12 +01:00
|
|
|
/**
|
|
|
|
* Move the given range of items of the given list to this one
|
|
|
|
* before the given position.
|
|
|
|
*/
|
|
|
|
void splice(iterator position, IntrusiveList &from,
|
|
|
|
iterator _begin, iterator _end, size_type n) noexcept {
|
|
|
|
if (_begin == _end)
|
|
|
|
return;
|
|
|
|
|
|
|
|
auto &next_node = ToNode(*position);
|
|
|
|
auto &prev_node = ToNode(*std::prev(position));
|
|
|
|
|
|
|
|
auto &first_node = ToNode(*_begin);
|
|
|
|
auto &before_first_node = ToNode(*std::prev(_begin));
|
|
|
|
auto &last_node = ToNode(*std::prev(_end));
|
|
|
|
auto &after_last_node = ToNode(*_end);
|
|
|
|
|
|
|
|
/* remove from the other list */
|
|
|
|
IntrusiveListNode::Connect(before_first_node, after_last_node);
|
|
|
|
from.counter -= n;
|
|
|
|
|
|
|
|
/* insert into this list */
|
|
|
|
IntrusiveListNode::Connect(prev_node, first_node);
|
|
|
|
IntrusiveListNode::Connect(last_node, next_node);
|
|
|
|
counter += n;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Move all items of the given list to this one before the
|
|
|
|
* given position.
|
|
|
|
*/
|
|
|
|
void splice(iterator position, IntrusiveList &from) noexcept {
|
2022-11-11 10:11:41 +01:00
|
|
|
splice(position, from, from.begin(), from.end(),
|
2022-11-11 17:15:15 +01:00
|
|
|
constant_time_size ? from.size() : 1);
|
2022-11-10 15:57:12 +01:00
|
|
|
}
|
2020-10-18 19:21:11 +02:00
|
|
|
};
|