lean2/src/util/list.h
2014-10-17 17:08:39 -07:00

210 lines
7.4 KiB
C++

/*
Copyright (c) 2013 Microsoft Corporation. All rights reserved.
Released under Apache 2.0 license as described in the file LICENSE.
Author: Leonardo de Moura
*/
#pragma once
#include <iostream>
#include <iterator>
#include "util/rc.h"
#include "util/debug.h"
#include "util/optional.h"
#include "util/memory_pool.h"
namespace lean {
/**
\brief Basic list template.
*/
template<typename T>
class list {
public:
class cell {
MK_LEAN_RC()
T m_head;
list m_tail;
template<typename... Fields>
cell(bool, list const & t, Fields&&... head):m_rc(1), m_head(head...), m_tail(t) {}
public:
cell(T const & h, list const & t):m_rc(1), m_head(h), m_tail(t) {}
~cell() {}
T const & head() const { return m_head; }
list const & tail() const { return m_tail; }
void dealloc();
};
private:
static memory_pool & get_allocator() {
LEAN_THREAD_PTR(memory_pool, g_allocator);
if (!g_allocator)
g_allocator = allocate_thread_memory_pool(sizeof(cell));
return *g_allocator;
}
cell * m_ptr;
cell * steal_ptr() { cell * r = m_ptr; m_ptr = nullptr; return r; }
public:
list():m_ptr(nullptr) {}
list(T const & h, list const & t):m_ptr(new (get_allocator().allocate()) cell(h, t)) {}
list(T const & h):m_ptr(new (get_allocator().allocate()) cell(h, list())) {}
list(list const & s):m_ptr(s.m_ptr) { if (m_ptr) m_ptr->inc_ref(); }
list(list&& s):m_ptr(s.m_ptr) { s.m_ptr = nullptr; }
list(std::initializer_list<T> const & l):list() {
auto it = l.end();
while (it != l.begin()) {
--it;
*this = list(*it, *this);
}
}
explicit list(cell * c):m_ptr(c) { if (m_ptr) m_ptr->inc_ref(); }
~list() {
if (m_ptr && m_ptr->dec_ref_core()) {
cell * it = m_ptr;
while (true) {
lean_assert(it);
lean_assert(it->get_rc() == 0);
cell * next = it->m_tail.steal_ptr();
it->dealloc();
if (next && next->dec_ref_core()) {
it = next;
} else {
break;
}
}
}
}
list & operator=(list const & s) { LEAN_COPY_REF(s); }
list & operator=(list && s) { LEAN_MOVE_REF(s); }
/**
\brief Return internal representation.
This method is useful when we know the list is not going to be deleted and
we want to save a temporary reference to it. Use raw() prevents us from updating the
reference counters.
\warning Use it with care. The main risk of storing references to cell is that
the list may be deleted.
*/
cell * raw() const { return m_ptr; }
/** \brief Return true iff it is not the empty/nil list. */
explicit operator bool() const { return m_ptr != nullptr; }
friend bool is_nil(list const & l) { return l.m_ptr == nullptr; }
friend bool empty(list const & l) { return is_nil(l); }
friend T const & head(list const & l) { lean_assert(!is_nil(l)); return l.m_ptr->m_head; }
friend list const & tail(list const & l) { lean_assert(!is_nil(l)); return l.m_ptr->m_tail; }
/** \brief Pointer equality. Return true iff \c l1 and \c l2 point to the same memory location. */
friend bool is_eqp(list const & l1, list const & l2) { return l1.m_ptr == l2.m_ptr; }
friend bool is_eqp(list const & l1, cell const * l2) { return l1.m_ptr == l2; }
template<typename... Args>
void emplace_front(Args&&... args) {
cell * new_ptr = new (get_allocator().allocate()) cell(true, *this, args...);
if (m_ptr) m_ptr->dec_ref();
m_ptr = new_ptr;
}
/** \brief Structural equality. */
friend bool operator==(list const & l1, list const & l2) {
cell * it1 = l1.m_ptr;
cell * it2 = l2.m_ptr;
while (it1 != nullptr && it2 != nullptr) {
if (it1 == it2)
return true;
if (it1->m_head != it2->m_head)
return false;
it1 = it1->m_tail.m_ptr;
it2 = it2->m_tail.m_ptr;
}
return it1 == nullptr && it2 == nullptr;
}
friend bool operator!=(list const & l1, list const & l2) { return !(l1 == l2); }
/** \brief List iterator object. */
class iterator {
friend class list;
cell const * m_it;
iterator(cell const * it):m_it(it) {}
public:
typedef std::forward_iterator_tag iterator_category;
typedef T value_type;
typedef unsigned difference_type;
typedef T const * pointer;
typedef T const & reference;
iterator(iterator const & s):m_it(s.m_it) {}
iterator & operator++() { m_it = m_it->m_tail.m_ptr; return *this; }
iterator operator++(int) { iterator tmp(*this); operator++(); return tmp; }
bool operator==(iterator const & s) const { return m_it == s.m_it; }
bool operator!=(iterator const & s) const { return !operator==(s); }
T const & operator*() { lean_assert(m_it); return m_it->m_head; }
T const * operator->() { lean_assert(m_it); return &(m_it->m_head); }
};
iterator begin() const { return iterator(m_ptr); }
iterator end() const { return iterator(nullptr); }
};
template<typename T>
void list<T>::cell::dealloc() {
this->~cell();
get_allocator().recycle(this);
}
template<typename T> inline list<T> cons(T const & h, list<T> const & t) { return list<T>(h, t); }
template<typename T> inline T const & car(list<T> const & l) { return head(l); }
template<typename T> inline list<T> const & cdr(list<T> const & l) { return tail(l); }
template<typename T> inline list<T> to_list(T const & v) { return list<T>(v); }
template<typename T> inline list<T> to_list(optional<T> const & v) { return v ? to_list(*v) : list<T>(); }
template<typename T> inline list<T> to_list(T const * v) { return v ? to_list(*v) : list<T>(); }
template<typename T> inline list<T> ptr_to_list(list<T> const * l) { return l ? *l : list<T>(); }
template<typename T> inline std::ostream & operator<<(std::ostream & out, list<T> const & l) {
out << "(";
bool first = true;
list<T> const * ptr = &l;
while (*ptr) {
if (first)
first = false;
else
out << " ";
out << head(*ptr);
ptr = &tail(*ptr);
}
out << ")";
return out;
}
template<typename T> unsigned length(list<T> const & l) {
unsigned r = 0;
list<T> const * it = &l;
while (*it) {
r++;
it = &tail(*it);
}
return r;
}
/** \brief Return a list containing the elements in the subrange <tt>[begin, end)</tt>. */
template<typename It> list<typename std::iterator_traits<It>::value_type>
to_list(It const & begin, It const & end,
list<typename std::iterator_traits<It>::value_type> const & ls = list<typename std::iterator_traits<It>::value_type>()) {
list<typename std::iterator_traits<It>::value_type> r = ls;
auto it = end;
while (it != begin) {
--it;
r = cons(*it, r);
}
return r;
}
/** \brief Return <tt>reverse(to_list(begin, end))</tt> */
template<typename It> list<typename std::iterator_traits<It>::value_type> reverse_to_list(It const & begin, It const & end) {
list<typename std::iterator_traits<It>::value_type> r;
for (auto it = begin; it != end; ++it)
r = cons(*it, r);
return r;
}
}