Return error if Get() fails in Prefetching Filter blocks (#7463)

Summary:
Right now all I/O failures under
PartitionFilterBlock::CacheDependencies() is swallowed. Return error in
case prefetch fails.

Pull Request resolved: https://github.com/facebook/rocksdb/pull/7463

Test Plan: make check -j64

Reviewed By: anand1976

Differential Revision: D24008226

Pulled By: akankshamahajan15

fbshipit-source-id: b65d63b2d01465db92500b78de7ad58650ec9b3b
main
Akanksha Mahajan 4 years ago committed by Facebook GitHub Bot
parent 668ee08915
commit 7d503e66a9
  1. 12
      file/file_prefetch_buffer.cc
  2. 9
      table/block_based/block_based_table_reader.cc
  3. 4
      table/block_based/filter_block.h
  4. 27
      table/block_based/partitioned_filter_block.cc
  5. 2
      table/block_based/partitioned_filter_block.h

@ -91,17 +91,19 @@ Status FilePrefetchBuffer::Prefetch(const IOOptions& opts,
size_t read_len = static_cast<size_t>(roundup_len - chunk_len); size_t read_len = static_cast<size_t>(roundup_len - chunk_len);
s = reader->Read(opts, rounddown_offset + chunk_len, read_len, &result, s = reader->Read(opts, rounddown_offset + chunk_len, read_len, &result,
buffer_.BufferStart() + chunk_len, nullptr, for_compaction); buffer_.BufferStart() + chunk_len, nullptr, for_compaction);
if (!s.ok()) {
return s;
}
#ifndef NDEBUG #ifndef NDEBUG
if (!s.ok() || result.size() < read_len) { if (result.size() < read_len) {
// Fake an IO error to force db_stress fault injection to ignore // Fake an IO error to force db_stress fault injection to ignore
// truncated read errors // truncated read errors
IGNORE_STATUS_IF_ERROR(Status::IOError()); IGNORE_STATUS_IF_ERROR(Status::IOError());
} }
#endif #endif
if (s.ok()) { buffer_offset_ = rounddown_offset;
buffer_offset_ = rounddown_offset; buffer_.Size(static_cast<size_t>(chunk_len) + result.size());
buffer_.Size(static_cast<size_t>(chunk_len) + result.size());
}
return s; return s;
} }

@ -1038,13 +1038,16 @@ Status BlockBasedTable::PrefetchIndexAndFilterBlocks(
auto filter = new_table->CreateFilterBlockReader( auto filter = new_table->CreateFilterBlockReader(
ro, prefetch_buffer, use_cache, prefetch_filter, pin_filter, ro, prefetch_buffer, use_cache, prefetch_filter, pin_filter,
lookup_context); lookup_context);
if (filter) { if (filter) {
rep_->filter = std::move(filter);
// Refer to the comment above about paritioned indexes always being cached // Refer to the comment above about paritioned indexes always being cached
if (prefetch_all) { if (prefetch_all) {
filter->CacheDependencies(ro, pin_all); s = rep_->filter->CacheDependencies(ro, pin_all);
if (!s.ok()) {
return s;
}
} }
rep_->filter = std::move(filter);
} }
} }

