2015-11-23 04:35:40 +08:00
|
|
|
#pragma once
|
2015-06-23 21:17:10 +08:00
|
|
|
|
|
|
|
#include <algorithm>
|
2015-11-23 04:35:40 +08:00
|
|
|
#include <memory>
|
2015-12-08 04:50:07 +08:00
|
|
|
#include <cassert>
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
#include "threading/sync/lockable.hpp"
|
|
|
|
#include "threading/sync/spinlock.hpp"
|
|
|
|
|
|
|
|
#include "utils/random/fast_binomial.hpp"
|
2015-12-08 04:50:07 +08:00
|
|
|
#include "memory/lazy_gc.hpp"
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-06-28 17:43:52 +08:00
|
|
|
// concurrent skiplist based on the implementation described in
|
|
|
|
// "A Provably Correct Scalable Concurrent Skip List"
|
|
|
|
// https://www.cs.tau.ac.il/~shanir/nir-pubs-web/Papers/OPODIS2006-BA.pdf
|
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
template <class K, class T, size_t H=32, class lock_t=SpinLock>
|
2015-12-08 04:50:07 +08:00
|
|
|
class SkipList : LazyGC<SkipList<K, T, H, lock_t>>, Lockable<lock_t>
|
2015-06-23 21:17:10 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
public:
|
2015-11-23 04:35:40 +08:00
|
|
|
static thread_local FastBinomial<H> rnd;
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
using data_t = std::pair<const K, T>;
|
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
struct Flags
|
2015-06-28 17:43:52 +08:00
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
enum node_flags : uint8_t {
|
|
|
|
MARKED = 0x01,
|
|
|
|
FULLY_LINKED = 0x10,
|
|
|
|
};
|
|
|
|
|
|
|
|
bool is_marked() const
|
2015-06-28 17:43:52 +08:00
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
return flags.load(std::memory_order_acquire) & MARKED;
|
2015-06-28 17:43:52 +08:00
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
void set_marked()
|
|
|
|
{
|
|
|
|
flags.fetch_or(MARKED, std::memory_order_release);
|
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
bool is_fully_linked() const
|
|
|
|
{
|
|
|
|
return flags.load(std::memory_order_acquire) & FULLY_LINKED;
|
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
void set_fully_linked()
|
|
|
|
{
|
|
|
|
flags.fetch_or(FULLY_LINKED, std::memory_order_release);
|
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
private:
|
|
|
|
std::atomic<uint8_t> flags {0};
|
|
|
|
};
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
class Node : Lockable<lock_t>
|
2015-06-23 21:17:10 +08:00
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
public:
|
2015-12-08 04:50:07 +08:00
|
|
|
friend class SkipList;
|
2015-11-23 04:35:40 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
data_t data;
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
const uint8_t height;
|
|
|
|
Flags flags;
|
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
static Node* create(const K& key, const T& item, uint8_t height)
|
|
|
|
{
|
|
|
|
return create({key, item}, height);
|
|
|
|
}
|
|
|
|
|
|
|
|
static Node* create(data_t&& data, uint8_t height)
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
|
|
|
// [ Node ][Node*][Node*][Node*]...[Node*]
|
|
|
|
// | | | | |
|
|
|
|
// | 0 1 2 height-1
|
|
|
|
// |----------------||-----------------------------|
|
|
|
|
// space for Node space for tower pointers
|
|
|
|
// structure right after the Node
|
|
|
|
// structure
|
|
|
|
auto size = sizeof(Node) + height * sizeof(std::atomic<Node*>);
|
|
|
|
auto node = static_cast<Node*>(std::malloc(size));
|
|
|
|
|
|
|
|
// we have raw memory and we need to construct an object
|
|
|
|
// of type Node on it
|
2015-12-08 04:50:07 +08:00
|
|
|
return new (node) Node(std::move(data), height);
|
2015-06-28 17:43:52 +08:00
|
|
|
}
|
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
static void destroy(Node* node)
|
|
|
|
{
|
|
|
|
node->~Node();
|
|
|
|
free(node);
|
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
Node* forward(size_t level) const
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
return tower[level].load(std::memory_order_acquire);
|
|
|
|
}
|
2015-11-23 04:35:40 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
void forward(size_t level, Node* next)
|
|
|
|
{
|
|
|
|
tower[level].store(next, std::memory_order_release);
|
2015-11-23 04:35:40 +08:00
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
private:
|
2015-12-08 04:50:07 +08:00
|
|
|
Node(data_t data, uint8_t height)
|
|
|
|
: data(data), height(height)
|
2015-06-28 17:43:52 +08:00
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
// here we assume, that the memory for N towers (N = height) has
|
|
|
|
// been allocated right after the Node structure so we need to
|
|
|
|
// initialize that memory
|
|
|
|
for(auto i = 0; i < height; ++i)
|
|
|
|
new (&tower[i]) std::atomic<Node*> {nullptr};
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
~Node()
|
|
|
|
{
|
|
|
|
for(auto i = 0; i < height; ++i)
|
|
|
|
tower[i].~atomic();
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
// this creates an array of the size zero. we can't put any sensible
|
|
|
|
// value here since we don't know what size it will be untill the
|
|
|
|
// node is allocated. we could make it a Node** but then we would
|
|
|
|
// have two memory allocations, one for node and one for the forward
|
|
|
|
// list. this way we avoid expensive malloc/free calls and also cache
|
|
|
|
// thrashing when following a pointer on the heap
|
|
|
|
std::atomic<Node*> tower[0];
|
|
|
|
};
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
public:
|
2015-12-08 04:50:07 +08:00
|
|
|
class ConstIterator
|
|
|
|
{
|
|
|
|
friend class SkipList;
|
|
|
|
ConstIterator(Node* node) : node(node) {}
|
|
|
|
|
|
|
|
public:
|
|
|
|
ConstIterator() = default;
|
|
|
|
ConstIterator(const ConstIterator&) = default;
|
|
|
|
|
|
|
|
const data_t& operator*()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
return node->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
const data_t* operator->()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
return &node->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
operator const data_t()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
return node->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator& operator++()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
node = node->forward(0);
|
|
|
|
return *this;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator& operator++(int)
|
|
|
|
{
|
|
|
|
return operator++();
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator==(const ConstIterator& a, const ConstIterator& b)
|
|
|
|
{
|
|
|
|
return a.node == b.node;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator!=(const ConstIterator& a, const ConstIterator& b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
Node* node {nullptr};
|
|
|
|
};
|
|
|
|
|
|
|
|
class Iterator
|
2015-06-23 21:17:10 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
friend class SkipList;
|
|
|
|
Iterator(Node* node) : node(node) {}
|
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
public:
|
2015-12-08 04:50:07 +08:00
|
|
|
Iterator() = default;
|
|
|
|
Iterator(const Iterator&) = default;
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
data_t& operator*()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
return node->data;
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
data_t* operator->()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
|
|
|
return &node->data;
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
operator data_t()
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
assert(node != nullptr);
|
|
|
|
return node->data;
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator& operator++()
|
|
|
|
{
|
|
|
|
assert(node != nullptr);
|
2015-11-23 04:35:40 +08:00
|
|
|
node = node->forward(0);
|
|
|
|
return *this;
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
Iterator& operator++(int)
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
|
|
|
return operator++();
|
2015-06-28 17:43:52 +08:00
|
|
|
}
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
friend bool operator==(const Iterator& a, const Iterator& b)
|
|
|
|
{
|
|
|
|
return a.node == b.node;
|
|
|
|
}
|
|
|
|
|
|
|
|
friend bool operator!=(const Iterator& a, const Iterator& b)
|
|
|
|
{
|
|
|
|
return !(a == b);
|
|
|
|
}
|
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
private:
|
2015-12-08 04:50:07 +08:00
|
|
|
Node* node {nullptr};
|
2015-11-23 04:35:40 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
SkipList() : header(Node::create(K(), T(), H)) {}
|
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
friend class Accessor;
|
|
|
|
|
|
|
|
class Accessor
|
|
|
|
{
|
|
|
|
friend class SkipList;
|
|
|
|
|
|
|
|
Accessor(SkipList& skiplist) : skiplist(skiplist) {}
|
|
|
|
public:
|
|
|
|
Accessor(const Accessor&) = delete;
|
|
|
|
Accessor(Accessor&&) = default;
|
|
|
|
|
|
|
|
Iterator begin()
|
|
|
|
{
|
|
|
|
return skiplist.begin();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator begin() const
|
|
|
|
{
|
|
|
|
return skiplist.cbegin();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator cbegin() const
|
|
|
|
{
|
|
|
|
return skiplist.cbegin();
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator end()
|
|
|
|
{
|
|
|
|
return skiplist.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator end() const
|
|
|
|
{
|
|
|
|
return skiplist.cend();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator cend() const
|
|
|
|
{
|
|
|
|
return skiplist.cend();
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<Iterator, bool> insert_unique(const K& key, const T& item)
|
|
|
|
{
|
|
|
|
return skiplist.insert({key, item}, preds, succs);
|
|
|
|
}
|
|
|
|
|
|
|
|
std::pair<Iterator, bool> insert_unique(const K& key, T&& item)
|
|
|
|
{
|
|
|
|
return skiplist.insert({key, std::move(item)}, preds, succs);
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator find(const K& key) const
|
|
|
|
{
|
|
|
|
return skiplist.find(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator find(const K& key)
|
|
|
|
{
|
|
|
|
return skiplist.find(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
bool remove(const K& key)
|
|
|
|
{
|
|
|
|
return skiplist.remove(key, preds, succs);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
SkipList& skiplist;
|
|
|
|
Node* preds[H], *succs[H];
|
|
|
|
};
|
|
|
|
|
|
|
|
Accessor access()
|
|
|
|
{
|
|
|
|
return Accessor(*this);
|
|
|
|
}
|
|
|
|
|
|
|
|
private:
|
|
|
|
using guard_t = std::unique_lock<lock_t>;
|
|
|
|
|
|
|
|
Iterator begin()
|
|
|
|
{
|
|
|
|
return Iterator(header->forward(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator begin() const
|
|
|
|
{
|
|
|
|
return ConstIterator(header->forward(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator cbegin() const
|
|
|
|
{
|
|
|
|
return ConstIterator(header->forward(0));
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator end()
|
|
|
|
{
|
|
|
|
return Iterator();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator end() const
|
|
|
|
{
|
|
|
|
return ConstIterator();
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator cend() const
|
|
|
|
{
|
|
|
|
return ConstIterator();
|
|
|
|
}
|
2015-11-23 04:35:40 +08:00
|
|
|
|
|
|
|
size_t size() const
|
2015-06-28 17:43:52 +08:00
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
return count.load(std::memory_order_acquire);
|
2015-06-28 17:43:52 +08:00
|
|
|
}
|
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
bool greater(const K& key, const Node* const node)
|
2015-06-28 17:43:52 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
return node && key > node->data.first;
|
|
|
|
}
|
2015-11-23 04:35:40 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
bool less(const K& key, const Node* const node)
|
|
|
|
{
|
|
|
|
return (node == nullptr) || key < node->data.first;
|
|
|
|
}
|
|
|
|
|
|
|
|
ConstIterator find(const K& key) const
|
|
|
|
{
|
|
|
|
return find_node<ConstIterator>(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
Iterator find(const K& key)
|
|
|
|
{
|
|
|
|
return find_node<Iterator>(key);
|
|
|
|
}
|
|
|
|
|
|
|
|
template <class It>
|
|
|
|
It find_node(const K& key)
|
|
|
|
{
|
|
|
|
Node* node, *pred = header;
|
|
|
|
int h = static_cast<int>(pred->height) - 1;
|
2015-06-28 17:43:52 +08:00
|
|
|
|
|
|
|
while(true)
|
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
// try to descend down first the next key on this layer overshoots
|
|
|
|
for(; h >= 0 && less(key, node = pred->forward(h)); --h) {}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
// if we overshoot at every layer, item doesn't exist
|
|
|
|
if(h < 0)
|
2015-12-08 04:50:07 +08:00
|
|
|
return It();
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
// the item is farther to the right, continue going right as long
|
|
|
|
// as the key is greater than the current node's key
|
|
|
|
while(greater(key, node))
|
|
|
|
pred = node, node = node->forward(h);
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-11-23 04:35:40 +08:00
|
|
|
// check if we have a hit. if not, we need to descend down again
|
2015-12-08 04:50:07 +08:00
|
|
|
if(!less(key, node) && !node->flags.is_marked())
|
|
|
|
return It(node);
|
2015-11-23 04:35:40 +08:00
|
|
|
}
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
int find_path(Node* from, int start, const K& key,
|
|
|
|
Node* preds[], Node* succs[])
|
|
|
|
{
|
|
|
|
int level_found = -1;
|
|
|
|
Node* pred = from;
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
for(int level = start; level >= 0; --level)
|
|
|
|
{
|
|
|
|
Node* node = pred->forward(level);
|
|
|
|
|
|
|
|
while(greater(key, node))
|
|
|
|
pred = node, node = pred->forward(level);
|
|
|
|
|
|
|
|
if(level_found == -1 && !less(key, node))
|
|
|
|
level_found = level;
|
|
|
|
|
|
|
|
preds[level] = pred;
|
|
|
|
succs[level] = node;
|
|
|
|
}
|
|
|
|
|
|
|
|
return level_found;
|
|
|
|
}
|
|
|
|
|
|
|
|
template <bool ADDING>
|
|
|
|
bool lock_nodes(uint8_t height, guard_t guards[],
|
|
|
|
Node* preds[], Node* succs[])
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
Node *prepred, *pred, *succ = nullptr;
|
|
|
|
bool valid = true;
|
|
|
|
|
|
|
|
for(int level = 0; valid && level < height; ++level)
|
|
|
|
{
|
|
|
|
pred = preds[level], succ = succs[level];
|
|
|
|
|
|
|
|
if(pred != prepred)
|
|
|
|
guards[level] = pred->acquire_unique(), prepred = pred;
|
|
|
|
|
|
|
|
valid = !pred->flags.is_marked() && pred->forward(level) == succ;
|
|
|
|
|
|
|
|
if(ADDING)
|
|
|
|
valid = valid && (succ == nullptr || !succ->flags.is_marked());
|
|
|
|
}
|
|
|
|
|
|
|
|
return valid;
|
2015-11-23 04:35:40 +08:00
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
std::pair<Iterator, bool>
|
|
|
|
insert(data_t&& data, Node* preds[], Node* succs[])
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
while(true)
|
|
|
|
{
|
|
|
|
auto level = find_path(header, H - 1, data.first, preds, succs);
|
|
|
|
|
|
|
|
if(level != -1)
|
|
|
|
{
|
|
|
|
auto found = succs[level];
|
|
|
|
|
|
|
|
if(found->flags.is_marked())
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while(!found->flags.is_fully_linked())
|
|
|
|
usleep(250);
|
|
|
|
|
|
|
|
return {Iterator {succs[level]}, false};
|
|
|
|
}
|
|
|
|
|
|
|
|
auto height = rnd();
|
|
|
|
guard_t guards[H];
|
|
|
|
|
|
|
|
// try to acquire the locks for predecessors up to the height of
|
|
|
|
// the new node. release the locks and try again if someone else
|
|
|
|
// has the locks
|
|
|
|
if(!lock_nodes<true>(height, guards, preds, succs))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
// you have the locks, create a new node
|
|
|
|
auto new_node = Node::create(std::move(data), height);
|
|
|
|
|
|
|
|
// link the predecessors and successors, e.g.
|
|
|
|
//
|
|
|
|
// 4 HEAD ... P ------------------------> S ... NULL
|
|
|
|
// 3 HEAD ... ... P -----> NEW ---------> S ... NULL
|
|
|
|
// 2 HEAD ... ... P -----> NEW -----> S ... ... NULL
|
|
|
|
// 1 HEAD ... ... ... P -> NEW -> S ... ... ... NULL
|
|
|
|
for(uint8_t level = 0; level < height; ++level)
|
|
|
|
{
|
|
|
|
new_node->forward(level, succs[level]);
|
|
|
|
preds[level]->forward(level, new_node);
|
|
|
|
}
|
|
|
|
|
|
|
|
new_node->flags.set_fully_linked();
|
|
|
|
count.fetch_add(1, std::memory_order_relaxed);
|
|
|
|
|
|
|
|
return {Iterator {new_node}, true};
|
|
|
|
}
|
2015-11-23 04:35:40 +08:00
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
bool ok_delete(Node* node, int level)
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
return node->flags.is_fully_linked()
|
|
|
|
&& node->height - 1 == level
|
|
|
|
&& !node->flags.is_marked();
|
|
|
|
}
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
bool remove(const K& key, Node* preds[], Node* succs[])
|
|
|
|
{
|
|
|
|
Node* node = nullptr;
|
|
|
|
guard_t node_guard;
|
|
|
|
bool marked = false;
|
|
|
|
int height = 0;
|
|
|
|
|
|
|
|
while(true)
|
2015-11-23 04:35:40 +08:00
|
|
|
{
|
2015-12-08 04:50:07 +08:00
|
|
|
auto level = find_path(header, H - 1, key, preds, succs);
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
if(!marked && (level == -1 || !ok_delete(succs[level], level)))
|
|
|
|
return false;
|
2015-06-28 17:43:52 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
if(!marked)
|
|
|
|
{
|
|
|
|
node = succs[level];
|
|
|
|
height = node->height;
|
|
|
|
node_guard = node->acquire_unique();
|
2015-06-25 09:06:00 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
if(node->flags.is_marked())
|
|
|
|
return false;
|
2015-06-23 21:17:10 +08:00
|
|
|
|
2015-12-08 04:50:07 +08:00
|
|
|
node->flags.set_marked();
|
|
|
|
}
|
|
|
|
|
|
|
|
guard_t guards[H];
|
|
|
|
|
|
|
|
if(!lock_nodes<false>(height, guards, preds, succs))
|
|
|
|
continue;
|
|
|
|
|
|
|
|
for(int level = height - 1; level >= 0; --level)
|
|
|
|
preds[level]->forward(level, node->forward(level));
|
|
|
|
|
|
|
|
// TODO recycle(node)
|
|
|
|
count.fetch_sub(1, std::memory_order_relaxed);
|
|
|
|
return true;
|
2015-11-23 04:35:40 +08:00
|
|
|
}
|
2015-12-08 04:50:07 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
guard_t gc_lock_acquire()
|
|
|
|
{
|
|
|
|
return this->acquire_unique();
|
|
|
|
}
|
|
|
|
|
|
|
|
void vacuum()
|
|
|
|
{
|
2015-11-23 04:35:40 +08:00
|
|
|
|
|
|
|
}
|
2015-12-08 04:50:07 +08:00
|
|
|
|
|
|
|
std::atomic<size_t> count {0};
|
|
|
|
Node* header;
|
2015-11-23 04:35:40 +08:00
|
|
|
};
|
2015-12-08 04:50:07 +08:00
|
|
|
|
|
|
|
template <class K, class T, size_t H, class lock_t>
|
|
|
|
thread_local FastBinomial<H> SkipList<K, T, H, lock_t>::rnd;
|
|
|
|
|