// 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 #include #include #include "cache/cache_reservation_manager.h" #include "cache/lru_cache.h" #include "memory/memory_allocator_impl.h" #include "rocksdb/secondary_cache.h" #include "rocksdb/slice.h" #include "rocksdb/status.h" #include "util/compression.h" #include "util/mutexlock.h" namespace ROCKSDB_NAMESPACE { class CompressedSecondaryCacheResultHandle : public SecondaryCacheResultHandle { public: CompressedSecondaryCacheResultHandle(Cache::ObjectPtr value, size_t size) : value_(value), size_(size) {} ~CompressedSecondaryCacheResultHandle() override = default; CompressedSecondaryCacheResultHandle( const CompressedSecondaryCacheResultHandle&) = delete; CompressedSecondaryCacheResultHandle& operator=( const CompressedSecondaryCacheResultHandle&) = delete; bool IsReady() override { return true; } void Wait() override {} Cache::ObjectPtr Value() override { return value_; } size_t Size() override { return size_; } private: Cache::ObjectPtr value_; size_t size_; }; // The CompressedSecondaryCache is a concrete implementation of // rocksdb::SecondaryCache. // // When a block is found from CompressedSecondaryCache::Lookup, we check whether // there is a dummy block with the same key in the primary cache. // 1. If the dummy block exits, we erase the block from // CompressedSecondaryCache and insert it into the primary cache. // 2. If not, we just insert a dummy block into the primary cache // (charging the actual size of the block) and don not erase the block from // CompressedSecondaryCache. A standalone handle is returned to the caller. // // When a block is evicted from the primary cache, we check whether // there is a dummy block with the same key in CompressedSecondaryCache. // 1. If the dummy block exits, the block is inserted into // CompressedSecondaryCache. // 2. If not, we just insert a dummy block (size 0) in CompressedSecondaryCache. // // Users can also cast a pointer to CompressedSecondaryCache and call methods on // it directly, especially custom methods that may be added // in the future. For example - // std::unique_ptr cache = // NewCompressedSecondaryCache(opts); // static_cast(cache.get())->Erase(key); class CompressedSecondaryCache : public SecondaryCache { public: explicit CompressedSecondaryCache( const CompressedSecondaryCacheOptions& opts); ~CompressedSecondaryCache() override; const char* Name() const override { return "CompressedSecondaryCache"; } Status Insert(const Slice& key, Cache::ObjectPtr value, const Cache::CacheItemHelper* helper, bool force_insert) override; Status InsertSaved(const Slice& key, const Slice& saved, CompressionType type, CacheTier source) override; std::unique_ptr Lookup( const Slice& key, const Cache::CacheItemHelper* helper, Cache::CreateContext* create_context, bool /*wait*/, bool advise_erase, Statistics* stats, bool& kept_in_sec_cache) override; bool SupportForceErase() const override { return true; } void Erase(const Slice& key) override; void WaitAll(std::vector /*handles*/) override {} Status SetCapacity(size_t capacity) override; Status GetCapacity(size_t& capacity) override; Status Deflate(size_t decrease) override; Status Inflate(size_t increase) override; std::string GetPrintableOptions() const override; size_t TEST_GetUsage() { return cache_->GetUsage(); } private: friend class CompressedSecondaryCacheTestBase; static constexpr std::array malloc_bin_sizes_{ 128, 256, 512, 1024, 2048, 4096, 8192, 16384}; struct CacheValueChunk { // TODO try "CacheAllocationPtr next;". CacheValueChunk* next; size_t size; // Beginning of the chunk data (MUST BE THE LAST FIELD IN THIS STRUCT!) char data[1]; void Free() { delete[] reinterpret_cast(this); } }; // Split value into chunks to better fit into jemalloc bins. The chunks // are stored in CacheValueChunk and extra charge is needed for each chunk, // so the cache charge is recalculated here. CacheValueChunk* SplitValueIntoChunks(const Slice& value, CompressionType compression_type, size_t& charge); // After merging chunks, the extra charge for each chunk is removed, so // the charge is recalculated. CacheAllocationPtr MergeChunksIntoValue(const void* chunks_head, size_t& charge); bool MaybeInsertDummy(const Slice& key); Status InsertInternal(const Slice& key, Cache::ObjectPtr value, const Cache::CacheItemHelper* helper, CompressionType type, CacheTier source); size_t TEST_GetCharge(const Slice& key); // TODO: clean up to use cleaner interfaces in typed_cache.h const Cache::CacheItemHelper* GetHelper(bool enable_custom_split_merge) const; std::shared_ptr cache_; CompressedSecondaryCacheOptions cache_options_; mutable port::Mutex capacity_mutex_; std::shared_ptr cache_res_mgr_; bool disable_cache_; }; } // namespace ROCKSDB_NAMESPACE