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
use crate::emit::{Emit, EmitContext};
use crate::ir::Value;
use crate::parse::IndicesToIds;
use crate::{GlobalId, Result};
use failure::bail;
#[derive(Debug, Copy, Clone)]
pub enum InitExpr {
Value(Value),
Global(GlobalId),
}
impl InitExpr {
pub(crate) fn eval(init: &wasmparser::InitExpr, ids: &IndicesToIds) -> Result<InitExpr> {
use wasmparser::Operator::*;
let mut reader = init.get_operators_reader();
let val = match reader.read()? {
I32Const { value } => InitExpr::Value(Value::I32(value)),
I64Const { value } => InitExpr::Value(Value::I64(value)),
F32Const { value } => InitExpr::Value(Value::F32(f32::from_bits(value.bits()))),
F64Const { value } => InitExpr::Value(Value::F64(f64::from_bits(value.bits()))),
GetGlobal { global_index } => InitExpr::Global(ids.get_global(global_index)?),
_ => bail!("invalid constant expression"),
};
match reader.read()? {
End => {}
_ => bail!("invalid constant expression"),
}
reader.ensure_end()?;
Ok(val)
}
}
impl Emit for InitExpr {
fn emit(&self, cx: &mut EmitContext) {
match *self {
InitExpr::Value(val) => val.emit(&mut cx.encoder),
InitExpr::Global(id) => {
let idx = cx.indices.get_global_index(id);
cx.encoder.byte(0x23);
cx.encoder.u32(idx);
}
}
cx.encoder.byte(0x0b);
}
}