From 9a8e3f73ed44ec383b59b88a9448bf833767c75a Mon Sep 17 00:00:00 2001 From: sdong Date: Wed, 23 Dec 2015 17:30:10 -0800 Subject: [PATCH] plain table reader: non-mmap mode to keep two recent buffers Summary: In plain table reader's non-mmap mode, we only keep the most recent read buffer. However, for binary search, it is likely we come back to a location to read. To avoid one pread in such a case, we keep two read buffers. It should cover most of the cases. Test Plan: 1. run tests 2. check the optimization works through strace when running ./table_reader_bench -mmap_read=false --num_keys2=1 -num_keys1=5000 -table_factory=plain_table --iterator --through_db Reviewers: anthony, rven, kradhakrishnan, igor, yhchiang, IslamAbdelRahman Reviewed By: IslamAbdelRahman Subscribers: leveldb, dhruba Differential Revision: https://reviews.facebook.net/D51171 --- db/plain_table_db_test.cc | 54 ++++++++++++++++ table/plain_table_key_coding.cc | 93 +++++++++++++++------------ table/plain_table_key_coding.h | 107 +++++++++++++++++++++----------- table/table_reader_bench.cc | 3 +- util/testutil.h | 9 ++- 5 files changed, 188 insertions(+), 78 deletions(-) diff --git a/db/plain_table_db_test.cc b/db/plain_table_db_test.cc index d9c00823679..ff07def9690 100644 --- a/db/plain_table_db_test.cc +++ b/db/plain_table_db_test.cc @@ -29,6 +29,7 @@ #include "table/bloom_block.h" #include "table/table_builder.h" #include "table/plain_table_factory.h" +#include "table/plain_table_key_coding.h" #include "table/plain_table_reader.h" #include "util/hash.h" #include "util/logging.h" @@ -41,6 +42,59 @@ using std::unique_ptr; namespace rocksdb { +class PlainTableKeyDecoderTest : public testing::Test {}; + +TEST_F(PlainTableKeyDecoderTest, ReadNonMmap) { + std::string tmp; + Random rnd(301); + const uint32_t kLength = 2222; + Slice contents = test::RandomString(&rnd, kLength, &tmp); + test::StringSource* string_source = + new test::StringSource(contents, 0, false); + + unique_ptr file_reader( + test::GetRandomAccessFileReader(string_source)); + unique_ptr file_info(new PlainTableReaderFileInfo( + std::move(file_reader), EnvOptions(), kLength)); + + { + PlainTableFileReader reader(file_info.get()); + + const uint32_t kReadSize = 77; + for (uint32_t pos = 0; pos < kLength; pos += kReadSize) { + uint32_t read_size = std::min(kLength - pos, kReadSize); + Slice out; + ASSERT_TRUE(reader.Read(pos, read_size, &out)); + ASSERT_EQ(0, out.compare(tmp.substr(pos, read_size))); + } + + ASSERT_LT(string_source->total_reads(), kLength / kReadSize / 2); + } + + std::vector>> reads = { + {{600, 30}, {590, 30}, {600, 20}, {600, 40}}, + {{800, 20}, {100, 20}, {500, 20}, {1500, 20}, {100, 20}, {80, 20}}, + {{1000, 20}, {500, 20}, {1000, 50}}, + {{1000, 20}, {500, 20}, {500, 20}}, + {{1000, 20}, {500, 20}, {200, 20}, {500, 20}}, + {{1000, 20}, {500, 20}, {200, 20}, {1000, 50}}, + {{600, 500}, {610, 20}, {100, 20}}, + {{500, 100}, {490, 100}, {550, 50}}, + }; + + std::vector num_file_reads = {2, 6, 2, 2, 4, 3, 2, 2}; + + for (size_t i = 0; i < reads.size(); i++) { + string_source->set_total_reads(0); + PlainTableFileReader reader(file_info.get()); + for (auto p : reads[i]) { + Slice out; + ASSERT_TRUE(reader.Read(p.first, p.second, &out)); + ASSERT_EQ(0, out.compare(tmp.substr(p.first, p.second))); + } + ASSERT_EQ(num_file_reads[i], string_source->total_reads()); + } +} class PlainTableDBTest : public testing::Test, public testing::WithParamInterface { diff --git a/table/plain_table_key_coding.cc b/table/plain_table_key_coding.cc index 057c7f90ffb..b1aa14d8d45 100644 --- a/table/plain_table_key_coding.cc +++ b/table/plain_table_key_coding.cc @@ -164,47 +164,62 @@ Status PlainTableKeyEncoder::AppendKey(const Slice& key, return Status::OK(); } -inline bool PlainTableKeyDecoder::FileReader::Read(uint32_t file_offset, - uint32_t len, Slice* out) { - if (file_info_->is_mmap_mode) { - assert(file_offset + len <= file_info_->data_end_offset); - *out = Slice(file_info_->file_data.data() + file_offset, len); - return true; - } else { - return ReadNonMmap(file_offset, len, out); - } +Slice PlainTableFileReader::GetFromBuffer(Buffer* buffer, uint32_t file_offset, + uint32_t len) { + assert(file_offset + len <= file_info_->data_end_offset); + return Slice(buffer->buf.get() + (file_offset - buffer->buf_start_offset), + len); } -bool PlainTableKeyDecoder::FileReader::ReadNonMmap(uint32_t file_offset, - uint32_t len, Slice* out) { +bool PlainTableFileReader::ReadNonMmap(uint32_t file_offset, uint32_t len, + Slice* out) { const uint32_t kPrefetchSize = 256u; - if (file_offset < buf_start_offset_ || - file_offset + len > buf_start_offset_ + buf_len_) { - // Load buffer - assert(file_offset + len <= file_info_->data_end_offset); - uint32_t size_to_read = std::min(file_info_->data_end_offset - file_offset, - std::max(kPrefetchSize, len)); - if (size_to_read > buf_capacity_) { - buf_.reset(new char[size_to_read]); - buf_capacity_ = size_to_read; - buf_len_ = 0; - } - Slice read_result; - Status s = file_info_->file->Read(file_offset, size_to_read, &read_result, - buf_.get()); - if (!s.ok()) { - status_ = s; - return false; + + // Try to read from buffers. + for (uint32_t i = 0; i < num_buf_; i++) { + Buffer* buffer = buffers_[num_buf_ - 1 - i].get(); + if (file_offset >= buffer->buf_start_offset && + file_offset + len <= buffer->buf_start_offset + buffer->buf_len) { + *out = GetFromBuffer(buffer, file_offset, len); + return true; } - buf_start_offset_ = file_offset; - buf_len_ = size_to_read; } - *out = Slice(buf_.get() + (file_offset - buf_start_offset_), len); + + Buffer* new_buffer; + // Data needed is not in any of the buffer. Allocate a new buffer. + if (num_buf_ < buffers_.size()) { + // Add a new buffer + new_buffer = new Buffer(); + buffers_[num_buf_++].reset(new_buffer); + } else { + // Now simply replace the last buffer. Can improve the placement policy + // if needed. + new_buffer = buffers_[num_buf_ - 1].get(); + } + + assert(file_offset + len <= file_info_->data_end_offset); + uint32_t size_to_read = std::min(file_info_->data_end_offset - file_offset, + std::max(kPrefetchSize, len)); + if (size_to_read > new_buffer->buf_capacity) { + new_buffer->buf.reset(new char[size_to_read]); + new_buffer->buf_capacity = size_to_read; + new_buffer->buf_len = 0; + } + Slice read_result; + Status s = file_info_->file->Read(file_offset, size_to_read, &read_result, + new_buffer->buf.get()); + if (!s.ok()) { + status_ = s; + return false; + } + new_buffer->buf_start_offset = file_offset; + new_buffer->buf_len = size_to_read; + *out = GetFromBuffer(new_buffer, file_offset, len); return true; } -inline bool PlainTableKeyDecoder::FileReader::ReadVarint32( - uint32_t offset, uint32_t* out, uint32_t* bytes_read) { +inline bool PlainTableFileReader::ReadVarint32(uint32_t offset, uint32_t* out, + uint32_t* bytes_read) { if (file_info_->is_mmap_mode) { const char* start = file_info_->file_data.data() + offset; const char* limit = @@ -218,8 +233,8 @@ inline bool PlainTableKeyDecoder::FileReader::ReadVarint32( } } -bool PlainTableKeyDecoder::FileReader::ReadVarint32NonMmap( - uint32_t offset, uint32_t* out, uint32_t* bytes_read) { +bool PlainTableFileReader::ReadVarint32NonMmap(uint32_t offset, uint32_t* out, + uint32_t* bytes_read) { const char* start; const char* limit; const uint32_t kMaxVarInt32Size = 6u; @@ -298,7 +313,7 @@ Status PlainTableKeyDecoder::NextPlainEncodingKey(uint32_t start_offset, if (!s.ok()) { return s; } - if (!file_reader_.file_info_->is_mmap_mode) { + if (!file_reader_.file_info()->is_mmap_mode) { cur_key_.SetInternalKey(*parsed_key); parsed_key->user_key = Slice(cur_key_.GetKey().data(), user_key_size); if (internal_key != nullptr) { @@ -348,14 +363,14 @@ Status PlainTableKeyDecoder::NextPrefixEncodingKey( if (!s.ok()) { return s; } - if (!file_reader_.file_info_->is_mmap_mode || + if (!file_reader_.file_info()->is_mmap_mode || (internal_key != nullptr && !decoded_internal_key_valid)) { // In non-mmap mode, always need to make a copy of keys returned to // users, because after reading value for the key, the key might // be invalid. cur_key_.SetInternalKey(*parsed_key); saved_user_key_ = cur_key_.GetKey(); - if (!file_reader_.file_info_->is_mmap_mode) { + if (!file_reader_.file_info()->is_mmap_mode) { parsed_key->user_key = Slice(cur_key_.GetKey().data(), size); } if (internal_key != nullptr) { @@ -394,7 +409,7 @@ Status PlainTableKeyDecoder::NextPrefixEncodingKey( if (!s.ok()) { return s; } - if (!file_reader_.file_info_->is_mmap_mode) { + if (!file_reader_.file_info()->is_mmap_mode) { // In non-mmap mode, we need to make a copy of keys returned to // users, because after reading value for the key, the key might // be invalid. diff --git a/table/plain_table_key_coding.h b/table/plain_table_key_coding.h index a98010d5b75..c44ddbdebdd 100644 --- a/table/plain_table_key_coding.h +++ b/table/plain_table_key_coding.h @@ -8,6 +8,7 @@ #include "rocksdb/slice.h" #include "db/dbformat.h" +#include "table/plain_table_reader.h" namespace rocksdb { @@ -51,6 +52,74 @@ class PlainTableKeyEncoder { IterKey pre_prefix_; }; +class PlainTableFileReader { + public: + explicit PlainTableFileReader(const PlainTableReaderFileInfo* _file_info) + : file_info_(_file_info), num_buf_(0) {} + // In mmaped mode, the results point to mmaped area of the file, which + // means it is always valid before closing the file. + // In non-mmap mode, the results point to an internal buffer. If the caller + // makes another read call, the results may not be valid. So callers should + // make a copy when needed. + // In order to save read calls to files, we keep two internal buffers: + // the first read and the most recent read. This is efficient because it + // columns these two common use cases: + // (1) hash index only identify one location, we read the key to verify + // the location, and read key and value if it is the right location. + // (2) after hash index checking, we identify two locations (because of + // hash bucket conflicts), we binary search the two location to see + // which one is what we need and start to read from the location. + // These two most common use cases will be covered by the two buffers + // so that we don't need to re-read the same location. + // Currently we keep a fixed size buffer. If a read doesn't exactly fit + // the buffer, we replace the second buffer with the location user reads. + // + // If return false, status code is stored in status_. + bool Read(uint32_t file_offset, uint32_t len, Slice* out) { + if (file_info_->is_mmap_mode) { + assert(file_offset + len <= file_info_->data_end_offset); + *out = Slice(file_info_->file_data.data() + file_offset, len); + return true; + } else { + return ReadNonMmap(file_offset, len, out); + } + } + + // If return false, status code is stored in status_. + bool ReadNonMmap(uint32_t file_offset, uint32_t len, Slice* output); + + // *bytes_read = 0 means eof. false means failure and status is saved + // in status_. Not directly returning Status to save copying status + // object to map previous performance of mmap mode. + inline bool ReadVarint32(uint32_t offset, uint32_t* output, + uint32_t* bytes_read); + + bool ReadVarint32NonMmap(uint32_t offset, uint32_t* output, + uint32_t* bytes_read); + + Status status() const { return status_; } + + const PlainTableReaderFileInfo* file_info() { return file_info_; } + + private: + const PlainTableReaderFileInfo* file_info_; + + struct Buffer { + Buffer() : buf_start_offset(0), buf_len(0), buf_capacity(0) {} + std::unique_ptr buf; + uint32_t buf_start_offset; + uint32_t buf_len; + uint32_t buf_capacity; + }; + + // Keep buffers for two recent reads. + std::array, 2> buffers_; + uint32_t num_buf_; + Status status_; + + Slice GetFromBuffer(Buffer* buf, uint32_t file_offset, uint32_t len); +}; + // A helper class to decode keys from input buffer // Actual data format of the key is documented in plain_table_factory.h class PlainTableKeyDecoder { @@ -82,43 +151,7 @@ class PlainTableKeyDecoder { Slice* internal_key, uint32_t* bytes_read, bool* seekable = nullptr); - class FileReader { - public: - explicit FileReader(const PlainTableReaderFileInfo* file_info) - : file_info_(file_info), - buf_start_offset_(0), - buf_len_(0), - buf_capacity_(0) {} - // In mmaped mode, the results point to mmaped area of the file, which - // means it is always valid before closing the file. - // In non-mmap mode, the results point to an internal buffer. If the caller - // makes another read call, the results will not be valid. So callers should - // make a copy when needed. - // If return false, status code is stored in status_. - inline bool Read(uint32_t file_offset, uint32_t len, Slice* output); - - // If return false, status code is stored in status_. - bool ReadNonMmap(uint32_t file_offset, uint32_t len, Slice* output); - - // *bytes_read = 0 means eof. false means failure and status is saved - // in status_. Not directly returning Status to save copying status - // object to map previous performance of mmap mode. - inline bool ReadVarint32(uint32_t offset, uint32_t* output, - uint32_t* bytes_read); - - bool ReadVarint32NonMmap(uint32_t offset, uint32_t* output, - uint32_t* bytes_read); - - Status status() const { return status_; } - - const PlainTableReaderFileInfo* file_info_; - std::unique_ptr buf_; - uint32_t buf_start_offset_; - uint32_t buf_len_; - uint32_t buf_capacity_; - Status status_; - }; - FileReader file_reader_; + PlainTableFileReader file_reader_; EncodingType encoding_type_; uint32_t prefix_len_; uint32_t fixed_user_key_len_; diff --git a/table/table_reader_bench.cc b/table/table_reader_bench.cc index c93e2c5f415..cee0d10e39c 100644 --- a/table/table_reader_bench.cc +++ b/table/table_reader_bench.cc @@ -204,7 +204,8 @@ void TableReaderBenchmark(Options& opts, EnvOptions& env_options, } // verify key; total_time += Now(env, measured_by_nanosecond) - start_time; - assert(Slice(MakeKey(r1, r2 + count, through_db)) == iter->key()); + assert(Slice(MakeKey(r1, r2 + count, through_db)) == + (through_db ? iter->key() : iiter->key())); start_time = Now(env, measured_by_nanosecond); if (++count >= r2_len) { break; diff --git a/util/testutil.h b/util/testutil.h index 68e2f8c9e84..d8b4f0ca212 100644 --- a/util/testutil.h +++ b/util/testutil.h @@ -237,7 +237,8 @@ class StringSource: public RandomAccessFile { bool mmap = false) : contents_(contents.data(), contents.size()), uniq_id_(uniq_id), - mmap_(mmap) {} + mmap_(mmap), + total_reads_(0) {} virtual ~StringSource() { } @@ -245,6 +246,7 @@ class StringSource: public RandomAccessFile { virtual Status Read(uint64_t offset, size_t n, Slice* result, char* scratch) const override { + total_reads_++; if (offset > contents_.size()) { return Status::InvalidArgument("invalid Read offset"); } @@ -271,10 +273,15 @@ class StringSource: public RandomAccessFile { return static_cast(rid-id); } + int total_reads() const { return total_reads_; } + + void set_total_reads(int tr) { total_reads_ = tr; } + private: std::string contents_; uint64_t uniq_id_; bool mmap_; + mutable int total_reads_; }; class NullLogger : public Logger {