use crate::analysis::cfg::CfgState;
use crate::analysis::cfg::PcodeCfg;
use crate::analysis::cpa::residue::Residue;
use crate::analysis::cpa::state::LocationState;
use crate::analysis::pcode_store::PcodeOpRef;
use jingle_sleigh::PcodeOperation;
pub struct CfgReducer<N>
where
N: LocationState + CfgState,
{
pub cfg: PcodeCfg<N, PcodeOperation>,
}
impl<N> CfgReducer<N>
where
N: LocationState + CfgState,
{
pub fn take_cfg(&mut self) -> PcodeCfg<N, PcodeOperation> {
std::mem::take(&mut self.cfg)
}
}
impl<N> Residue<N> for CfgReducer<N>
where
N: LocationState + CfgState,
{
type Output = PcodeCfg<N>;
fn new_state(&mut self, state: &N, dest_state: &N, op: &Option<PcodeOpRef<'_>>) {
self.cfg.add_node(state);
if let Some(op) = op {
let owned_op = op.as_ref().clone();
self.cfg.add_edge(state, dest_state, owned_op);
}
}
fn merged_state(
&mut self,
state: &N,
dest_state: &N,
merged_state: &N,
op: &Option<PcodeOpRef<'_>>,
) {
tracing::debug!("merged called: dest_state and merged_state provided");
self.cfg.replace_and_combine_nodes(dest_state, merged_state);
if let Some(op) = op {
let owned_op = op.as_ref().clone();
self.cfg.add_edge(state, merged_state, owned_op);
}
}
fn new() -> Self {
Self {
cfg: PcodeCfg::new(),
}
}
fn finalize(self) -> Self::Output {
self.cfg
}
}