mirror of
https://github.com/lightningnetwork/lnd.git
synced 2025-01-19 05:45:21 +01:00
303 lines
8.0 KiB
Go
303 lines
8.0 KiB
Go
// Copyright (c) 2009 The Go Authors. All rights reserved.
|
|
// Copyright (c) 2024 Lightning Labs and the Lightning Network Developers
|
|
|
|
// 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.
|
|
// * Neither the name of Google Inc. nor the names of its
|
|
// contributors may be used to endorse or promote products derived from
|
|
// this software without specific prior written permission.
|
|
|
|
// 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 COPYRIGHT
|
|
// OWNER 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.
|
|
package fn
|
|
|
|
type Node[A any] struct {
|
|
// prev is a pointer to the previous node in the List.
|
|
prev *Node[A]
|
|
|
|
// next is a pointer to the next node in the List.
|
|
next *Node[A]
|
|
|
|
// list is the root pointer to the List in which this node is located.
|
|
list *List[A]
|
|
|
|
// Value is the actual data contained within the Node.
|
|
Value A
|
|
}
|
|
|
|
// Next returns the next list node or nil.
|
|
func (e *Node[A]) Next() *Node[A] {
|
|
if e.list == nil {
|
|
return nil
|
|
}
|
|
|
|
if e.next == &e.list.root {
|
|
return nil
|
|
}
|
|
|
|
return e.next
|
|
}
|
|
|
|
// Prev returns the previous list node or nil.
|
|
func (e *Node[A]) Prev() *Node[A] {
|
|
if e.list == nil {
|
|
return nil
|
|
}
|
|
|
|
if e.prev == &e.list.root {
|
|
return nil
|
|
}
|
|
|
|
return e.prev
|
|
}
|
|
|
|
// List represents a doubly linked list.
|
|
// The zero value for List is an empty list ready to use.
|
|
type List[A any] struct {
|
|
// root is a sentinal Node to identify the head and tail of the list.
|
|
// root.prev is the tail, root.next is the head. For the purposes of
|
|
// elegance, the absence of a next or prev node is encoded as the
|
|
// address of the root node.
|
|
root Node[A]
|
|
|
|
// len is the current length of the list.
|
|
len int
|
|
}
|
|
|
|
// Init intializes or clears the List l.
|
|
func (l *List[A]) Init() *List[A] {
|
|
l.root.next = &l.root
|
|
l.root.prev = &l.root
|
|
l.len = 0
|
|
|
|
return l
|
|
}
|
|
|
|
// lazyInit lazily initializes a zero List value. It is called by other public
|
|
// functions that could feasibly be called on a List that was initialized by the
|
|
// raw List[A]{} constructor.
|
|
func (l *List[A]) lazyInit() {
|
|
if l.root.next == nil {
|
|
l.Init()
|
|
}
|
|
}
|
|
|
|
// insert inserts n after predecessor, increments l.len, and returns n.
|
|
func (l *List[A]) insert(n *Node[A], predecessor *Node[A]) *Node[A] {
|
|
// Make n point to correct neighborhood.
|
|
n.prev = predecessor
|
|
n.next = predecessor.next
|
|
|
|
// Make neighborhood point to n.
|
|
n.prev.next = n
|
|
n.next.prev = n
|
|
|
|
// Make n part of the list.
|
|
n.list = l
|
|
|
|
// Increment list length.
|
|
l.len++
|
|
|
|
return n
|
|
}
|
|
|
|
// insertVal is a convenience wrapper for
|
|
// insert(&Node[A]{Value: v}, predecessor).
|
|
func (l *List[A]) insertVal(a A, predecessor *Node[A]) *Node[A] {
|
|
return l.insert(&Node[A]{Value: a}, predecessor)
|
|
}
|
|
|
|
// move removes n from its current position and inserts it as the successor to
|
|
// predecessor.
|
|
func (l *List[A]) move(n *Node[A], predecessor *Node[A]) {
|
|
if n == predecessor {
|
|
return // Can't move after itself.
|
|
}
|
|
|
|
if predecessor.next == n {
|
|
return // Nothing to be done.
|
|
}
|
|
|
|
// Bind previous and next to each other.
|
|
n.prev.next = n.next
|
|
n.next.prev = n.prev
|
|
|
|
// Make n point to new neighborhood.
|
|
n.prev = predecessor
|
|
n.next = predecessor.next
|
|
|
|
// Make new neighborhood point to n.
|
|
n.prev.next = n
|
|
n.next.prev = n
|
|
}
|
|
|
|
// New returns an initialized List.
|
|
func NewList[A any]() *List[A] {
|
|
l := List[A]{}
|
|
return l.Init()
|
|
}
|
|
|
|
// Len returns the number of elements of List l.
|
|
// The complexity is O(1).
|
|
func (l *List[A]) Len() int {
|
|
return l.len
|
|
}
|
|
|
|
// Front returns the first Node of List l or nil if the list is empty.
|
|
func (l *List[A]) Front() *Node[A] {
|
|
if l.len == 0 {
|
|
return nil
|
|
}
|
|
|
|
return l.root.next
|
|
}
|
|
|
|
// Back returns the last Node of List l or nil if the list is empty.
|
|
func (l *List[A]) Back() *Node[A] {
|
|
if l.len == 0 {
|
|
return nil
|
|
}
|
|
|
|
return l.root.prev
|
|
}
|
|
|
|
// Remove removes Node n from List l if n is an element of List l.
|
|
// It returns the Node value e.Value.
|
|
// The Node must not be nil.
|
|
func (l *List[A]) Remove(n *Node[A]) A {
|
|
if n.list == l {
|
|
n.prev.next = n.next
|
|
n.next.prev = n.prev
|
|
l.len--
|
|
|
|
v := n.Value
|
|
// Set all node data to nil to prevent dangling references.
|
|
*n = Node[A]{Value: v}
|
|
|
|
return v
|
|
}
|
|
|
|
return n.Value
|
|
}
|
|
|
|
// PushFront inserts a new Node n with value a at the front of List l and
|
|
// returns n.
|
|
func (l *List[A]) PushFront(a A) *Node[A] {
|
|
l.lazyInit()
|
|
return l.insertVal(a, &l.root)
|
|
}
|
|
|
|
// PushBack inserts a new Node n with value a at the back of List l and returns
|
|
// n.
|
|
func (l *List[A]) PushBack(a A) *Node[A] {
|
|
l.lazyInit()
|
|
return l.insertVal(a, l.root.prev)
|
|
}
|
|
|
|
// InsertBefore inserts a new Node n with value a immediately before successor
|
|
// and returns n. If successor is not an element of l, the list is not
|
|
// modified. The successor must not be nil.
|
|
func (l *List[A]) InsertBefore(a A, successor *Node[A]) *Node[A] {
|
|
if successor == nil {
|
|
return l.insertVal(a, &l.root)
|
|
}
|
|
|
|
if successor.list != l {
|
|
return nil
|
|
}
|
|
|
|
return l.insertVal(a, successor.prev)
|
|
}
|
|
|
|
// InsertAfter inserts a new Node n with value a immediately after and returns
|
|
// e. If predecessor is not an element of l, the list is not modified. The
|
|
// predecessor must not be nil.
|
|
func (l *List[A]) InsertAfter(a A, predecessor *Node[A]) *Node[A] {
|
|
if predecessor == nil {
|
|
return l.insertVal(a, l.root.prev)
|
|
}
|
|
|
|
if predecessor.list != l {
|
|
return nil
|
|
}
|
|
|
|
return l.insertVal(a, predecessor)
|
|
}
|
|
|
|
// MoveToFront moves Node n to the front of List l.
|
|
// If n is not an element of l, the list is not modified.
|
|
// The Node must not be nil.
|
|
func (l *List[A]) MoveToFront(n *Node[A]) {
|
|
if n.list == l {
|
|
l.move(n, &l.root)
|
|
}
|
|
}
|
|
|
|
// MoveToBack moves Node n to the back of List l.
|
|
// If n is not an element of l, the list is not modified.
|
|
// The Node must not be nil.
|
|
func (l *List[A]) MoveToBack(n *Node[A]) {
|
|
if n.list == l {
|
|
l.move(n, l.root.prev)
|
|
}
|
|
}
|
|
|
|
// MoveBefore moves Node n to its new position before successor.
|
|
// If n or successor is not an element of l, or n == successor, the list is not
|
|
// modified. The Node and successor must not be nil.
|
|
func (l *List[A]) MoveBefore(n, successor *Node[A]) {
|
|
if n.list == l && successor.list == l {
|
|
l.move(n, successor.prev)
|
|
}
|
|
}
|
|
|
|
// MoveAfter moves Node n to its new position after predecessor.
|
|
// If n or predecessor is not an element of l, or n == predecessor, the list is
|
|
// not modified. The Node and predecessor must not be nil.
|
|
func (l *List[A]) MoveAfter(n, predecessor *Node[A]) {
|
|
if n.list == l && predecessor.list == l {
|
|
l.move(n, predecessor)
|
|
}
|
|
}
|
|
|
|
// PushBackList inserts a copy of List other at the back of List l.
|
|
// The Lists l and other may be the same. They must not be nil.
|
|
func (l *List[A]) PushBackList(other *List[A]) {
|
|
l.lazyInit()
|
|
n := other.Front()
|
|
sz := other.Len()
|
|
for i := 0; i < sz; i++ {
|
|
l.insertVal(n.Value, l.root.prev)
|
|
n = n.Next()
|
|
}
|
|
}
|
|
|
|
// PushFrontList inserts a copy of List other at the front of List l.
|
|
// The Lists l and other may be the same. They must not be nil.
|
|
func (l *List[A]) PushFrontList(other *List[A]) {
|
|
l.lazyInit()
|
|
n := other.Back()
|
|
sz := other.Len()
|
|
for i := 0; i < sz; i++ {
|
|
l.insertVal(n.Value, &l.root)
|
|
n = n.Prev()
|
|
}
|
|
}
|