use crate::prelude::*;
use std::collections::HashMap;
use std::fmt;
use wasm_encoder::Encode;
#[derive(Hash, PartialEq, Eq, Copy, Clone)]
pub enum Trap {
CannotLeave,
CannotEnter,
UnalignedPointer,
InvalidDiscriminant,
InvalidChar,
ListByteLengthOverflow,
StringLengthTooBig,
StringLengthOverflow,
AssertFailed(&'static str),
}
#[derive(Default)]
pub struct TrapSection {
trap_to_index: HashMap<Trap, usize>,
trap_list: Vec<Trap>,
function_traps: Vec<(u32, Vec<(usize, usize)>)>,
}
impl TrapSection {
pub fn append(&mut self, func: u32, traps: Vec<(usize, Trap)>) {
if traps.is_empty() {
return;
}
let traps = traps
.into_iter()
.map(|(offset, trap)| {
let trap = *self.trap_to_index.entry(trap).or_insert_with(|| {
let idx = self.trap_list.len();
self.trap_list.push(trap);
idx
});
(offset, trap)
})
.collect();
self.function_traps.push((func, traps));
}
pub fn finish(self) -> Vec<u8> {
let mut data = Vec::new();
self.trap_list.len().encode(&mut data);
for trap in self.trap_list.iter() {
trap.to_string().encode(&mut data);
}
self.function_traps.len().encode(&mut data);
for (func, traps) in self.function_traps.iter() {
func.encode(&mut data);
traps.len().encode(&mut data);
for (func_offset, trap_message) in traps {
func_offset.encode(&mut data);
trap_message.encode(&mut data);
}
}
data
}
}
impl fmt::Display for Trap {
fn fmt(&self, f: &mut fmt::Formatter<'_>) -> fmt::Result {
match self {
Trap::CannotLeave => "cannot leave instance".fmt(f),
Trap::CannotEnter => "cannot enter instance".fmt(f),
Trap::UnalignedPointer => "pointer not aligned correctly".fmt(f),
Trap::InvalidDiscriminant => "invalid variant discriminant".fmt(f),
Trap::InvalidChar => "invalid char value specified".fmt(f),
Trap::ListByteLengthOverflow => "byte size of list too large for i32".fmt(f),
Trap::StringLengthTooBig => "string byte size exceeds maximum".fmt(f),
Trap::StringLengthOverflow => "string byte size overflows i32".fmt(f),
Trap::AssertFailed(s) => write!(f, "assertion failure: {}", s),
}
}
}