fork of https://github.com/rust-rocksdb/rust-rocksdb for nextgraph
				
			
			
		
			You can not select more than 25 topics
			Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
		
		
		
		
		
			
		
			
				
					
					
						
							413 lines
						
					
					
						
							14 KiB
						
					
					
				
			
		
		
	
	
							413 lines
						
					
					
						
							14 KiB
						
					
					
				| //  Copyright (c) 2013, 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.
 | |
| 
 | |
| #include "utilities/persistent_cache/persistent_cache_test.h"
 | |
| 
 | |
| #include <functional>
 | |
| #include <memory>
 | |
| #include <thread>
 | |
| 
 | |
| #include "file/file_util.h"
 | |
| #include "utilities/persistent_cache/block_cache_tier.h"
 | |
| 
 | |
| namespace ROCKSDB_NAMESPACE {
 | |
| 
 | |
| static const double kStressFactor = .125;
 | |
| 
 | |
| #ifdef OS_LINUX
 | |
| static void OnOpenForRead(void* arg) {
 | |
|   int* val = static_cast<int*>(arg);
 | |
|   *val &= ~O_DIRECT;
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "NewRandomAccessFile:O_DIRECT",
 | |
|       std::bind(OnOpenForRead, std::placeholders::_1));
 | |
| }
 | |
| 
 | |
| static void OnOpenForWrite(void* arg) {
 | |
|   int* val = static_cast<int*>(arg);
 | |
|   *val &= ~O_DIRECT;
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "NewWritableFile:O_DIRECT",
 | |
|       std::bind(OnOpenForWrite, std::placeholders::_1));
 | |
| }
 | |
| #endif
 | |
| 
 | |
| static void OnDeleteDir(void* arg) {
 | |
|   char* dir = static_cast<char*>(arg);
 | |
|   ASSERT_OK(DestroyDir(Env::Default(), std::string(dir)));
 | |
| }
 | |
| 
 | |
| //
 | |
| // Simple logger that prints message on stdout
 | |
| //
 | |
| class ConsoleLogger : public Logger {
 | |
|  public:
 | |
|   using Logger::Logv;
 | |
|   ConsoleLogger() : Logger(InfoLogLevel::ERROR_LEVEL) {}
 | |
| 
 | |
|   void Logv(const char* format, va_list ap) override {
 | |
|     MutexLock _(&lock_);
 | |
|     vprintf(format, ap);
 | |
|     printf("\n");
 | |
|   }
 | |
| 
 | |
|   port::Mutex lock_;
 | |
| };
 | |
| 
 | |
| // construct a tiered RAM+Block cache
 | |
| std::unique_ptr<PersistentTieredCache> NewTieredCache(
 | |
|     const size_t mem_size, const PersistentCacheConfig& opt) {
 | |
|   std::unique_ptr<PersistentTieredCache> tcache(new PersistentTieredCache());
 | |
|   // create primary tier
 | |
|   assert(mem_size);
 | |
|   auto pcache = std::shared_ptr<PersistentCacheTier>(new VolatileCacheTier(
 | |
|       /*is_compressed*/ true, mem_size));
 | |
|   tcache->AddTier(pcache);
 | |
|   // create secondary tier
 | |
|   auto scache = std::shared_ptr<PersistentCacheTier>(new BlockCacheTier(opt));
 | |
|   tcache->AddTier(scache);
 | |
| 
 | |
|   Status s = tcache->Open();
 | |
|   assert(s.ok());
 | |
|   return tcache;
 | |
| }
 | |
| 
 | |
| // create block cache
 | |
