arrays are now lists structs are now records unions are now variants this ruins some of my union puns, oh well
256 lines
7.6 KiB
Rust
256 lines
7.6 KiB
Rust
use proptest::prelude::*;
|
|
use wiggle::{GuestMemory, GuestType};
|
|
use wiggle_test::{impl_errno, HostMemory, MemArea, WasiCtx};
|
|
|
|
wiggle::from_witx!({
|
|
witx: ["$CARGO_MANIFEST_DIR/tests/variant.witx"],
|
|
ctx: WasiCtx,
|
|
});
|
|
|
|
impl_errno!(types::Errno, types::GuestErrorConversion);
|
|
|
|
// Avoid panics on overflow
|
|
fn mult_lose_overflow(a: i32, b: u32) -> i32 {
|
|
let a_64: i64 = a as i64;
|
|
let b_64: i64 = b as i64;
|
|
let product = a_64 * b_64;
|
|
product as i32
|
|
}
|
|
|
|
// Avoid assert_eq(NaN, NaN) failures
|
|
fn mult_zero_nan(a: f32, b: u32) -> f32 {
|
|
if a.is_nan() {
|
|
0.0
|
|
} else {
|
|
let product = a * b as f32;
|
|
if product.is_nan() {
|
|
0.0
|
|
} else {
|
|
product
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a> variant_example::VariantExample for WasiCtx<'a> {
|
|
fn get_tag(&self, u: &types::Reason) -> Result<types::Excuse, types::Errno> {
|
|
println!("GET TAG: {:?}", u);
|
|
match u {
|
|
types::Reason::DogAte { .. } => Ok(types::Excuse::DogAte),
|
|
types::Reason::Traffic { .. } => Ok(types::Excuse::Traffic),
|
|
types::Reason::Sleeping { .. } => Ok(types::Excuse::Sleeping),
|
|
}
|
|
}
|
|
fn reason_mult(&self, u: &types::ReasonMut<'_>, multiply_by: u32) -> Result<(), types::Errno> {
|
|
match u {
|
|
types::ReasonMut::DogAte(fptr) => {
|
|
let val = fptr.read().expect("valid pointer");
|
|
println!("REASON MULT DogAte({})", val);
|
|
fptr.write(mult_zero_nan(val, multiply_by))
|
|
.expect("valid pointer");
|
|
}
|
|
types::ReasonMut::Traffic(iptr) => {
|
|
let val = iptr.read().expect("valid pointer");
|
|
println!("REASON MULT Traffic({})", val);
|
|
iptr.write(mult_lose_overflow(val, multiply_by))
|
|
.expect("valid pointer");
|
|
}
|
|
types::ReasonMut::Sleeping => {
|
|
println!("REASON MULT Sleeping");
|
|
}
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
fn reason_strat() -> impl Strategy<Value = types::Reason> {
|
|
prop_oneof![
|
|
prop::num::f32::ANY.prop_map(|v| types::Reason::DogAte(v)),
|
|
prop::num::i32::ANY.prop_map(|v| types::Reason::Traffic(v)),
|
|
Just(types::Reason::Sleeping),
|
|
]
|
|
.boxed()
|
|
}
|
|
|
|
fn reason_tag(r: &types::Reason) -> types::Excuse {
|
|
match r {
|
|
types::Reason::DogAte { .. } => types::Excuse::DogAte,
|
|
types::Reason::Traffic { .. } => types::Excuse::Traffic,
|
|
types::Reason::Sleeping { .. } => types::Excuse::Sleeping,
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct GetTagExercise {
|
|
pub input: types::Reason,
|
|
pub input_loc: MemArea,
|
|
pub return_loc: MemArea,
|
|
}
|
|
|
|
impl GetTagExercise {
|
|
pub fn strat() -> BoxedStrategy<Self> {
|
|
(
|
|
reason_strat(),
|
|
HostMemory::mem_area_strat(types::Reason::guest_size()),
|
|
HostMemory::mem_area_strat(types::Excuse::guest_size()),
|
|
)
|
|
.prop_map(|(input, input_loc, return_loc)| GetTagExercise {
|
|
input,
|
|
input_loc,
|
|
return_loc,
|
|
})
|
|
.prop_filter("non-overlapping pointers", |e| {
|
|
MemArea::non_overlapping_set(&[e.input_loc, e.return_loc])
|
|
})
|
|
.boxed()
|
|
}
|
|
|
|
pub fn test(&self) {
|
|
let ctx = WasiCtx::new();
|
|
let host_memory = HostMemory::new();
|
|
|
|
let discriminant = reason_tag(&self.input) as u8;
|
|
host_memory
|
|
.ptr(self.input_loc.ptr)
|
|
.write(discriminant)
|
|
.expect("input discriminant ptr");
|
|
match self.input {
|
|
types::Reason::DogAte(f) => {
|
|
host_memory
|
|
.ptr(self.input_loc.ptr + 4)
|
|
.write(f)
|
|
.expect("input contents ref_mut");
|
|
}
|
|
types::Reason::Traffic(v) => host_memory
|
|
.ptr(self.input_loc.ptr + 4)
|
|
.write(v)
|
|
.expect("input contents ref_mut"),
|
|
types::Reason::Sleeping => {} // Do nothing
|
|
}
|
|
let e = variant_example::get_tag(
|
|
&ctx,
|
|
&host_memory,
|
|
self.input_loc.ptr as i32,
|
|
self.return_loc.ptr as i32,
|
|
);
|
|
|
|
assert_eq!(e, Ok(types::Errno::Ok as i32), "get_tag errno");
|
|
|
|
let return_val: types::Excuse = host_memory
|
|
.ptr(self.return_loc.ptr)
|
|
.read()
|
|
.expect("return ref");
|
|
|
|
assert_eq!(return_val, reason_tag(&self.input), "get_tag return value");
|
|
}
|
|
}
|
|
|
|
proptest! {
|
|
#[test]
|
|
fn get_tag(e in GetTagExercise::strat()) {
|
|
e.test();
|
|
}
|
|
}
|
|
|
|
#[derive(Debug)]
|
|
struct ReasonMultExercise {
|
|
pub input: types::Reason,
|
|
pub input_loc: MemArea,
|
|
pub input_pointee_loc: MemArea,
|
|
pub multiply_by: u32,
|
|
}
|
|
|
|
impl ReasonMultExercise {
|
|
pub fn strat() -> BoxedStrategy<Self> {
|
|
(
|
|
reason_strat(),
|
|
HostMemory::mem_area_strat(types::Reason::guest_size()),
|
|
HostMemory::mem_area_strat(4),
|
|
prop::num::u32::ANY,
|
|
)
|
|
.prop_map(
|
|
|(input, input_loc, input_pointee_loc, multiply_by)| ReasonMultExercise {
|
|
input,
|
|
input_loc,
|
|
input_pointee_loc,
|
|
multiply_by,
|
|
},
|
|
)
|
|
.prop_filter("non-overlapping pointers", |e| {
|
|
MemArea::non_overlapping_set(&[e.input_loc, e.input_pointee_loc])
|
|
})
|
|
.boxed()
|
|
}
|
|
|
|
pub fn test(&self) {
|
|
let ctx = WasiCtx::new();
|
|
let host_memory = HostMemory::new();
|
|
|
|
let discriminant = reason_tag(&self.input) as u8;
|
|
host_memory
|
|
.ptr(self.input_loc.ptr)
|
|
.write(discriminant)
|
|
.expect("input discriminant ref_mut");
|
|
host_memory
|
|
.ptr(self.input_loc.ptr + 4)
|
|
.write(self.input_pointee_loc.ptr)
|
|
.expect("input pointer ref_mut");
|
|
|
|
match self.input {
|
|
types::Reason::DogAte(f) => {
|
|
host_memory
|
|
.ptr(self.input_pointee_loc.ptr)
|
|
.write(f)
|
|
.expect("input contents ref_mut");
|
|
}
|
|
types::Reason::Traffic(v) => {
|
|
host_memory
|
|
.ptr(self.input_pointee_loc.ptr)
|
|
.write(v)
|
|
.expect("input contents ref_mut");
|
|
}
|
|
types::Reason::Sleeping => {} // Do nothing
|
|
}
|
|
let e = variant_example::reason_mult(
|
|
&ctx,
|
|
&host_memory,
|
|
self.input_loc.ptr as i32,
|
|
self.multiply_by as i32,
|
|
);
|
|
|
|
assert_eq!(e, Ok(types::Errno::Ok as i32), "reason_mult errno");
|
|
|
|
match self.input {
|
|
types::Reason::DogAte(f) => {
|
|
let f_result: f32 = host_memory
|
|
.ptr(self.input_pointee_loc.ptr)
|
|
.read()
|
|
.expect("input contents ref_mut");
|
|
assert_eq!(
|
|
mult_zero_nan(f, self.multiply_by),
|
|
f_result,
|
|
"DogAte result"
|
|
)
|
|
}
|
|
types::Reason::Traffic(v) => {
|
|
let v_result: i32 = host_memory
|
|
.ptr(self.input_pointee_loc.ptr)
|
|
.read()
|
|
.expect("input contents ref_mut");
|
|
assert_eq!(
|
|
mult_lose_overflow(v, self.multiply_by),
|
|
v_result,
|
|
"Traffic result"
|
|
)
|
|
}
|
|
types::Reason::Sleeping => {} // Do nothing
|
|
}
|
|
}
|
|
}
|
|
|
|
proptest! {
|
|
#[test]
|
|
fn reason_mult(e in ReasonMultExercise::strat()) {
|
|
e.test();
|
|
}
|
|
}
|