@ -153,7 +153,9 @@ class FilterBlockReader {
return error_msg; return error_msg;
} }
virtual void CacheDependencies(const ReadOptions& /*ro*/, bool /*pin*/) {} virtual Status CacheDependencies(const ReadOptions& /*ro*/, bool /*pin*/) {
return Status::OK();
}
virtual bool RangeMayExist(const Slice* /*iterate_upper_bound*/, virtual bool RangeMayExist(const Slice* /*iterate_upper_bound*/,
const Slice& user_key, const Slice& user_key,

@ -412,8 +412,8 @@ size_t PartitionedFilterBlockReader::ApproximateMemoryUsage() const {
} }
// TODO(myabandeh): merge this with the same function in IndexReader // TODO(myabandeh): merge this with the same function in IndexReader
void PartitionedFilterBlockReader::CacheDependencies(const ReadOptions& ro, Status PartitionedFilterBlockReader::CacheDependencies(const ReadOptions& ro,
bool pin) { bool pin) {
assert(table()); assert(table());
const BlockBasedTable::Rep* const rep = table()->get_rep(); const BlockBasedTable::Rep* const rep = table()->get_rep();
@ -426,12 +426,11 @@ void PartitionedFilterBlockReader::CacheDependencies(const ReadOptions& ro,
Status s = GetOrReadFilterBlock(false /* no_io */, nullptr /* get_context */, Status s = GetOrReadFilterBlock(false /* no_io */, nullptr /* get_context */,
&lookup_context, &filter_block); &lookup_context, &filter_block);
if (!s.ok()) { if (!s.ok()) {
ROCKS_LOG_WARN(rep->ioptions.info_log, ROCKS_LOG_ERROR(rep->ioptions.info_log,
"Error retrieving top-level filter block while trying to " "Error retrieving top-level filter block while trying to "
"cache filter partitions: %s", "cache filter partitions: %s",
s.ToString().c_str()); s.ToString().c_str());
IGNORE_STATUS_IF_ERROR(s); return s;
return;
} }
// Before read partitions, prefetch them to avoid lots of IOs // Before read partitions, prefetch them to avoid lots of IOs
@ -465,6 +464,9 @@ void PartitionedFilterBlockReader::CacheDependencies(const ReadOptions& ro,
s = prefetch_buffer->Prefetch(opts, rep->file.get(), prefetch_off, s = prefetch_buffer->Prefetch(opts, rep->file.get(), prefetch_off,
static_cast<size_t>(prefetch_len)); static_cast<size_t>(prefetch_len));
} }
if (!s.ok()) {
return s;
}
// After prefetch, read the partitions one by one // After prefetch, read the partitions one by one
for (biter.SeekToFirst(); biter.Valid(); biter.Next()) { for (biter.SeekToFirst(); biter.Valid(); biter.Next()) {
@ -477,17 +479,20 @@ void PartitionedFilterBlockReader::CacheDependencies(const ReadOptions& ro,
prefetch_buffer.get(), ro, handle, UncompressionDict::GetEmptyDict(), prefetch_buffer.get(), ro, handle, UncompressionDict::GetEmptyDict(),
&block, BlockType::kFilter, nullptr /* get_context */, &lookup_context, &block, BlockType::kFilter, nullptr /* get_context */, &lookup_context,
nullptr /* contents */); nullptr /* contents */);
if (!s.ok()) {
return s;
}
assert(s.ok() || block.GetValue() == nullptr); assert(s.ok() || block.GetValue() == nullptr);
if (s.ok() && block.GetValue() != nullptr) {
if (block.GetValue() != nullptr) {
if (block.IsCached()) { if (block.IsCached()) {
if (pin) { if (pin) {
filter_map_[handle.offset()] = std::move(block); filter_map_[handle.offset()] = std::move(block);
} }
} }
} }
IGNORE_STATUS_IF_ERROR(s);
} }
return biter.status();
} }
const InternalKeyComparator* PartitionedFilterBlockReader::internal_comparator() const InternalKeyComparator* PartitionedFilterBlockReader::internal_comparator()

@ -130,7 +130,7 @@ class PartitionedFilterBlockReader : public FilterBlockReaderCommon<Block> {
uint64_t block_offset, BlockHandle filter_handle, uint64_t block_offset, BlockHandle filter_handle,
bool no_io, BlockCacheLookupContext* lookup_context, bool no_io, BlockCacheLookupContext* lookup_context,
FilterManyFunction filter_function) const; FilterManyFunction filter_function) const;
void CacheDependencies(const ReadOptions& ro, bool pin) override; Status CacheDependencies(const ReadOptions& ro, bool pin) override;
const InternalKeyComparator* internal_comparator() const; const InternalKeyComparator* internal_comparator() const;
bool index_key_includes_seq() const; bool index_key_includes_seq() const;

Loading…
Cancel
Save