2019-08-29 17:33:08 +03:00
|
|
|
#![feature(test)]
|
2021-05-28 09:07:26 +03:00
|
|
|
#![feature(bench_black_box)]
|
2019-08-29 17:33:08 +03:00
|
|
|
|
|
|
|
extern crate rand;
|
|
|
|
extern crate rand_pcg;
|
2021-05-28 08:56:21 +03:00
|
|
|
extern crate test;
|
2019-08-29 17:33:08 +03:00
|
|
|
|
|
|
|
use evalexpr::build_operator_tree;
|
2021-05-28 08:56:21 +03:00
|
|
|
use rand::{distributions::Uniform, seq::SliceRandom, Rng, SeedableRng};
|
|
|
|
use rand_pcg::Pcg32;
|
2019-08-29 17:33:08 +03:00
|
|
|
use std::hint::black_box;
|
2021-05-28 08:56:21 +03:00
|
|
|
use test::Bencher;
|
2019-08-29 17:33:08 +03:00
|
|
|
|
|
|
|
const BENCHMARK_LEN: usize = 100_000;
|
2019-08-29 18:19:05 +03:00
|
|
|
const EXPONENTIAL_TUPLE_ITERATIONS: usize = 12;
|
2019-08-29 17:33:08 +03:00
|
|
|
|
|
|
|
fn generate_expression<Gen: Rng>(len: usize, gen: &mut Gen) -> String {
|
2019-08-29 18:19:05 +03:00
|
|
|
let int_distribution = Uniform::new_inclusive(1, 100);
|
2019-08-29 17:33:08 +03:00
|
|
|
let whitespaces = vec![" ", "", "", " ", " \n", " "];
|
|
|
|
let operators = vec!["+", "-", "*", "/", "%", "^"];
|
|
|
|
let mut result = String::new();
|
|
|
|
result.push_str(&format!("{}", gen.sample(int_distribution)));
|
|
|
|
|
|
|
|
while result.len() < len {
|
|
|
|
result.push_str(whitespaces.choose(gen).unwrap());
|
|
|
|
result.push_str(operators.choose(gen).unwrap());
|
|
|
|
result.push_str(whitespaces.choose(gen).unwrap());
|
|
|
|
result.push_str(&format!("{}", gen.sample(int_distribution)));
|
|
|
|
}
|
|
|
|
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_expression_chain<Gen: Rng>(len: usize, gen: &mut Gen) -> String {
|
|
|
|
let mut chain = generate_expression(10, gen);
|
|
|
|
while chain.len() < len {
|
|
|
|
chain.push_str("; ");
|
|
|
|
chain.push_str(&generate_expression(10, gen));
|
|
|
|
}
|
|
|
|
chain
|
|
|
|
}
|
|
|
|
|
|
|
|
fn generate_small_expressions<Gen: Rng>(len: usize, gen: &mut Gen) -> Vec<String> {
|
|
|
|
let mut result = Vec::new();
|
|
|
|
let mut result_len = 0;
|
|
|
|
while result_len < len {
|
|
|
|
let expression = generate_expression(10, gen);
|
|
|
|
result_len += expression.len();
|
|
|
|
result.push(expression);
|
|
|
|
}
|
|
|
|
result
|
|
|
|
}
|
|
|
|
|
2019-08-29 18:19:05 +03:00
|
|
|
fn generate_large_tuple_expression<Gen: Rng>(iterations: usize, gen: &mut Gen) -> String {
|
|
|
|
let mut result = String::from("a=(");
|
|
|
|
result.push_str(&generate_expression(0, gen));
|
2021-06-22 10:38:12 +02:00
|
|
|
result.push(',');
|
2019-08-29 18:19:05 +03:00
|
|
|
result.push_str(&generate_expression(0, gen));
|
2021-06-22 10:38:12 +02:00
|
|
|
result.push(')');
|
2019-08-29 18:19:05 +03:00
|
|
|
for _ in 0..iterations {
|
|
|
|
result.push_str(";a=(a,a)")
|
|
|
|
}
|
|
|
|
dbg!(result)
|
|
|
|
}
|
|
|
|
|
2019-08-29 17:33:08 +03:00
|
|
|
#[bench]
|
|
|
|
fn bench_parse_long_expression_chains(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(0);
|
|
|
|
let long_expression_chain = generate_expression_chain(BENCHMARK_LEN, &mut gen);
|
|
|
|
|
2021-05-28 08:56:21 +03:00
|
|
|
bencher.iter(|| build_operator_tree(&long_expression_chain).unwrap());
|
2019-08-29 17:33:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_parse_deep_expression_trees(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(15);
|
|
|
|
let deep_expression_tree = generate_expression(BENCHMARK_LEN, &mut gen);
|
|
|
|
|
2021-05-28 08:56:21 +03:00
|
|
|
bencher.iter(|| build_operator_tree(&deep_expression_tree).unwrap());
|
2019-08-29 17:33:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_parse_many_small_expressions(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(33);
|
|
|
|
let small_expressions = generate_small_expressions(BENCHMARK_LEN, &mut gen);
|
|
|
|
|
|
|
|
bencher.iter(|| {
|
|
|
|
for expression in &small_expressions {
|
2021-06-22 10:38:12 +02:00
|
|
|
black_box(build_operator_tree(expression).unwrap());
|
2019-08-29 17:33:08 +03:00
|
|
|
}
|
|
|
|
});
|
2019-08-29 18:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_evaluate_long_expression_chains(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(0);
|
2021-05-28 08:56:21 +03:00
|
|
|
let long_expression_chain =
|
|
|
|
build_operator_tree(&generate_expression_chain(BENCHMARK_LEN, &mut gen)).unwrap();
|
2019-08-29 18:19:05 +03:00
|
|
|
|
2021-05-28 08:56:21 +03:00
|
|
|
bencher.iter(|| long_expression_chain.eval().unwrap());
|
2019-08-29 18:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_evaluate_deep_expression_trees(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(15);
|
2021-05-28 08:56:21 +03:00
|
|
|
let deep_expression_tree =
|
|
|
|
build_operator_tree(&generate_expression(BENCHMARK_LEN, &mut gen)).unwrap();
|
2019-08-29 18:19:05 +03:00
|
|
|
|
2021-05-28 08:56:21 +03:00
|
|
|
bencher.iter(|| deep_expression_tree.eval().unwrap());
|
2019-08-29 18:19:05 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_evaluate_many_small_expressions(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(33);
|
2021-05-28 08:56:21 +03:00
|
|
|
let small_expressions: Vec<_> = generate_small_expressions(BENCHMARK_LEN, &mut gen)
|
|
|
|
.iter()
|
2021-06-22 10:38:12 +02:00
|
|
|
.map(|expression| build_operator_tree(expression).unwrap())
|
2021-05-28 08:56:21 +03:00
|
|
|
.collect();
|
2019-08-29 18:19:05 +03:00
|
|
|
|
|
|
|
bencher.iter(|| {
|
|
|
|
for expression in &small_expressions {
|
|
|
|
black_box(expression.eval().unwrap());
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
#[bench]
|
|
|
|
fn bench_evaluate_large_tuple_expression(bencher: &mut Bencher) {
|
|
|
|
let mut gen = Pcg32::seed_from_u64(44);
|
2021-05-28 08:56:21 +03:00
|
|
|
let large_tuple_expression = build_operator_tree(&generate_large_tuple_expression(
|
|
|
|
EXPONENTIAL_TUPLE_ITERATIONS,
|
|
|
|
&mut gen,
|
|
|
|
))
|
|
|
|
.unwrap();
|
2019-08-29 18:19:05 +03:00
|
|
|
dbg!(&large_tuple_expression);
|
|
|
|
|
2021-05-28 08:56:21 +03:00
|
|
|
bencher.iter(|| large_tuple_expression.eval().unwrap());
|
|
|
|
}
|