use crate::{
BinaryReader, ComponentExternalKind, ComponentValType, FromReader, Result, SectionLimited,
};
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum TypeBounds {
Eq(u32),
SubResource,
}
impl<'a> FromReader<'a> for TypeBounds {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
Ok(match reader.read_u8()? {
0x00 => TypeBounds::Eq(reader.read()?),
0x01 => TypeBounds::SubResource,
x => return reader.invalid_leading_byte(x, "type bound"),
})
}
}
#[derive(Clone, Copy, Debug, Eq, PartialEq)]
pub enum ComponentTypeRef {
Module(u32),
Func(u32),
Value(ComponentValType),
Type(TypeBounds),
Instance(u32),
Component(u32),
}
impl ComponentTypeRef {
pub fn kind(&self) -> ComponentExternalKind {
match self {
ComponentTypeRef::Module(_) => ComponentExternalKind::Module,
ComponentTypeRef::Func(_) => ComponentExternalKind::Func,
ComponentTypeRef::Value(_) => ComponentExternalKind::Value,
ComponentTypeRef::Type(..) => ComponentExternalKind::Type,
ComponentTypeRef::Instance(_) => ComponentExternalKind::Instance,
ComponentTypeRef::Component(_) => ComponentExternalKind::Component,
}
}
}
impl<'a> FromReader<'a> for ComponentTypeRef {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
Ok(match reader.read()? {
ComponentExternalKind::Module => ComponentTypeRef::Module(reader.read()?),
ComponentExternalKind::Func => ComponentTypeRef::Func(reader.read_var_u32()?),
ComponentExternalKind::Value => ComponentTypeRef::Value(reader.read()?),
ComponentExternalKind::Type => ComponentTypeRef::Type(reader.read()?),
ComponentExternalKind::Instance => ComponentTypeRef::Instance(reader.read()?),
ComponentExternalKind::Component => ComponentTypeRef::Component(reader.read()?),
})
}
}
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
pub struct ComponentImport<'a> {
pub name: ComponentExternName<'a>,
pub ty: ComponentTypeRef,
}
impl<'a> FromReader<'a> for ComponentImport<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
Ok(ComponentImport {
name: reader.read()?,
ty: reader.read()?,
})
}
}
pub type ComponentImportSectionReader<'a> = SectionLimited<'a, ComponentImport<'a>>;
#[derive(Debug, Copy, Clone, Eq, PartialEq)]
#[allow(missing_docs)]
pub struct ComponentExternName<'a> {
pub name: &'a str,
pub implements: Option<&'a str>,
}
impl<'a> FromReader<'a> for ComponentExternName<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
let has_options = match reader.read_u8()? {
0x00 => false,
0x01 => false,
0x02 => {
if reader.cm_implements() {
true
} else {
bail!(
reader.original_position() - 1,
"the `cm-implements` feature is not active"
)
}
}
x => return reader.invalid_leading_byte(x, "component name"),
};
let mut ret = ComponentExternName {
name: reader.read_string()?,
implements: None,
};
if has_options {
for _ in 0..reader.read_var_u32()? {
let pos = reader.original_position();
match reader.read()? {
ComponentNameOpt::Implements(name) => {
if ret.implements.is_some() {
bail!(pos, "duplicate 'implements' option in name");
}
ret.implements = Some(name);
}
}
}
}
Ok(ret)
}
}
enum ComponentNameOpt<'a> {
Implements(&'a str),
}
impl<'a> FromReader<'a> for ComponentNameOpt<'a> {
fn from_reader(reader: &mut BinaryReader<'a>) -> Result<Self> {
match reader.read_u8()? {
0x00 => Ok(ComponentNameOpt::Implements(reader.read()?)),
x => return reader.invalid_leading_byte(x, "name option"),
}
}
}