#![allow(
clippy::large_enum_variant,
clippy::type_complexity,
clippy::too_many_arguments
)]
use criterion::{black_box, criterion_group, criterion_main, Criterion};
use mpstthree::binary::struct_trait::{end::End, recv::Recv, send::Send, session::Session};
use mpstthree::role::broadcast::RoleBroadcast;
use mpstthree::role::end::RoleEnd;
use mpstthree::{
bundle_struct_fork_close_multi, choose_mpst_multi_to_all, create_multiple_normal_name,
create_multiple_normal_role, create_recv_mpst_session_bundle, create_send_mpst_session_bundle,
offer_mpst,
};
use std::error::Error;
use std::marker;
bundle_struct_fork_close_multi!(close_mpst_multi, fork_mpst, MeshedChannels, 2);
create_multiple_normal_role!(
RoleA, RoleADual |
RoleB, RoleBDual |
);
create_multiple_normal_name!(NameA, NameB);
create_send_mpst_session_bundle!(
send_mpst_a_to_b, RoleB, 1 | =>
NameA, MeshedChannels, 2
);
create_send_mpst_session_bundle!(
send_mpst_b_to_a, RoleA, 1 | =>
NameB, MeshedChannels, 2
);
create_recv_mpst_session_bundle!(
recv_mpst_a_from_b, RoleB, 1 | =>
NameA, MeshedChannels, 2
);
create_recv_mpst_session_bundle!(
recv_mpst_b_from_a, RoleA, 1 | =>
NameB, MeshedChannels, 2
);
type Choose0fromAtoB<N> = <RecursBtoA<N> as Session>::Dual;
enum Branching0fromAtoB<N: marker::Send> {
More(MeshedChannels<Recv<N, Send<N, RecursBtoA<N>>>, RoleA<RoleA<RoleA<RoleEnd>>>, NameB>),
Done(MeshedChannels<End, RoleEnd, NameB>),
}
type RecursBtoA<N> = Recv<Branching0fromAtoB<N>, End>;
type EndpointA<N> = MeshedChannels<Choose0fromAtoB<N>, RoleBroadcast, NameA>;
type EndpointB<N> = MeshedChannels<RecursBtoA<N>, RoleA<RoleEnd>, NameB>;
fn endpoint_a(s: EndpointA<i64>) -> Result<(), Box<dyn Error>> {
recurs_a(s, LOOPS, 1)
}
fn recurs_a(s: EndpointA<i64>, index: i64, old: i64) -> Result<(), Box<dyn Error>> {
match index {
0 => {
let s = choose_mpst_multi_to_all!(
s,
Branching0fromAtoB::<i64>::Done, =>
NameA,
MeshedChannels,
1
);
close_mpst_multi(s)
}
i => {
let s = choose_mpst_multi_to_all!(
s,
Branching0fromAtoB::<i64>::More, =>
NameA,
MeshedChannels,
1
);
let s = send_mpst_a_to_b(old, s);
let (new, s) = recv_mpst_a_from_b(s)?;
recurs_a(s, i - 1, new)
}
}
}
fn endpoint_b(s: EndpointB<i64>) -> Result<(), Box<dyn Error>> {
recurs_b(s, 0)
}
fn recurs_b(s: EndpointB<i64>, old: i64) -> Result<(), Box<dyn Error>> {
offer_mpst!(s, recv_mpst_b_from_a, {
Branching0fromAtoB::Done(s) => {
close_mpst_multi(s)
},
Branching0fromAtoB::More(s) => {
let (new, s) = recv_mpst_b_from_a(s)?;
let s = send_mpst_b_to_a(new + old, s);
recurs_b(s, new + old)
},
})
}
fn aux() {
let (thread_a, thread_b) = fork_mpst(black_box(endpoint_a), black_box(endpoint_b));
thread_a.join().unwrap();
thread_b.join().unwrap();
}
static LOOPS: i64 = 20;
pub fn fib(c: &mut Criterion) {
c.bench_function(&format!("Fibo {LOOPS} basic"), |b| b.iter(aux));
}
criterion_group! {
name = bench;
config = Criterion::default().significance_level(0.05).without_plots().sample_size(100000);
targets = fib,
}
criterion_main! {
bench
}