From a4d9c02511d143d96a0c91e822449ec437f7483b Mon Sep 17 00:00:00 2001 From: Andrew Kryczka Date: Fri, 2 Jun 2017 12:08:01 -0700 Subject: [PATCH] Pass CF ID to MemTableRepFactory Summary: Some users want to monitor column family activity in their custom memtable implementations. Previously there was no way to figure out with which column family a memtable is associated. This diff: - adds an overload to MemTableRepFactory::CreateMemTableRep() that provides the CF ID. For compatibility, its default implementation calls the old overload. - updates MemTable to create MemTableRep's using the new overload. Closes https://github.com/facebook/rocksdb/pull/2346 Differential Revision: D5108061 Pulled By: ajkr fbshipit-source-id: 3a1921214a348dd8ea0f54e1cab3b71c3d46d616 --- db/column_family.cc | 2 +- db/db_memtable_test.cc | 31 +++++++++++++++++++++++++++++++ db/db_test_util.h | 1 + db/memtable.cc | 8 ++++---- db/memtable.h | 2 +- db/memtable_list_test.cc | 12 ++++++------ db/write_batch_test.cc | 2 +- include/rocksdb/memtablerep.h | 12 ++++++++++++ memtable/hash_cuckoo_rep.h | 1 + memtable/hash_linklist_rep.h | 1 + memtable/hash_skiplist_rep.h | 1 + table/table_test.cc | 9 +++++---- 12 files changed, 65 insertions(+), 17 deletions(-) diff --git a/db/column_family.cc b/db/column_family.cc index 18899bbbc6d..8cfcdac18a8 100644 --- a/db/column_family.cc +++ b/db/column_family.cc @@ -785,7 +785,7 @@ uint64_t ColumnFamilyData::GetTotalSstFilesSize() const { MemTable* ColumnFamilyData::ConstructNewMemtable( const MutableCFOptions& mutable_cf_options, SequenceNumber earliest_seq) { return new MemTable(internal_comparator_, ioptions_, mutable_cf_options, - write_buffer_manager_, earliest_seq); + write_buffer_manager_, earliest_seq, id_); } void ColumnFamilyData::CreateNewMemtable( diff --git a/db/db_memtable_test.cc b/db/db_memtable_test.cc index 7cbb3e23c43..220ceac7ed7 100644 --- a/db/db_memtable_test.cc +++ b/db/db_memtable_test.cc @@ -84,14 +84,27 @@ class MockMemTableRepFactory : public MemTableRepFactory { return mock_rep_; } + virtual MemTableRep* CreateMemTableRep(const MemTableRep::KeyComparator& cmp, + MemTableAllocator* allocator, + const SliceTransform* transform, + Logger* logger, + uint32_t column_family_id) override { + last_column_family_id_ = column_family_id; + return CreateMemTableRep(cmp, allocator, transform, logger); + } + virtual const char* Name() const override { return "MockMemTableRepFactory"; } MockMemTableRep* rep() { return mock_rep_; } bool IsInsertConcurrentlySupported() const override { return false; } + uint32_t GetLastColumnFamilyId() { return last_column_family_id_; } + private: MockMemTableRep* mock_rep_; + // workaround since there's no port::kMaxUint32 yet. + uint32_t last_column_family_id_ = static_cast(-1); }; class TestPrefixExtractor : public SliceTransform { @@ -157,6 +170,24 @@ TEST_F(DBMemTableTest, InsertWithHint) { ASSERT_EQ("vvv", Get("whitelisted")); } +TEST_F(DBMemTableTest, ColumnFamilyId) { + // Verifies MemTableRepFactory is told the right column family id. + Options options; + options.allow_concurrent_memtable_write = false; + options.create_if_missing = true; + options.memtable_factory.reset(new MockMemTableRepFactory()); + DestroyAndReopen(options); + CreateAndReopenWithCF({"pikachu"}, options); + + for (int cf = 0; cf < 2; ++cf) { + ASSERT_OK(Put(cf, "key", "val")); + ASSERT_OK(Flush(cf)); + ASSERT_EQ( + cf, static_cast(options.memtable_factory.get()) + ->GetLastColumnFamilyId()); + } +} + } // namespace rocksdb int main(int argc, char** argv) { diff --git a/db/db_test_util.h b/db/db_test_util.h index 574dc9ef6e8..36fdecf429e 100644 --- a/db/db_test_util.h +++ b/db/db_test_util.h @@ -184,6 +184,7 @@ class SpecialSkipListFactory : public MemTableRepFactory { explicit SpecialSkipListFactory(int num_entries_flush) : num_entries_flush_(num_entries_flush) {} + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep( const MemTableRep::KeyComparator& compare, MemTableAllocator* allocator, const SliceTransform* transform, Logger* logger) override { diff --git a/db/memtable.cc b/db/memtable.cc index 6ee0b603ecd..bfb562fede8 100644 --- a/db/memtable.cc +++ b/db/memtable.cc @@ -63,7 +63,7 @@ MemTable::MemTable(const InternalKeyComparator& cmp, const ImmutableCFOptions& ioptions, const MutableCFOptions& mutable_cf_options, WriteBufferManager* write_buffer_manager, - SequenceNumber latest_seq) + SequenceNumber latest_seq, uint32_t column_family_id) : comparator_(cmp), moptions_(ioptions, mutable_cf_options), refs_(0), @@ -73,10 +73,10 @@ MemTable::MemTable(const InternalKeyComparator& cmp, allocator_(&arena_, write_buffer_manager), table_(ioptions.memtable_factory->CreateMemTableRep( comparator_, &allocator_, ioptions.prefix_extractor, - ioptions.info_log)), + ioptions.info_log, column_family_id)), range_del_table_(SkipListFactory().CreateMemTableRep( - comparator_, &allocator_, nullptr /* transform */, - ioptions.info_log)), + comparator_, &allocator_, nullptr /* transform */, ioptions.info_log, + column_family_id)), is_range_del_table_empty_(true), data_size_(0), num_entries_(0), diff --git a/db/memtable.h b/db/memtable.h index e4c35b43777..260a1232aca 100644 --- a/db/memtable.h +++ b/db/memtable.h @@ -103,7 +103,7 @@ class MemTable { const ImmutableCFOptions& ioptions, const MutableCFOptions& mutable_cf_options, WriteBufferManager* write_buffer_manager, - SequenceNumber earliest_seq); + SequenceNumber earliest_seq, uint32_t column_family_id); // Do not delete this MemTable unless Unref() indicates it not in use. ~MemTable(); diff --git a/db/memtable_list_test.cc b/db/memtable_list_test.cc index b07852c6c2d..081d61bffc7 100644 --- a/db/memtable_list_test.cc +++ b/db/memtable_list_test.cc @@ -136,7 +136,7 @@ TEST_F(MemTableListTest, GetTest) { WriteBufferManager wb(options.db_write_buffer_size); MemTable* mem = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem->Ref(); // Write some keys to this memtable. @@ -175,7 +175,7 @@ TEST_F(MemTableListTest, GetTest) { // Create another memtable and write some keys to it WriteBufferManager wb2(options.db_write_buffer_size); MemTable* mem2 = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb2, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem2->Ref(); mem2->Add(++seq, kTypeDeletion, "key1", ""); @@ -243,7 +243,7 @@ TEST_F(MemTableListTest, GetFromHistoryTest) { WriteBufferManager wb(options.db_write_buffer_size); MemTable* mem = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem->Ref(); // Write some keys to this memtable. @@ -321,7 +321,7 @@ TEST_F(MemTableListTest, GetFromHistoryTest) { // Create another memtable and write some keys to it WriteBufferManager wb2(options.db_write_buffer_size); MemTable* mem2 = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb2, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem2->Ref(); mem2->Add(++seq, kTypeDeletion, "key1", ""); @@ -346,7 +346,7 @@ TEST_F(MemTableListTest, GetFromHistoryTest) { // Add a third memtable to push the first memtable out of the history WriteBufferManager wb3(options.db_write_buffer_size); MemTable* mem3 = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb3, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem3->Ref(); list.Add(mem3, &to_delete); ASSERT_EQ(1, list.NumNotFlushed()); @@ -420,7 +420,7 @@ TEST_F(MemTableListTest, FlushPendingTest) { MutableCFOptions mutable_cf_options(options); for (int i = 0; i < num_tables; i++) { MemTable* mem = new MemTable(cmp, ioptions, mutable_cf_options, &wb, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem->Ref(); std::string value; diff --git a/db/write_batch_test.cc b/db/write_batch_test.cc index 2c44b7ffad1..511fec154bf 100644 --- a/db/write_batch_test.cc +++ b/db/write_batch_test.cc @@ -34,7 +34,7 @@ static std::string PrintContents(WriteBatch* b) { ImmutableCFOptions ioptions(options); WriteBufferManager wb(options.db_write_buffer_size); MemTable* mem = new MemTable(cmp, ioptions, MutableCFOptions(options), &wb, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); mem->Ref(); std::string state; ColumnFamilyMemTablesDefault cf_mems_default(mem); diff --git a/include/rocksdb/memtablerep.h b/include/rocksdb/memtablerep.h index 3ae89a9800a..3753d9bc399 100644 --- a/include/rocksdb/memtablerep.h +++ b/include/rocksdb/memtablerep.h @@ -216,10 +216,18 @@ class MemTableRep { class MemTableRepFactory { public: virtual ~MemTableRepFactory() {} + virtual MemTableRep* CreateMemTableRep(const MemTableRep::KeyComparator&, MemTableAllocator*, const SliceTransform*, Logger* logger) = 0; + virtual MemTableRep* CreateMemTableRep( + const MemTableRep::KeyComparator& key_cmp, MemTableAllocator* allocator, + const SliceTransform* slice_transform, Logger* logger, + uint32_t /* column_family_id */) { + return CreateMemTableRep(key_cmp, allocator, slice_transform, logger); + } + virtual const char* Name() const = 0; // Return true if the current MemTableRep supports concurrent inserts @@ -238,6 +246,7 @@ class SkipListFactory : public MemTableRepFactory { public: explicit SkipListFactory(size_t lookahead = 0) : lookahead_(lookahead) {} + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep(const MemTableRep::KeyComparator&, MemTableAllocator*, const SliceTransform*, @@ -264,10 +273,13 @@ class VectorRepFactory : public MemTableRepFactory { public: explicit VectorRepFactory(size_t count = 0) : count_(count) { } + + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep(const MemTableRep::KeyComparator&, MemTableAllocator*, const SliceTransform*, Logger* logger) override; + virtual const char* Name() const override { return "VectorRepFactory"; } diff --git a/memtable/hash_cuckoo_rep.h b/memtable/hash_cuckoo_rep.h index 173a907b4e1..fdb977c2496 100644 --- a/memtable/hash_cuckoo_rep.h +++ b/memtable/hash_cuckoo_rep.h @@ -28,6 +28,7 @@ class HashCuckooRepFactory : public MemTableRepFactory { virtual ~HashCuckooRepFactory() {} + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep( const MemTableRep::KeyComparator& compare, MemTableAllocator* allocator, const SliceTransform* transform, Logger* logger) override; diff --git a/memtable/hash_linklist_rep.h b/memtable/hash_linklist_rep.h index 5197e7cfbbc..f792b5b88b7 100644 --- a/memtable/hash_linklist_rep.h +++ b/memtable/hash_linklist_rep.h @@ -28,6 +28,7 @@ class HashLinkListRepFactory : public MemTableRepFactory { virtual ~HashLinkListRepFactory() {} + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep( const MemTableRep::KeyComparator& compare, MemTableAllocator* allocator, const SliceTransform* transform, Logger* logger) override; diff --git a/memtable/hash_skiplist_rep.h b/memtable/hash_skiplist_rep.h index 56a289c4b1d..e26e0f57b83 100644 --- a/memtable/hash_skiplist_rep.h +++ b/memtable/hash_skiplist_rep.h @@ -25,6 +25,7 @@ class HashSkipListRepFactory : public MemTableRepFactory { virtual ~HashSkipListRepFactory() {} + using MemTableRepFactory::CreateMemTableRep; virtual MemTableRep* CreateMemTableRep( const MemTableRep::KeyComparator& compare, MemTableAllocator* allocator, const SliceTransform* transform, Logger* logger) override; diff --git a/table/table_test.cc b/table/table_test.cc index 184ea30c964..d606de14034 100644 --- a/table/table_test.cc +++ b/table/table_test.cc @@ -429,7 +429,7 @@ class MemTableConstructor: public Constructor { ImmutableCFOptions ioptions(options_); memtable_ = new MemTable(internal_comparator_, ioptions, MutableCFOptions(options_), - wb, kMaxSequenceNumber); + wb, kMaxSequenceNumber, 0 /* column_family_id */); memtable_->Ref(); } ~MemTableConstructor() { @@ -443,7 +443,7 @@ class MemTableConstructor: public Constructor { ImmutableCFOptions mem_ioptions(ioptions); memtable_ = new MemTable(internal_comparator_, mem_ioptions, MutableCFOptions(options_), write_buffer_manager_, - kMaxSequenceNumber); + kMaxSequenceNumber, 0 /* column_family_id */); memtable_->Ref(); int seq = 1; for (const auto kv : kv_map) { @@ -2594,8 +2594,9 @@ TEST_F(MemTableTest, Simple) { options.memtable_factory = table_factory; ImmutableCFOptions ioptions(options); WriteBufferManager wb(options.db_write_buffer_size); - MemTable* memtable = new MemTable(cmp, ioptions, MutableCFOptions(options), - &wb, kMaxSequenceNumber); + MemTable* memtable = + new MemTable(cmp, ioptions, MutableCFOptions(options), &wb, + kMaxSequenceNumber, 0 /* column_family_id */); memtable->Ref(); WriteBatch batch; WriteBatchInternal::SetSequence(&batch, 100);