use quote::quote_spanned;
use super::{
DelayType, OperatorCategory, OperatorConstraints, OperatorWriteOutput, RANGE_0,
RANGE_1, WriteContextArgs,
};
pub const FOLD_NO_REPLAY: OperatorConstraints = OperatorConstraints {
name: "fold_no_replay",
categories: &[OperatorCategory::Fold],
hard_range_inn: RANGE_1,
soft_range_inn: RANGE_1,
hard_range_out: &(0..=1),
soft_range_out: &(0..=1),
num_args: 2,
persistence_args: &(0..=1),
type_args: RANGE_0,
is_external_input: false,
has_singleton_output: true,
flo_type: None,
ports_inn: None,
ports_out: None,
input_delaytype_fn: |_| Some(DelayType::Stratum),
write_fn: |wc @ &WriteContextArgs {
root,
context,
df_ident,
op_span,
work_fn,
work_fn_async,
ident,
is_pull,
inputs,
outputs,
singleton_output_ident,
arguments,
..
},
diagnostics| {
let init_fn = &arguments[0];
let func = &arguments[1];
let initializer_func_ident = wc.make_ident("initializer_func");
let init = quote_spanned! {op_span=>
(#initializer_func_ident)()
};
let [persistence] = wc.persistence_args_disallow_mutable(diagnostics);
let input = &inputs[0];
let accumulator_ident = wc.make_ident("accumulator");
let item_ident = wc.make_ident("item");
let write_prologue = quote_spanned! {op_span=>
#[allow(unused_mut, reason = "for if `Fn` instead of `FnMut`.")]
let mut #initializer_func_ident = #init_fn;
#[allow(clippy::redundant_closure_call)]
let #singleton_output_ident = #df_ident.add_state(::std::cell::RefCell::new(#init));
};
let write_prologue_after = wc
.persistence_as_state_lifespan(persistence)
.map(|lifespan| quote_spanned! {op_span=>
#[allow(clippy::redundant_closure_call)]
#df_ident.set_state_lifespan_hook(
#singleton_output_ident, #lifespan, move |rcell| { rcell.replace(#init); },
);
}).unwrap_or_default();
let assign_accum_ident = quote_spanned! {op_span=>
#[allow(unused_mut)]
let mut #accumulator_ident = unsafe {
#context.state_ref_unchecked(#singleton_output_ident)
}.borrow_mut();
};
let foreach_body = quote_spanned! {op_span=>
#[inline(always)]
fn call_comb_type<Accum, Item>(
accum: &mut Accum,
item: Item,
mut func: impl FnMut(&mut Accum, Item),
) {
(func)(accum, item);
}
#[allow(clippy::redundant_closure_call)]
call_comb_type(&mut *#accumulator_ident, #item_ident, #func);
};
let write_iterator = if is_pull {
quote_spanned! {op_span=>
#assign_accum_ident
let mut __was_updated = false;
{
let __fut = #root::dfir_pipes::pull::Pull::for_each(#input, |#item_ident| {
#foreach_body
__was_updated = true;
});
let () = #work_fn_async(__fut).await;
}
let #ident = if __was_updated || (#context.current_tick().0 == 0 && #context.is_first_run_this_tick()) {
#work_fn(
|| #root::dfir_pipes::pull::iter(
::std::option::Option::Some(::std::clone::Clone::clone(&*#accumulator_ident))
)
)
} else {
#work_fn(
|| #root::dfir_pipes::pull::iter(::std::option::Option::None)
)
};
}
} else {
assert_eq!(0, outputs.len());
quote_spanned! {op_span=>
let #ident = #root::dfir_pipes::push::for_each(|#item_ident| {
#assign_accum_ident
#foreach_body
});
}
};
Ok(OperatorWriteOutput {
write_prologue,
write_prologue_after,
write_iterator,
write_iterator_after: Default::default(),
})
},
};