1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
// This file contains code from external sources.
// Attributions: https://github.com/wasmerio/wasmer/blob/master/ATTRIBUTIONS.md

use crate::{wasm_unsupported, WasmResult};
use std::boxed::Box;
use wasmer_types::entity::PrimaryMap;
use wasmer_types::SignatureIndex;

/// Map of signatures to a function's parameter and return types.
pub(crate) type WasmTypes =
    PrimaryMap<SignatureIndex, (Box<[wasmparser::Type]>, Box<[wasmparser::Type]>)>;

/// Contains information decoded from the Wasm module that must be referenced
/// during each Wasm function's translation.
///
/// This is only for data that is maintained by `wasmer-compiler` itself, as
/// opposed to being maintained by the embedder. Data that is maintained by the
/// embedder is represented with `ModuleEnvironment`.
#[derive(Debug)]
pub struct ModuleTranslationState {
    /// A map containing a Wasm module's original, raw signatures.
    ///
    /// This is used for translating multi-value Wasm blocks inside functions,
    /// which are encoded to refer to their type signature via index.
    pub(crate) wasm_types: WasmTypes,
}

impl ModuleTranslationState {
    /// Creates a new empty ModuleTranslationState.
    pub fn new() -> Self {
        Self {
            wasm_types: PrimaryMap::new(),
        }
    }

    /// Get the parameter and result types for the given Wasm blocktype.
    pub fn blocktype_params_results(
        &self,
        ty_or_ft: wasmparser::TypeOrFuncType,
    ) -> WasmResult<(&[wasmparser::Type], &[wasmparser::Type])> {
        Ok(match ty_or_ft {
            wasmparser::TypeOrFuncType::Type(ty) => match ty {
                wasmparser::Type::I32 => (&[], &[wasmparser::Type::I32]),
                wasmparser::Type::I64 => (&[], &[wasmparser::Type::I64]),
                wasmparser::Type::F32 => (&[], &[wasmparser::Type::F32]),
                wasmparser::Type::F64 => (&[], &[wasmparser::Type::F64]),
                wasmparser::Type::V128 => (&[], &[wasmparser::Type::V128]),
                wasmparser::Type::ExternRef => (&[], &[wasmparser::Type::ExternRef]),
                wasmparser::Type::FuncRef => (&[], &[wasmparser::Type::FuncRef]),
                wasmparser::Type::EmptyBlockType => (&[], &[]),
                ty => return Err(wasm_unsupported!("blocktype_params_results: type {:?}", ty)),
            },
            wasmparser::TypeOrFuncType::FuncType(ty_index) => {
                let sig_idx = SignatureIndex::from_u32(ty_index);
                let (ref params, ref results) = self.wasm_types[sig_idx];
                (&*params, &*results)
            }
        })
    }
}