/* This file is part of TON Blockchain Library. TON Blockchain Library is free software: you can redistribute it and/or modify it under the terms of the GNU Lesser General Public License as published by the Free Software Foundation, either version 2 of the License, or (at your option) any later version. TON Blockchain Library 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 Lesser General Public License for more details. You should have received a copy of the GNU Lesser General Public License along with TON Blockchain Library. If not, see . Copyright 2017-2020 Telegram Systems LLP */ #pragma once #include "td/utils/common.h" namespace td { struct ListNode { ListNode *next; ListNode *prev; ListNode() { clear(); } ~ListNode() { remove(); } ListNode(const ListNode &) = delete; ListNode &operator=(const ListNode &) = delete; ListNode(ListNode &&other) { if (other.empty()) { clear(); } else { init_from(std::move(other)); } } ListNode &operator=(ListNode &&other) { if (this == &other) { return *this; } this->remove(); if (!other.empty()) { init_from(std::move(other)); } return *this; } void connect(ListNode *to) { CHECK(to != nullptr); next = to; to->prev = this; } void remove() { prev->connect(next); clear(); } void put(ListNode *other) { DCHECK(other->empty()); put_unsafe(other); } void put_back(ListNode *other) { DCHECK(other->empty()); prev->connect(other); other->connect(this); } ListNode *get() { ListNode *result = prev; if (result == this) { return nullptr; } result->prev->connect(this); result->clear(); // this->connect(result->next); return result; } bool empty() const { return next == this; } ListNode *begin() { return next; } ListNode *end() { return this; } ListNode *get_next() { return next; } ListNode *get_prev() { return prev; } protected: void clear() { next = this; prev = this; } void init_from(ListNode &&other) { ListNode *head = other.prev; other.remove(); head->put_unsafe(this); } void put_unsafe(ListNode *other) { other->connect(next); this->connect(other); } }; } // namespace td