2022-06-02 09:29:35 -04:00
|
|
|
/*
|
|
|
|
* SPDX-FileCopyrightText: 2015-2022 Espressif Systems (Shanghai) CO LTD
|
|
|
|
*
|
|
|
|
* SPDX-License-Identifier: Apache-2.0
|
|
|
|
*/
|
2016-08-17 11:08:22 -04:00
|
|
|
#ifndef intrusive_list_h
|
|
|
|
#define intrusive_list_h
|
|
|
|
|
|
|
|
#include <cassert>
|
2018-07-02 07:10:43 -04:00
|
|
|
#include <unordered_map>
|
2022-06-02 09:29:35 -04:00
|
|
|
#include <cstddef>
|
2016-08-17 11:08:22 -04:00
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class intrusive_list;
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class intrusive_list_node
|
|
|
|
{
|
|
|
|
protected:
|
|
|
|
friend class intrusive_list<T>;
|
|
|
|
T* mPrev = nullptr;
|
|
|
|
T* mNext = nullptr;
|
|
|
|
};
|
|
|
|
|
|
|
|
template <typename T>
|
|
|
|
class intrusive_list
|
|
|
|
{
|
|
|
|
typedef intrusive_list_node<T> TNode;
|
|
|
|
static_assert(std::is_base_of<TNode, T>::value, "");
|
|
|
|
|
|
|
|
public:
|
|
|
|
|
2022-08-15 17:12:20 -04:00
|
|
|
class iterator
|
2016-08-17 11:08:22 -04:00
|
|
|
{
|
|
|
|
public:
|
2022-08-15 17:12:20 -04:00
|
|
|
using iterator_category = std::forward_iterator_tag;
|
|
|
|
using value_type = T;
|
|
|
|
using difference_type = ptrdiff_t;
|
|
|
|
using pointer = T*;
|
|
|
|
using reference = T&;
|
2016-08-17 11:08:22 -04:00
|
|
|
|
|
|
|
iterator() : mPos(nullptr) {}
|
|
|
|
|
|
|
|
iterator(T* pos) : mPos(pos) {}
|
|
|
|
|
|
|
|
iterator operator++(int)
|
|
|
|
{
|
|
|
|
auto result = *this;
|
|
|
|
mPos = mPos->mNext;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator operator--(int)
|
|
|
|
{
|
|
|
|
auto result = *this;
|
|
|
|
mPos = mPos->mPrev;
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator& operator++()
|
|
|
|
{
|
|
|
|
mPos = mPos->mNext;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator& operator--()
|
|
|
|
{
|
|
|
|
mPos = mPos->mPrev;
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool operator==(const iterator& other) const
|
|
|
|
{
|
|
|
|
return mPos == other.mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool operator!=(const iterator& other) const
|
|
|
|
{
|
|
|
|
return !(*this == other);
|
|
|
|
}
|
|
|
|
|
|
|
|
T& operator*()
|
|
|
|
{
|
|
|
|
return *mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
const T& operator*() const
|
|
|
|
{
|
|
|
|
return *mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
T* operator->()
|
|
|
|
{
|
|
|
|
return mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
const T* operator->() const
|
|
|
|
{
|
|
|
|
return mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
operator T*()
|
|
|
|
{
|
|
|
|
return mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
operator const T*() const
|
|
|
|
{
|
|
|
|
return mPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
protected:
|
|
|
|
T* mPos;
|
|
|
|
};
|
|
|
|
|
|
|
|
void push_back(T* node)
|
|
|
|
{
|
|
|
|
if (mLast) {
|
|
|
|
mLast->mNext = node;
|
|
|
|
}
|
|
|
|
node->mPrev = mLast;
|
|
|
|
node->mNext = nullptr;
|
|
|
|
mLast = node;
|
|
|
|
if (mFirst == nullptr) {
|
|
|
|
mFirst = node;
|
|
|
|
}
|
|
|
|
++mSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
void push_front(T* node)
|
|
|
|
{
|
|
|
|
node->mPrev = nullptr;
|
|
|
|
node->mNext = mFirst;
|
|
|
|
if (mFirst) {
|
|
|
|
mFirst->mPrev = node;
|
|
|
|
}
|
|
|
|
mFirst = node;
|
|
|
|
if (mLast == nullptr) {
|
|
|
|
mLast = node;
|
|
|
|
}
|
|
|
|
++mSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
T& back()
|
|
|
|
{
|
|
|
|
return *mLast;
|
|
|
|
}
|
|
|
|
|
|
|
|
const T& back() const
|
|
|
|
{
|
|
|
|
return *mLast;
|
|
|
|
}
|
|
|
|
|
|
|
|
T& front()
|
|
|
|
{
|
|
|
|
return *mFirst;
|
|
|
|
}
|
|
|
|
|
|
|
|
const T& front() const
|
|
|
|
{
|
|
|
|
return *mFirst;
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop_front()
|
|
|
|
{
|
|
|
|
erase(mFirst);
|
|
|
|
}
|
|
|
|
|
|
|
|
void pop_back()
|
|
|
|
{
|
|
|
|
erase(mLast);
|
|
|
|
}
|
|
|
|
|
|
|
|
void insert(iterator next, T* node)
|
|
|
|
{
|
|
|
|
if (static_cast<T*>(next) == nullptr) {
|
|
|
|
push_back(node);
|
|
|
|
} else {
|
|
|
|
auto prev = next->mPrev;
|
|
|
|
if (!prev) {
|
|
|
|
push_front(node);
|
|
|
|
} else {
|
|
|
|
prev->mNext = node;
|
|
|
|
next->mPrev = node;
|
|
|
|
node->mNext = next;
|
|
|
|
node->mPrev = &(*prev);
|
|
|
|
++mSize;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void erase(iterator it)
|
|
|
|
{
|
|
|
|
auto prev = it->mPrev;
|
|
|
|
auto next = it->mNext;
|
|
|
|
|
|
|
|
if (prev) {
|
|
|
|
prev->mNext = next;
|
|
|
|
} else {
|
|
|
|
mFirst = next;
|
|
|
|
}
|
|
|
|
if (next) {
|
|
|
|
next->mPrev = prev;
|
|
|
|
} else {
|
|
|
|
mLast = prev;
|
|
|
|
}
|
|
|
|
--mSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator begin()
|
|
|
|
{
|
|
|
|
return iterator(mFirst);
|
|
|
|
}
|
|
|
|
|
|
|
|
iterator end()
|
|
|
|
{
|
|
|
|
return iterator(nullptr);
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t size() const
|
|
|
|
{
|
|
|
|
return mSize;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool empty() const
|
|
|
|
{
|
|
|
|
return mSize == 0;
|
|
|
|
}
|
2018-04-02 06:44:59 -04:00
|
|
|
|
2016-08-17 11:08:22 -04:00
|
|
|
void clear()
|
|
|
|
{
|
|
|
|
while (mFirst) {
|
|
|
|
erase(mFirst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-02 06:44:59 -04:00
|
|
|
void clearAndFreeNodes()
|
|
|
|
{
|
|
|
|
while (mFirst) {
|
|
|
|
auto tmp = mFirst;
|
|
|
|
erase(mFirst);
|
|
|
|
delete tmp;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-08-17 11:08:22 -04:00
|
|
|
protected:
|
|
|
|
T* mFirst = nullptr;
|
|
|
|
T* mLast = nullptr;
|
|
|
|
size_t mSize = 0;
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
#endif /* intrusive_list_h */
|