aboutsummaryrefslogtreecommitdiff
path: root/incria/benches/spreadsheet.rs
blob: ada4b2286fa65ed2c6c72d933793228c1d7d5004 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
use std::{future::Future, pin::Pin, sync::OnceLock};

use criterion::{black_box, criterion_group, criterion_main, Criterion};
use incria::{
    deps,
    thunk::{Thunk, ThunkMapper},
    Mapper,
};

type Key = (usize, usize);
type Value = usize;

#[derive(Debug, Default)]
struct CellThunk;

impl Thunk<Key, Value> for CellThunk {
    fn compute(&self, key: Key) -> Pin<Box<dyn Future<Output = Value> + Send + '_>> {
        Box::pin(async move {
            if key.0 > 0 {
                let prev1 = *cell_mapping().get(&(key.0 - 1, key.1 - 1)).await;
                let prev2 = *cell_mapping().get(&(key.0 - 1, key.1)).await;
                let prev3 = *cell_mapping().get(&(key.0 - 1, key.1 + 1)).await;
                let val = prev1 + prev2 + prev3;

                if val > 10_000 {
                    1
                } else {
                    val
                }
            } else {
                1
            }
        })
    }
}

fn calc_raw(key: Key) -> Value {
    if key.0 > 0 {
        let prev1 = calc_raw((key.0 - 1, key.1 - 1));
        let prev2 = calc_raw((key.0 - 1, key.1));
        let prev3 = calc_raw((key.0 - 1, key.1 + 1));
        let val = prev1 + prev2 + prev3;

        if val > 10_000 {
            1
        } else {
            val
        }
    } else {
        1
    }
}

type CellMapping = ThunkMapper<Key, Value, CellThunk>;
static CELL_MAPPING: OnceLock<CellMapping> = OnceLock::new();
fn cell_mapping() -> &'static CellMapping {
    CELL_MAPPING.get_or_init(CellMapping::default)
}

fn criterion_benchmark(c: &mut Criterion) {
    c.bench_function("spreadsheet non-incria (n = 5)", |b| {
        b.iter(|| calc_raw(black_box((5, 5))));
    });
    c.bench_function("spreadsheet fresh (n = 5)", |b| {
        b.to_async(tokio::runtime::Runtime::new().unwrap())
            .iter(|| deps::with_node_id(deps::next_node_id(), cell_mapping().get(&(5, 5))));
    });
}

criterion_group!(benches, criterion_benchmark);
criterion_main!(benches);