no prepared keys
This commit is contained in:
parent
fe6b1de1f6
commit
f71be68c38
@ -133,19 +133,19 @@ int main(int argc, char *argv[]) {
|
|||||||
rocksdb::ColumnFamilyOptions cfo{};
|
rocksdb::ColumnFamilyOptions cfo{};
|
||||||
cfo.enable_blob_files = true;
|
cfo.enable_blob_files = true;
|
||||||
cfo.min_blob_size = args.blob_size;
|
cfo.min_blob_size = args.blob_size;
|
||||||
// rocksdb::BlockBasedTableOptions top{};
|
cfo.disable_auto_compactions = true;
|
||||||
// top.use_delta_encoding = false;
|
cfo.max_compaction_bytes = (1ULL << 60);
|
||||||
// cfo.table_factory.reset(rocksdb::NewBlockBasedTableFactory(top));
|
cfo.level0_stop_writes_trigger = 100000;
|
||||||
|
cfo.level0_slowdown_writes_trigger = 100000;
|
||||||
|
cfo.level0_file_num_compaction_trigger = 100000;
|
||||||
|
cfo.write_buffer_size = 64 << 20;
|
||||||
|
cfo.max_write_buffer_number = 64;
|
||||||
|
|
||||||
// use 3GB block cache
|
// use 3GB block cache
|
||||||
auto cache = rocksdb::NewLRUCache(3 << 30);
|
auto cache = rocksdb::NewLRUCache(3 << 30);
|
||||||
rocksdb::BlockBasedTableOptions table_options{};
|
rocksdb::BlockBasedTableOptions table_options{};
|
||||||
table_options.block_cache = cache;
|
table_options.block_cache = cache;
|
||||||
cfo.table_factory.reset(NewBlockBasedTableFactory(table_options));
|
cfo.table_factory.reset(NewBlockBasedTableFactory(table_options));
|
||||||
// the following three options makes it not trigger GC in test
|
|
||||||
cfo.level0_file_num_compaction_trigger = 10000;
|
|
||||||
cfo.write_buffer_size = 64 << 20;
|
|
||||||
cfo.max_write_buffer_number = 16;
|
|
||||||
|
|
||||||
std::vector<rocksdb::ColumnFamilyDescriptor> cfd{};
|
std::vector<rocksdb::ColumnFamilyDescriptor> cfd{};
|
||||||
cfd.push_back(rocksdb::ColumnFamilyDescriptor("default", cfo));
|
cfd.push_back(rocksdb::ColumnFamilyDescriptor("default", cfo));
|
||||||
@ -160,7 +160,6 @@ int main(int argc, char *argv[]) {
|
|||||||
wopt.no_slowdown = true;
|
wopt.no_slowdown = true;
|
||||||
// wopt.disableWAL = true;
|
// wopt.disableWAL = true;
|
||||||
std::vector<std::thread> wg;
|
std::vector<std::thread> wg;
|
||||||
std::vector<std::vector<std::string>> keys{};
|
|
||||||
std::atomic<uint64_t> total_op{0};
|
std::atomic<uint64_t> total_op{0};
|
||||||
rocksdb::OptimisticTransactionDB *db;
|
rocksdb::OptimisticTransactionDB *db;
|
||||||
auto b = nm::Instant::now();
|
auto b = nm::Instant::now();
|
||||||
@ -174,32 +173,16 @@ int main(int argc, char *argv[]) {
|
|||||||
std::mt19937 gen(rd());
|
std::mt19937 gen(rd());
|
||||||
|
|
||||||
std::string val(args.value_size, 'x');
|
std::string val(args.value_size, 'x');
|
||||||
std::vector<size_t> key_counts(args.threads, args.iterations / args.threads);
|
|
||||||
for (size_t i = 0; i < args.iterations % args.threads; ++i) {
|
|
||||||
key_counts[i] += 1;
|
|
||||||
}
|
|
||||||
keys.reserve(args.threads);
|
|
||||||
for (size_t tid = 0; tid < args.threads; ++tid) {
|
|
||||||
std::vector<std::string> key{};
|
|
||||||
key.reserve(key_counts[tid]);
|
|
||||||
for (size_t i = 0; i < key_counts[tid]; ++i) {
|
|
||||||
auto tmp = std::format("key_{}_{}", tid, i);
|
|
||||||
tmp.resize(args.key_size, 'x');
|
|
||||||
key.emplace_back(std::move(tmp));
|
|
||||||
}
|
|
||||||
if (args.mode == "get" || args.random) {
|
|
||||||
std::shuffle(key.begin(), key.end(), gen);
|
|
||||||
}
|
|
||||||
keys.emplace_back(std::move(key));
|
|
||||||
}
|
|
||||||
|
|
||||||
auto *handle = handles[0];
|
auto *handle = handles[0];
|
||||||
|
|
||||||
if (args.mode == "get" || args.mode == "scan") {
|
if (args.mode == "get" || args.mode == "scan") {
|
||||||
auto *kv = db->BeginTransaction(wopt);
|
auto *kv = db->BeginTransaction(wopt);
|
||||||
for (size_t tid = 0; tid < args.threads; ++tid) {
|
for (size_t tid = 0; tid < args.threads; ++tid) {
|
||||||
auto *tk = &keys[tid];
|
size_t count = args.iterations / args.threads + (tid < args.iterations % args.threads ? 1 : 0);
|
||||||
for (auto &key: *tk) {
|
for (size_t i = 0; i < count; ++i) {
|
||||||
|
auto key = std::format("key_{}_{}", tid, i);
|
||||||
|
key.resize(args.key_size, 'x');
|
||||||
kv->Put(handle, key, val);
|
kv->Put(handle, key, val);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -214,16 +197,15 @@ int main(int argc, char *argv[]) {
|
|||||||
|
|
||||||
handle = handles[0];
|
handle = handles[0];
|
||||||
|
|
||||||
// simulate common use cases
|
|
||||||
std::uniform_int_distribution<size_t> tid_dist(0, args.threads - 1);
|
std::uniform_int_distribution<size_t> tid_dist(0, args.threads - 1);
|
||||||
for (size_t i = 0; i < args.iterations; ++i) {
|
for (size_t i = 0; i < args.iterations; ++i) {
|
||||||
auto tid = tid_dist(gen);
|
auto tid = tid_dist(gen);
|
||||||
if (keys[tid].empty()) {
|
size_t count = args.iterations / args.threads + (tid < args.iterations % args.threads ? 1 : 0);
|
||||||
continue;
|
std::uniform_int_distribution<size_t> key_dist(0, count - 1);
|
||||||
}
|
auto idx = key_dist(gen);
|
||||||
std::uniform_int_distribution<size_t> key_dist(0, keys[tid].size() - 1);
|
auto key = std::format("key_{}_{}", tid, idx);
|
||||||
const auto &k = keys[tid][key_dist(gen)];
|
key.resize(args.key_size, 'x');
|
||||||
auto s = db->Get(rocksdb::ReadOptions(), k, &val);
|
auto s = db->Get(rocksdb::ReadOptions(), key, &val);
|
||||||
if (!s.ok()) {
|
if (!s.ok()) {
|
||||||
std::terminate();
|
std::terminate();
|
||||||
}
|
}
|
||||||
@ -243,18 +225,21 @@ int main(int argc, char *argv[]) {
|
|||||||
if (!upper_bound.empty()) {
|
if (!upper_bound.empty()) {
|
||||||
ropt.iterate_upper_bound = &upper_bound_slice;
|
ropt.iterate_upper_bound = &upper_bound_slice;
|
||||||
}
|
}
|
||||||
auto *tk = &keys[tid];
|
|
||||||
ropt.prefix_same_as_start = true;
|
ropt.prefix_same_as_start = true;
|
||||||
ropt.snapshot = snapshot;
|
ropt.snapshot = snapshot;
|
||||||
size_t round = 0;
|
size_t round = 0;
|
||||||
std::mt19937 mixed_gen(static_cast<uint32_t>(base_seed) ^ static_cast<uint32_t>(0x9e3779b9U * (tid + 1)));
|
std::mt19937 mixed_gen(static_cast<uint32_t>(base_seed) ^ static_cast<uint32_t>(0x9e3779b9U * (tid + 1)));
|
||||||
std::uniform_int_distribution<int> mixed_dist(0, 99);
|
std::uniform_int_distribution<int> mixed_dist(0, 99);
|
||||||
|
|
||||||
|
size_t key_count = args.iterations / args.threads + (tid < args.iterations % args.threads ? 1 : 0);
|
||||||
|
|
||||||
ready_barrier.arrive_and_wait();
|
ready_barrier.arrive_and_wait();
|
||||||
start_barrier.arrive_and_wait();
|
start_barrier.arrive_and_wait();
|
||||||
|
|
||||||
if (args.mode == "insert") {
|
if (args.mode == "insert") {
|
||||||
for (auto &key: *tk) {
|
for (size_t i = 0; i < key_count; ++i) {
|
||||||
|
auto key = std::format("key_{}_{}", tid, i);
|
||||||
|
key.resize(args.key_size, 'x');
|
||||||
round += 1;
|
round += 1;
|
||||||
auto *kv = db->BeginTransaction(wopt);
|
auto *kv = db->BeginTransaction(wopt);
|
||||||
kv->Put(handle, key, val);
|
kv->Put(handle, key, val);
|
||||||
@ -263,7 +248,9 @@ int main(int argc, char *argv[]) {
|
|||||||
}
|
}
|
||||||
|
|
||||||
} else if (args.mode == "get") {
|
} else if (args.mode == "get") {
|
||||||
for (auto &key: *tk) {
|
for (size_t i = 0; i < key_count; ++i) {
|
||||||
|
auto key = std::format("key_{}_{}", tid, i);
|
||||||
|
key.resize(args.key_size, 'x');
|
||||||
round += 1;
|
round += 1;
|
||||||
auto *kv = db->BeginTransaction(wopt);
|
auto *kv = db->BeginTransaction(wopt);
|
||||||
kv->Get(ropt, handle, key, &rval);
|
kv->Get(ropt, handle, key, &rval);
|
||||||
@ -271,20 +258,21 @@ int main(int argc, char *argv[]) {
|
|||||||
delete kv;
|
delete kv;
|
||||||
}
|
}
|
||||||
} else if (args.mode == "mixed") {
|
} else if (args.mode == "mixed") {
|
||||||
for (auto &key: *tk) {
|
for (size_t i = 0; i < key_count; ++i) {
|
||||||
|
auto key = std::format("key_{}_{}", tid, i);
|
||||||
|
key.resize(args.key_size, 'x');
|
||||||
round += 1;
|
round += 1;
|
||||||
auto is_insert = mixed_dist(mixed_gen) < static_cast<int>(args.insert_ratio);
|
auto is_insert = mixed_dist(mixed_gen) < static_cast<int>(args.insert_ratio);
|
||||||
auto *kv = db->BeginTransaction(wopt);
|
auto *kv = db->BeginTransaction(wopt);
|
||||||
if (is_insert) {
|
if (is_insert) {
|
||||||
kv->Put(handle, key, val);
|
kv->Put(handle, key, val);
|
||||||
} else {
|
} else {
|
||||||
kv->Get(ropt, handle, key, &rval); // not found
|
kv->Get(ropt, handle, key, &rval);
|
||||||
}
|
}
|
||||||
kv->Commit();
|
kv->Commit();
|
||||||
delete kv;
|
delete kv;
|
||||||
}
|
}
|
||||||
} else if (args.mode == "scan") {
|
} else if (args.mode == "scan") {
|
||||||
// ropt.pin_data = true;
|
|
||||||
auto *iter = db->NewIterator(ropt);
|
auto *iter = db->NewIterator(ropt);
|
||||||
iter->Seek(prefix);
|
iter->Seek(prefix);
|
||||||
size_t n = 0;
|
size_t n = 0;
|
||||||
|
|||||||
88
src/main.rs
88
src/main.rs
@ -52,7 +52,7 @@ fn main() {
|
|||||||
Logger::init().add_file("/tmp/x.log", true);
|
Logger::init().add_file("/tmp/x.log", true);
|
||||||
log::set_max_level(log::LevelFilter::Info);
|
log::set_max_level(log::LevelFilter::Info);
|
||||||
}
|
}
|
||||||
let mut args = Args::parse();
|
let args = Args::parse();
|
||||||
|
|
||||||
let path = Path::new(&args.path);
|
let path = Path::new(&args.path);
|
||||||
|
|
||||||
@ -86,64 +86,66 @@ fn main() {
|
|||||||
exit(1);
|
exit(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut keys: Vec<Vec<Vec<u8>>> = Vec::with_capacity(args.threads);
|
|
||||||
let mut opt = Options::new(path);
|
let mut opt = Options::new(path);
|
||||||
opt.sync_on_write = false;
|
opt.sync_on_write = false;
|
||||||
opt.over_provision = true; // large value will use lots of memeory
|
opt.over_provision = false;
|
||||||
opt.inline_size = args.blob_size;
|
opt.inline_size = args.blob_size;
|
||||||
opt.tmp_store = args.mode != "get" && args.mode != "scan";
|
opt.tmp_store = args.mode != "get" && args.mode != "scan";
|
||||||
opt.cache_capacity = 3 << 30;
|
opt.cache_capacity = 3 << 30;
|
||||||
|
opt.data_file_size = 64 << 20;
|
||||||
|
opt.max_log_size = 1 << 30;
|
||||||
let mut saved = opt.clone();
|
let mut saved = opt.clone();
|
||||||
saved.tmp_store = false;
|
saved.tmp_store = false;
|
||||||
let mut db = Mace::new(opt.validate().unwrap()).unwrap();
|
let mut db = Mace::new(opt.validate().unwrap()).unwrap();
|
||||||
db.disable_gc();
|
db.disable_gc();
|
||||||
let mut bkt = db.new_bucket("default").unwrap();
|
let mut bkt = db.new_bucket("default").unwrap();
|
||||||
|
|
||||||
let mut rng = rand::rng();
|
|
||||||
let value = Arc::new(vec![b'0'; args.value_size]);
|
let value = Arc::new(vec![b'0'; args.value_size]);
|
||||||
let mut key_counts = vec![args.iterations / args.threads; args.threads];
|
|
||||||
for cnt in key_counts.iter_mut().take(args.iterations % args.threads) {
|
|
||||||
*cnt += 1;
|
|
||||||
}
|
|
||||||
for tid in 0..args.threads {
|
|
||||||
let mut tk = Vec::with_capacity(key_counts[tid]);
|
|
||||||
for i in 0..key_counts[tid] {
|
|
||||||
let mut key = format!("key_{tid}_{i}").into_bytes();
|
|
||||||
key.resize(args.key_size, b'x');
|
|
||||||
tk.push(key);
|
|
||||||
}
|
|
||||||
if args.random || args.mode == "get" {
|
|
||||||
tk.shuffle(&mut rng);
|
|
||||||
}
|
|
||||||
keys.push(tk);
|
|
||||||
}
|
|
||||||
|
|
||||||
if args.mode == "get" || args.mode == "scan" {
|
if args.mode == "get" || args.mode == "scan" {
|
||||||
let pre_tx = bkt.begin().unwrap();
|
let pre_tx = bkt.begin().unwrap();
|
||||||
(0..args.threads).for_each(|tid| {
|
for tid in 0..args.threads {
|
||||||
for k in &keys[tid] {
|
let count = args.iterations / args.threads
|
||||||
pre_tx.put(k, &*value).unwrap();
|
+ if tid < args.iterations % args.threads {
|
||||||
|
1
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
};
|
||||||
|
for i in 0..count {
|
||||||
|
let mut key = format!("key_{tid}_{i}").into_bytes();
|
||||||
|
key.resize(args.key_size, b'x');
|
||||||
|
pre_tx.put(&key, &*value).unwrap();
|
||||||
}
|
}
|
||||||
});
|
}
|
||||||
pre_tx.commit().unwrap();
|
pre_tx.commit().unwrap();
|
||||||
drop(bkt);
|
drop(bkt);
|
||||||
drop(db);
|
drop(db);
|
||||||
// re-open db
|
|
||||||
saved.tmp_store = true;
|
saved.tmp_store = true;
|
||||||
db = Mace::new(saved.validate().unwrap()).unwrap();
|
db = Mace::new(saved.validate().unwrap()).unwrap();
|
||||||
bkt = db.get_bucket("default").unwrap();
|
bkt = db.get_bucket("default").unwrap();
|
||||||
|
|
||||||
// simulate common use cases
|
let mut rng = rand::rng();
|
||||||
for _ in 0..args.iterations {
|
for _ in 0..args.iterations {
|
||||||
let tid = rng.random_range(0..args.threads);
|
let tid = rng.random_range(0..args.threads);
|
||||||
let Some(k) = keys[tid].choose(&mut rng) else {
|
let count = args.iterations / args.threads
|
||||||
continue;
|
+ if tid < args.iterations % args.threads {
|
||||||
};
|
1
|
||||||
|
} else {
|
||||||
|
0
|
||||||
|
};
|
||||||
|
let idx = rng.random_range(0..count);
|
||||||
|
let mut key = format!("key_{tid}_{idx}").into_bytes();
|
||||||
|
key.resize(args.key_size, b'x');
|
||||||
let view = bkt.view().unwrap();
|
let view = bkt.view().unwrap();
|
||||||
view.get(k).unwrap();
|
view.get(&key).unwrap();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
let mut key_counts = vec![args.iterations / args.threads; args.threads];
|
||||||
|
for cnt in key_counts.iter_mut().take(args.iterations % args.threads) {
|
||||||
|
*cnt += 1;
|
||||||
|
}
|
||||||
|
|
||||||
let ready_barrier = Arc::new(std::sync::Barrier::new(args.threads + 1));
|
let ready_barrier = Arc::new(std::sync::Barrier::new(args.threads + 1));
|
||||||
let start_barrier = Arc::new(std::sync::Barrier::new(args.threads + 1));
|
let start_barrier = Arc::new(std::sync::Barrier::new(args.threads + 1));
|
||||||
let total_ops = Arc::new(std::sync::atomic::AtomicUsize::new(0));
|
let total_ops = Arc::new(std::sync::atomic::AtomicUsize::new(0));
|
||||||
@ -151,13 +153,14 @@ fn main() {
|
|||||||
let h: Vec<JoinHandle<()>> = (0..args.threads)
|
let h: Vec<JoinHandle<()>> = (0..args.threads)
|
||||||
.map(|tid| {
|
.map(|tid| {
|
||||||
let db = bkt.clone();
|
let db = bkt.clone();
|
||||||
let tk: &Vec<Vec<u8>> = unsafe { std::mem::transmute(&keys[tid]) };
|
|
||||||
let total_ops = total_ops.clone();
|
let total_ops = total_ops.clone();
|
||||||
let ready_barrier = Arc::clone(&ready_barrier);
|
let ready_barrier = Arc::clone(&ready_barrier);
|
||||||
let start_barrier = Arc::clone(&start_barrier);
|
let start_barrier = Arc::clone(&start_barrier);
|
||||||
let mode = args.mode.clone();
|
let mode = args.mode.clone();
|
||||||
let insert_ratio = args.insert_ratio;
|
let insert_ratio = args.insert_ratio;
|
||||||
let val = value.clone();
|
let val = value.clone();
|
||||||
|
let key_count = key_counts[tid];
|
||||||
|
let key_size = args.key_size;
|
||||||
let prefix = format!("key_{tid}_");
|
let prefix = format!("key_{tid}_");
|
||||||
|
|
||||||
std::thread::spawn(move || {
|
std::thread::spawn(move || {
|
||||||
@ -167,7 +170,9 @@ fn main() {
|
|||||||
start_barrier.wait();
|
start_barrier.wait();
|
||||||
match mode.as_str() {
|
match mode.as_str() {
|
||||||
"insert" => {
|
"insert" => {
|
||||||
for key in tk {
|
for i in 0..key_count {
|
||||||
|
let mut key = format!("key_{tid}_{i}").into_bytes();
|
||||||
|
key.resize(key_size, b'x');
|
||||||
round += 1;
|
round += 1;
|
||||||
let tx = db.begin().unwrap();
|
let tx = db.begin().unwrap();
|
||||||
tx.put(key.as_slice(), val.as_slice()).unwrap();
|
tx.put(key.as_slice(), val.as_slice()).unwrap();
|
||||||
@ -175,7 +180,9 @@ fn main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
"get" => {
|
"get" => {
|
||||||
for key in tk {
|
for i in 0..key_count {
|
||||||
|
let mut key = format!("key_{tid}_{i}").into_bytes();
|
||||||
|
key.resize(key_size, b'x');
|
||||||
round += 1;
|
round += 1;
|
||||||
let tx = db.view().unwrap();
|
let tx = db.view().unwrap();
|
||||||
let x = tx.get(key).unwrap();
|
let x = tx.get(key).unwrap();
|
||||||
@ -183,7 +190,9 @@ fn main() {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
"mixed" => {
|
"mixed" => {
|
||||||
for key in tk {
|
for i in 0..key_count {
|
||||||
|
let mut key = format!("key_{tid}_{i}").into_bytes();
|
||||||
|
key.resize(key_size, b'x');
|
||||||
let is_insert = rand::random_range(0..100) < insert_ratio;
|
let is_insert = rand::random_range(0..100) < insert_ratio;
|
||||||
round += 1;
|
round += 1;
|
||||||
|
|
||||||
@ -193,7 +202,7 @@ fn main() {
|
|||||||
tx.commit().unwrap();
|
tx.commit().unwrap();
|
||||||
} else {
|
} else {
|
||||||
let tx = db.view().unwrap();
|
let tx = db.view().unwrap();
|
||||||
let x = tx.get(key); // not found
|
let x = tx.get(key);
|
||||||
let _ = std::hint::black_box(x);
|
let _ = std::hint::black_box(x);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -233,16 +242,17 @@ fn main() {
|
|||||||
} else {
|
} else {
|
||||||
0
|
0
|
||||||
};
|
};
|
||||||
if args.mode == "insert" {
|
let mut mode = args.mode.clone();
|
||||||
|
if mode == "insert" {
|
||||||
if args.random {
|
if args.random {
|
||||||
args.mode = "random_insert".into();
|
mode = "random_insert".into();
|
||||||
} else {
|
} else {
|
||||||
args.mode = "sequential_insert".into();
|
mode = "sequential_insert".into();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
eprintln!(
|
eprintln!(
|
||||||
"{},{},{},{},{},{},{}",
|
"{},{},{},{},{},{},{}",
|
||||||
args.mode,
|
mode,
|
||||||
args.threads,
|
args.threads,
|
||||||
args.key_size,
|
args.key_size,
|
||||||
args.value_size,
|
args.value_size,
|
||||||
|
|||||||
Loading…
Reference in New Issue
Block a user