mirror of
https://github.com/inspircd/inspircd.git
synced 2025-03-08 17:19:01 -05:00
171 lines
3.1 KiB
C++
171 lines
3.1 KiB
C++
/*
|
|
* InspIRCd -- Internet Relay Chat Daemon
|
|
*
|
|
* Copyright (C) 2020, 2022 Sadie Powell <sadie@witchery.services>
|
|
* Copyright (C) 2014 Attila Molnar <attilamolnar@hush.com>
|
|
*
|
|
* This file is part of InspIRCd. InspIRCd is free software: you can
|
|
* redistribute it and/or modify it under the terms of the GNU General Public
|
|
* License as published by the Free Software Foundation, version 2.
|
|
*
|
|
* This program is distributed in the hope that it will be useful, but WITHOUT
|
|
* ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
|
|
* FOR A PARTICULAR PURPOSE. See the GNU General Public License for more
|
|
* details.
|
|
*
|
|
* You should have received a copy of the GNU General Public License
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
|
*/
|
|
|
|
|
|
namespace insp
|
|
{
|
|
|
|
template <typename T, typename Tag>
|
|
class INSPIRCD_INTRUSIVE_LIST_NAME final
|
|
{
|
|
public:
|
|
class iterator final
|
|
{
|
|
T* curr;
|
|
|
|
public:
|
|
using difference_type = std::ptrdiff_t;
|
|
using iterator_category = std::bidirectional_iterator_tag;
|
|
using pointer = T**;
|
|
using reference = T*&;
|
|
using value_type = T*;
|
|
|
|
iterator(T* i = nullptr)
|
|
: curr(i)
|
|
{
|
|
}
|
|
|
|
iterator& operator++()
|
|
{
|
|
curr = curr->intrusive_list_node<T, Tag>::ptr_next;
|
|
return *this;
|
|
}
|
|
|
|
iterator operator++(int)
|
|
{
|
|
iterator ret(*this);
|
|
operator++();
|
|
return ret;
|
|
}
|
|
|
|
iterator& operator--()
|
|
{
|
|
curr = curr->intrusive_list_node<T, Tag>::ptr_prev;
|
|
return *this;
|
|
}
|
|
|
|
iterator operator--(int)
|
|
{
|
|
iterator ret(*this);
|
|
operator--();
|
|
return ret;
|
|
}
|
|
|
|
bool operator==(const iterator& other) const { return (curr == other.curr); }
|
|
bool operator!=(const iterator& other) const { return (curr != other.curr); }
|
|
T* operator*() const { return curr; }
|
|
};
|
|
|
|
typedef iterator const_iterator;
|
|
|
|
bool empty() const
|
|
{
|
|
return (size() == 0);
|
|
}
|
|
|
|
size_t size() const
|
|
{
|
|
return listsize;
|
|
}
|
|
|
|
iterator begin() const
|
|
{
|
|
return iterator(listhead);
|
|
}
|
|
|
|
iterator end() const
|
|
{
|
|
return iterator();
|
|
}
|
|
|
|
void pop_front()
|
|
{
|
|
erase(listhead);
|
|
}
|
|
|
|
T* front() const
|
|
{
|
|
return listhead;
|
|
}
|
|
|
|
void push_front(T* x)
|
|
{
|
|
if (listsize++)
|
|
{
|
|
x->intrusive_list_node<T, Tag>::ptr_next = listhead;
|
|
listhead->intrusive_list_node<T, Tag>::ptr_prev = x;
|
|
}
|
|
#ifdef INSPIRCD_INTRUSIVE_LIST_HAS_TAIL
|
|
else
|
|
listtail = x;
|
|
#endif
|
|
listhead = x;
|
|
}
|
|
|
|
#ifdef INSPIRCD_INTRUSIVE_LIST_HAS_TAIL
|
|
T* back() const
|
|
{
|
|
return listtail;
|
|
}
|
|
|
|
void push_back(T* x)
|
|
{
|
|
if (listsize++)
|
|
{
|
|
x->intrusive_list_node<T, Tag>::ptr_prev = listtail;
|
|
listtail->intrusive_list_node<T, Tag>::ptr_next = x;
|
|
}
|
|
else
|
|
listhead = x;
|
|
listtail = x;
|
|
}
|
|
|
|
void pop_back()
|
|
{
|
|
erase(listtail);
|
|
}
|
|
#endif
|
|
|
|
void erase(const iterator& it)
|
|
{
|
|
erase(*it);
|
|
}
|
|
|
|
void erase(T* x)
|
|
{
|
|
if (listhead == x)
|
|
listhead = x->intrusive_list_node<T, Tag>::ptr_next;
|
|
#ifdef INSPIRCD_INTRUSIVE_LIST_HAS_TAIL
|
|
if (listtail == x)
|
|
listtail = x->intrusive_list_node<T, Tag>::ptr_prev;
|
|
#endif
|
|
x->intrusive_list_node<T, Tag>::unlink();
|
|
listsize--;
|
|
}
|
|
|
|
private:
|
|
T* listhead = nullptr;
|
|
#ifdef INSPIRCD_INTRUSIVE_LIST_HAS_TAIL
|
|
T* listtail = nullptr;
|
|
#endif
|
|
size_t listsize = 0;
|
|
};
|
|
|
|
} // namespace insp
|