risingwave_frontend/optimizer/plan_expr_rewriter/
cse_rewriter.rsuse std::collections::HashMap;
use crate::expr::{Expr, ExprImpl, ExprRewriter, FunctionCall, InputRef};
use crate::optimizer::plan_expr_visitor::CseExprCounter;
#[derive(Default)]
pub struct CseRewriter {
pub expr_counter: CseExprCounter,
pub cse_input_ref_offset: usize,
pub cse_mapping: HashMap<FunctionCall, InputRef>,
}
impl CseRewriter {
pub fn new(expr_counter: CseExprCounter, cse_input_ref_offset: usize) -> Self {
Self {
expr_counter,
cse_input_ref_offset,
cse_mapping: HashMap::default(),
}
}
}
impl ExprRewriter for CseRewriter {
fn rewrite_function_call(&mut self, func_call: FunctionCall) -> ExprImpl {
if let Some(count) = self.expr_counter.counter.get(&func_call)
&& *count > 1
{
if let Some(expr) = self.cse_mapping.get(&func_call) {
let expr: ExprImpl = ExprImpl::InputRef(expr.clone().into());
return expr;
}
let input_ref = InputRef::new(self.cse_input_ref_offset, func_call.return_type());
self.cse_input_ref_offset += 1;
self.cse_mapping.insert(func_call, input_ref.clone());
return ExprImpl::InputRef(input_ref.into());
}
let (func_type, inputs, ret) = func_call.decompose();
let inputs = inputs
.into_iter()
.map(|expr| self.rewrite_expr(expr))
.collect();
FunctionCall::new_unchecked(func_type, inputs, ret).into()
}
}