use crate::{
ComponentExportKind, ComponentSection, ComponentSectionId, ComponentValType, Encode,
encode_section,
};
use alloc::borrow::Cow;
use alloc::string::String;
use alloc::vec::Vec;
#[derive(Clone, Copy, Debug, Eq, PartialEq, Hash)]
pub enum TypeBounds {
Eq(u32),
SubResource,
}
impl Encode for TypeBounds {
fn encode(&self, sink: &mut Vec<u8>) {
match self {
Self::Eq(i) => {
sink.push(0x00);
i.encode(sink);
}
Self::SubResource => sink.push(0x01),
}
}
}
#[derive(Clone, Copy, Debug, Eq, PartialEq, Hash)]
pub enum ComponentTypeRef {
Module(u32),
Func(u32),
Value(ComponentValType),
Type(TypeBounds),
Instance(u32),
Component(u32),
}
impl ComponentTypeRef {
pub fn kind(&self) -> ComponentExportKind {
match self {
Self::Module(_) => ComponentExportKind::Module,
Self::Func(_) => ComponentExportKind::Func,
Self::Value(_) => ComponentExportKind::Value,
Self::Type(..) => ComponentExportKind::Type,
Self::Instance(_) => ComponentExportKind::Instance,
Self::Component(_) => ComponentExportKind::Component,
}
}
}
impl Encode for ComponentTypeRef {
fn encode(&self, sink: &mut Vec<u8>) {
self.kind().encode(sink);
match self {
Self::Module(idx) | Self::Func(idx) | Self::Instance(idx) | Self::Component(idx) => {
idx.encode(sink);
}
Self::Value(ty) => ty.encode(sink),
Self::Type(bounds) => bounds.encode(sink),
}
}
}
#[derive(Clone, Debug, Default)]
pub struct ComponentImportSection {
bytes: Vec<u8>,
num_added: u32,
}
impl ComponentImportSection {
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 import<'a>(
&mut self,
name: impl Into<ComponentExternName<'a>>,
ty: ComponentTypeRef,
) -> &mut Self {
name.into().encode(&mut self.bytes);
ty.encode(&mut self.bytes);
self.num_added += 1;
self
}
}
impl Encode for ComponentImportSection {
fn encode(&self, sink: &mut Vec<u8>) {
encode_section(sink, self.num_added, &self.bytes);
}
}
impl ComponentSection for ComponentImportSection {
fn id(&self) -> u8 {
ComponentSectionId::Import.into()
}
}
#[derive(Debug, Clone)]
pub struct ComponentExternName<'a> {
pub name: Cow<'a, str>,
pub implements: Option<Cow<'a, str>>,
}
impl Encode for ComponentExternName<'_> {
fn encode(&self, bytes: &mut Vec<u8>) {
let mut options = Vec::new();
if let Some(s) = &self.implements {
options.push((0x00, s.as_bytes()));
}
if options.is_empty() {
bytes.push(0x00);
} else {
bytes.push(0x02);
}
self.name.encode(bytes);
if !options.is_empty() {
options.len().encode(bytes);
for (kind, val) in options {
bytes.push(kind);
val.encode(bytes);
}
}
}
}
impl<'a> From<&'a str> for ComponentExternName<'a> {
fn from(name: &'a str) -> Self {
ComponentExternName {
name: Cow::Borrowed(name),
implements: None,
}
}
}
impl<'a> From<&'a String> for ComponentExternName<'a> {
fn from(name: &'a String) -> Self {
ComponentExternName {
name: Cow::Borrowed(name),
implements: None,
}
}
}
impl<'a> From<String> for ComponentExternName<'a> {
fn from(name: String) -> Self {
ComponentExternName {
name: Cow::Owned(name),
implements: None,
}
}
}
#[cfg(feature = "wasmparser")]
impl<'a> From<wasmparser::ComponentExternName<'a>> for ComponentExternName<'a> {
fn from(name: wasmparser::ComponentExternName<'a>) -> Self {
let wasmparser::ComponentExternName { name, implements } = name;
ComponentExternName {
name: name.into(),
implements: implements.map(|s| s.into()),
}
}
}