#ifndef ROCKSDB_LITE
#include "util/hash_skiplist_rep.h"
#include "rocksdb/memtablerep.h"
#include "util/arena.h"
#include "rocksdb/slice.h"
#include "rocksdb/slice_transform.h"
#include "port/port.h"
#include "port/atomic_pointer.h"
#include "util/murmurhash.h"
#include "db/memtable.h"
#include "db/skiplist.h"
namespace
rocksdb {
namespace
{
class
HashSkipListRep :
public
MemTableRep {
public
:
HashSkipListRep(
const
MemTableRep::KeyComparator& compare, Arena* arena,
const
SliceTransform* transform,
size_t
bucket_size,
int32_t skiplist_height, int32_t skiplist_branching_factor);
virtual
void
Insert(KeyHandle handle) override;
virtual
bool
Contains(
const
char
* key)
const
override;
virtual
size_t
ApproximateMemoryUsage() override;
virtual
void
Get(
const
LookupKey& k,
void
* callback_args,
bool
(*callback_func)(
void
* arg,
const
char
* entry)) override;
virtual
~HashSkipListRep();
virtual
MemTableRep::Iterator* GetIterator(Arena* arena =
nullptr
) override;
virtual
MemTableRep::Iterator* GetDynamicPrefixIterator(
Arena* arena =
nullptr
) override;
private
:
friend
class
DynamicIterator;
typedef
SkipList<
const
char
*,
const
MemTableRep::KeyComparator&> Bucket;
size_t
bucket_size_;
const
int32_t skiplist_height_;
const
int32_t skiplist_branching_factor_;
port::AtomicPointer* buckets_;
const
SliceTransform* transform_;
const
MemTableRep::KeyComparator& compare_;
Arena*
const
arena_;
inline
size_t
GetHash(
const
Slice& slice)
const
{
return
MurmurHash(slice.data(), slice.size(), 0) % bucket_size_;
}
inline
Bucket* GetBucket(
size_t
i)
const
{
return
static_cast
<Bucket*>(buckets_[i].Acquire_Load());
}
inline
Bucket* GetBucket(
const
Slice& slice)
const
{
return
GetBucket(GetHash(slice));
}
Bucket* GetInitializedBucket(
const
Slice& transformed);
class
Iterator :
public
MemTableRep::Iterator {
public
:
explicit
Iterator(Bucket* list,
bool
own_list =
true
,
Arena* arena =
nullptr
)
: list_(list), iter_(list), own_list_(own_list), arena_(arena) {}
virtual
~Iterator() {
if
(own_list_) {
assert
(list_ !=
nullptr
);
delete
list_;
}
}
virtual
bool
Valid()
const
{
return
list_ !=
nullptr
&& iter_.Valid();
}
virtual
const
char
* key()
const
{
assert
(Valid());
return
iter_.key();
}
virtual
void
Next() {
assert
(Valid());
iter_.Next();
}
virtual
void
Prev() {
assert
(Valid());
iter_.Prev();
}
virtual
void
Seek(
const
Slice& internal_key,
const
char
* memtable_key) {
if
(list_ !=
nullptr
) {
const
char
* encoded_key =
(memtable_key !=
nullptr
) ?
memtable_key : EncodeKey(&tmp_, internal_key);
iter_.Seek(encoded_key);
}
}
virtual
void
SeekToFirst() {
if
(list_ !=
nullptr
) {
iter_.SeekToFirst();
}
}
virtual
void
SeekToLast() {
if
(list_ !=
nullptr
) {
iter_.SeekToLast();
}
}
protected
:
void
Reset(Bucket* list) {
if
(own_list_) {
assert
(list_ !=
nullptr
);
delete
list_;
}
list_ = list;
iter_.SetList(list);
own_list_ =
false
;
}
private
:
Bucket* list_;
Bucket::Iterator iter_;
bool
own_list_;
std::unique_ptr<Arena> arena_;
std::string tmp_;
};
class
DynamicIterator :
public
HashSkipListRep::Iterator {
public
:
explicit
DynamicIterator(
const
HashSkipListRep& memtable_rep)
: HashSkipListRep::Iterator(
nullptr
,
false
),
memtable_rep_(memtable_rep) {}
virtual
void
Seek(
const
Slice& k,
const
char
* memtable_key) {
auto
transformed = memtable_rep_.transform_->Transform(ExtractUserKey(k));
Reset(memtable_rep_.GetBucket(transformed));
HashSkipListRep::Iterator::Seek(k, memtable_key);
}
virtual
void
SeekToFirst() {
Reset(
nullptr
);
}
virtual
void
SeekToLast() {
Reset(
nullptr
);
}
private
:
const
HashSkipListRep& memtable_rep_;
};
class
EmptyIterator :
public
MemTableRep::Iterator {
public
:
EmptyIterator() { }
virtual
bool
Valid()
const
{
return
false
;
}
virtual
const
char
* key()
const
{
assert
(
false
);
return
nullptr
;
}
virtual
void
Next() { }
virtual
void
Prev() { }
virtual
void
Seek(
const
Slice& internal_key,
const
char
* memtable_key) { }
virtual
void
SeekToFirst() { }
virtual
void
SeekToLast() { }
private
:
};
};
HashSkipListRep::HashSkipListRep(
const
MemTableRep::KeyComparator& compare,
Arena* arena,
const
SliceTransform* transform,
size_t
bucket_size, int32_t skiplist_height,
int32_t skiplist_branching_factor)
: MemTableRep(arena),
bucket_size_(bucket_size),
skiplist_height_(skiplist_height),
skiplist_branching_factor_(skiplist_branching_factor),
transform_(transform),
compare_(compare),
arena_(arena) {
auto
mem =
arena->AllocateAligned(
sizeof
(port::AtomicPointer) * bucket_size);
buckets_ =
new
(mem) port::AtomicPointer[bucket_size];
for
(
size_t
i = 0; i < bucket_size_; ++i) {
buckets_[i].NoBarrier_Store(
nullptr
);
}
}
HashSkipListRep::~HashSkipListRep() {
}
HashSkipListRep::Bucket* HashSkipListRep::GetInitializedBucket(
const
Slice& transformed) {
size_t
hash = GetHash(transformed);
auto
bucket = GetBucket(hash);
if
(bucket ==
nullptr
) {
auto
addr = arena_->AllocateAligned(
sizeof
(Bucket));
bucket =
new
(addr) Bucket(compare_, arena_, skiplist_height_,
skiplist_branching_factor_);
buckets_[hash].Release_Store(
static_cast
<
void
*>(bucket));
}
return
bucket;
}
void
HashSkipListRep::Insert(KeyHandle handle) {
auto
* key =
static_cast
<
char
*>(handle);
assert
(!Contains(key));
auto
transformed = transform_->Transform(UserKey(key));
auto
bucket = GetInitializedBucket(transformed);
bucket->Insert(key);
}
bool
HashSkipListRep::Contains(
const
char
* key)
const
{
auto
transformed = transform_->Transform(UserKey(key));
auto
bucket = GetBucket(transformed);
if
(bucket ==
nullptr
) {
return
false
;
}
return
bucket->Contains(key);
}
size_t
HashSkipListRep::ApproximateMemoryUsage() {
return
0;
}
void
HashSkipListRep::Get(
const
LookupKey& k,
void
* callback_args,
bool
(*callback_func)(
void
* arg,
const
char
* entry)) {
auto
transformed = transform_->Transform(k.user_key());
auto
bucket = GetBucket(transformed);
if
(bucket !=
nullptr
) {
Bucket::Iterator iter(bucket);
for
(iter.Seek(k.memtable_key().data());
iter.Valid() && callback_func(callback_args, iter.key());
iter.Next()) {
}
}
}
MemTableRep::Iterator* HashSkipListRep::GetIterator(Arena* arena) {
Arena* new_arena =
new
Arena(arena_->BlockSize());
auto
list =
new
Bucket(compare_, new_arena);
for
(
size_t
i = 0; i < bucket_size_; ++i) {
auto
bucket = GetBucket(i);
if
(bucket !=
nullptr
) {
Bucket::Iterator itr(bucket);
for
(itr.SeekToFirst(); itr.Valid(); itr.Next()) {
list->Insert(itr.key());
}
}
}
if
(arena ==
nullptr
) {
return
new
Iterator(list,
true
, new_arena);
}
else
{
auto
mem = arena->AllocateAligned(
sizeof
(Iterator));
return
new
(mem) Iterator(list,
true
, new_arena);
}
}
MemTableRep::Iterator* HashSkipListRep::GetDynamicPrefixIterator(Arena* arena) {
if
(arena ==
nullptr
) {
return
new
DynamicIterator(*
this
);
}
else
{
auto
mem = arena->AllocateAligned(
sizeof
(DynamicIterator));
return
new
(mem) DynamicIterator(*
this
);
}
}
}
MemTableRep* HashSkipListRepFactory::CreateMemTableRep(
const
MemTableRep::KeyComparator& compare, Arena* arena,
const
SliceTransform* transform, Logger* logger) {
return
new
HashSkipListRep(compare, arena, transform, bucket_count_,
skiplist_height_, skiplist_branching_factor_);
}
MemTableRepFactory* NewHashSkipListRepFactory(
size_t
bucket_count, int32_t skiplist_height,
int32_t skiplist_branching_factor) {
return
new
HashSkipListRepFactory(bucket_count, skiplist_height,
skiplist_branching_factor);
}
}
#endif // ROCKSDB_LITE