blob: e6dd6a03127f1240d97dc3bb1e912ba4ae20f83f [file] [log] [blame]
/* -*- Mode:C++; c-file-style:"gnu"; indent-tabs-mode:nil; -*- */
/**
* Copyright (c) 2014, Regents of the University of California.
*
* This file is part of NDN repo-ng (Next generation of NDN repository).
* See AUTHORS.md for complete list of repo-ng authors and contributors.
*
* repo-ng 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,
* either version 3 of the License, or (at your option) any later version.
*
* repo-ng 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
* repo-ng, e.g., in COPYING.md file. If not, see <http://www.gnu.org/licenses/>.
*/
#ifndef REPO_STORAGE_SKIPLIST_HPP
#define REPO_STORAGE_SKIPLIST_HPP
#include "common.hpp"
namespace repo {
class SkipList32Levels25Probabilty
{
public:
static size_t
getMaxLevels()
{
return 32;
}
static double
getProbability()
{
return 0.25; // 25%
}
};
template<typename T>
struct SkipListNode
{
typedef SkipListNode<T>* SkipListNodePointer;
T data;
std::vector<SkipListNodePointer> prevs;
std::vector<SkipListNodePointer> nexts;
};
template<typename T, class Ref, class Ptr>
class SkipListIterator : public std::iterator<std::bidirectional_iterator_tag, T>
{
public:
typedef SkipListNode<T>* NodePointer;
NodePointer node;
typedef SkipListIterator<T, Ref, Ptr> Self;
typedef T value_type;
typedef Ptr pointer;
typedef Ref reference;
typedef ptrdiff_t difference_type;
typedef SkipListIterator<T, const T&, const T*> const_iterator;
/// alias of const_iterator
typedef const_iterator iterator;
public:
// constructor
SkipListIterator()
{
}
explicit
SkipListIterator(NodePointer x)
: node(x)
{
}
SkipListIterator(const SkipListIterator& x)
: node(x.node)
{
}
bool
operator==(const const_iterator& x) const
{
return node == x.node;
}
bool
operator!=(const const_iterator& x) const
{
return node != x.node;
}
reference
operator*() const
{
return (*node).data;
}
pointer
operator->() const
{
return &(operator*());
}
Self&
operator++()
{
node = node->nexts[0];
return *this;
}
Self
operator++(int)
{
Self tmp = *this;
++*this;
return tmp;
}
Self&
operator--()
{
node = node->prevs[0];
return *this;
}
Self
operator--(int)
{
Self tmp = *this;
--*this;
return tmp;
}
};
/*
* @brief SkipList
*
* Examples of internal structure:
* "A <-i-> B" means A->nexts[i] = B and B->prevs[i] = A
*
* case 1: an empty skip list
* m_head <-0-> m_head
*
* case 2: a skip list with only one level and only one node
* m_head <-0-> A <-0-> m_head
*
* case 3: a skip list with three nodes, node A and B appear on one level,
* node C appears on two levels
* m_head <-1-> C <-1-> m_head
* m_head <-0-> A <-0-> B <-0-> C <-0-> m_head
*/
template<typename T, typename Compare = std::less<T>,
typename Traits = SkipList32Levels25Probabilty>
class SkipList
{
public:
typedef T value_type;
typedef value_type* pointer;
typedef const value_type* const_pointer;
typedef value_type& reference;
typedef const value_type& const_reference;
typedef SkipListNode<T> Node;
typedef Node* NodePointer;
typedef size_t size_type;
typedef ptrdiff_t difference_type;
typedef SkipListIterator<T, const T&, const T*> const_iterator;
/// alias of const_iterator
typedef const_iterator iterator;
public:
explicit
SkipList()
{
initializeHead();
}
~SkipList()
{
clear();
deallocateNode(m_head);
}
const_iterator
begin() const
{
return const_iterator(*(*m_head).nexts.begin());
}
const_iterator
end() const
{
return const_iterator(m_head);
}
bool
empty() const
{
return *(m_head->nexts.begin()) == m_head;
}
size_t
size() const
{
return m_size;
}
const_iterator
lower_bound(const T& x) const;
const_iterator
find(const T& x) const;
std::pair<const_iterator, bool>
insert(const T& x);
const_iterator
erase(const_iterator it);
protected:
/*
* @brief allocate memory for node
*/
NodePointer
allocateNode()
{
return m_skiplistAllocator.allocate(sizeof(Node));
}
/*
* @brief deallocate memory of node
*/
void
deallocateNode(NodePointer p)
{
m_skiplistAllocator.deallocate(p, sizeof(Node));
}
/*
* @brief initialize the node
*/
NodePointer
createNode()
{
NodePointer p = allocateNode();
Node node;
m_skiplistAllocator.construct(p, node);
return p;
}
/*
* @brief initialize the node with given value
* @para to be set to the value of node
*/
NodePointer
createNode(const T& x)
{
NodePointer p = allocateNode();
Node node;
m_skiplistAllocator.construct(p, node);
m_dataAllocator.construct(&(p->data), x);
return p;
}
/*
* @brief destructror of the node
* @para given pointer of node to be destructed
*/
void
destroyNode(NodePointer p)
{
m_skiplistAllocator.destroy(p);
deallocateNode(p);
}
/*
* @brief initialize the head
*/
void
initializeHead()
{
m_head = createNode();
m_head->prevs.push_back(m_head);
m_head->nexts.push_back(m_head);
m_size = 0;
}
/*
* @brief destroy all the nodes of skiplist except the head
*/
void
clear()
{
NodePointer cur = m_head->nexts[0];
while (cur != m_head) {
NodePointer tmp = cur;
cur = cur->nexts[0];
destroyNode(tmp);
}
m_head->nexts[0] = m_head;
m_head->prevs[0] = m_head;
}
/*
* @brief pick a random height for inserted skiplist entry
*/
size_t
pickRandomLevel() const
{
static boost::random::mt19937 gen;
static boost::random::geometric_distribution<size_t> dist(Traits::getProbability());
return std::min(dist(gen), Traits::getMaxLevels());
}
protected:
NodePointer m_head;
std::allocator<Node> m_skiplistAllocator;
std::allocator<T> m_dataAllocator;
Compare m_compare;
size_t m_size;
};
template<typename T, typename Compare, typename Traits>
typename SkipList<T, Compare, Traits>::const_iterator
SkipList<T, Compare, Traits>::lower_bound(const T& x) const
{
size_t nLevels = m_head->nexts.size();
NodePointer p = m_head;
NodePointer q = p->nexts[nLevels - 1];
for (int i = nLevels - 1; i >= 0; --i) {
q = p->nexts[i];
if (q != m_head) {
while (m_compare(q->data, x)) {
p = p->nexts[i];
q = p->nexts[i];
if (q == m_head) {
break;
}
}
}
}
return const_iterator(q);
}
template<typename T, typename Compare, typename Traits>
typename SkipList<T, Compare, Traits>::const_iterator
SkipList<T, Compare, Traits>::find(const T& x) const
{
const_iterator it = this->lower_bound(x);
if (it == this->end() || *it != x)
return this->end();
return it;
}
template<typename T, typename Compare, typename Traits>
std::pair<typename SkipList<T, Compare, Traits>::const_iterator, bool>
SkipList<T, Compare, Traits>::insert(const T& x)
{
size_t nLevels = m_head->nexts.size();
// 1. find insert position
std::vector<NodePointer> insertPositions(nLevels);
NodePointer p = m_head;
NodePointer q = p->nexts[nLevels - 1];
for (int i = nLevels - 1; i >= 0; --i) {
q = p->nexts[i];
if (q != m_head) {
while (m_compare(q->data, x)) {
p = p->nexts[i];
q = p->nexts[i];
if (q == m_head) {
break;
}
}
}
insertPositions[i] = p;
}
// 2. whether q->data == x?
if (q != m_head)
if (!m_compare(q->data, x) && !m_compare(x, q->data)) {
return std::pair<const_iterator, bool>(const_iterator(q), false);
}
// 3. construct new node;
NodePointer newNode = createNode(x);
// 4. pick random nLevels
size_t newLevel = pickRandomLevel();
// 5. insert the new node
newNode->nexts.resize(newLevel + 1);
newNode->prevs.resize(newLevel + 1);
if (newLevel > nLevels - 1) {
m_head->nexts.resize(newLevel + 1, m_head);
m_head->prevs.resize(newLevel + 1, m_head);
insertPositions.resize(newLevel + 1, m_head);
}
for (size_t i = 0; i <= newLevel; i++) {
newNode->nexts[i] = insertPositions[i]->nexts[i];
newNode->prevs[i] = insertPositions[i];
insertPositions[i]->nexts[i] = newNode;
newNode->nexts[i]->prevs[i] = newNode;
}
++m_size;
return std::pair<const_iterator, bool>(const_iterator(newNode), true);
}
template<typename T, typename Compare, typename Traits>
typename SkipList<T, Compare, Traits>::const_iterator
SkipList<T, Compare, Traits>::erase(typename SkipList<T, Compare, Traits>::const_iterator it)
{
NodePointer eraseNode = it.node;
if (!empty() && eraseNode != m_head) {
NodePointer returnNode = eraseNode->nexts[0];
size_t nLevels = eraseNode->nexts.size();
for (int i = nLevels - 1; i >= 0; --i) {
eraseNode->nexts[i]->prevs[i] = eraseNode->prevs[i];
eraseNode->prevs[i]->nexts[i] = eraseNode->nexts[i];
// clear empty nLevels
if ((eraseNode->nexts[i] == eraseNode->prevs[i]) && i > 0) {
m_head->nexts.pop_back();
m_head->prevs.pop_back();
}
}
destroyNode(eraseNode);
--m_size;
return const_iterator(returnNode);
}
else {
return end();
}
}
} // namespace repo
#endif // REPO_STORAGE_SKIPLIST_HPP