364 lines
12 KiB
C
364 lines
12 KiB
C
|
// Copyright (c) Facebook, Inc. and its affiliates. All Rights Reserved.
|
||
|
// This source code is licensed under both the GPLv2 (found in the
|
||
|
// COPYING file in the root directory) and Apache 2.0 License
|
||
|
// (found in the LICENSE.Apache file in the root directory).
|
||
|
|
||
|
#pragma once
|
||
|
#ifndef ROCKSDB_LITE
|
||
|
|
||
|
#include <unordered_map>
|
||
|
|
||
|
#include "file/random_access_file_reader.h"
|
||
|
#include "file/writable_file_writer.h"
|
||
|
#include "rocksdb/utilities/cache_dump_load.h"
|
||
|
#include "table/block_based/block.h"
|
||
|
#include "table/block_based/block_like_traits.h"
|
||
|
#include "table/block_based/block_type.h"
|
||
|
#include "table/block_based/cachable_entry.h"
|
||
|
#include "table/block_based/parsed_full_filter_block.h"
|
||
|
#include "table/block_based/reader_common.h"
|
||
|
|
||
|
namespace ROCKSDB_NAMESPACE {
|
||
|
|
||
|
// the read buffer size of for the default CacheDumpReader
|
||
|
const unsigned int kDumpReaderBufferSize = 1024; // 1KB
|
||
|
static const unsigned int kSizePrefixLen = 4;
|
||
|
|
||
|
enum CacheDumpUnitType : unsigned char {
|
||
|
kHeader = 1,
|
||
|
kFooter = 2,
|
||
|
kData = 3,
|
||
|
kFilter = 4,
|
||
|
kProperties = 5,
|
||
|
kCompressionDictionary = 6,
|
||
|
kRangeDeletion = 7,
|
||
|
kHashIndexPrefixes = 8,
|
||
|
kHashIndexMetadata = 9,
|
||
|
kMetaIndex = 10,
|
||
|
kIndex = 11,
|
||
|
kDeprecatedFilterBlock = 12,
|
||
|
kFilterMetaBlock = 13,
|
||
|
kBlockTypeMax,
|
||
|
};
|
||
|
|
||
|
// The metadata of a dump unit. After it is serilized, its size is fixed 16
|
||
|
// bytes.
|
||
|
struct DumpUnitMeta {
|
||
|
// sequence number is a monotonically increasing number to indicate the order
|
||
|
// of the blocks being written. Header is 0.
|
||
|
uint32_t sequence_num;
|
||
|
// The Crc32c checksum of its dump unit.
|
||
|
uint32_t dump_unit_checksum;
|
||
|
// The dump unit size after the dump unit is serilized to a string.
|
||
|
uint64_t dump_unit_size;
|
||
|
|
||
|
void reset() {
|
||
|
sequence_num = 0;
|
||
|
dump_unit_checksum = 0;
|
||
|
dump_unit_size = 0;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// The data structure to hold a block and its information.
|
||
|
struct DumpUnit {
|
||
|
// The timestamp when the block is identified, copied, and dumped from block
|
||
|
// cache
|
||
|
uint64_t timestamp;
|
||
|
// The type of the block
|
||
|
CacheDumpUnitType type;
|
||
|
// The key of this block when the block is referenced by this Cache
|
||
|
Slice key;
|
||
|
// The block size
|
||
|
size_t value_len;
|
||
|
// The Crc32c checksum of the block
|
||
|
uint32_t value_checksum;
|
||
|
// Pointer to the block. Note that, in the dump process, it points to a memory
|
||
|
// buffer copied from cache block. The buffer is freed when we process the
|
||
|
// next block. In the load process, we use an std::string to store the
|
||
|
// serilized dump_unit read from the reader. So it points to the memory
|
||
|
// address of the begin of the block in this string.
|
||
|
void* value;
|
||
|
|
||
|
void reset() {
|
||
|
timestamp = 0;
|
||
|
type = CacheDumpUnitType::kBlockTypeMax;
|
||
|
key.clear();
|
||
|
value_len = 0;
|
||
|
value_checksum = 0;
|
||
|
value = nullptr;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
// The default implementation of the Cache Dumper
|
||
|
class CacheDumperImpl : public CacheDumper {
|
||
|
public:
|
||
|
CacheDumperImpl(const CacheDumpOptions& dump_options,
|
||
|
const std::shared_ptr<Cache>& cache,
|
||
|
std::unique_ptr<CacheDumpWriter>&& writer)
|
||
|
: options_(dump_options), cache_(cache), writer_(std::move(writer)) {}
|
||
|
~CacheDumperImpl() { writer_.reset(); }
|
||
|
Status SetDumpFilter(std::vector<DB*> db_list) override;
|
||
|
IOStatus DumpCacheEntriesToWriter() override;
|
||
|
|
||
|
private:
|
||
|
IOStatus WriteRawBlock(uint64_t timestamp, CacheDumpUnitType type,
|
||
|
const Slice& key, void* value, size_t len,
|
||
|
uint32_t checksum);
|
||
|
|
||
|
IOStatus WriteHeader();
|
||
|
|
||
|
IOStatus WriteCacheBlock(const CacheDumpUnitType type, const Slice& key,
|
||
|
void* value, size_t len);
|
||
|
IOStatus WriteFooter();
|
||
|
bool ShouldFilterOut(const Slice& key);
|
||
|
std::function<void(const Slice&, void*, size_t, Cache::DeleterFn)>
|
||
|
DumpOneBlockCallBack();
|
||
|
|
||
|
CacheDumpOptions options_;
|
||
|
std::shared_ptr<Cache> cache_;
|
||
|
std::unique_ptr<CacheDumpWriter> writer_;
|
||
|
std::unordered_map<Cache::DeleterFn, CacheEntryRole> role_map_;
|
||
|
SystemClock* clock_;
|
||
|
uint32_t sequence_num_;
|
||
|
// The cache key prefix filter. Currently, we use db_session_id as the prefix,
|
||
|
// so using std::set to store the prefixes as filter is enough. Further
|
||
|
// improvement can be applied like BloomFilter or others to speedup the
|
||
|
// filtering.
|
||
|
std::set<std::string> prefix_filter_;
|
||
|
};
|
||
|
|
||
|
// The default implementation of CacheDumpedLoader
|
||
|
class CacheDumpedLoaderImpl : public CacheDumpedLoader {
|
||
|
public:
|
||
|
CacheDumpedLoaderImpl(const CacheDumpOptions& dump_options,
|
||
|
const BlockBasedTableOptions& toptions,
|
||
|
const std::shared_ptr<SecondaryCache>& secondary_cache,
|
||
|
std::unique_ptr<CacheDumpReader>&& reader)
|
||
|
: options_(dump_options),
|
||
|
toptions_(toptions),
|
||
|
secondary_cache_(secondary_cache),
|
||
|
reader_(std::move(reader)) {}
|
||
|
~CacheDumpedLoaderImpl() {}
|
||
|
IOStatus RestoreCacheEntriesToSecondaryCache() override;
|
||
|
|
||
|
private:
|
||
|
IOStatus ReadDumpUnitMeta(std::string* data, DumpUnitMeta* unit_meta);
|
||
|
IOStatus ReadDumpUnit(size_t len, std::string* data, DumpUnit* unit);
|
||
|
IOStatus ReadHeader(std::string* data, DumpUnit* dump_unit);
|
||
|
IOStatus ReadCacheBlock(std::string* data, DumpUnit* dump_unit);
|
||
|
|
||
|
CacheDumpOptions options_;
|
||
|
const BlockBasedTableOptions& toptions_;
|
||
|
std::shared_ptr<SecondaryCache> secondary_cache_;
|
||
|
std::unique_ptr<CacheDumpReader> reader_;
|
||
|
std::unordered_map<Cache::DeleterFn, CacheEntryRole> role_map_;
|
||
|
};
|
||
|
|
||
|
// The default implementation of CacheDumpWriter. We write the blocks to a file
|
||
|
// sequentially.
|
||
|
class ToFileCacheDumpWriter : public CacheDumpWriter {
|
||
|
public:
|
||
|
explicit ToFileCacheDumpWriter(
|
||
|
std::unique_ptr<WritableFileWriter>&& file_writer)
|
||
|
: file_writer_(std::move(file_writer)) {}
|
||
|
|
||
|
~ToFileCacheDumpWriter() { Close().PermitUncheckedError(); }
|
||
|
|
||
|
// Write the serilized metadata to the file
|
||
|
virtual IOStatus WriteMetadata(const Slice& metadata) override {
|
||
|
assert(file_writer_ != nullptr);
|
||
|
std::string prefix;
|
||
|
PutFixed32(&prefix, static_cast<uint32_t>(metadata.size()));
|
||
|
IOStatus io_s = file_writer_->Append(Slice(prefix));
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
io_s = file_writer_->Append(metadata);
|
||
|
return io_s;
|
||
|
}
|
||
|
|
||
|
// Write the serilized data to the file
|
||
|
virtual IOStatus WritePacket(const Slice& data) override {
|
||
|
assert(file_writer_ != nullptr);
|
||
|
std::string prefix;
|
||
|
PutFixed32(&prefix, static_cast<uint32_t>(data.size()));
|
||
|
IOStatus io_s = file_writer_->Append(Slice(prefix));
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
io_s = file_writer_->Append(data);
|
||
|
return io_s;
|
||
|
}
|
||
|
|
||
|
// Reset the writer
|
||
|
virtual IOStatus Close() override {
|
||
|
file_writer_.reset();
|
||
|
return IOStatus::OK();
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
std::unique_ptr<WritableFileWriter> file_writer_;
|
||
|
};
|
||
|
|
||
|
// The default implementation of CacheDumpReader. It is implemented based on
|
||
|
// RandomAccessFileReader. Note that, we keep an internal variable to remember
|
||
|
// the current offset.
|
||
|
class FromFileCacheDumpReader : public CacheDumpReader {
|
||
|
public:
|
||
|
explicit FromFileCacheDumpReader(
|
||
|
std::unique_ptr<RandomAccessFileReader>&& reader)
|
||
|
: file_reader_(std::move(reader)),
|
||
|
offset_(0),
|
||
|
buffer_(new char[kDumpReaderBufferSize]) {}
|
||
|
|
||
|
~FromFileCacheDumpReader() { delete[] buffer_; }
|
||
|
|
||
|
virtual IOStatus ReadMetadata(std::string* metadata) override {
|
||
|
uint32_t metadata_len = 0;
|
||
|
IOStatus io_s = ReadSizePrefix(&metadata_len);
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
return Read(metadata_len, metadata);
|
||
|
}
|
||
|
|
||
|
virtual IOStatus ReadPacket(std::string* data) override {
|
||
|
uint32_t data_len = 0;
|
||
|
IOStatus io_s = ReadSizePrefix(&data_len);
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
return Read(data_len, data);
|
||
|
}
|
||
|
|
||
|
private:
|
||
|
IOStatus ReadSizePrefix(uint32_t* len) {
|
||
|
std::string prefix;
|
||
|
IOStatus io_s = Read(kSizePrefixLen, &prefix);
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
Slice encoded_slice(prefix);
|
||
|
if (!GetFixed32(&encoded_slice, len)) {
|
||
|
return IOStatus::Corruption("Decode size prefix string failed");
|
||
|
}
|
||
|
return IOStatus::OK();
|
||
|
}
|
||
|
|
||
|
IOStatus Read(size_t len, std::string* data) {
|
||
|
assert(file_reader_ != nullptr);
|
||
|
IOStatus io_s;
|
||
|
|
||
|
unsigned int bytes_to_read = static_cast<unsigned int>(len);
|
||
|
unsigned int to_read = bytes_to_read > kDumpReaderBufferSize
|
||
|
? kDumpReaderBufferSize
|
||
|
: bytes_to_read;
|
||
|
|
||
|
while (to_read > 0) {
|
||
|
io_s = file_reader_->Read(IOOptions(), offset_, to_read, &result_,
|
||
|
buffer_, nullptr);
|
||
|
if (!io_s.ok()) {
|
||
|
return io_s;
|
||
|
}
|
||
|
if (result_.size() < to_read) {
|
||
|
return IOStatus::Corruption("Corrupted cache dump file.");
|
||
|
}
|
||
|
data->append(result_.data(), result_.size());
|
||
|
|
||
|
offset_ += to_read;
|
||
|
bytes_to_read -= to_read;
|
||
|
to_read = bytes_to_read > kDumpReaderBufferSize ? kDumpReaderBufferSize
|
||
|
: bytes_to_read;
|
||
|
}
|
||
|
return io_s;
|
||
|
}
|
||
|
std::unique_ptr<RandomAccessFileReader> file_reader_;
|
||
|
Slice result_;
|
||
|
size_t offset_;
|
||
|
char* buffer_;
|
||
|
};
|
||
|
|
||
|
// The cache dump and load helper class
|
||
|
class CacheDumperHelper {
|
||
|
public:
|
||
|
// serilize the dump_unit_meta to a string, it is fixed 16 bytes size.
|
||
|
static void EncodeDumpUnitMeta(const DumpUnitMeta& meta, std::string* data) {
|
||
|
assert(data);
|
||
|
PutFixed32(data, static_cast<uint32_t>(meta.sequence_num));
|
||
|
PutFixed32(data, static_cast<uint32_t>(meta.dump_unit_checksum));
|
||
|
PutFixed64(data, meta.dump_unit_size);
|
||
|
}
|
||
|
|
||
|
// Serilize the dump_unit to a string.
|
||
|
static void EncodeDumpUnit(const DumpUnit& dump_unit, std::string* data) {
|
||
|
assert(data);
|
||
|
PutFixed64(data, dump_unit.timestamp);
|
||
|
data->push_back(dump_unit.type);
|
||
|
PutLengthPrefixedSlice(data, dump_unit.key);
|
||
|
PutFixed32(data, static_cast<uint32_t>(dump_unit.value_len));
|
||
|
PutFixed32(data, dump_unit.value_checksum);
|
||
|
PutLengthPrefixedSlice(data,
|
||
|
Slice((char*)dump_unit.value, dump_unit.value_len));
|
||
|
}
|
||
|
|
||
|
// Deserilize the dump_unit_meta from a string
|
||
|
static Status DecodeDumpUnitMeta(const std::string& encoded_data,
|
||
|
DumpUnitMeta* unit_meta) {
|
||
|
assert(unit_meta != nullptr);
|
||
|
Slice encoded_slice = Slice(encoded_data);
|
||
|
if (!GetFixed32(&encoded_slice, &(unit_meta->sequence_num))) {
|
||
|
return Status::Incomplete("Decode dumped unit meta sequence_num failed");
|
||
|
}
|
||
|
if (!GetFixed32(&encoded_slice, &(unit_meta->dump_unit_checksum))) {
|
||
|
return Status::Incomplete(
|
||
|
"Decode dumped unit meta dump_unit_checksum failed");
|
||
|
}
|
||
|
if (!GetFixed64(&encoded_slice, &(unit_meta->dump_unit_size))) {
|
||
|
return Status::Incomplete(
|
||
|
"Decode dumped unit meta dump_unit_size failed");
|
||
|
}
|
||
|
return Status::OK();
|
||
|
}
|
||
|
|
||
|
// Deserilize the dump_unit from a string.
|
||
|
static Status DecodeDumpUnit(const std::string& encoded_data,
|
||
|
DumpUnit* dump_unit) {
|
||
|
assert(dump_unit != nullptr);
|
||
|
Slice encoded_slice = Slice(encoded_data);
|
||
|
|
||
|
// Decode timestamp
|
||
|
if (!GetFixed64(&encoded_slice, &dump_unit->timestamp)) {
|
||
|
return Status::Incomplete("Decode dumped unit string failed");
|
||
|
}
|
||
|
// Decode the block type
|
||
|
dump_unit->type = static_cast<CacheDumpUnitType>(encoded_slice[0]);
|
||
|
encoded_slice.remove_prefix(1);
|
||
|
// Decode the key
|
||
|
if (!GetLengthPrefixedSlice(&encoded_slice, &(dump_unit->key))) {
|
||
|
return Status::Incomplete("Decode dumped unit string failed");
|
||
|
}
|
||
|
// Decode the value size
|
||
|
uint32_t value_len;
|
||
|
if (!GetFixed32(&encoded_slice, &value_len)) {
|
||
|
return Status::Incomplete("Decode dumped unit string failed");
|
||
|
}
|
||
|
dump_unit->value_len = static_cast<size_t>(value_len);
|
||
|
// Decode the value checksum
|
||
|
if (!GetFixed32(&encoded_slice, &(dump_unit->value_checksum))) {
|
||
|
return Status::Incomplete("Decode dumped unit string failed");
|
||
|
}
|
||
|
// Decode the block content and copy to the memory space whose pointer
|
||
|
// will be managed by the cache finally.
|
||
|
Slice block;
|
||
|
if (!GetLengthPrefixedSlice(&encoded_slice, &block)) {
|
||
|
return Status::Incomplete("Decode dumped unit string failed");
|
||
|
}
|
||
|
dump_unit->value = (void*)block.data();
|
||
|
assert(block.size() == dump_unit->value_len);
|
||
|
return Status::OK();
|
||
|
}
|
||
|
};
|
||
|
|
||
|
} // namespace ROCKSDB_NAMESPACE
|
||
|
#endif // ROCKSDB_LITE
|