2011-03-18 23:37:00 +01:00
|
|
|
// 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.
|
|
|
|
|
2011-03-30 20:35:40 +02:00
|
|
|
#include "leveldb/options.h"
|
2011-03-18 23:37:00 +01:00
|
|
|
|
2013-01-11 02:18:50 +01:00
|
|
|
#include <limits>
|
|
|
|
|
2013-03-21 23:59:47 +01:00
|
|
|
#include "leveldb/cache.h"
|
2013-05-12 11:36:59 +02:00
|
|
|
#include "leveldb/compaction_filter.h"
|
2011-03-30 20:35:40 +02:00
|
|
|
#include "leveldb/comparator.h"
|
|
|
|
#include "leveldb/env.h"
|
2012-08-22 20:43:53 +02:00
|
|
|
#include "leveldb/filter_policy.h"
|
2013-03-21 23:59:47 +01:00
|
|
|
#include "leveldb/merge_operator.h"
|
2013-07-23 23:42:27 +02:00
|
|
|
#include "db/skiplistrep.h"
|
2011-03-18 23:37:00 +01:00
|
|
|
|
|
|
|
namespace leveldb {
|
|
|
|
|
|
|
|
Options::Options()
|
|
|
|
: comparator(BytewiseComparator()),
|
2013-03-21 23:59:47 +01:00
|
|
|
merge_operator(nullptr),
|
2013-05-12 11:36:59 +02:00
|
|
|
compaction_filter(nullptr),
|
2011-03-18 23:37:00 +01:00
|
|
|
create_if_missing(false),
|
|
|
|
error_if_exists(false),
|
|
|
|
paranoid_checks(false),
|
|
|
|
env(Env::Default()),
|
2013-03-01 03:04:58 +01:00
|
|
|
info_log(nullptr),
|
2011-04-12 21:38:58 +02:00
|
|
|
write_buffer_size(4<<20),
|
2012-10-19 23:00:53 +02:00
|
|
|
max_write_buffer_number(2),
|
2013-06-11 23:23:58 +02:00
|
|
|
min_write_buffer_number_to_merge(1),
|
2011-03-18 23:37:00 +01:00
|
|
|
max_open_files(1000),
|
2011-04-12 21:38:58 +02:00
|
|
|
block_size(4096),
|
2011-03-18 23:37:00 +01:00
|
|
|
block_restart_interval(16),
|
2012-04-17 17:36:46 +02:00
|
|
|
compression(kSnappyCompression),
|
2013-03-01 03:04:58 +01:00
|
|
|
filter_policy(nullptr),
|
2012-06-23 04:30:03 +02:00
|
|
|
num_levels(7),
|
|
|
|
level0_file_num_compaction_trigger(4),
|
|
|
|
level0_slowdown_writes_trigger(8),
|
|
|
|
level0_stop_writes_trigger(12),
|
|
|
|
max_mem_compaction_level(2),
|
|
|
|
target_file_size_base(2 * 1048576),
|
2012-08-22 02:33:46 +02:00
|
|
|
target_file_size_multiplier(1),
|
2012-07-03 02:30:32 +02:00
|
|
|
max_bytes_for_level_base(10 * 1048576),
|
2012-06-23 04:30:03 +02:00
|
|
|
max_bytes_for_level_multiplier(10),
|
2013-05-21 20:37:06 +02:00
|
|
|
max_bytes_for_level_multiplier_additional(num_levels, 1),
|
2012-06-23 04:30:03 +02:00
|
|
|
expanded_compaction_factor(25),
|
2012-11-21 08:07:41 +01:00
|
|
|
source_compaction_factor(1),
|
2012-06-23 04:30:03 +02:00
|
|
|
max_grandparent_overlap_factor(10),
|
2012-08-15 00:20:36 +02:00
|
|
|
disableDataSync(false),
|
2012-08-27 21:10:26 +02:00
|
|
|
use_fsync(false),
|
2012-09-06 02:44:13 +02:00
|
|
|
db_stats_log_interval(1800),
|
2012-09-17 22:35:57 +02:00
|
|
|
db_log_dir(""),
|
2012-10-16 17:53:46 +02:00
|
|
|
disable_seek_compaction(false),
|
2012-10-26 22:37:21 +02:00
|
|
|
delete_obsolete_files_period_micros(0),
|
2012-11-06 01:51:55 +01:00
|
|
|
max_background_compactions(1),
|
2012-10-29 22:18:00 +01:00
|
|
|
max_log_file_size(0),
|
2013-02-05 04:42:40 +01:00
|
|
|
log_file_time_to_roll(0),
|
|
|
|
keep_log_file_num(1000),
|
2013-08-06 00:43:49 +02:00
|
|
|
soft_rate_limit(0.0),
|
|
|
|
hard_rate_limit(0.0),
|
|
|
|
rate_limit_delay_max_milliseconds(1000),
|
2013-01-11 02:18:50 +01:00
|
|
|
max_manifest_file_size(std::numeric_limits<uint64_t>::max()),
|
2012-11-06 21:02:18 +01:00
|
|
|
no_block_cache(false),
|
|
|
|
table_cache_numshardbits(4),
|
Make arena block size configurable
Summary:
Add an option for arena block size, default value 4096 bytes. Arena will allocate blocks with such size.
I am not sure about passing parameter to skiplist in the new virtualized framework, though I talked to Jim a bit. So add Jim as reviewer.
Test Plan:
new unit test, I am running db_test.
For passing paramter from configured option to Arena, I tried tests like:
TEST(DBTest, Arena_Option) {
std::string dbname = test::TmpDir() + "/db_arena_option_test";
DestroyDB(dbname, Options());
DB* db = nullptr;
Options opts;
opts.create_if_missing = true;
opts.arena_block_size = 1000000; // tested 99, 999999
Status s = DB::Open(opts, dbname, &db);
db->Put(WriteOptions(), "a", "123");
}
and printed some debug info. The results look good. Any suggestion for such a unit-test?
Reviewers: haobo, dhruba, emayanke, jpaton
Reviewed By: dhruba
CC: leveldb, zshao
Differential Revision: https://reviews.facebook.net/D11799
2013-07-31 21:42:23 +02:00
|
|
|
arena_block_size(0),
|
2012-11-26 22:56:45 +01:00
|
|
|
disable_auto_compactions(false),
|
2013-01-15 23:05:42 +01:00
|
|
|
WAL_ttl_seconds(0),
|
2013-02-28 23:09:30 +01:00
|
|
|
manifest_preallocation_size(4 * 1024 * 1024),
|
2013-03-15 01:00:04 +01:00
|
|
|
purge_redundant_kvs_while_flush(true),
|
|
|
|
allow_os_buffer(true),
|
|
|
|
allow_mmap_reads(false),
|
2013-04-10 04:42:07 +02:00
|
|
|
allow_mmap_writes(true),
|
2013-05-21 20:53:33 +02:00
|
|
|
is_fd_close_on_exec(true),
|
2013-05-11 00:21:04 +02:00
|
|
|
skip_log_error_on_recovery(false),
|
2013-05-15 19:34:02 +02:00
|
|
|
stats_dump_period_sec(3600),
|
2013-05-18 00:53:01 +02:00
|
|
|
block_size_deviation (10),
|
|
|
|
advise_random_on_open(true),
|
2013-06-01 01:30:17 +02:00
|
|
|
access_hint_on_compaction_start(NORMAL),
|
2013-06-14 07:49:46 +02:00
|
|
|
use_adaptive_mutex(false),
|
2013-07-06 03:49:18 +02:00
|
|
|
bytes_per_sync(0),
|
2013-07-18 00:08:56 +02:00
|
|
|
compaction_style(kCompactionStyleLevel),
|
2013-07-23 23:42:27 +02:00
|
|
|
filter_deletes(false),
|
2013-08-13 19:56:20 +02:00
|
|
|
memtable_factory(std::shared_ptr<SkipListFactory>(new SkipListFactory)),
|
|
|
|
compaction_filter_factory(
|
|
|
|
std::shared_ptr<CompactionFilterFactory>(
|
|
|
|
new DefaultCompactionFilterFactory())) {
|
|
|
|
|
2013-07-23 23:42:27 +02:00
|
|
|
assert(memtable_factory.get() != nullptr);
|
2011-03-18 23:37:00 +01:00
|
|
|
}
|
|
|
|
|
2013-05-18 00:53:01 +02:00
|
|
|
static const char* const access_hints[] = {
|
|
|
|
"NONE", "NORMAL", "SEQUENTIAL", "WILLNEED"
|
|
|
|
};
|
|
|
|
|
2012-08-22 20:43:53 +02:00
|
|
|
void
|
2013-01-20 11:07:13 +01:00
|
|
|
Options::Dump(Logger* log) const
|
2012-08-22 20:43:53 +02:00
|
|
|
{
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.comparator: %s", comparator->Name());
|
2013-03-21 23:59:47 +01:00
|
|
|
Log(log," Options.merge_operator: %s",
|
|
|
|
merge_operator? merge_operator->Name() : "None");
|
2013-05-12 11:36:59 +02:00
|
|
|
Log(log," Options.compaction_filter: %s",
|
|
|
|
compaction_filter? compaction_filter->Name() : "None");
|
2013-08-13 19:56:20 +02:00
|
|
|
Log(log," Options.compaction_filter_factory: %s",
|
|
|
|
compaction_filter_factory->Name());
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.error_if_exists: %d", error_if_exists);
|
|
|
|
Log(log," Options.paranoid_checks: %d", paranoid_checks);
|
|
|
|
Log(log," Options.env: %p", env);
|
2013-01-20 11:07:13 +01:00
|
|
|
Log(log," Options.info_log: %p", info_log.get());
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.write_buffer_size: %zd", write_buffer_size);
|
2012-11-08 00:11:37 +01:00
|
|
|
Log(log," Options.max_write_buffer_number: %d", max_write_buffer_number);
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.max_open_files: %d", max_open_files);
|
2013-01-20 11:07:13 +01:00
|
|
|
Log(log," Options.block_cache: %p", block_cache.get());
|
2012-11-01 01:02:24 +01:00
|
|
|
if (block_cache) {
|
|
|
|
Log(log," Options.block_cache_size: %zd",
|
|
|
|
block_cache->GetCapacity());
|
|
|
|
}
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.block_size: %zd", block_size);
|
|
|
|
Log(log," Options.block_restart_interval: %d", block_restart_interval);
|
2013-01-24 19:54:26 +01:00
|
|
|
if (!compression_per_level.empty()) {
|
2013-03-15 02:32:01 +01:00
|
|
|
for (unsigned int i = 0; i < compression_per_level.size(); i++) {
|
2012-11-01 18:50:08 +01:00
|
|
|
Log(log," Options.compression[%d]: %d",
|
2012-10-28 07:13:17 +01:00
|
|
|
i, compression_per_level[i]);
|
|
|
|
}
|
|
|
|
} else {
|
2012-10-29 22:18:00 +01:00
|
|
|
Log(log," Options.compression: %d", compression);
|
2012-10-28 07:13:17 +01:00
|
|
|
}
|
2012-08-27 21:10:26 +02:00
|
|
|
Log(log," Options.filter_policy: %s",
|
2013-03-01 03:04:58 +01:00
|
|
|
filter_policy == nullptr ? "nullptr" : filter_policy->Name());
|
2012-08-27 21:10:26 +02:00
|
|
|
Log(log," Options.num_levels: %d", num_levels);
|
|
|
|
Log(log," Options.disableDataSync: %d", disableDataSync);
|
|
|
|
Log(log," Options.use_fsync: %d", use_fsync);
|
2012-11-06 21:02:18 +01:00
|
|
|
Log(log," Options.max_log_file_size: %ld", max_log_file_size);
|
2013-01-11 02:18:50 +01:00
|
|
|
Log(log,"Options.max_manifest_file_size: %ld",
|
|
|
|
max_manifest_file_size);
|
2013-02-05 04:42:40 +01:00
|
|
|
Log(log," Options.log_file_time_to_roll: %ld", log_file_time_to_roll);
|
|
|
|
Log(log," Options.keep_log_file_num: %ld", keep_log_file_num);
|
2012-10-26 23:55:02 +02:00
|
|
|
Log(log," Options.db_stats_log_interval: %d",
|
2012-08-27 21:10:26 +02:00
|
|
|
db_stats_log_interval);
|
2013-03-15 01:00:04 +01:00
|
|
|
Log(log," Options.allow_os_buffer: %d", allow_os_buffer);
|
|
|
|
Log(log," Options.allow_mmap_reads: %d", allow_mmap_reads);
|
|
|
|
Log(log," Options.allow_mmap_writes: %d", allow_mmap_writes);
|
2013-06-11 23:23:58 +02:00
|
|
|
Log(log," Options.min_write_buffer_number_to_merge: %d",
|
|
|
|
min_write_buffer_number_to_merge);
|
2013-02-28 23:09:30 +01:00
|
|
|
Log(log," Options.purge_redundant_kvs_while_flush: %d",
|
|
|
|
purge_redundant_kvs_while_flush);
|
2012-11-01 18:50:08 +01:00
|
|
|
Log(log," Options.compression_opts.window_bits: %d",
|
|
|
|
compression_opts.window_bits);
|
|
|
|
Log(log," Options.compression_opts.level: %d",
|
|
|
|
compression_opts.level);
|
|
|
|
Log(log," Options.compression_opts.strategy: %d",
|
|
|
|
compression_opts.strategy);
|
2012-08-27 21:10:26 +02:00
|
|
|
Log(log," Options.level0_file_num_compaction_trigger: %d",
|
|
|
|
level0_file_num_compaction_trigger);
|
|
|
|
Log(log," Options.level0_slowdown_writes_trigger: %d",
|
|
|
|
level0_slowdown_writes_trigger);
|
|
|
|
Log(log," Options.level0_stop_writes_trigger: %d",
|
|
|
|
level0_stop_writes_trigger);
|
|
|
|
Log(log," Options.max_mem_compaction_level: %d",
|
|
|
|
max_mem_compaction_level);
|
|
|
|
Log(log," Options.target_file_size_base: %d",
|
|
|
|
target_file_size_base);
|
|
|
|
Log(log," Options.target_file_size_multiplier: %d",
|
|
|
|
target_file_size_multiplier);
|
2012-11-09 03:45:19 +01:00
|
|
|
Log(log," Options.max_bytes_for_level_base: %ld",
|
2012-08-27 21:10:26 +02:00
|
|
|
max_bytes_for_level_base);
|
|
|
|
Log(log," Options.max_bytes_for_level_multiplier: %d",
|
|
|
|
max_bytes_for_level_multiplier);
|
2013-05-21 20:37:06 +02:00
|
|
|
for (int i = 0; i < num_levels; i++) {
|
|
|
|
Log(log,"Options.max_bytes_for_level_multiplier_addtl[%d]: %d",
|
|
|
|
i, max_bytes_for_level_multiplier_additional[i]);
|
|
|
|
}
|
2012-08-27 21:10:26 +02:00
|
|
|
Log(log," Options.expanded_compaction_factor: %d",
|
|
|
|
expanded_compaction_factor);
|
2012-11-21 08:07:41 +01:00
|
|
|
Log(log," Options.source_compaction_factor: %d",
|
|
|
|
source_compaction_factor);
|
2012-08-27 21:10:26 +02:00
|
|
|
Log(log," Options.max_grandparent_overlap_factor: %d",
|
|
|
|
max_grandparent_overlap_factor);
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.db_log_dir: %s",
|
2012-09-06 02:44:13 +02:00
|
|
|
db_log_dir.c_str());
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.disable_seek_compaction: %d",
|
2012-09-17 22:35:57 +02:00
|
|
|
disable_seek_compaction);
|
2012-11-01 01:02:24 +01:00
|
|
|
Log(log," Options.no_block_cache: %d",
|
|
|
|
no_block_cache);
|
|
|
|
Log(log," Options.table_cache_numshardbits: %d",
|
2013-05-11 00:21:04 +02:00
|
|
|
table_cache_numshardbits);
|
Make arena block size configurable
Summary:
Add an option for arena block size, default value 4096 bytes. Arena will allocate blocks with such size.
I am not sure about passing parameter to skiplist in the new virtualized framework, though I talked to Jim a bit. So add Jim as reviewer.
Test Plan:
new unit test, I am running db_test.
For passing paramter from configured option to Arena, I tried tests like:
TEST(DBTest, Arena_Option) {
std::string dbname = test::TmpDir() + "/db_arena_option_test";
DestroyDB(dbname, Options());
DB* db = nullptr;
Options opts;
opts.create_if_missing = true;
opts.arena_block_size = 1000000; // tested 99, 999999
Status s = DB::Open(opts, dbname, &db);
db->Put(WriteOptions(), "a", "123");
}
and printed some debug info. The results look good. Any suggestion for such a unit-test?
Reviewers: haobo, dhruba, emayanke, jpaton
Reviewed By: dhruba
CC: leveldb, zshao
Differential Revision: https://reviews.facebook.net/D11799
2013-07-31 21:42:23 +02:00
|
|
|
Log(log," Options.arena_block_size: %ld",
|
|
|
|
arena_block_size);
|
2012-10-19 21:16:18 +02:00
|
|
|
Log(log," Options.delete_obsolete_files_period_micros: %ld",
|
|
|
|
delete_obsolete_files_period_micros);
|
2012-10-19 23:00:53 +02:00
|
|
|
Log(log," Options.max_background_compactions: %d",
|
|
|
|
max_background_compactions);
|
2013-08-06 00:43:49 +02:00
|
|
|
Log(log," Options.hard_rate_limit: %.2f",
|
|
|
|
hard_rate_limit);
|
2013-08-02 20:46:47 +02:00
|
|
|
Log(log," Options.rate_limit_delay_max_milliseconds: %u",
|
2013-08-06 00:43:49 +02:00
|
|
|
rate_limit_delay_max_milliseconds);
|
2012-11-21 00:45:41 +01:00
|
|
|
Log(log," Options.disable_auto_compactions: %d",
|
|
|
|
disable_auto_compactions);
|
2013-01-15 23:05:42 +01:00
|
|
|
Log(log," Options.WAL_ttl_seconds: %ld",
|
2012-11-26 22:56:45 +01:00
|
|
|
WAL_ttl_seconds);
|
2013-01-15 23:05:42 +01:00
|
|
|
Log(log," Options.manifest_preallocation_size: %ld",
|
|
|
|
manifest_preallocation_size);
|
2013-04-10 04:42:07 +02:00
|
|
|
Log(log," Options.purge_redundant_kvs_while_flush: %d",
|
|
|
|
purge_redundant_kvs_while_flush);
|
|
|
|
Log(log," Options.allow_os_buffer: %d",
|
|
|
|
allow_os_buffer);
|
|
|
|
Log(log," Options.allow_mmap_reads: %d",
|
|
|
|
allow_mmap_reads);
|
|
|
|
Log(log," Options.allow_mmap_writes: %d",
|
|
|
|
allow_mmap_writes);
|
|
|
|
Log(log," Options.is_fd_close_on_exec: %d",
|
|
|
|
is_fd_close_on_exec);
|
2013-05-21 20:53:33 +02:00
|
|
|
Log(log," Options.skip_log_error_on_recovery: %d",
|
|
|
|
skip_log_error_on_recovery);
|
2013-08-02 20:46:47 +02:00
|
|
|
Log(log," Options.stats_dump_period_sec: %u",
|
2013-05-11 00:21:04 +02:00
|
|
|
stats_dump_period_sec);
|
2013-05-15 19:34:02 +02:00
|
|
|
Log(log," Options.block_size_deviation: %d",
|
|
|
|
block_size_deviation);
|
2013-05-18 00:53:01 +02:00
|
|
|
Log(log," Options.advise_random_on_open: %d",
|
|
|
|
advise_random_on_open);
|
|
|
|
Log(log," Options.access_hint_on_compaction_start: %s",
|
|
|
|
access_hints[access_hint_on_compaction_start]);
|
2013-06-01 01:30:17 +02:00
|
|
|
Log(log," Options.use_adaptive_mutex: %d",
|
|
|
|
use_adaptive_mutex);
|
2013-06-14 07:49:46 +02:00
|
|
|
Log(log," Options.bytes_per_sync: %ld",
|
|
|
|
bytes_per_sync);
|
2013-07-13 01:56:52 +02:00
|
|
|
Log(log," Options.filter_deletes: %d",
|
|
|
|
filter_deletes);
|
2013-07-04 00:32:49 +02:00
|
|
|
Log(log," Options.compaction_style: %d",
|
|
|
|
compaction_style);
|
2013-08-02 20:46:47 +02:00
|
|
|
Log(log," Options.compaction_options_universal.size_ratio: %u",
|
2013-07-04 00:32:49 +02:00
|
|
|
compaction_options_universal.size_ratio);
|
2013-08-02 20:46:47 +02:00
|
|
|
Log(log," Options.compaction_options_universal.min_merge_width: %u",
|
2013-07-04 00:32:49 +02:00
|
|
|
compaction_options_universal.min_merge_width);
|
2013-08-02 20:46:47 +02:00
|
|
|
Log(log," Options.compaction_options_universal.max_merge_width: %u",
|
2013-07-04 00:32:49 +02:00
|
|
|
compaction_options_universal.max_merge_width);
|
2012-08-22 20:43:53 +02:00
|
|
|
} // Options::Dump
|
|
|
|
|
2013-03-08 18:19:24 +01:00
|
|
|
//
|
|
|
|
// The goal of this method is to create a configuration that
|
|
|
|
// allows an application to write all files into L0 and
|
|
|
|
// then do a single compaction to output all files into L1.
|
2013-02-26 07:57:37 +01:00
|
|
|
Options*
|
|
|
|
Options::PrepareForBulkLoad()
|
|
|
|
{
|
2013-03-08 18:19:24 +01:00
|
|
|
// never slowdown ingest.
|
2013-02-26 07:57:37 +01:00
|
|
|
level0_file_num_compaction_trigger = (1<<30);
|
|
|
|
level0_slowdown_writes_trigger = (1<<30);
|
|
|
|
level0_stop_writes_trigger = (1<<30);
|
2013-03-08 18:19:24 +01:00
|
|
|
|
|
|
|
// no auto compactions please. The application should issue a
|
|
|
|
// manual compaction after all data is loaded into L0.
|
2013-02-26 07:57:37 +01:00
|
|
|
disable_auto_compactions = true;
|
|
|
|
disable_seek_compaction = true;
|
|
|
|
disableDataSync = true;
|
2013-03-08 18:19:24 +01:00
|
|
|
|
|
|
|
// A manual compaction run should pick all files in L0 in
|
|
|
|
// a single compaction run.
|
2013-02-26 07:57:37 +01:00
|
|
|
source_compaction_factor = (1<<30);
|
|
|
|
|
2013-03-08 18:19:24 +01:00
|
|
|
// It is better to have only 2 levels, otherwise a manual
|
|
|
|
// compaction would compact at every possible level, thereby
|
|
|
|
// increasing the total time needed for compactions.
|
|
|
|
num_levels = 2;
|
|
|
|
|
|
|
|
// Prevent a memtable flush to automatically promote files
|
|
|
|
// to L1. This is helpful so that all files that are
|
|
|
|
// input to the manual compaction are all at L0.
|
|
|
|
max_background_compactions = 2;
|
|
|
|
|
|
|
|
// The compaction would create large files in L1.
|
|
|
|
target_file_size_base = 256 * 1024 * 1024;
|
2013-02-26 07:57:37 +01:00
|
|
|
return this;
|
|
|
|
}
|
|
|
|
|
2011-10-31 18:22:06 +01:00
|
|
|
} // namespace leveldb
|