Refactor cache.cc

Summary: Refactor cache.cc so that I can plugin clock cache (D55581). Mainly move `ShardedCache` to separate file, move `LRUHandle` back to cache.cc and rename it lru_cache.cc.

Test Plan:
    make check -j64

Reviewers: lightmark, sdong

Reviewed By: sdong

Subscribers: andrewkr, dhruba, leveldb

Differential Revision: https://reviews.facebook.net/D59655
main
Yi Wu 9 years ago
parent c6a8665b37
commit 4b95253587
  1. 3
      CMakeLists.txt
  2. 18
      include/rocksdb/cache.h
  3. 1
      include/rocksdb/utilities/sim_cache.h
  4. 11
      src.mk
  5. 2
      util/hash.h
  6. 259
      util/lru_cache.cc
  7. 71
      util/lru_cache_handle.h
  8. 117
      util/sharded_cache.cc
  9. 93
      util/sharded_cache.h
  10. 29
      utilities/simulator_cache/sim_cache.cc

@ -193,7 +193,6 @@ set(SOURCES
util/arena.cc util/arena.cc
util/bloom.cc util/bloom.cc
util/build_version.cc util/build_version.cc
util/cache.cc
util/coding.cc util/coding.cc
util/compaction_job_stats_impl.cc util/compaction_job_stats_impl.cc
util/comparator.cc util/comparator.cc
@ -214,6 +213,7 @@ set(SOURCES
util/histogram_windowing.cc util/histogram_windowing.cc
util/instrumented_mutex.cc util/instrumented_mutex.cc
util/iostats_context.cc util/iostats_context.cc
util/lru_cache.cc
tools/ldb_cmd.cc tools/ldb_cmd.cc
tools/ldb_tool.cc tools/ldb_tool.cc
util/logging.cc util/logging.cc
@ -229,6 +229,7 @@ set(SOURCES
util/perf_level.cc util/perf_level.cc
util/random.cc util/random.cc
util/rate_limiter.cc util/rate_limiter.cc
util/sharded_cache.cc
util/slice.cc util/slice.cc
util/statistics.cc util/statistics.cc
util/status.cc util/status.cc

@ -34,13 +34,9 @@ class Cache;
// Create a new cache with a fixed size capacity. The cache is sharded // Create a new cache with a fixed size capacity. The cache is sharded
// to 2^num_shard_bits shards, by hash of the key. The total capacity // to 2^num_shard_bits shards, by hash of the key. The total capacity
// is divided and evenly assigned to each shard. // is divided and evenly assigned to each shard.
// extern std::shared_ptr<Cache> NewLRUCache(size_t capacity,
// The parameter num_shard_bits defaults to 4, and strict_capacity_limit int num_shard_bits = 6,
// defaults to false. bool strict_capacity_limit = false);
extern std::shared_ptr<Cache> NewLRUCache(size_t capacity);
extern std::shared_ptr<Cache> NewLRUCache(size_t capacity, int num_shard_bits);
extern std::shared_ptr<Cache> NewLRUCache(size_t capacity, int num_shard_bits,
bool strict_capacity_limit);
class Cache { class Cache {
public: public:
@ -112,8 +108,8 @@ class Cache {
// capacity. // capacity.
virtual void SetStrictCapacityLimit(bool strict_capacity_limit) = 0; virtual void SetStrictCapacityLimit(bool strict_capacity_limit) = 0;
// Set whether to return error on insertion when cache reaches its full // Get the flag whether to return error on insertion when cache reaches its
// capacity. // full capacity.
virtual bool HasStrictCapacityLimit() const = 0; virtual bool HasStrictCapacityLimit() const = 0;
// returns the maximum configured capacity of the cache // returns the maximum configured capacity of the cache
@ -148,10 +144,6 @@ class Cache {
virtual void EraseUnRefEntries() = 0; virtual void EraseUnRefEntries() = 0;
private: private:
void LRU_Remove(Handle* e);
void LRU_Append(Handle* e);
void Unref(Handle* e);
// No copying allowed // No copying allowed
Cache(const Cache&); Cache(const Cache&);
Cache& operator=(const Cache&); Cache& operator=(const Cache&);

@ -11,7 +11,6 @@
#include "rocksdb/cache.h" #include "rocksdb/cache.h"
#include "rocksdb/slice.h" #include "rocksdb/slice.h"
#include "rocksdb/status.h" #include "rocksdb/status.h"
#include "util/lru_cache_handle.h"
namespace rocksdb { namespace rocksdb {

@ -88,7 +88,6 @@ LIB_SOURCES = \
util/arena.cc \ util/arena.cc \
util/bloom.cc \ util/bloom.cc \
util/build_version.cc \ util/build_version.cc \
util/cache.cc \
util/coding.cc \ util/coding.cc \
util/comparator.cc \ util/comparator.cc \
util/compaction_job_stats_impl.cc \ util/compaction_job_stats_impl.cc \
@ -100,10 +99,6 @@ LIB_SOURCES = \
util/env_chroot.cc \ util/env_chroot.cc \
util/env_hdfs.cc \ util/env_hdfs.cc \
util/env_posix.cc \ util/env_posix.cc \
util/io_posix.cc \
util/threadpool.cc \
util/transaction_test_util.cc \
util/sst_file_manager_impl.cc \
util/file_util.cc \ util/file_util.cc \
util/file_reader_writer.cc \ util/file_reader_writer.cc \
util/filter_policy.cc \ util/filter_policy.cc \
@ -112,6 +107,12 @@ LIB_SOURCES = \
util/histogram_windowing.cc \ util/histogram_windowing.cc \
util/instrumented_mutex.cc \ util/instrumented_mutex.cc \
util/iostats_context.cc \ util/iostats_context.cc \
util/io_posix.cc \
util/lru_cache.cc \
util/threadpool.cc \
util/transaction_test_util.cc \
util/sharded_cache.cc \
util/sst_file_manager_impl.cc \
utilities/backupable/backupable_db.cc \ utilities/backupable/backupable_db.cc \
utilities/convenience/info_log_finder.cc \ utilities/convenience/info_log_finder.cc \
utilities/checkpoint/checkpoint.cc \ utilities/checkpoint/checkpoint.cc \

@ -13,6 +13,8 @@
#include <stddef.h> #include <stddef.h>
#include <stdint.h> #include <stdint.h>
#include "rocksdb/slice.h"
namespace rocksdb { namespace rocksdb {
extern uint32_t Hash(const char* data, size_t n, uint32_t seed); extern uint32_t Hash(const char* data, size_t n, uint32_t seed);

@ -12,11 +12,9 @@
#include <stdlib.h> #include <stdlib.h>
#include "port/port.h" #include "port/port.h"
#include "rocksdb/cache.h"
#include "util/autovector.h" #include "util/autovector.h"
#include "util/hash.h"
#include "util/lru_cache_handle.h"
#include "util/mutexlock.h" #include "util/mutexlock.h"
#include "util/sharded_cache.h"
namespace rocksdb { namespace rocksdb {
@ -24,6 +22,60 @@ namespace {
// LRU cache implementation // LRU cache implementation
// An entry is a variable length heap-allocated structure.
// Entries are referenced by cache and/or by any external entity.
// The cache keeps all its entries in table. Some elements
// are also stored on LRU list.
//
// LRUHandle can be in these states:
// 1. Referenced externally AND in hash table.
// In that case the entry is *not* in the LRU. (refs > 1 && in_cache == true)
// 2. Not referenced externally and in hash table. In that case the entry is
// in the LRU and can be freed. (refs == 1 && in_cache == true)
// 3. Referenced externally and not in hash table. In that case the entry is
// in not on LRU and not in table. (refs >= 1 && in_cache == false)
//
// All newly created LRUHandles are in state 1. If you call
// LRUCacheShard::Release
// on entry in state 1, it will go into state 2. To move from state 1 to
// state 3, either call LRUCacheShard::Erase or LRUCacheShard::Insert with the
// same key.
// To move from state 2 to state 1, use LRUCacheShard::Lookup.
// Before destruction, make sure that no handles are in state 1. This means
// that any successful LRUCacheShard::Lookup/LRUCacheShard::Insert have a
// matching
// RUCache::Release (to move into state 2) or LRUCacheShard::Erase (for state 3)
struct LRUHandle {
void* value;
void (*deleter)(const Slice&, void* value);
LRUHandle* next_hash;
LRUHandle* next;
LRUHandle* prev;
size_t charge; // TODO(opt): Only allow uint32_t?
size_t key_length;
uint32_t refs; // a number of refs to this entry
// cache itself is counted as 1
bool in_cache; // true, if this entry is referenced by the hash table
uint32_t hash; // Hash of key(); used for fast sharding and comparisons
char key_data[1]; // Beginning of key
Slice key() const {
// For cheaper lookups, we allow a temporary Handle object
// to store a pointer to a key in "value".
if (next == this) {
return *(reinterpret_cast<Slice*>(value));
} else {
return Slice(key_data, key_length);
}
}
void Free() {
assert((refs == 1 && in_cache) || (refs == 0 && !in_cache));
(*deleter)(key(), value);
delete[] reinterpret_cast<char*>(this);
}
};
// We provide our own simple hash table since it removes a whole bunch // We provide our own simple hash table since it removes a whole bunch
// of porting hacks and is also faster than some of the built-in hash // of porting hacks and is also faster than some of the built-in hash
// table implementations in some of the compiler/runtime combinations // table implementations in some of the compiler/runtime combinations
@ -131,46 +183,47 @@ class HandleTable {
}; };
// A single shard of sharded cache. // A single shard of sharded cache.
class LRUCache { class LRUCacheShard : public CacheShard {
public: public:
LRUCache(); LRUCacheShard();
~LRUCache(); virtual ~LRUCacheShard();
// Separate from constructor so caller can easily make an array of LRUCache // Separate from constructor so caller can easily make an array of LRUCache
// if current usage is more than new capacity, the function will attempt to // if current usage is more than new capacity, the function will attempt to
// free the needed space // free the needed space
void SetCapacity(size_t capacity); virtual void SetCapacity(size_t capacity) override;
// Set the flag to reject insertion if cache if full. // Set the flag to reject insertion if cache if full.
void SetStrictCapacityLimit(bool strict_capacity_limit); virtual void SetStrictCapacityLimit(bool strict_capacity_limit) override;
// Like Cache methods, but with an extra "hash" parameter. // Like Cache methods, but with an extra "hash" parameter.
Status Insert(const Slice& key, uint32_t hash, void* value, size_t charge, virtual Status Insert(const Slice& key, uint32_t hash, void* value,
size_t charge,
void (*deleter)(const Slice& key, void* value), void (*deleter)(const Slice& key, void* value),
Cache::Handle** handle); Cache::Handle** handle) override;
Cache::Handle* Lookup(const Slice& key, uint32_t hash); virtual Cache::Handle* Lookup(const Slice& key, uint32_t hash) override;
void Release(Cache::Handle* handle); virtual void Release(Cache::Handle* handle) override;
void Erase(const Slice& key, uint32_t hash); virtual void Erase(const Slice& key, uint32_t hash) override;
// Although in some platforms the update of size_t is atomic, to make sure // Although in some platforms the update of size_t is atomic, to make sure
// GetUsage() and GetPinnedUsage() work correctly under any platform, we'll // GetUsage() and GetPinnedUsage() work correctly under any platform, we'll
// protect them with mutex_. // protect them with mutex_.
size_t GetUsage() const { virtual size_t GetUsage() const override {
MutexLock l(&mutex_); MutexLock l(&mutex_);
return usage_; return usage_;
} }
size_t GetPinnedUsage() const { virtual size_t GetPinnedUsage() const override {
MutexLock l(&mutex_); MutexLock l(&mutex_);
assert(usage_ >= lru_usage_); assert(usage_ >= lru_usage_);
return usage_ - lru_usage_; return usage_ - lru_usage_;
} }
void ApplyToAllCacheEntries(void (*callback)(void*, size_t), virtual void ApplyToAllCacheEntries(void (*callback)(void*, size_t),
bool thread_safe); bool thread_safe) override;
void EraseUnRefEntries(); virtual void EraseUnRefEntries() override;
private: private:
void LRU_Remove(LRUHandle* e); void LRU_Remove(LRUHandle* e);
@ -210,15 +263,15 @@ class LRUCache {
HandleTable table_; HandleTable table_;
}; };
LRUCache::LRUCache() : usage_(0), lru_usage_(0) { LRUCacheShard::LRUCacheShard() : usage_(0), lru_usage_(0) {
// Make empty circular linked list // Make empty circular linked list
lru_.next = &lru_; lru_.next = &lru_;
lru_.prev = &lru_; lru_.prev = &lru_;
} }
LRUCache::~LRUCache() {} LRUCacheShard::~LRUCacheShard() {}
bool LRUCache::Unref(LRUHandle* e) { bool LRUCacheShard::Unref(LRUHandle* e) {
assert(e->refs > 0); assert(e->refs > 0);
e->refs--; e->refs--;
return e->refs == 0; return e->refs == 0;
@ -226,7 +279,7 @@ bool LRUCache::Unref(LRUHandle* e) {
// Call deleter and free // Call deleter and free
void LRUCache::EraseUnRefEntries() { void LRUCacheShard::EraseUnRefEntries() {
autovector<LRUHandle*> last_reference_list; autovector<LRUHandle*> last_reference_list;
{ {
MutexLock l(&mutex_); MutexLock l(&mutex_);
@ -249,7 +302,7 @@ void LRUCache::EraseUnRefEntries() {
} }
} }
void LRUCache::ApplyToAllCacheEntries(void (*callback)(void*, size_t), void LRUCacheShard::ApplyToAllCacheEntries(void (*callback)(void*, size_t),
bool thread_safe) { bool thread_safe) {
if (thread_safe) { if (thread_safe) {
mutex_.Lock(); mutex_.Lock();
@ -261,7 +314,7 @@ void LRUCache::ApplyToAllCacheEntries(void (*callback)(void*, size_t),
} }
} }
void LRUCache::LRU_Remove(LRUHandle* e) { void LRUCacheShard::LRU_Remove(LRUHandle* e) {
assert(e->next != nullptr); assert(e->next != nullptr);
assert(e->prev != nullptr); assert(e->prev != nullptr);
e->next->prev = e->prev; e->next->prev = e->prev;
@ -270,7 +323,7 @@ void LRUCache::LRU_Remove(LRUHandle* e) {
lru_usage_ -= e->charge; lru_usage_ -= e->charge;
} }
void LRUCache::LRU_Append(LRUHandle* e) { void LRUCacheShard::LRU_Append(LRUHandle* e) {
// Make "e" newest entry by inserting just before lru_ // Make "e" newest entry by inserting just before lru_
assert(e->next == nullptr); assert(e->next == nullptr);
assert(e->prev == nullptr); assert(e->prev == nullptr);
@ -281,7 +334,8 @@ void LRUCache::LRU_Append(LRUHandle* e) {
lru_usage_ += e->charge; lru_usage_ += e->charge;
} }
void LRUCache::EvictFromLRU(size_t charge, autovector<LRUHandle*>* deleted) { void LRUCacheShard::EvictFromLRU(size_t charge,
autovector<LRUHandle*>* deleted) {
while (usage_ + charge > capacity_ && lru_.next != &lru_) { while (usage_ + charge > capacity_ && lru_.next != &lru_) {
LRUHandle* old = lru_.next; LRUHandle* old = lru_.next;
assert(old->in_cache); assert(old->in_cache);
@ -295,7 +349,7 @@ void LRUCache::EvictFromLRU(size_t charge, autovector<LRUHandle*>* deleted) {
} }
} }
void LRUCache::SetCapacity(size_t capacity) { void LRUCacheShard::SetCapacity(size_t capacity) {
autovector<LRUHandle*> last_reference_list; autovector<LRUHandle*> last_reference_list;
{ {
MutexLock l(&mutex_); MutexLock l(&mutex_);
@ -309,12 +363,12 @@ void LRUCache::SetCapacity(size_t capacity) {
} }
} }
void LRUCache::SetStrictCapacityLimit(bool strict_capacity_limit) { void LRUCacheShard::SetStrictCapacityLimit(bool strict_capacity_limit) {
MutexLock l(&mutex_); MutexLock l(&mutex_);
strict_capacity_limit_ = strict_capacity_limit; strict_capacity_limit_ = strict_capacity_limit;
} }
Cache::Handle* LRUCache::Lookup(const Slice& key, uint32_t hash) { Cache::Handle* LRUCacheShard::Lookup(const Slice& key, uint32_t hash) {
MutexLock l(&mutex_); MutexLock l(&mutex_);
LRUHandle* e = table_.Lookup(key, hash); LRUHandle* e = table_.Lookup(key, hash);
if (e != nullptr) { if (e != nullptr) {
@ -327,7 +381,7 @@ Cache::Handle* LRUCache::Lookup(const Slice& key, uint32_t hash) {
return reinterpret_cast<Cache::Handle*>(e); return reinterpret_cast<Cache::Handle*>(e);
} }
void LRUCache::Release(Cache::Handle* handle) { void LRUCacheShard::Release(Cache::Handle* handle) {
if (handle == nullptr) { if (handle == nullptr) {
return; return;
} }
@ -364,7 +418,7 @@ void LRUCache::Release(Cache::Handle* handle) {
} }
} }
Status LRUCache::Insert(const Slice& key, uint32_t hash, void* value, Status LRUCacheShard::Insert(const Slice& key, uint32_t hash, void* value,
size_t charge, size_t charge,
void (*deleter)(const Slice& key, void* value), void (*deleter)(const Slice& key, void* value),
Cache::Handle** handle) { Cache::Handle** handle) {
@ -437,7 +491,7 @@ Status LRUCache::Insert(const Slice& key, uint32_t hash, void* value,
return s; return s;
} }
void LRUCache::Erase(const Slice& key, uint32_t hash) { void LRUCacheShard::Erase(const Slice& key, uint32_t hash) {
LRUHandle* e; LRUHandle* e;
bool last_reference = false; bool last_reference = false;
{ {
@ -462,149 +516,52 @@ void LRUCache::Erase(const Slice& key, uint32_t hash) {
} }
} }
static int kNumShardBits = 6; // default values, can be overridden class LRUCache : public ShardedCache {
public:
class ShardedLRUCache : public Cache { LRUCache(size_t capacity, int num_shard_bits, bool strict_capacity_limit)
private: : ShardedCache(capacity, num_shard_bits, strict_capacity_limit) {
LRUCache* shards_; int num_shards = 1 << num_shard_bits;
port::Mutex id_mutex_; shards_ = new LRUCacheShard[num_shards];
port::Mutex capacity_mutex_; SetCapacity(capacity);
uint64_t last_id_; SetStrictCapacityLimit(strict_capacity_limit);
int num_shard_bits_;
size_t capacity_;
bool strict_capacity_limit_;
static inline uint32_t HashSlice(const Slice& s) {
return Hash(s.data(), s.size(), 0);
} }
uint32_t Shard(uint32_t hash) { virtual ~LRUCache() { delete[] shards_; }
// Note, hash >> 32 yields hash in gcc, not the zero we expect!
return (num_shard_bits_ > 0) ? (hash >> (32 - num_shard_bits_)) : 0;
}
public: virtual CacheShard* GetShard(int shard) override {
ShardedLRUCache(size_t capacity, int num_shard_bits, return reinterpret_cast<CacheShard*>(&shards_[shard]);
bool strict_capacity_limit)
: last_id_(0),
num_shard_bits_(num_shard_bits),
capacity_(capacity),
strict_capacity_limit_(strict_capacity_limit) {
int num_shards = 1 << num_shard_bits_;
shards_ = new LRUCache[num_shards];
const size_t per_shard = (capacity + (num_shards - 1)) / num_shards;
for (int s = 0; s < num_shards; s++) {
shards_[s].SetCapacity(per_shard);
shards_[s].SetStrictCapacityLimit(strict_capacity_limit);
}
}
virtual ~ShardedLRUCache() { delete[] shards_; }
virtual void SetCapacity(size_t capacity) override {
int num_shards = 1 << num_shard_bits_;
const size_t per_shard = (capacity + (num_shards - 1)) / num_shards;
MutexLock l(&capacity_mutex_);
for (int s = 0; s < num_shards; s++) {
shards_[s].SetCapacity(per_shard);
}
capacity_ = capacity;
}
virtual void SetStrictCapacityLimit(bool strict_capacity_limit) override {
int num_shards = 1 << num_shard_bits_;
for (int s = 0; s < num_shards; s++) {
shards_[s].SetStrictCapacityLimit(strict_capacity_limit);
}
strict_capacity_limit_ = strict_capacity_limit;
}
virtual Status Insert(const Slice& key, void* value, size_t charge,
void (*deleter)(const Slice& key, void* value),
Handle** handle) override {
const uint32_t hash = HashSlice(key);
return shards_[Shard(hash)].Insert(key, hash, value, charge, deleter,
handle);
}
virtual Handle* Lookup(const Slice& key) override {
const uint32_t hash = HashSlice(key);
return shards_[Shard(hash)].Lookup(key, hash);
} }
virtual void Release(Handle* handle) override {
LRUHandle* h = reinterpret_cast<LRUHandle*>(handle);
shards_[Shard(h->hash)].Release(handle);
}
virtual void Erase(const Slice& key) override {
const uint32_t hash = HashSlice(key);
shards_[Shard(hash)].Erase(key, hash);
}
virtual void* Value(Handle* handle) override {
return reinterpret_cast<LRUHandle*>(handle)->value;
}
virtual uint64_t NewId() override {
MutexLock l(&id_mutex_);
return ++(last_id_);
}
virtual size_t GetCapacity() const override { return capacity_; }
virtual bool HasStrictCapacityLimit() const override { virtual const CacheShard* GetShard(int shard) const override {
return strict_capacity_limit_; return reinterpret_cast<CacheShard*>(&shards_[shard]);
} }
virtual size_t GetUsage() const override { virtual void* Value(Handle* handle) override {
// We will not lock the cache when getting the usage from shards. return reinterpret_cast<const LRUHandle*>(handle)->value;
int num_shards = 1 << num_shard_bits_;
size_t usage = 0;
for (int s = 0; s < num_shards; s++) {
usage += shards_[s].GetUsage();
}
return usage;
} }
virtual size_t GetUsage(Handle* handle) const override { virtual size_t GetCharge(Handle* handle) const override {
return reinterpret_cast<LRUHandle*>(handle)->charge; return reinterpret_cast<const LRUHandle*>(handle)->charge;
} }
virtual size_t GetPinnedUsage() const override { virtual uint32_t GetHash(Handle* handle) const override {
// We will not lock the cache when getting the usage from shards. return reinterpret_cast<const LRUHandle*>(handle)->hash;
int num_shards = 1 << num_shard_bits_;
size_t usage = 0;
for (int s = 0; s < num_shards; s++) {
usage += shards_[s].GetPinnedUsage();
}
return usage;
} }
virtual void DisownData() override { shards_ = nullptr; } virtual void DisownData() override { shards_ = nullptr; }
virtual void ApplyToAllCacheEntries(void (*callback)(void*, size_t), private:
bool thread_safe) override { LRUCacheShard* shards_;
int num_shards = 1 << num_shard_bits_;
for (int s = 0; s < num_shards; s++) {
shards_[s].ApplyToAllCacheEntries(callback, thread_safe);
}
}
virtual void EraseUnRefEntries() override {
int num_shards = 1 << num_shard_bits_;
for (int s = 0; s < num_shards; s++) {
shards_[s].EraseUnRefEntries();
}
}
}; };
} // end anonymous namespace } // end anonymous namespace
std::shared_ptr<Cache> NewLRUCache(size_t capacity) {
return NewLRUCache(capacity, kNumShardBits, false);
}
std::shared_ptr<Cache> NewLRUCache(size_t capacity, int num_shard_bits) {
return NewLRUCache(capacity, num_shard_bits, false);
}
std::shared_ptr<Cache> NewLRUCache(size_t capacity, int num_shard_bits, std::shared_ptr<Cache> NewLRUCache(size_t capacity, int num_shard_bits,
bool strict_capacity_limit) { bool strict_capacity_limit) {
if (num_shard_bits >= 20) { if (num_shard_bits >= 20) {
return nullptr; // the cache cannot be sharded into too many fine pieces return nullptr; // the cache cannot be sharded into too many fine pieces
} }
return std::make_shared<ShardedLRUCache>(capacity, num_shard_bits, return std::make_shared<LRUCache>(capacity, num_shard_bits,
strict_capacity_limit); strict_capacity_limit);
} }

@ -1,71 +0,0 @@
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
// This source code is licensed under the BSD-style license found in the
// LICENSE file in the root directory of this source tree. An additional grant
// of patent rights can be found in the PATENTS file in the same directory.
//
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. See the AUTHORS file for names of contributors.
#pragma once
#include <cassert>
#include <cstdint>
#include "port/port.h"
#include "rocksdb/slice.h"
namespace rocksdb {
// An entry is a variable length heap-allocated structure.
// Entries are referenced by cache and/or by any external entity.
// The cache keeps all its entries in table. Some elements
// are also stored on LRU list.
//
// LRUHandle can be in these states:
// 1. Referenced externally AND in hash table.
// In that case the entry is *not* in the LRU. (refs > 1 && in_cache == true)
// 2. Not referenced externally and in hash table. In that case the entry is
// in the LRU and can be freed. (refs == 1 && in_cache == true)
// 3. Referenced externally and not in hash table. In that case the entry is
// in not on LRU and not in table. (refs >= 1 && in_cache == false)
//
// All newly created LRUHandles are in state 1. If you call LRUCache::Release
// on entry in state 1, it will go into state 2. To move from state 1 to
// state 3, either call LRUCache::Erase or LRUCache::Insert with the same key.
// To move from state 2 to state 1, use LRUCache::Lookup.
// Before destruction, make sure that no handles are in state 1. This means
// that any successful LRUCache::Lookup/LRUCache::Insert have a matching
// RUCache::Release (to move into state 2) or LRUCache::Erase (for state 3)
struct LRUHandle {
void* value;
void (*deleter)(const Slice&, void* value);
LRUHandle* next_hash;
LRUHandle* next;
LRUHandle* prev;
size_t charge; // TODO(opt): Only allow uint32_t?
size_t key_length;
uint32_t refs; // a number of refs to this entry
// cache itself is counted as 1
bool in_cache; // true, if this entry is referenced by the hash table
uint32_t hash; // Hash of key(); used for fast sharding and comparisons
char key_data[1]; // Beginning of key
Slice key() const {
// For cheaper lookups, we allow a temporary Handle object
// to store a pointer to a key in "value".
if (next == this) {
return *(reinterpret_cast<Slice*>(value));
} else {
return Slice(key_data, key_length);
}
}
void Free() {
assert((refs == 1 && in_cache) || (refs == 0 && !in_cache));
(*deleter)(key(), value);
delete[] reinterpret_cast<char*>(this);
}
};
} // end namespace rocksdb

@ -0,0 +1,117 @@
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
// This source code is licensed under the BSD-style license found in the
// LICENSE file in the root directory of this source tree. An additional grant
// of patent rights can be found in the PATENTS file in the same directory.
//
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. See the AUTHORS file for names of contributors.
#include "util/sharded_cache.h"
#include "util/mutexlock.h"
namespace rocksdb {
ShardedCache::ShardedCache(size_t capacity, int num_shard_bits,
bool strict_capacity_limit)
: num_shard_bits_(num_shard_bits),
capacity_(capacity),
strict_capacity_limit_(strict_capacity_limit),
last_id_(1) {}
void ShardedCache::SetCapacity(size_t capacity) {
int num_shards = 1 << num_shard_bits_;
const size_t per_shard = (capacity + (num_shards - 1)) / num_shards;
MutexLock l(&capacity_mutex_);
for (int s = 0; s < num_shards; s++) {
GetShard(s)->SetCapacity(per_shard);
}
capacity_ = capacity;
}
void ShardedCache::SetStrictCapacityLimit(bool strict_capacity_limit) {
int num_shards = 1 << num_shard_bits_;
MutexLock l(&capacity_mutex_);
for (int s = 0; s < num_shards; s++) {
GetShard(s)->SetStrictCapacityLimit(strict_capacity_limit);
}
strict_capacity_limit_ = strict_capacity_limit;
}
Status ShardedCache::Insert(const Slice& key, void* value, size_t charge,
void (*deleter)(const Slice& key, void* value),
Handle** handle) {
uint32_t hash = HashSlice(key);
return GetShard(Shard(hash))
->Insert(key, hash, value, charge, deleter, handle);
}
Cache::Handle* ShardedCache::Lookup(const Slice& key) {
uint32_t hash = HashSlice(key);
return GetShard(Shard(hash))->Lookup(key, hash);
}
void ShardedCache::Release(Handle* handle) {
uint32_t hash = GetHash(handle);
GetShard(Shard(hash))->Release(handle);
}
void ShardedCache::Erase(const Slice& key) {
uint32_t hash = HashSlice(key);
GetShard(Shard(hash))->Erase(key, hash);
}
uint64_t ShardedCache::NewId() {
return last_id_.fetch_add(1, std::memory_order_relaxed);
}
size_t ShardedCache::GetCapacity() const {
MutexLock l(&capacity_mutex_);
return capacity_;
}
bool ShardedCache::HasStrictCapacityLimit() const {
MutexLock l(&capacity_mutex_);
return strict_capacity_limit_;
}
size_t ShardedCache::GetUsage() const {
// We will not lock the cache when getting the usage from shards.
int num_shards = 1 << num_shard_bits_;
size_t usage = 0;
for (int s = 0; s < num_shards; s++) {
usage += GetShard(s)->GetUsage();
}
return usage;
}
size_t ShardedCache::GetUsage(Handle* handle) const {
return GetCharge(handle);
}
size_t ShardedCache::GetPinnedUsage() const {
// We will not lock the cache when getting the usage from shards.
int num_shards = 1 << num_shard_bits_;
size_t usage = 0;
for (int s = 0; s < num_shards; s++) {
usage += GetShard(s)->GetPinnedUsage();
}
return usage;
}
void ShardedCache::ApplyToAllCacheEntries(void (*callback)(void*, size_t),
bool thread_safe) {
int num_shards = 1 << num_shard_bits_;
for (int s = 0; s < num_shards; s++) {
GetShard(s)->ApplyToAllCacheEntries(callback, thread_safe);
}
}
void ShardedCache::EraseUnRefEntries() {
int num_shards = 1 << num_shard_bits_;
for (int s = 0; s < num_shards; s++) {
GetShard(s)->EraseUnRefEntries();
}
}
} // namespace rocksdb

@ -0,0 +1,93 @@
// Copyright (c) 2011-present, Facebook, Inc. All rights reserved.
// This source code is licensed under the BSD-style license found in the
// LICENSE file in the root directory of this source tree. An additional grant
// of patent rights can be found in the PATENTS file in the same directory.
//
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file. See the AUTHORS file for names of contributors.
#pragma once
#include <atomic>
#include "port/port.h"
#include "rocksdb/cache.h"
#include "util/hash.h"
namespace rocksdb {
// Single cache shard interface.
class CacheShard {
public:
CacheShard() = default;
virtual ~CacheShard() = default;
virtual Status Insert(const Slice& key, uint32_t hash, void* value,
size_t charge,
void (*deleter)(const Slice& key, void* value),
Cache::Handle** handle) = 0;
virtual Cache::Handle* Lookup(const Slice& key, uint32_t hash) = 0;
virtual void Release(Cache::Handle* handle) = 0;
virtual void Erase(const Slice& key, uint32_t hash) = 0;
virtual void SetCapacity(size_t capacity) = 0;
virtual void SetStrictCapacityLimit(bool strict_capacity_limit) = 0;
virtual size_t GetUsage() const = 0;
virtual size_t GetPinnedUsage() const = 0;
virtual void ApplyToAllCacheEntries(void (*callback)(void*, size_t),
bool thread_safe) = 0;
virtual void EraseUnRefEntries() = 0;
};
// Generic cache interface which shards cache by hash of keys. 2^num_shard_bits
// shards will be created, with capacity split evenly to each of the shards.
// Keys are sharded by the highest num_shard_bits bits of hash value.
class ShardedCache : public Cache {
public:
ShardedCache(size_t capacity, int num_shard_bits, bool strict_capacity_limit);
virtual ~ShardedCache() = default;
virtual CacheShard* GetShard(int shard) = 0;
virtual const CacheShard* GetShard(int shard) const = 0;
virtual void* Value(Handle* handle) override = 0;
virtual size_t GetCharge(Handle* handle) const = 0;
virtual uint32_t GetHash(Handle* handle) const = 0;
virtual void DisownData() override = 0;
virtual void SetCapacity(size_t capacity) override;
virtual void SetStrictCapacityLimit(bool strict_capacity_limit) override;
virtual Status Insert(const Slice& key, void* value, size_t charge,
void (*deleter)(const Slice& key, void* value),
Handle** handle) override;
virtual Handle* Lookup(const Slice& key) override;
virtual void Release(Handle* handle) override;
virtual void Erase(const Slice& key) override;
virtual uint64_t NewId() override;
virtual size_t GetCapacity() const override;
virtual bool HasStrictCapacityLimit() const override;
virtual size_t GetUsage() const override;
virtual size_t GetUsage(Handle* handle) const override;
virtual size_t GetPinnedUsage() const override;
virtual void ApplyToAllCacheEntries(void (*callback)(void*, size_t),
bool thread_safe) override;
virtual void EraseUnRefEntries() override;
private:
static inline uint32_t HashSlice(const Slice& s) {
return Hash(s.data(), s.size(), 0);
}
uint32_t Shard(uint32_t hash) {
// Note, hash >> 32 yields hash in gcc, not the zero we expect!
return (num_shard_bits_ > 0) ? (hash >> (32 - num_shard_bits_)) : 0;
}
int num_shard_bits_;
mutable port::Mutex capacity_mutex_;
size_t capacity_;
bool strict_capacity_limit_;
std::atomic<uint64_t> last_id_;
};
} // namespace rocksdb

@ -65,9 +65,7 @@ class SimCacheImpl : public SimCache {
key_only_cache_->Erase(key); key_only_cache_->Erase(key);
} }
virtual void* Value(Handle* handle) override { virtual void* Value(Handle* handle) override { return cache_->Value(handle); }
return reinterpret_cast<LRUHandle*>(handle)->value;
}
virtual uint64_t NewId() override { return cache_->NewId(); } virtual uint64_t NewId() override { return cache_->NewId(); }
@ -80,7 +78,7 @@ class SimCacheImpl : public SimCache {
virtual size_t GetUsage() const override { return cache_->GetUsage(); } virtual size_t GetUsage() const override { return cache_->GetUsage(); }
virtual size_t GetUsage(Handle* handle) const override { virtual size_t GetUsage(Handle* handle) const override {
return reinterpret_cast<LRUHandle*>(handle)->charge; return cache_->GetUsage(handle);
} }
virtual size_t GetPinnedUsage() const override { virtual size_t GetPinnedUsage() const override {
@ -113,12 +111,21 @@ class SimCacheImpl : public SimCache {
key_only_cache_->SetCapacity(capacity); key_only_cache_->SetCapacity(capacity);
} }
virtual uint64_t get_lookup_counter() const override { return lookup_times_; } virtual uint64_t get_lookup_counter() const override {
virtual uint64_t get_hit_counter() const override { return hit_times_; } return lookup_times_.load(std::memory_order_relaxed);
}
virtual uint64_t get_hit_counter() const override {
return hit_times_.load(std::memory_order_relaxed);
}
virtual double get_hit_rate() const override { virtual double get_hit_rate() const override {
return hit_times_ * 1.0f / lookup_times_; return get_hit_counter() * 1.0f / get_lookup_counter();
}
virtual void reset_counter() override {
lookup_times_.store(0, std::memory_order_relaxed);
hit_times_.store(0, std::memory_order_relaxed);
} }
virtual void reset_counter() override { hit_times_ = lookup_times_ = 0; }
virtual std::string ToString() const override { virtual std::string ToString() const override {
std::string res; std::string res;
@ -137,8 +144,10 @@ class SimCacheImpl : public SimCache {
std::shared_ptr<Cache> key_only_cache_; std::shared_ptr<Cache> key_only_cache_;
std::atomic<uint64_t> lookup_times_; std::atomic<uint64_t> lookup_times_;
std::atomic<uint64_t> hit_times_; std::atomic<uint64_t> hit_times_;
void inc_lookup_counter() { lookup_times_++; } void inc_lookup_counter() {
void inc_hit_counter() { hit_times_++; } lookup_times_.fetch_add(1, std::memory_order_relaxed);
}
void inc_hit_counter() { hit_times_.fetch_add(1, std::memory_order_relaxed); }
}; };
} // end anonymous namespace } // end anonymous namespace

Loading…
Cancel
Save