2017-12-11 15:16:37 -08:00
|
|
|
// 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).
|
|
|
|
//
|
|
|
|
// Copyright (c) 2011 The LevelDB Authors. All rights reserved.
|
|
|
|
// Use of this source code is governed by a BSD-style license that can be
|
|
|
|
// found in the LICENSE file. See the AUTHORS file for names of contributors.
|
|
|
|
|
|
|
|
#pragma once
|
2019-05-30 17:39:43 -07:00
|
|
|
#include "memory/memory_allocator.h"
|
2019-05-30 14:47:29 -07:00
|
|
|
#include "table/block_based/block.h"
|
2019-06-18 19:00:03 -07:00
|
|
|
#include "table/block_based/block_type.h"
|
2017-12-11 15:16:37 -08:00
|
|
|
#include "table/format.h"
|
2018-10-02 17:21:54 -07:00
|
|
|
|
2020-02-20 12:07:53 -08:00
|
|
|
namespace ROCKSDB_NAMESPACE {
|
2019-05-23 14:19:12 -07:00
|
|
|
|
|
|
|
// Retrieves a single block of a given file. Utilizes the prefetch buffer and/or
|
|
|
|
// persistent cache provided (if any) to try to avoid reading from the file
|
|
|
|
// directly. Note that both the prefetch buffer and the persistent cache are
|
|
|
|
// optional; also, note that the persistent cache may be configured to store either
|
|
|
|
// compressed or uncompressed blocks.
|
|
|
|
//
|
|
|
|
// If the retrieved block is compressed and the do_uncompress flag is set,
|
|
|
|
// BlockFetcher uncompresses the block (using the uncompression dictionary,
|
|
|
|
// if provided, to prime the compression algorithm), and returns the resulting
|
|
|
|
// uncompressed block data. Otherwise, it returns the original block.
|
|
|
|
//
|
|
|
|
// Two read options affect the behavior of BlockFetcher: if verify_checksums is
|
|
|
|
// true, the checksum of the (original) block is checked; if fill_cache is true,
|
|
|
|
// the block is added to the persistent cache if needed.
|
|
|
|
//
|
|
|
|
// Memory for uncompressed and compressed blocks is allocated as needed
|
|
|
|
// using memory_allocator and memory_allocator_compressed, respectively
|
|
|
|
// (if provided; otherwise, the default allocator is used).
|
|
|
|
|
2017-12-11 15:16:37 -08:00
|
|
|
class BlockFetcher {
|
|
|
|
public:
|
|
|
|
BlockFetcher(RandomAccessFileReader* file,
|
|
|
|
FilePrefetchBuffer* prefetch_buffer, const Footer& footer,
|
|
|
|
const ReadOptions& read_options, const BlockHandle& handle,
|
2021-05-05 13:59:21 -07:00
|
|
|
BlockContents* contents, const ImmutableOptions& ioptions,
|
2019-06-18 19:00:03 -07:00
|
|
|
bool do_uncompress, bool maybe_compressed, BlockType block_type,
|
2019-01-23 18:11:08 -08:00
|
|
|
const UncompressionDict& uncompression_dict,
|
2018-06-27 17:09:29 -07:00
|
|
|
const PersistentCacheOptions& cache_options,
|
2018-11-28 17:58:08 -08:00
|
|
|
MemoryAllocator* memory_allocator = nullptr,
|
2019-06-19 14:07:36 -07:00
|
|
|
MemoryAllocator* memory_allocator_compressed = nullptr,
|
|
|
|
bool for_compaction = false)
|
2017-12-11 15:16:37 -08:00
|
|
|
: file_(file),
|
|
|
|
prefetch_buffer_(prefetch_buffer),
|
|
|
|
footer_(footer),
|
|
|
|
read_options_(read_options),
|
|
|
|
handle_(handle),
|
|
|
|
contents_(contents),
|
|
|
|
ioptions_(ioptions),
|
|
|
|
do_uncompress_(do_uncompress),
|
2018-11-28 17:58:08 -08:00
|
|
|
maybe_compressed_(maybe_compressed),
|
2019-06-18 19:00:03 -07:00
|
|
|
block_type_(block_type),
|
2020-04-24 15:30:12 -07:00
|
|
|
block_size_(static_cast<size_t>(handle_.size())),
|
|
|
|
block_size_with_trailer_(block_size(handle_)),
|
2019-01-23 18:11:08 -08:00
|
|
|
uncompression_dict_(uncompression_dict),
|
2018-10-02 17:21:54 -07:00
|
|
|
cache_options_(cache_options),
|
2018-11-28 17:58:08 -08:00
|
|
|
memory_allocator_(memory_allocator),
|
2019-06-19 14:07:36 -07:00
|
|
|
memory_allocator_compressed_(memory_allocator_compressed),
|
|
|
|
for_compaction_(for_compaction) {}
|
|
|
|
|
2021-04-01 10:06:55 -07:00
|
|
|
IOStatus ReadBlockContents();
|
2018-11-13 17:00:49 -08:00
|
|
|
CompressionType get_compression_type() const { return compression_type_; }
|
2017-12-11 15:16:37 -08:00
|
|
|
|
2020-04-24 15:30:12 -07:00
|
|
|
#ifndef NDEBUG
|
|
|
|
int TEST_GetNumStackBufMemcpy() const { return num_stack_buf_memcpy_; }
|
|
|
|
int TEST_GetNumHeapBufMemcpy() const { return num_heap_buf_memcpy_; }
|
|
|
|
int TEST_GetNumCompressedBufMemcpy() const {
|
|
|
|
return num_compressed_buf_memcpy_;
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2017-12-11 15:16:37 -08:00
|
|
|
private:
|
2020-04-24 15:30:12 -07:00
|
|
|
#ifndef NDEBUG
|
|
|
|
int num_stack_buf_memcpy_ = 0;
|
|
|
|
int num_heap_buf_memcpy_ = 0;
|
|
|
|
int num_compressed_buf_memcpy_ = 0;
|
|
|
|
|
|
|
|
#endif
|
2017-12-11 15:16:37 -08:00
|
|
|
static const uint32_t kDefaultStackBufferSize = 5000;
|
|
|
|
|
|
|
|
RandomAccessFileReader* file_;
|
|
|
|
FilePrefetchBuffer* prefetch_buffer_;
|
|
|
|
const Footer& footer_;
|
2017-12-12 12:08:52 -08:00
|
|
|
const ReadOptions read_options_;
|
2017-12-11 15:16:37 -08:00
|
|
|
const BlockHandle& handle_;
|
|
|
|
BlockContents* contents_;
|
2021-05-05 13:59:21 -07:00
|
|
|
const ImmutableOptions& ioptions_;
|
2020-04-24 15:30:12 -07:00
|
|
|
const bool do_uncompress_;
|
|
|
|
const bool maybe_compressed_;
|
|
|
|
const BlockType block_type_;
|
|
|
|
const size_t block_size_;
|
|
|
|
const size_t block_size_with_trailer_;
|
2019-01-23 18:11:08 -08:00
|
|
|
const UncompressionDict& uncompression_dict_;
|
2017-12-11 15:16:37 -08:00
|
|
|
const PersistentCacheOptions& cache_options_;
|
2018-11-28 17:58:08 -08:00
|
|
|
MemoryAllocator* memory_allocator_;
|
|
|
|
MemoryAllocator* memory_allocator_compressed_;
|
2021-04-01 10:06:55 -07:00
|
|
|
IOStatus io_status_;
|
2017-12-11 15:16:37 -08:00
|
|
|
Slice slice_;
|
|
|
|
char* used_buf_ = nullptr;
|
2020-04-24 15:30:12 -07:00
|
|
|
AlignedBuf direct_io_buf_;
|
2018-10-02 17:21:54 -07:00
|
|
|
CacheAllocationPtr heap_buf_;
|
2018-11-28 17:58:08 -08:00
|
|
|
CacheAllocationPtr compressed_buf_;
|
2017-12-11 15:16:37 -08:00
|
|
|
char stack_buf_[kDefaultStackBufferSize];
|
|
|
|
bool got_from_prefetch_buffer_ = false;
|
2020-04-24 15:30:12 -07:00
|
|
|
CompressionType compression_type_;
|
2019-06-19 14:07:36 -07:00
|
|
|
bool for_compaction_ = false;
|
2017-12-11 15:16:37 -08:00
|
|
|
|
|
|
|
// return true if found
|
|
|
|
bool TryGetUncompressBlockFromPersistentCache();
|
|
|
|
// return true if found
|
|
|
|
bool TryGetFromPrefetchBuffer();
|
|
|
|
bool TryGetCompressedBlockFromPersistentCache();
|
|
|
|
void PrepareBufferForBlockFromFile();
|
2020-04-24 15:30:12 -07:00
|
|
|
// Copy content from used_buf_ to new heap_buf_.
|
|
|
|
void CopyBufferToHeapBuf();
|
|
|
|
// Copy content from used_buf_ to new compressed_buf_.
|
|
|
|
void CopyBufferToCompressedBuf();
|
2017-12-11 15:16:37 -08:00
|
|
|
void GetBlockContents();
|
|
|
|
void InsertCompressedBlockToPersistentCacheIfNeeded();
|
|
|
|
void InsertUncompressedBlockToPersistentCacheIfNeeded();
|
|
|
|
void CheckBlockChecksum();
|
|
|
|
};
|
2020-02-20 12:07:53 -08:00
|
|
|
} // namespace ROCKSDB_NAMESPACE
|