Store the filter bits reader alongside the filter block contents (#5936)
Summary: Amongst other things, PR https://github.com/facebook/rocksdb/issues/5504 refactored the filter block readers so that only the filter block contents are stored in the block cache (as opposed to the earlier design where the cache stored the filter block reader itself, leading to potentially dangling pointers and concurrency bugs). However, this change introduced a performance hit since with the new code, the metadata fields are re-parsed upon every access. This patch reunites the block contents with the filter bits reader to eliminate this overhead; since this is still a self-contained pure data object, it is safe to store it in the cache. (Note: this is similar to how the zstd digest is handled.) Pull Request resolved: https://github.com/facebook/rocksdb/pull/5936 Test Plan: make asan_check filter_bench results for the old code: ``` $ ./filter_bench -quick WARNING: Assertions are enabled; benchmarks unnecessarily slow Building... Build avg ns/key: 26.7153 Number of filters: 16669 Total memory (MB): 200.009 Bits/key actual: 10.0647 ---------------------------- Inside queries... Dry run (46b) ns/op: 33.4258 Single filter ns/op: 42.5974 Random filter ns/op: 217.861 ---------------------------- Outside queries... Dry run (25d) ns/op: 32.4217 Single filter ns/op: 50.9855 Random filter ns/op: 219.167 Average FP rate %: 1.13993 ---------------------------- Done. (For more info, run with -legend or -help.) $ ./filter_bench -quick -use_full_block_reader WARNING: Assertions are enabled; benchmarks unnecessarily slow Building... Build avg ns/key: 26.5172 Number of filters: 16669 Total memory (MB): 200.009 Bits/key actual: 10.0647 ---------------------------- Inside queries... Dry run (46b) ns/op: 32.3556 Single filter ns/op: 83.2239 Random filter ns/op: 370.676 ---------------------------- Outside queries... Dry run (25d) ns/op: 32.2265 Single filter ns/op: 93.5651 Random filter ns/op: 408.393 Average FP rate %: 1.13993 ---------------------------- Done. (For more info, run with -legend or -help.) ``` With the new code: ``` $ ./filter_bench -quick WARNING: Assertions are enabled; benchmarks unnecessarily slow Building... Build avg ns/key: 25.4285 Number of filters: 16669 Total memory (MB): 200.009 Bits/key actual: 10.0647 ---------------------------- Inside queries... Dry run (46b) ns/op: 31.0594 Single filter ns/op: 43.8974 Random filter ns/op: 226.075 ---------------------------- Outside queries... Dry run (25d) ns/op: 31.0295 Single filter ns/op: 50.3824 Random filter ns/op: 226.805 Average FP rate %: 1.13993 ---------------------------- Done. (For more info, run with -legend or -help.) $ ./filter_bench -quick -use_full_block_reader WARNING: Assertions are enabled; benchmarks unnecessarily slow Building... Build avg ns/key: 26.5308 Number of filters: 16669 Total memory (MB): 200.009 Bits/key actual: 10.0647 ---------------------------- Inside queries... Dry run (46b) ns/op: 33.2968 Single filter ns/op: 58.6163 Random filter ns/op: 291.434 ---------------------------- Outside queries... Dry run (25d) ns/op: 32.1839 Single filter ns/op: 66.9039 Random filter ns/op: 292.828 Average FP rate %: 1.13993 ---------------------------- Done. (For more info, run with -legend or -help.) ``` Differential Revision: D17991712 Pulled By: ltamasi fbshipit-source-id: 7ea205550217bfaaa1d5158ebd658e5832e60f29main
parent
c53db172a1
commit
29ccf2075c
@ -0,0 +1,22 @@ |
||||
// Copyright (c) 2011-present, Facebook, Inc. 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).
|
||||
//
|
||||
|
||||
#include "table/block_based/parsed_full_filter_block.h" |
||||
#include "rocksdb/filter_policy.h" |
||||
|
||||
namespace rocksdb { |
||||
|
||||
ParsedFullFilterBlock::ParsedFullFilterBlock(const FilterPolicy* filter_policy, |
||||
BlockContents&& contents) |
||||
: block_contents_(std::move(contents)), |
||||
filter_bits_reader_( |
||||
!block_contents_.data.empty() |
||||
? filter_policy->GetFilterBitsReader(block_contents_.data) |
||||
: nullptr) {} |
||||
|
||||
ParsedFullFilterBlock::~ParsedFullFilterBlock() = default; |
||||
|
||||
} // namespace rocksdb
|
@ -0,0 +1,40 @@ |
||||
// Copyright (c) 2011-present, Facebook, Inc. 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 |
||||
|
||||
#include <memory> |
||||
|
||||
#include "table/format.h" |
||||
|
||||
namespace rocksdb { |
||||
|
||||
class FilterBitsReader; |
||||
class FilterPolicy; |
||||
|
||||
// The sharable/cachable part of the full filter.
|
||||
class ParsedFullFilterBlock { |
||||
public: |
||||
ParsedFullFilterBlock(const FilterPolicy* filter_policy, |
||||
BlockContents&& contents); |
||||
~ParsedFullFilterBlock(); |
||||
|
||||
FilterBitsReader* filter_bits_reader() const { |
||||
return filter_bits_reader_.get(); |
||||
} |
||||
|
||||
// TODO: consider memory usage of FullFilterBitsReader
|
||||
size_t ApproximateMemoryUsage() const { |
||||
return block_contents_.ApproximateMemoryUsage(); |
||||
} |
||||
|
||||
bool own_bytes() const { return block_contents_.own_bytes(); } |
||||
|
||||
private: |
||||
BlockContents block_contents_; |
||||
std::unique_ptr<FilterBitsReader> filter_bits_reader_; |
||||
}; |
||||
|
||||
} // namespace rocksdb
|
Loading…
Reference in new issue