use std::marker::PhantomData;
use crate::analysis::cpa::residue::Residue;
use crate::analysis::cpa::state::AbstractState;
pub struct TerminatingReducer<S>
where
S: AbstractState,
{
all_states: Vec<S>,
non_terminating_states: Vec<S>,
_phantom: PhantomData<S>,
}
impl<S> TerminatingReducer<S>
where
S: AbstractState,
{
pub fn new_with_capacity(cap: usize) -> Self {
Self {
all_states: Vec::with_capacity(cap),
non_terminating_states: Vec::with_capacity(cap),
_phantom: Default::default(),
}
}
fn compute_terminating_states(self) -> Vec<S> {
let mut terminating_states = Vec::new();
for state in self.all_states {
if !self.non_terminating_states.iter().any(|s| s == &state) {
terminating_states.push(state);
}
}
terminating_states
}
}
impl<S> Default for TerminatingReducer<S>
where
S: AbstractState,
{
fn default() -> Self {
Self {
all_states: Vec::new(),
non_terminating_states: Vec::new(),
_phantom: Default::default(),
}
}
}
impl<'a, S> Residue<'a, S> for TerminatingReducer<S>
where
S: AbstractState,
{
type Output = Vec<S>;
fn new_state(
&mut self,
state: &S,
dest_state: &S,
_op: &Option<crate::analysis::pcode_store::PcodeOpRef<'a>>,
) {
if !self.non_terminating_states.iter().any(|s| s == state) {
self.non_terminating_states.push(state.clone());
}
if !self.all_states.iter().any(|s| s == dest_state) {
self.all_states.push(dest_state.clone());
}
}
fn merged_state(
&mut self,
curr_state: &S,
original_merged_state: &S,
merged_state: &S,
_op: &Option<crate::analysis::pcode_store::PcodeOpRef<'a>>,
) {
if !self.non_terminating_states.iter().any(|s| s == curr_state) {
self.non_terminating_states.push(curr_state.clone());
}
for state in &mut self.all_states {
if state == original_merged_state {
*state = merged_state.clone();
}
}
for state in &mut self.non_terminating_states {
if state == original_merged_state {
*state = merged_state.clone();
}
}
}
fn new() -> Self {
Self::default()
}
fn finalize(self) -> Self::Output {
self.compute_terminating_states()
}
}
#[derive(Debug, Clone, Copy)]
pub struct TerminatingReducerFactory;
impl TerminatingReducerFactory {
pub const fn new() -> Self {
TerminatingReducerFactory
}
}
impl Default for TerminatingReducerFactory {
fn default() -> Self {
Self::new()
}
}
pub const TERMINATING: TerminatingReducerFactory = TerminatingReducerFactory;
impl<A> crate::analysis::cpa::residue::ReducerFactoryForState<A> for TerminatingReducerFactory
where
A: crate::analysis::cpa::ConfigurableProgramAnalysis,
A::State: crate::analysis::cpa::state::AbstractState,
{
type Reducer<'op> = TerminatingReducer<A::State>;
fn make<'op>(&self) -> Self::Reducer<'op> {
TerminatingReducer::default()
}
}