use std::fmt;
use std::fmt::Debug;
use std::marker::PhantomData;
use libafl::prelude::mutational::MutatedTransform;
use libafl::prelude::*;
use libafl_bolts::prelude::*;
#[derive(Clone, Debug)]
pub struct PuffinMutationalStage<E, EM, I, M, Z> {
mutator: M,
#[allow(clippy::type_complexity)]
phantom: PhantomData<(E, EM, I, Z)>,
max_iterations_per_stage: u64,
}
impl<E, EM, I, M, Z> UsesState for PuffinMutationalStage<E, EM, I, M, Z>
where
E: UsesState<State = Z::State>,
EM: UsesState<State = Z::State>,
M: Mutator<I, Z::State>,
Z: Evaluator<E, EM>,
Z::State: HasClientPerfMonitor + HasCorpus + HasRand,
{
type State = Z::State;
}
impl<E, EM, I, M, Z> MutationalStage<E, EM, I, M, Z> for PuffinMutationalStage<E, EM, I, M, Z>
where
E: UsesState<State = Z::State>,
EM: UsesState<State = Z::State>,
M: Mutator<I, Z::State>,
Z: Evaluator<E, EM>,
Z::State: HasClientPerfMonitor + HasCorpus + HasRand,
I: MutatedTransform<Self::Input, Self::State> + Clone,
{
#[inline]
fn mutator(&self) -> &M {
&self.mutator
}
#[inline]
fn mutator_mut(&mut self) -> &mut M {
&mut self.mutator
}
fn iterations(&self, state: &mut Z::State, _corpus_idx: CorpusId) -> Result<u64, Error> {
Ok(1 + state.rand_mut().below(self.max_iterations_per_stage))
}
}
impl<E, EM, I, M, Z> Stage<E, EM, Z> for PuffinMutationalStage<E, EM, I, M, Z>
where
E: UsesState<State = Z::State>,
EM: UsesState<State = Z::State>,
M: Mutator<I, Z::State>,
Z: Evaluator<E, EM>,
Z::State: HasClientPerfMonitor + HasCorpus + HasRand,
I: MutatedTransform<Self::Input, Self::State> + Clone,
{
#[inline]
#[allow(clippy::let_and_return)]
fn perform(
&mut self,
fuzzer: &mut Z,
executor: &mut E,
state: &mut Z::State,
manager: &mut EM,
corpus_idx: CorpusId,
) -> Result<(), Error> {
let ret = self.perform_mutational(fuzzer, executor, state, manager, corpus_idx);
#[cfg(feature = "introspection")]
state.introspection_monitor_mut().finish_stage();
ret
}
}
impl<E, EM, I, M, Z> PuffinMutationalStage<E, EM, I, M, Z>
where
I: Input,
E: UsesState<State = Z::State>,
EM: UsesState<State = Z::State>,
M: Mutator<I, Z::State>,
Z: Evaluator<E, EM>,
{
#[allow(dead_code)]
pub const fn new(mutator: M, max_iterations_per_stage: u64) -> Self {
Self {
mutator,
phantom: PhantomData,
max_iterations_per_stage,
}
}
}
pub struct PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
mutations: MT,
phantom: PhantomData<(I, S)>,
max_mutations_per_iteration: u64,
}
impl<I, MT, S> Debug for PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(
f,
"StdScheduledMutator with {} mutations for Input type {}",
self.mutations.len(),
core::any::type_name::<I>()
)
}
}
impl<I, MT, S> Named for PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
fn name(&self) -> &str {
"PuffinScheduledMutator"
}
}
impl<I, MT, S> Mutator<I, S> for PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
#[inline]
fn mutate(
&mut self,
state: &mut S,
input: &mut I,
stage_idx: i32,
) -> Result<MutationResult, Error> {
self.scheduled_mutate(state, input, stage_idx)
}
}
impl<I, MT, S> ComposedByMutations<I, MT, S> for PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
#[inline]
fn mutations(&self) -> &MT {
&self.mutations
}
#[inline]
fn mutations_mut(&mut self) -> &mut MT {
&mut self.mutations
}
}
impl<I, MT, S> ScheduledMutator<I, MT, S> for PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
fn iterations(&self, state: &mut S, _: &I) -> u64 {
state.rand_mut().below(self.max_mutations_per_iteration)
}
fn schedule(&self, state: &mut S, _: &I) -> MutationId {
debug_assert!(!self.mutations().is_empty());
(state.rand_mut().below(self.mutations().len() as u64) as usize).into()
}
}
impl<I, MT, S> PuffinScheduledMutator<I, MT, S>
where
I: Input,
MT: MutatorsTuple<I, S>,
S: HasRand,
{
#[allow(dead_code)]
pub const fn new(mutations: MT, max_mutations_per_iteration: u64) -> Self {
Self {
mutations,
phantom: PhantomData,
max_mutations_per_iteration,
}
}
}