| std::unique_ptr<PersistentCacheTier> NewBlockCache(
 | |
|     Env* env, const std::string& path,
 | |
|     const uint64_t max_size = std::numeric_limits<uint64_t>::max(),
 | |
|     const bool enable_direct_writes = false) {
 | |
|   const uint32_t max_file_size =
 | |
|       static_cast<uint32_t>(12 * 1024 * 1024 * kStressFactor);
 | |
|   auto log = std::make_shared<ConsoleLogger>();
 | |
|   PersistentCacheConfig opt(env, path, max_size, log);
 | |
|   opt.cache_file_size = max_file_size;
 | |
|   opt.max_write_pipeline_backlog_size = std::numeric_limits<uint64_t>::max();
 | |
|   opt.enable_direct_writes = enable_direct_writes;
 | |
|   std::unique_ptr<PersistentCacheTier> scache(new BlockCacheTier(opt));
 | |
|   Status s = scache->Open();
 | |
|   assert(s.ok());
 | |
|   return scache;
 | |
| }
 | |
| 
 | |
| // create a new cache tier
 | |
| std::unique_ptr<PersistentTieredCache> NewTieredCache(
 | |
|     Env* env, const std::string& path, const uint64_t max_volatile_cache_size,
 | |
|     const uint64_t max_block_cache_size =
 | |
|         std::numeric_limits<uint64_t>::max()) {
 | |
|   const uint32_t max_file_size =
 | |
|       static_cast<uint32_t>(12 * 1024 * 1024 * kStressFactor);
 | |
|   auto log = std::make_shared<ConsoleLogger>();
 | |
|   auto opt = PersistentCacheConfig(env, path, max_block_cache_size, log);
 | |
|   opt.cache_file_size = max_file_size;
 | |
|   opt.max_write_pipeline_backlog_size = std::numeric_limits<uint64_t>::max();
 | |
|   // create tier out of the two caches
 | |
|   auto cache = NewTieredCache(max_volatile_cache_size, opt);
 | |
|   return cache;
 | |
| }
 | |
| 
 | |
| PersistentCacheTierTest::PersistentCacheTierTest()
 | |
|     : path_(test::PerThreadDBPath("cache_test")) {
 | |
| #ifdef OS_LINUX
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->EnableProcessing();
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "NewRandomAccessFile:O_DIRECT", OnOpenForRead);
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "NewWritableFile:O_DIRECT", OnOpenForWrite);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| // Block cache tests
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_BlockCacheInsertWithFileCreateError) {
 | |
|   cache_ = NewBlockCache(Env::Default(), path_,
 | |
|                          /*size=*/std::numeric_limits<uint64_t>::max(),
 | |
|                          /*direct_writes=*/false);
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "BlockCacheTier::NewCacheFile:DeleteDir", OnDeleteDir);
 | |
| 
 | |
|   RunNegativeInsertTest(/*nthreads=*/1,
 | |
|                         /*max_keys*/
 | |
|                         static_cast<size_t>(10 * 1024 * kStressFactor));
 | |
| 
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->ClearAllCallBacks();
 | |
| }
 | |
| 
 | |
| // Travis is unable to handle the normal version of the tests running out of
 | |
| // fds, out of space and timeouts. This is an easier version of the test
 | |
| // specifically written for Travis
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_BasicTest) {
 | |
|   cache_ = std::make_shared<VolatileCacheTier>();
 | |
|   RunInsertTest(/*nthreads=*/1, /*max_keys=*/1024);
 | |
| 
 | |
|   cache_ = NewBlockCache(Env::Default(), path_,
 | |
|                          /*size=*/std::numeric_limits<uint64_t>::max(),
 | |
|                          /*direct_writes=*/true);
 | |
|   RunInsertTest(/*nthreads=*/1, /*max_keys=*/1024);
 | |
| 
 | |
|   cache_ = NewTieredCache(Env::Default(), path_,
 | |
|                           /*memory_size=*/static_cast<size_t>(1 * 1024 * 1024));
 | |
|   RunInsertTest(/*nthreads=*/1, /*max_keys=*/1024);
 | |
| }
 | |
| 
 | |
| // Volatile cache tests
 | |
| // DISABLED for now (somewhat expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_VolatileCacheInsert) {
 | |
