use super::{AdapterModuleSection, SectionId, TypeRef};
use crate::{encoders, ValType};
const TYPE_INSTANCE: u8 = 0x7f;
const TYPE_MODULE: u8 = 0x7e;
const TYPE_FUNCTION: u8 = 0x7d;
const ALIAS_TYPE_INSTANCE_EXPORT: u8 = 0x00;
const ALIAS_TYPE_OUTER: u8 = 0x01;
const OUTER_ALIAS_MODULE: u8 = 0x01;
const OUTER_ALIAS_TYPE: u8 = 0x06;
const INSTANCE_TYPEDEF_TYPE: u8 = 0x01;
const INSTANCE_TYPEDEF_ALIAS: u8 = 0x05;
const INSTANCE_TYPEDEF_EXPORT: u8 = 0x06;
const MODULE_TYPEDEF_TYPE: u8 = 0x01;
const MODULE_TYPEDEF_ALIAS: u8 = 0x05;
const MODULE_TYPEDEF_EXPORT: u8 = 0x06;
const MODULE_TYPEDEF_IMPORT: u8 = 0x02;
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum DefinitionKind {
Instance = 0,
Module = 1,
Function = 2,
Table = 3,
Memory = 4,
Global = 5,
}
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum OuterAliasIndex {
Module(u32),
Type(u32),
}
impl OuterAliasIndex {
pub(crate) fn encode(&self, bytes: &mut Vec<u8>) {
match self {
Self::Module(index) => {
bytes.extend(encoders::u32(*index));
bytes.push(OUTER_ALIAS_MODULE);
}
Self::Type(index) => {
bytes.extend(encoders::u32(*index));
bytes.push(OUTER_ALIAS_TYPE);
}
}
}
}
#[derive(Debug, Clone, Default)]
pub struct InstanceType {
bytes: Vec<u8>,
num_added: u32,
}
impl InstanceType {
pub fn new() -> Self {
Self::default()
}
#[must_use = "the encoder must be used to encode the type"]
pub fn ty(&mut self) -> TypeEncoder {
self.bytes.push(INSTANCE_TYPEDEF_TYPE);
self.num_added += 1;
TypeEncoder(&mut self.bytes)
}
pub fn alias_instance_export(
&mut self,
instance: u32,
name: &str,
kind: DefinitionKind,
) -> &mut Self {
self.bytes.push(INSTANCE_TYPEDEF_ALIAS);
self.bytes.push(ALIAS_TYPE_INSTANCE_EXPORT);
self.bytes.extend(encoders::u32(instance));
self.bytes.extend(encoders::str(name));
self.bytes.push(kind as u8);
self.num_added += 1;
self
}
pub fn alias_outer_module(&mut self, count: u32, index: OuterAliasIndex) -> &mut Self {
self.bytes.push(INSTANCE_TYPEDEF_ALIAS);
self.bytes.push(ALIAS_TYPE_OUTER);
self.bytes.extend(encoders::u32(count));
index.encode(&mut self.bytes);
self.num_added += 1;
self
}
pub fn export(&mut self, name: &str, ty: TypeRef) -> &mut Self {
self.bytes.push(INSTANCE_TYPEDEF_EXPORT);
self.bytes.extend(encoders::str(name));
ty.encode(&mut self.bytes);
self.num_added += 1;
self
}
pub(crate) fn encode(&self, bytes: &mut Vec<u8>) {
bytes.extend(encoders::u32(self.num_added));
bytes.extend(self.bytes.iter().copied());
}
}
#[derive(Debug, Clone, Default)]
pub struct ModuleType {
bytes: Vec<u8>,
num_added: u32,
}
impl ModuleType {
pub fn new() -> Self {
Self::default()
}
#[must_use = "the encoder must be used to encode the type"]
pub fn ty(&mut self) -> TypeEncoder {
self.bytes.push(MODULE_TYPEDEF_TYPE);
self.num_added += 1;
TypeEncoder(&mut self.bytes)
}
pub fn alias_instance_export(
&mut self,
instance: u32,
name: &str,
kind: DefinitionKind,
) -> &mut Self {
self.bytes.push(MODULE_TYPEDEF_ALIAS);
self.bytes.push(ALIAS_TYPE_INSTANCE_EXPORT);
self.bytes.extend(encoders::u32(instance));
self.bytes.extend(encoders::str(name));
self.bytes.push(kind as u8);
self.num_added += 1;
self
}
pub fn alias_outer_module(&mut self, count: u32, index: OuterAliasIndex) -> &mut Self {
self.bytes.push(MODULE_TYPEDEF_ALIAS);
self.bytes.push(ALIAS_TYPE_OUTER);
self.bytes.extend(encoders::u32(count));
index.encode(&mut self.bytes);
self.num_added += 1;
self
}
pub fn export(&mut self, name: &str, ty: TypeRef) -> &mut Self {
self.bytes.push(MODULE_TYPEDEF_EXPORT);
self.bytes.extend(encoders::str(name));
ty.encode(&mut self.bytes);
self.num_added += 1;
self
}
pub fn import(&mut self, name: &str, ty: TypeRef) -> &mut Self {
self.bytes.push(MODULE_TYPEDEF_IMPORT);
self.bytes.extend(encoders::str(name));
ty.encode(&mut self.bytes);
self.num_added += 1;
self
}
pub(crate) fn encode(&self, bytes: &mut Vec<u8>) {
bytes.extend(encoders::u32(self.num_added));
bytes.extend(self.bytes.iter().copied());
}
}
#[derive(Debug)]
pub struct TypeEncoder<'a>(&'a mut Vec<u8>);
impl<'a> TypeEncoder<'a> {
pub fn instance(self, ty: &InstanceType) {
self.0.push(TYPE_INSTANCE);
ty.encode(self.0);
}
pub fn module(self, ty: &ModuleType) {
self.0.push(TYPE_MODULE);
ty.encode(self.0);
}
pub fn function(self, params: &[ValType], results: &[ValType]) {
self.0.push(TYPE_FUNCTION);
self.0
.extend(encoders::u32(u32::try_from(params.len()).unwrap()));
self.0.extend(params.iter().map(|p| u8::from(*p)));
self.0
.extend(encoders::u32(u32::try_from(results.len()).unwrap()));
self.0.extend(results.iter().map(|r| u8::from(*r)));
}
}
#[derive(Clone, Debug, Default)]
pub struct TypeSection {
bytes: Vec<u8>,
num_added: u32,
}
impl TypeSection {
pub fn new() -> Self {
Self::default()
}
pub fn len(&self) -> u32 {
self.num_added
}
pub fn is_empty(&self) -> bool {
self.num_added == 0
}
pub fn instance(&mut self, ty: &InstanceType) -> &mut Self {
let encoder = TypeEncoder(&mut self.bytes);
encoder.instance(ty);
self.num_added += 1;
self
}
pub fn module(&mut self, ty: &ModuleType) -> &mut Self {
let encoder = TypeEncoder(&mut self.bytes);
encoder.module(ty);
self.num_added += 1;
self
}
pub fn function(&mut self, params: &[ValType], results: &[ValType]) -> &mut Self {
let encoder = TypeEncoder(&mut self.bytes);
encoder.function(params, results);
self.num_added += 1;
self
}
}
impl AdapterModuleSection for TypeSection {
fn id(&self) -> u8 {
SectionId::Type.into()
}
fn encode<S>(&self, sink: &mut S)
where
S: Extend<u8>,
{
let num_added = encoders::u32(self.num_added);
let n = num_added.len();
sink.extend(
encoders::u32(u32::try_from(n + self.bytes.len()).unwrap())
.chain(num_added)
.chain(self.bytes.iter().copied()),
);
}
}