#![no_main]
use libfuzzer_sys::arbitrary::{Arbitrary, Unstructured};
use libfuzzer_sys::{fuzz_mutator, fuzz_target, fuzzer_mutate};
use mutatis::Session;
use postcard::{from_bytes, to_slice};
use rand::{Rng, SeedableRng};
use wasmtime_fuzzing::generators::GcOps;
use wasmtime_fuzzing::oracles::gc_ops;
fuzz_target!(|data: &[u8]| {
let Ok((seed, ops)) = postcard::from_bytes::<(u64, GcOps)>(data) else {
return;
};
let mut buf = [0u8; 1024];
let mut rng = rand::rngs::StdRng::seed_from_u64(seed);
rng.fill(&mut buf);
let u = Unstructured::new(&buf);
let Ok(config) = wasmtime_fuzzing::generators::Config::arbitrary_take_rest(u) else {
return;
};
let _ = gc_ops(config, ops);
});
fuzz_mutator!(|data: &mut [u8], size: usize, max_size: usize, seed: u32| {
let _ = env_logger::try_init();
if seed.count_ones() % 8 == 0 {
return fuzzer_mutate(data, size, max_size);
}
let mut tuple: (u64, GcOps) = from_bytes(&data[..size]).ok().unwrap_or_default();
let mut session = Session::new().seed(seed.into()).shrink(max_size < size);
if session.mutate(&mut tuple).is_ok() {
loop {
if let Ok(encoded) = to_slice(&tuple, data) {
return encoded.len();
}
if tuple.1.pop() {
continue;
}
break;
}
}
fuzzer_mutate(data, size, max_size)
});