|   for (auto nthreads : {1, 5}) {
 | |
|     for (auto max_keys :
 | |
|          {10 * 1024 * kStressFactor, 1 * 1024 * 1024 * kStressFactor}) {
 | |
|       cache_ = std::make_shared<VolatileCacheTier>();
 | |
|       RunInsertTest(nthreads, static_cast<size_t>(max_keys));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // DISABLED for now (somewhat expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_VolatileCacheInsertWithEviction) {
 | |
|   for (auto nthreads : {1, 5}) {
 | |
|     for (auto max_keys : {1 * 1024 * 1024 * kStressFactor}) {
 | |
|       cache_ = std::make_shared<VolatileCacheTier>(
 | |
|           /*compressed=*/true,
 | |
|           /*size=*/static_cast<size_t>(1 * 1024 * 1024 * kStressFactor));
 | |
|       RunInsertTestWithEviction(nthreads, static_cast<size_t>(max_keys));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Block cache tests
 | |
| // DISABLED for now (expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_BlockCacheInsert) {
 | |
|   for (auto direct_writes : {true, false}) {
 | |
|     for (auto nthreads : {1, 5}) {
 | |
|       for (auto max_keys :
 | |
|            {10 * 1024 * kStressFactor, 1 * 1024 * 1024 * kStressFactor}) {
 | |
|         cache_ = NewBlockCache(Env::Default(), path_,
 | |
|                                /*size=*/std::numeric_limits<uint64_t>::max(),
 | |
|                                direct_writes);
 | |
|         RunInsertTest(nthreads, static_cast<size_t>(max_keys));
 | |
|       }
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // DISABLED for now (somewhat expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_BlockCacheInsertWithEviction) {
 | |
|   for (auto nthreads : {1, 5}) {
 | |
|     for (auto max_keys : {1 * 1024 * 1024 * kStressFactor}) {
 | |
|       cache_ = NewBlockCache(
 | |
|           Env::Default(), path_,
 | |
|           /*max_size=*/static_cast<size_t>(200 * 1024 * 1024 * kStressFactor));
 | |
|       RunInsertTestWithEviction(nthreads, static_cast<size_t>(max_keys));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Tiered cache tests
 | |
| // DISABLED for now (expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_TieredCacheInsert) {
 | |
|   for (auto nthreads : {1, 5}) {
 | |
|     for (auto max_keys :
 | |
|          {10 * 1024 * kStressFactor, 1 * 1024 * 1024 * kStressFactor}) {
 | |
|       cache_ = NewTieredCache(
 | |
|           Env::Default(), path_,
 | |
|           /*memory_size=*/static_cast<size_t>(1 * 1024 * 1024 * kStressFactor));
 | |
|       RunInsertTest(nthreads, static_cast<size_t>(max_keys));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| // the tests causes a lot of file deletions which Travis limited testing
 | |
| // environment cannot handle
 | |
| // DISABLED for now (somewhat expensive)
 | |
| TEST_F(PersistentCacheTierTest, DISABLED_TieredCacheInsertWithEviction) {
 | |
|   for (auto nthreads : {1, 5}) {
 | |
|     for (auto max_keys : {1 * 1024 * 1024 * kStressFactor}) {
 | |
|       cache_ = NewTieredCache(
 | |
|           Env::Default(), path_,
 | |
|           /*memory_size=*/static_cast<size_t>(1 * 1024 * 1024 * kStressFactor),
 | |
|           /*block_cache_size*/
 | |
|           static_cast<size_t>(200 * 1024 * 1024 * kStressFactor));
 | |
|       RunInsertTestWithEviction(nthreads, static_cast<size_t>(max_keys));
 | |
|     }
 | |
|   }
 | |
| }
 | |
| 
 | |
| std::shared_ptr<PersistentCacheTier> MakeVolatileCache(
 | |
|     Env* /*env*/, const std::string& /*dbname*/) {
 | |
|   return std::make_shared<VolatileCacheTier>();
 | |
| }
 | |
| 
 | |
| std::shared_ptr<PersistentCacheTier> MakeBlockCache(Env* env,
 | |
|                                                     const std::string& dbname) {
 | |
|   return NewBlockCache(env, dbname);
 | |
| }
 | |
| 
 | |
| std::shared_ptr<PersistentCacheTier> MakeTieredCache(
 | |
|     Env* env, const std::string& dbname) {
 | |
|   const auto memory_size = 1 * 1024 * 1024 * kStressFactor;
 | |
|   return NewTieredCache(env, dbname, static_cast<size_t>(memory_size));
 | |
| }
 | |
| 
 | |
| #ifdef OS_LINUX
 | |
| static void UniqueIdCallback(void* arg) {
 | |
|   int* result = reinterpret_cast<int*>(arg);
 | |
|   if (*result == -1) {
 | |
|     *result = 0;
 | |
|   }
 | |
| 
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->ClearTrace();
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "GetUniqueIdFromFile:FS_IOC_GETVERSION", UniqueIdCallback);
 | |
| }
 | |
| #endif
 | |
| 
 | |
| TEST_F(PersistentCacheTierTest, FactoryTest) {
 | |
|   for (auto nvm_opt : {true, false}) {
 | |
|     ASSERT_FALSE(cache_);
 | |
|     auto log = std::make_shared<ConsoleLogger>();
 | |
|     std::shared_ptr<PersistentCache> cache;
 | |
|     ASSERT_OK(NewPersistentCache(Env::Default(), path_,
 | |
|                                  /*size=*/1 * 1024 * 1024 * 1024, log, nvm_opt,
 | |
|                                  &cache));
 | |
|     ASSERT_TRUE(cache);
 | |
|     ASSERT_EQ(cache->Stats().size(), 1);
 | |
|     ASSERT_TRUE(cache->Stats()[0].size());
 | |
|     cache.reset();
 | |
|   }
 | |
| }
 | |
| 
 | |
| PersistentCacheDBTest::PersistentCacheDBTest()
 | |
|     : DBTestBase("cache_test", /*env_do_fsync=*/true) {
 | |
| #ifdef OS_LINUX
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->EnableProcessing();
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "GetUniqueIdFromFile:FS_IOC_GETVERSION", UniqueIdCallback);
 | |
|   ROCKSDB_NAMESPACE::SyncPoint::GetInstance()->SetCallBack(
 | |
|       "NewRandomAccessFile:O_DIRECT", OnOpenForRead);
 | |
| #endif
 | |
| }
 | |
| 
 | |
| // test template
 | |
| void PersistentCacheDBTest::RunTest(
 | |
|     const std::function<std::shared_ptr<PersistentCacheTier>(bool)>& new_pcache,
 | |
|     const size_t max_keys = 100 * 1024, const size_t max_usecase = 3) {
 | |
|   // number of insertion interations
 | |
|   int num_iter = static_cast<int>(max_keys * kStressFactor);
 | |
| 
 | |
|   for (size_t iter = 0; iter < max_usecase; iter++) {
 | |
|     Options options;
 | |
|     options.write_buffer_size =
 | |
|         static_cast<size_t>(64 * 1024 * kStressFactor);  // small write buffer
 | |
|     options.statistics = ROCKSDB_NAMESPACE::CreateDBStatistics();
 | |
|     options = CurrentOptions(options);
 | |
| 
 | |
|     // setup page cache
 | |
|     std::shared_ptr<PersistentCacheTier> pcache;
 | |
|     BlockBasedTableOptions table_options;
 | |
|     table_options.cache_index_and_filter_blocks = true;
 | |
| 
 | |
|     const size_t size_max = std::numeric_limits<size_t>::max();
 | |
| 
 | |
|     switch (iter) {
 | |
|       case 0:
 | |
|         // page cache, block cache, no-compressed cache
 | |
|         pcache = new_pcache(/*is_compressed=*/true);
 | |
|         table_options.persistent_cache = pcache;
 | |
|         table_options.block_cache = NewLRUCache(size_max);
 | |
|         options.table_factory.reset(NewBlockBasedTableFactory(table_options));
 | |
|         break;
 | |
|       case 1:
 | |
|         // page cache, no block cache, no compressed cache
 | |
|         pcache = new_pcache(/*is_compressed=*/false);
 | |
|         table_options.persistent_cache = pcache;
 | |
|         table_options.block_cache = nullptr;
 | |
|         options.table_factory.reset(NewBlockBasedTableFactory(table_options));
 | |
|         break;
 | |
|       case 2:
 | |
|         // page cache, no block cache, no compressed cache
 | |
|         // Page cache caches compressed blocks
 | |
|         pcache = new_pcache(/*is_compressed=*/true);
 | |
|         table_options.persistent_cache = pcache;
 | |
|         table_options.block_cache = nullptr;
 | |
|         options.table_factory.reset(NewBlockBasedTableFactory(table_options));
 | |
|         break;
 | |
|       default:
 | |
|         FAIL();
 | |
|     }
 | |
| 
 | |
|     std::vector<std::string> values;
 | |
|     // insert data
 | |
|     Insert(options, table_options, num_iter, &values);
 | |
|     // flush all data in cache to device
 | |
|     pcache->TEST_Flush();
 | |
|     // verify data
 | |
|     Verify(num_iter, values);
 | |
| 
 | |
|     auto block_miss = TestGetTickerCount(options, BLOCK_CACHE_MISS);
 | |
|     auto page_hit = TestGetTickerCount(options, PERSISTENT_CACHE_HIT);
 | |
|     auto page_miss = TestGetTickerCount(options, PERSISTENT_CACHE_MISS);
 | |
| 
 | |
|     // check that we triggered the appropriate code paths in the cache
 | |
|     switch (iter) {
 | |
|       case 0:
 | |
|         // page cache, block cache, no-compressed cache
 | |
|         ASSERT_GT(page_miss, 0);
 | |
|         ASSERT_GT(page_hit, 0);
 | |
|         ASSERT_GT(block_miss, 0);
 | |
|         break;
 | |
|       case 1:
 | |
|       case 2:
 | |
|         // page cache, no block cache, no compressed cache
 | |
|         ASSERT_GT(page_miss, 0);
 | |
|         ASSERT_GT(page_hit, 0);
 | |
|         break;
 | |
|       default:
 | |
|         FAIL();
 | |
|     }
 | |
| 
 | |
|     options.create_if_missing = true;
 | |
|     DestroyAndReopen(options);
 | |
| 
 | |
|     ASSERT_OK(pcache->Close());
 | |
|   }
 | |
| }
 | |
| 
 | |
| // Travis is unable to handle the normal version of the tests running out of
 | |
| // fds, out of space and timeouts. This is an easier version of the test
 | |
| // specifically written for Travis.
 | |
| // Now used generally because main tests are too expensive as unit tests.
 | |
| TEST_F(PersistentCacheDBTest, BasicTest) {
 | |
|   RunTest(std::bind(&MakeBlockCache, env_, dbname_), /*max_keys=*/1024,
 | |
|           /*max_usecase=*/1);
 | |
| }
 | |
| 
 | |
| // test table with block page cache
 | |
| // DISABLED for now (very expensive, especially memory)
 | |
| TEST_F(PersistentCacheDBTest, DISABLED_BlockCacheTest) {
 | |
|   RunTest(std::bind(&MakeBlockCache, env_, dbname_));
 | |
| }
 | |
| 
 | |
| // test table with volatile page cache
 | |
| // DISABLED for now (very expensive, especially memory)
 | |
| TEST_F(PersistentCacheDBTest, DISABLED_VolatileCacheTest) {
 | |
|   RunTest(std::bind(&MakeVolatileCache, env_, dbname_));
 | |
| }
 | |
| 
 | |
| // test table with tiered page cache
 | |
| // DISABLED for now (very expensive, especially memory)
 | |
| TEST_F(PersistentCacheDBTest, DISABLED_TieredCacheTest) {
 | |
|   RunTest(std::bind(&MakeTieredCache, env_, dbname_));
 | |
| }
 | |
| 
 | |
| }  // namespace ROCKSDB_NAMESPACE
 | |
| 
 | |
| int main(int argc, char** argv) {
 | |
|   ROCKSDB_NAMESPACE::port::InstallStackTraceHandler();
 | |
|   ::testing::InitGoogleTest(&argc, argv);
 | |
|   return RUN_ALL_TESTS();
 | |
| }
 | |
| 
 |