2017-01-25 18:51:41 +01:00
|
|
|
// Copyright 2015-2017 Parity Technologies (UK) Ltd.
|
2016-02-05 13:40:41 +01:00
|
|
|
// This file is part of Parity.
|
|
|
|
|
|
|
|
// Parity is free software: you can redistribute it and/or modify
|
|
|
|
// it under the terms of the GNU General Public License as published by
|
|
|
|
// the Free Software Foundation, either version 3 of the License, or
|
|
|
|
// (at your option) any later version.
|
|
|
|
|
|
|
|
// Parity is distributed in the hope that it will be useful,
|
|
|
|
// but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
// MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
// GNU General Public License for more details.
|
|
|
|
|
|
|
|
// You should have received a copy of the GNU General Public License
|
|
|
|
// along with Parity. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
|
2015-12-04 18:05:59 +01:00
|
|
|
#![feature(test)]
|
|
|
|
|
|
|
|
extern crate test;
|
2017-09-03 09:11:14 +02:00
|
|
|
extern crate triehash;
|
2015-12-04 18:05:59 +01:00
|
|
|
extern crate ethcore_util;
|
2017-09-06 20:47:45 +02:00
|
|
|
extern crate ethcore_bytes;
|
2017-09-04 16:36:49 +02:00
|
|
|
extern crate ethcore_bigint;
|
2017-09-06 20:47:45 +02:00
|
|
|
extern crate memorydb;
|
2017-09-12 19:23:02 +02:00
|
|
|
extern crate patricia_trie as trie;
|
2015-12-04 18:05:59 +01:00
|
|
|
#[macro_use]
|
|
|
|
extern crate log;
|
2017-08-30 19:18:28 +02:00
|
|
|
extern crate hash;
|
2015-12-04 18:05:59 +01:00
|
|
|
|
2016-10-20 14:49:44 +02:00
|
|
|
use test::{Bencher, black_box};
|
2017-09-04 16:36:49 +02:00
|
|
|
use ethcore_bigint::hash::*;
|
2017-09-06 20:47:45 +02:00
|
|
|
use ethcore_bytes::*;
|
|
|
|
use trie::*;
|
|
|
|
use memorydb::*;
|
2017-09-03 09:11:14 +02:00
|
|
|
use triehash::*;
|
2017-08-30 19:18:28 +02:00
|
|
|
use hash::keccak;
|
2015-12-04 18:05:59 +01:00
|
|
|
|
|
|
|
fn random_word(alphabet: &[u8], min_count: usize, diff_count: usize, seed: &mut H256) -> Vec<u8> {
|
|
|
|
assert!(min_count + diff_count <= 32);
|
2017-08-30 19:18:28 +02:00
|
|
|
*seed = keccak(&seed);
|
2016-07-13 20:45:15 +02:00
|
|
|
let r = min_count + (seed[31] as usize % (diff_count + 1));
|
2015-12-04 18:05:59 +01:00
|
|
|
let mut ret: Vec<u8> = Vec::with_capacity(r);
|
|
|
|
for i in 0..r {
|
2016-07-13 20:45:15 +02:00
|
|
|
ret.push(alphabet[seed[i] as usize % alphabet.len()]);
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
ret
|
|
|
|
}
|
|
|
|
|
|
|
|
fn random_bytes(min_count: usize, diff_count: usize, seed: &mut H256) -> Vec<u8> {
|
|
|
|
assert!(min_count + diff_count <= 32);
|
2017-08-30 19:18:28 +02:00
|
|
|
*seed = keccak(&seed);
|
2016-07-13 20:45:15 +02:00
|
|
|
let r = min_count + (seed[31] as usize % (diff_count + 1));
|
2017-08-17 16:05:26 +02:00
|
|
|
seed[0..r].to_vec()
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
fn random_value(seed: &mut H256) -> Bytes {
|
2017-08-30 19:18:28 +02:00
|
|
|
*seed = keccak(&seed);
|
2016-07-13 20:45:15 +02:00
|
|
|
match seed[0] % 2 {
|
|
|
|
1 => vec![seed[31];1],
|
2017-08-17 16:05:26 +02:00
|
|
|
_ => seed.to_vec(),
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2016-02-02 01:59:14 +01:00
|
|
|
#[bench]
|
|
|
|
fn trie_insertions_32_mir_1k(b: &mut Bencher) {
|
|
|
|
let st = StandardMap {
|
|
|
|
alphabet: Alphabet::All,
|
|
|
|
min_key: 32,
|
|
|
|
journal_key: 0,
|
|
|
|
value_mode: ValueMode::Mirror,
|
|
|
|
count: 1000,
|
|
|
|
};
|
|
|
|
let d = st.make();
|
|
|
|
let mut hash_count = 0usize;
|
|
|
|
b.iter(&mut ||{
|
|
|
|
let mut memdb = MemoryDB::new();
|
|
|
|
let mut root = H256::new();
|
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2016-02-02 01:59:14 +01:00
|
|
|
}
|
|
|
|
hash_count = t.hash_count;
|
|
|
|
});
|
|
|
|
// println!("hash_count: {}", hash_count);
|
|
|
|
}
|
2016-10-20 14:49:44 +02:00
|
|
|
#[bench]
|
|
|
|
fn trie_iter(b: &mut Bencher) {
|
|
|
|
let st = StandardMap {
|
|
|
|
alphabet: Alphabet::All,
|
|
|
|
min_key: 32,
|
|
|
|
journal_key: 0,
|
|
|
|
value_mode: ValueMode::Mirror,
|
|
|
|
count: 1000,
|
|
|
|
};
|
|
|
|
let d = st.make();
|
|
|
|
let mut memdb = MemoryDB::new();
|
|
|
|
let mut root = H256::new();
|
|
|
|
{
|
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
|
|
|
for i in d.iter() {
|
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(&mut ||{
|
|
|
|
let t = TrieDB::new(&memdb, &root).unwrap();
|
|
|
|
for n in t.iter().unwrap() {
|
|
|
|
black_box(n).unwrap();
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2016-02-02 01:59:14 +01:00
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn triehash_insertions_32_mir_1k(b: &mut Bencher) {
|
|
|
|
let st = StandardMap {
|
|
|
|
alphabet: Alphabet::All,
|
|
|
|
min_key: 32,
|
|
|
|
journal_key: 0,
|
|
|
|
value_mode: ValueMode::Mirror,
|
|
|
|
count: 1000,
|
|
|
|
};
|
|
|
|
let d = st.make();
|
|
|
|
b.iter(&mut ||{
|
|
|
|
trie_root(d.clone()).clone();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn trie_insertions_32_ran_1k(b: &mut Bencher) {
|
|
|
|
let st = StandardMap {
|
|
|
|
alphabet: Alphabet::All,
|
|
|
|
min_key: 32,
|
|
|
|
journal_key: 0,
|
|
|
|
value_mode: ValueMode::Random,
|
|
|
|
count: 1000,
|
|
|
|
};
|
|
|
|
let d = st.make();
|
|
|
|
let mut hash_count = 0usize;
|
|
|
|
let mut r = H256::new();
|
|
|
|
b.iter(&mut ||{
|
|
|
|
let mut memdb = MemoryDB::new();
|
|
|
|
let mut root = H256::new();
|
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2016-02-02 01:59:14 +01:00
|
|
|
}
|
|
|
|
hash_count = t.hash_count;
|
|
|
|
r = t.root().clone();
|
|
|
|
});
|
|
|
|
// println!("result: {}", hash_count);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn triehash_insertions_32_ran_1k(b: &mut Bencher) {
|
|
|
|
let st = StandardMap {
|
|
|
|
alphabet: Alphabet::All,
|
|
|
|
min_key: 32,
|
|
|
|
journal_key: 0,
|
|
|
|
value_mode: ValueMode::Random,
|
|
|
|
count: 1000,
|
|
|
|
};
|
|
|
|
let d = st.make();
|
|
|
|
b.iter(&mut ||{
|
|
|
|
trie_root(d.clone()).clone();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2015-12-04 18:05:59 +01:00
|
|
|
#[bench]
|
2015-12-09 02:29:34 +01:00
|
|
|
fn trie_insertions_six_high(b: &mut Bencher) {
|
2015-12-04 18:05:59 +01:00
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_bytes(6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(||{
|
2015-12-10 15:57:47 +01:00
|
|
|
let mut memdb = MemoryDB::new();
|
2015-12-11 11:53:44 +01:00
|
|
|
let mut root = H256::new();
|
2015-12-17 12:43:50 +01:00
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
2015-12-04 18:05:59 +01:00
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2015-12-09 02:29:34 +01:00
|
|
|
fn triehash_insertions_six_high(b: &mut Bencher) {
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_bytes(6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(&||{
|
|
|
|
trie_root(d.clone());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn trie_insertions_six_mid(b: &mut Bencher) {
|
2015-12-04 18:05:59 +01:00
|
|
|
let alphabet = b"@QWERTYUIOPASDFGHJKLZXCVBNM[/]^_";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
b.iter(||{
|
2015-12-10 15:57:47 +01:00
|
|
|
let mut memdb = MemoryDB::new();
|
2015-12-11 11:53:44 +01:00
|
|
|
let mut root = H256::new();
|
2015-12-17 12:43:50 +01:00
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
2015-12-04 18:05:59 +01:00
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
debug!("hash_count={:?}", t.hash_count);
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2015-12-09 02:29:34 +01:00
|
|
|
fn triehash_insertions_six_mid(b: &mut Bencher) {
|
|
|
|
let alphabet = b"@QWERTYUIOPASDFGHJKLZXCVBNM[/]^_";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
b.iter(||{
|
|
|
|
trie_root(d.clone());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn trie_insertions_random_mid(b: &mut Bencher) {
|
2015-12-04 18:05:59 +01:00
|
|
|
let alphabet = b"@QWERTYUIOPASDFGHJKLZXCVBNM[/]^_";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 1, 5, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(||{
|
2015-12-10 15:57:47 +01:00
|
|
|
let mut memdb = MemoryDB::new();
|
2015-12-11 11:53:44 +01:00
|
|
|
let mut root = H256::new();
|
2015-12-17 12:43:50 +01:00
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
2015-12-04 18:05:59 +01:00
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
2015-12-09 02:29:34 +01:00
|
|
|
fn triehash_insertions_random_mid(b: &mut Bencher) {
|
|
|
|
let alphabet = b"@QWERTYUIOPASDFGHJKLZXCVBNM[/]^_";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 1, 5, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(||{
|
|
|
|
trie_root(d.clone());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn trie_insertions_six_low(b: &mut Bencher) {
|
2015-12-04 18:05:59 +01:00
|
|
|
let alphabet = b"abcdef";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(||{
|
2015-12-10 15:57:47 +01:00
|
|
|
let mut memdb = MemoryDB::new();
|
2015-12-11 11:53:44 +01:00
|
|
|
let mut root = H256::new();
|
2015-12-17 12:43:50 +01:00
|
|
|
let mut t = TrieDBMut::new(&mut memdb, &mut root);
|
2015-12-04 18:05:59 +01:00
|
|
|
for i in d.iter() {
|
2016-08-17 22:36:55 +02:00
|
|
|
t.insert(&i.0, &i.1).unwrap();
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-12-09 02:29:34 +01:00
|
|
|
#[bench]
|
|
|
|
fn triehash_insertions_six_low(b: &mut Bencher) {
|
|
|
|
let alphabet = b"abcdef";
|
|
|
|
let mut d: Vec<(Bytes, Bytes)> = Vec::new();
|
|
|
|
let mut seed = H256::new();
|
|
|
|
for _ in 0..1000 {
|
|
|
|
let k = random_word(alphabet, 6, 0, &mut seed);
|
|
|
|
let v = random_value(&mut seed);
|
|
|
|
d.push((k, v))
|
|
|
|
}
|
|
|
|
|
|
|
|
b.iter(||{
|
|
|
|
trie_root(d.clone());
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2015-12-04 18:05:59 +01:00
|
|
|
#[bench]
|
2017-08-30 19:18:28 +02:00
|
|
|
fn keccakx10000(b: &mut Bencher) {
|
2015-12-04 18:05:59 +01:00
|
|
|
b.iter(||{
|
|
|
|
let mut seed = H256::new();
|
2016-01-16 14:29:36 +01:00
|
|
|
for _ in 0..10000 {
|
2017-08-30 19:18:28 +02:00
|
|
|
seed = keccak(&seed);
|
2015-12-04 18:05:59 +01:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|