1#![warn(clippy::cast_possible_truncation, clippy::cast_sign_loss)]
9
10use cranelift_codegen::{
11 FinalizedMachReloc, FinalizedRelocTarget, MachTrap, binemit,
12 cursor::FuncCursor,
13 ir::{self, AbiParam, ArgumentPurpose, ExternalName, InstBuilder, Signature, TrapCode},
14 isa::{CallConv, TargetIsa},
15 settings,
16};
17use cranelift_entity::PrimaryMap;
18
19use target_lexicon::Architecture;
20use wasmtime_environ::{
21 BuiltinFunctionIndex, FlagValue, FuncIndex, RelocationTarget, Trap, TrapInformation, Tunables,
22 WasmFuncType, WasmHeapTopType, WasmHeapType, WasmValType,
23};
24
25pub use builder::builder;
26
27pub mod isa_builder;
28mod obj;
29pub use obj::*;
30mod compiled_function;
31pub use compiled_function::*;
32
33mod bounds_checks;
34mod builder;
35mod compiler;
36mod debug;
37mod func_environ;
38mod translate;
39
40use self::compiler::Compiler;
41
42const TRAP_INTERNAL_ASSERT: TrapCode = TrapCode::unwrap_user(1);
43const TRAP_OFFSET: u8 = 2;
44pub const TRAP_ALWAYS: TrapCode =
45 TrapCode::unwrap_user(Trap::AlwaysTrapAdapter as u8 + TRAP_OFFSET);
46pub const TRAP_CANNOT_ENTER: TrapCode =
47 TrapCode::unwrap_user(Trap::CannotEnterComponent as u8 + TRAP_OFFSET);
48pub const TRAP_INDIRECT_CALL_TO_NULL: TrapCode =
49 TrapCode::unwrap_user(Trap::IndirectCallToNull as u8 + TRAP_OFFSET);
50pub const TRAP_BAD_SIGNATURE: TrapCode =
51 TrapCode::unwrap_user(Trap::BadSignature as u8 + TRAP_OFFSET);
52pub const TRAP_NULL_REFERENCE: TrapCode =
53 TrapCode::unwrap_user(Trap::NullReference as u8 + TRAP_OFFSET);
54pub const TRAP_ALLOCATION_TOO_LARGE: TrapCode =
55 TrapCode::unwrap_user(Trap::AllocationTooLarge as u8 + TRAP_OFFSET);
56pub const TRAP_ARRAY_OUT_OF_BOUNDS: TrapCode =
57 TrapCode::unwrap_user(Trap::ArrayOutOfBounds as u8 + TRAP_OFFSET);
58pub const TRAP_UNREACHABLE: TrapCode =
59 TrapCode::unwrap_user(Trap::UnreachableCodeReached as u8 + TRAP_OFFSET);
60pub const TRAP_HEAP_MISALIGNED: TrapCode =
61 TrapCode::unwrap_user(Trap::HeapMisaligned as u8 + TRAP_OFFSET);
62pub const TRAP_TABLE_OUT_OF_BOUNDS: TrapCode =
63 TrapCode::unwrap_user(Trap::TableOutOfBounds as u8 + TRAP_OFFSET);
64pub const TRAP_CAST_FAILURE: TrapCode =
65 TrapCode::unwrap_user(Trap::CastFailure as u8 + TRAP_OFFSET);
66
67fn blank_sig(isa: &dyn TargetIsa, call_conv: CallConv) -> ir::Signature {
72 let pointer_type = isa.pointer_type();
73 let mut sig = ir::Signature::new(call_conv);
74 sig.params.push(ir::AbiParam::special(
76 pointer_type,
77 ir::ArgumentPurpose::VMContext,
78 ));
79 sig.params.push(ir::AbiParam::new(pointer_type));
80 return sig;
81}
82
83fn unbarriered_store_type_at_offset(
92 pos: &mut FuncCursor,
93 flags: ir::MemFlags,
94 base: ir::Value,
95 offset: i32,
96 value: ir::Value,
97) {
98 pos.ins().store(flags, value, base, offset);
99}
100
101fn unbarriered_load_type_at_offset(
111 isa: &dyn TargetIsa,
112 pos: &mut FuncCursor,
113 ty: WasmValType,
114 flags: ir::MemFlags,
115 base: ir::Value,
116 offset: i32,
117) -> ir::Value {
118 let ir_ty = value_type(isa, ty);
119 pos.ins().load(ir_ty, flags, base, offset)
120}
121
122fn value_type(isa: &dyn TargetIsa, ty: WasmValType) -> ir::types::Type {
124 match ty {
125 WasmValType::I32 => ir::types::I32,
126 WasmValType::I64 => ir::types::I64,
127 WasmValType::F32 => ir::types::F32,
128 WasmValType::F64 => ir::types::F64,
129 WasmValType::V128 => ir::types::I8X16,
130 WasmValType::Ref(rt) => reference_type(rt.heap_type, isa.pointer_type()),
131 }
132}
133
134fn array_call_signature(isa: &dyn TargetIsa) -> ir::Signature {
150 let mut sig = blank_sig(isa, CallConv::triple_default(isa.triple()));
151 sig.params.push(ir::AbiParam::new(isa.pointer_type()));
155 sig.params.push(ir::AbiParam::new(isa.pointer_type()));
156 sig.returns.push(ir::AbiParam::new(ir::types::I8));
158 sig
159}
160
161fn wasm_call_conv(isa: &dyn TargetIsa, tunables: &Tunables) -> CallConv {
163 if tunables.winch_callable {
173 assert!(
174 matches!(
175 isa.triple().architecture,
176 Architecture::X86_64 | Architecture::Aarch64(_)
177 ),
178 "The Winch calling convention is only implemented for x86_64 and aarch64"
179 );
180 CallConv::Winch
181 } else {
182 CallConv::Tail
183 }
184}
185
186fn wasm_call_signature(
188 isa: &dyn TargetIsa,
189 wasm_func_ty: &WasmFuncType,
190 tunables: &Tunables,
191) -> ir::Signature {
192 let call_conv = wasm_call_conv(isa, tunables);
193 let mut sig = blank_sig(isa, call_conv);
194 let cvt = |ty: &WasmValType| ir::AbiParam::new(value_type(isa, *ty));
195 sig.params.extend(wasm_func_ty.params().iter().map(&cvt));
196 sig.returns.extend(wasm_func_ty.returns().iter().map(&cvt));
197 sig
198}
199
200fn reference_type(wasm_ht: WasmHeapType, pointer_type: ir::Type) -> ir::Type {
202 match wasm_ht.top() {
203 WasmHeapTopType::Func => pointer_type,
204 WasmHeapTopType::Any | WasmHeapTopType::Extern => ir::types::I32,
205 WasmHeapTopType::Cont => todo!(), }
207}
208
209pub const NS_WASM_FUNC: u32 = 0;
214
215pub const NS_WASMTIME_BUILTIN: u32 = 1;
219
220pub const NS_PULLEY_HOSTCALL: u32 = 2;
226
227#[derive(Debug, Clone, PartialEq, Eq)]
229pub struct Relocation {
230 pub reloc: binemit::Reloc,
232 pub reloc_target: RelocationTarget,
234 pub offset: binemit::CodeOffset,
236 pub addend: binemit::Addend,
238}
239
240pub fn clif_flags_to_wasmtime(
242 flags: impl IntoIterator<Item = settings::Value>,
243) -> Vec<(&'static str, FlagValue<'static>)> {
244 flags
245 .into_iter()
246 .map(|val| (val.name, to_flag_value(&val)))
247 .collect()
248}
249
250fn to_flag_value(v: &settings::Value) -> FlagValue<'static> {
251 match v.kind() {
252 settings::SettingKind::Enum => FlagValue::Enum(v.as_enum().unwrap()),
253 settings::SettingKind::Num => FlagValue::Num(v.as_num().unwrap()),
254 settings::SettingKind::Bool => FlagValue::Bool(v.as_bool().unwrap()),
255 settings::SettingKind::Preset => unreachable!(),
256 }
257}
258
259pub fn mach_trap_to_trap(trap: &MachTrap) -> Option<TrapInformation> {
261 let &MachTrap { offset, code } = trap;
262 Some(TrapInformation {
263 code_offset: offset,
264 trap_code: clif_trap_to_env_trap(code)?,
265 })
266}
267
268fn clif_trap_to_env_trap(trap: ir::TrapCode) -> Option<Trap> {
269 Some(match trap {
270 ir::TrapCode::STACK_OVERFLOW => Trap::StackOverflow,
271 ir::TrapCode::HEAP_OUT_OF_BOUNDS => Trap::MemoryOutOfBounds,
272 ir::TrapCode::INTEGER_OVERFLOW => Trap::IntegerOverflow,
273 ir::TrapCode::INTEGER_DIVISION_BY_ZERO => Trap::IntegerDivisionByZero,
274 ir::TrapCode::BAD_CONVERSION_TO_INTEGER => Trap::BadConversionToInteger,
275
276 TRAP_INTERNAL_ASSERT => return None,
280
281 other => Trap::from_u8(other.as_raw().get() - TRAP_OFFSET).unwrap(),
282 })
283}
284
285fn mach_reloc_to_reloc(
288 reloc: &FinalizedMachReloc,
289 name_map: &PrimaryMap<ir::UserExternalNameRef, ir::UserExternalName>,
290) -> Relocation {
291 let &FinalizedMachReloc {
292 offset,
293 kind,
294 ref target,
295 addend,
296 } = reloc;
297 let reloc_target = match *target {
298 FinalizedRelocTarget::ExternalName(ExternalName::User(user_func_ref)) => {
299 let name = &name_map[user_func_ref];
300 match name.namespace {
301 NS_WASM_FUNC => RelocationTarget::Wasm(FuncIndex::from_u32(name.index)),
302 NS_WASMTIME_BUILTIN => {
303 RelocationTarget::Builtin(BuiltinFunctionIndex::from_u32(name.index))
304 }
305 NS_PULLEY_HOSTCALL => RelocationTarget::PulleyHostcall(name.index),
306 _ => panic!("unknown namespace {}", name.namespace),
307 }
308 }
309 FinalizedRelocTarget::ExternalName(ExternalName::LibCall(libcall)) => {
310 panic!("unexpected libcall {libcall:?}");
313 }
314 _ => panic!("unrecognized external name"),
315 };
316 Relocation {
317 reloc: kind,
318 reloc_target,
319 offset,
320 addend,
321 }
322}
323
324struct BuiltinFunctionSignatures {
326 pointer_type: ir::Type,
327
328 host_call_conv: CallConv,
329 wasm_call_conv: CallConv,
330 argument_extension: ir::ArgumentExtension,
331}
332
333impl BuiltinFunctionSignatures {
334 fn new(compiler: &Compiler) -> Self {
335 Self {
336 pointer_type: compiler.isa().pointer_type(),
337 host_call_conv: CallConv::triple_default(compiler.isa().triple()),
338 wasm_call_conv: wasm_call_conv(compiler.isa(), compiler.tunables()),
339 argument_extension: compiler.isa().default_argument_extension(),
340 }
341 }
342
343 fn vmctx(&self) -> AbiParam {
344 AbiParam::special(self.pointer_type, ArgumentPurpose::VMContext)
345 }
346
347 fn pointer(&self) -> AbiParam {
348 AbiParam::new(self.pointer_type)
349 }
350
351 fn u32(&self) -> AbiParam {
352 AbiParam::new(ir::types::I32)
353 }
354
355 fn u64(&self) -> AbiParam {
356 AbiParam::new(ir::types::I64)
357 }
358
359 fn f32(&self) -> AbiParam {
360 AbiParam::new(ir::types::F32)
361 }
362
363 fn f64(&self) -> AbiParam {
364 AbiParam::new(ir::types::F64)
365 }
366
367 fn u8(&self) -> AbiParam {
368 AbiParam::new(ir::types::I8)
369 }
370
371 fn i8x16(&self) -> AbiParam {
372 AbiParam::new(ir::types::I8X16)
373 }
374
375 fn f32x4(&self) -> AbiParam {
376 AbiParam::new(ir::types::F32X4)
377 }
378
379 fn f64x2(&self) -> AbiParam {
380 AbiParam::new(ir::types::F64X2)
381 }
382
383 fn bool(&self) -> AbiParam {
384 AbiParam::new(ir::types::I8)
385 }
386
387 fn wasm_signature(&self, builtin: BuiltinFunctionIndex) -> Signature {
388 let mut _cur = 0;
389 macro_rules! iter {
390 (
391 $(
392 $( #[$attr:meta] )*
393 $name:ident( $( $pname:ident: $param:ident ),* ) $( -> $result:ident )?;
394 )*
395 ) => {
396 $(
397 $( #[$attr] )*
398 if _cur == builtin.index() {
399 return Signature {
400 params: vec![ $( self.$param() ),* ],
401 returns: vec![ $( self.$result() )? ],
402 call_conv: self.wasm_call_conv,
403 };
404 }
405 _cur += 1;
406 )*
407 };
408 }
409
410 wasmtime_environ::foreach_builtin_function!(iter);
411
412 unreachable!();
413 }
414
415 fn host_signature(&self, builtin: BuiltinFunctionIndex) -> Signature {
416 let mut sig = self.wasm_signature(builtin);
417 sig.call_conv = self.host_call_conv;
418
419 for arg in sig.params.iter_mut().chain(sig.returns.iter_mut()) {
423 if arg.value_type.is_int() {
424 arg.extension = self.argument_extension;
425 }
426 }
427
428 sig
429 }
430}
431
432const I31_REF_DISCRIMINANT: u32 = 1;
438
439#[derive(PartialEq, Eq)]
445#[must_use]
446enum Reachability<T> {
447 Reachable(T),
449 Unreachable,
453}