098da83483
Test Plan: USE_CLANG=1 make all check -j32 Reviewers: andrewkr, IslamAbdelRahman Reviewed By: IslamAbdelRahman Subscribers: leveldb, andrewkr, dhruba Differential Revision: https://reviews.facebook.net/D59313
153 lines
4.4 KiB
C++
153 lines
4.4 KiB
C++
// Copyright (c) 2013, 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.
|
|
//
|
|
#ifndef ROCKSDB_LITE
|
|
|
|
#include "utilities/persistent_cache/volatile_tier_impl.h"
|
|
|
|
#include <string>
|
|
|
|
namespace rocksdb {
|
|
|
|
void VolatileCacheTier::DeleteCacheData(VolatileCacheTier::CacheData* data) {
|
|
assert(data);
|
|
delete data;
|
|
}
|
|
|
|
VolatileCacheTier::~VolatileCacheTier() { index_.Clear(&DeleteCacheData); }
|
|
|
|
std::vector<PersistentCacheTier::TierStats> VolatileCacheTier::Stats() {
|
|
PersistentCacheTier::TierStats stat;
|
|
stat.insert({"persistent_cache.volatile_cache.hits", stats_.cache_hits_});
|
|
stat.insert({"persistent_cache.volatile_cache.misses", stats_.cache_misses_});
|
|
stat.insert(
|
|
{"persistent_cache.volatile_cache.inserts", stats_.cache_inserts_});
|
|
stat.insert({"persistent_cache.volatile_cache.evicts", stats_.cache_evicts_});
|
|
stat.insert(
|
|
{"persistent_cache.volatile_cache.hit_pct", stats_.CacheHitPct()});
|
|
stat.insert(
|
|
{"persistent_cache.volatile_cache.miss_pct", stats_.CacheMissPct()});
|
|
|
|
std::vector<PersistentCacheTier::TierStats> tier_stats;
|
|
if (next_tier()) {
|
|
tier_stats = next_tier()->Stats();
|
|
}
|
|
tier_stats.push_back(stat);
|
|
return tier_stats;
|
|
}
|
|
|
|
std::string VolatileCacheTier::PrintStats() {
|
|
std::ostringstream ss;
|
|
ss << "pagecache.volatilecache.hits: " << stats_.cache_hits_ << std::endl
|
|
<< "pagecache.volatilecache.misses: " << stats_.cache_misses_ << std::endl
|
|
<< "pagecache.volatilecache.inserts: " << stats_.cache_inserts_
|
|
<< std::endl
|
|
<< "pagecache.volatilecache.evicts: " << stats_.cache_evicts_ << std::endl
|
|
<< "pagecache.volatilecache.hit_pct: " << stats_.CacheHitPct() << std::endl
|
|
<< "pagecache.volatilecache.miss_pct: " << stats_.CacheMissPct()
|
|
<< std::endl
|
|
<< PersistentCacheTier::PrintStats();
|
|
return ss.str();
|
|
}
|
|
|
|
Status VolatileCacheTier::Insert(const Slice& page_key, const char* data,
|
|
const size_t size) {
|
|
// precondition
|
|
assert(data);
|
|
assert(size);
|
|
|
|
// increment the size
|
|
size_ += size;
|
|
|
|
// check if we have overshot the limit, if so evict some space
|
|
while (size_ > max_size_) {
|
|
if (!Evict()) {
|
|
// unable to evict data, we give up so we don't spike read
|
|
// latency
|
|
assert(size_ >= size);
|
|
size_ -= size;
|
|
return Status::TryAgain("Unable to evict any data");
|
|
}
|
|
}
|
|
|
|
assert(size_ >= size);
|
|
|
|
// insert order: LRU, followed by index
|
|
std::string key(page_key.data(), page_key.size());
|
|
std::string value(data, size);
|
|
std::unique_ptr<CacheData> cache_data(
|
|
new CacheData(std::move(key), std::move(value)));
|
|
bool ok = index_.Insert(cache_data.get());
|
|
if (!ok) {
|
|
// decrement the size that we incremented ahead of time
|
|
assert(size_ >= size);
|
|
size_ -= size;
|
|
// failed to insert to cache, block already in cache
|
|
return Status::TryAgain("key already exists in volatile cache");
|
|
}
|
|
|
|
cache_data.release();
|
|
stats_.cache_inserts_++;
|
|
return Status::OK();
|
|
}
|
|
|
|
Status VolatileCacheTier::Lookup(const Slice& page_key,
|
|
std::unique_ptr<char[]>* result,
|
|
size_t* size) {
|
|
CacheData key(std::move(page_key.ToString()));
|
|
CacheData* kv;
|
|
bool ok = index_.Find(&key, &kv);
|
|
if (ok) {
|
|
// set return data
|
|
result->reset(new char[kv->value.size()]);
|
|
memcpy(result->get(), kv->value.c_str(), kv->value.size());
|
|
*size = kv->value.size();
|
|
// drop the reference on cache data
|
|
kv->refs_--;
|
|
// update stats
|
|
stats_.cache_hits_++;
|
|
return Status::OK();
|
|
}
|
|
|
|
stats_.cache_misses_++;
|
|
|
|
if (next_tier()) {
|
|
return next_tier()->Lookup(page_key, result, size);
|
|
}
|
|
|
|
return Status::NotFound("key not found in volatile cache");
|
|
}
|
|
|
|
bool VolatileCacheTier::Erase(const Slice& key) {
|
|
assert(!"not supported");
|
|
return true;
|
|
}
|
|
|
|
bool VolatileCacheTier::Evict() {
|
|
CacheData* edata = index_.Evict();
|
|
if (!edata) {
|
|
// not able to evict any object
|
|
return false;
|
|
}
|
|
|
|
stats_.cache_evicts_++;
|
|
|
|
// push the evicted object to the next level
|
|
if (next_tier()) {
|
|
next_tier()->Insert(Slice(edata->key), edata->value.c_str(),
|
|
edata->value.size());
|
|
}
|
|
|
|
// adjust size and destroy data
|
|
size_ -= edata->value.size();
|
|
delete edata;
|
|
|
|
return true;
|
|
}
|
|
|
|
} // namespace rocksdb
|
|
|
|
#endif
|