1#![warn(clippy::cast_possible_truncation, clippy::cast_sign_loss)]
9
10use cranelift_codegen::{
11 binemit,
12 cursor::FuncCursor,
13 ir::{self, AbiParam, ArgumentPurpose, ExternalName, InstBuilder, Signature, TrapCode},
14 isa::{CallConv, TargetIsa},
15 settings, FinalizedMachReloc, FinalizedRelocTarget, MachTrap,
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 builder;
34mod compiler;
35mod debug;
36mod func_environ;
37mod gc;
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 }
206}
207
208pub const NS_WASM_FUNC: u32 = 0;
213
214pub const NS_WASMTIME_BUILTIN: u32 = 1;
218
219pub const NS_PULLEY_HOSTCALL: u32 = 2;
225
226#[derive(Debug, Clone, PartialEq, Eq)]
228pub struct Relocation {
229 pub reloc: binemit::Reloc,
231 pub reloc_target: RelocationTarget,
233 pub offset: binemit::CodeOffset,
235 pub addend: binemit::Addend,
237}
238
239pub fn clif_flags_to_wasmtime(
241 flags: impl IntoIterator<Item = settings::Value>,
242) -> Vec<(&'static str, FlagValue<'static>)> {
243 flags
244 .into_iter()
245 .map(|val| (val.name, to_flag_value(&val)))
246 .collect()
247}
248
249fn to_flag_value(v: &settings::Value) -> FlagValue<'static> {
250 match v.kind() {
251 settings::SettingKind::Enum => FlagValue::Enum(v.as_enum().unwrap()),
252 settings::SettingKind::Num => FlagValue::Num(v.as_num().unwrap()),
253 settings::SettingKind::Bool => FlagValue::Bool(v.as_bool().unwrap()),
254 settings::SettingKind::Preset => unreachable!(),
255 }
256}
257
258pub fn mach_trap_to_trap(trap: &MachTrap) -> Option<TrapInformation> {
260 let &MachTrap { offset, code } = trap;
261 Some(TrapInformation {
262 code_offset: offset,
263 trap_code: clif_trap_to_env_trap(code)?,
264 })
265}
266
267fn clif_trap_to_env_trap(trap: ir::TrapCode) -> Option<Trap> {
268 Some(match trap {
269 ir::TrapCode::STACK_OVERFLOW => Trap::StackOverflow,
270 ir::TrapCode::HEAP_OUT_OF_BOUNDS => Trap::MemoryOutOfBounds,
271 ir::TrapCode::INTEGER_OVERFLOW => Trap::IntegerOverflow,
272 ir::TrapCode::INTEGER_DIVISION_BY_ZERO => Trap::IntegerDivisionByZero,
273 ir::TrapCode::BAD_CONVERSION_TO_INTEGER => Trap::BadConversionToInteger,
274
275 TRAP_INTERNAL_ASSERT => return None,
279
280 other => Trap::from_u8(other.as_raw().get() - TRAP_OFFSET).unwrap(),
281 })
282}
283
284fn mach_reloc_to_reloc(
287 reloc: &FinalizedMachReloc,
288 name_map: &PrimaryMap<ir::UserExternalNameRef, ir::UserExternalName>,
289) -> Relocation {
290 let &FinalizedMachReloc {
291 offset,
292 kind,
293 ref target,
294 addend,
295 } = reloc;
296 let reloc_target = match *target {
297 FinalizedRelocTarget::ExternalName(ExternalName::User(user_func_ref)) => {
298 let name = &name_map[user_func_ref];
299 match name.namespace {
300 NS_WASM_FUNC => RelocationTarget::Wasm(FuncIndex::from_u32(name.index)),
301 NS_WASMTIME_BUILTIN => {
302 RelocationTarget::Builtin(BuiltinFunctionIndex::from_u32(name.index))
303 }
304 NS_PULLEY_HOSTCALL => RelocationTarget::PulleyHostcall(name.index),
305 _ => panic!("unknown namespace {}", name.namespace),
306 }
307 }
308 FinalizedRelocTarget::ExternalName(ExternalName::LibCall(libcall)) => {
309 let libcall = libcall_cranelift_to_wasmtime(libcall);
310 RelocationTarget::HostLibcall(libcall)
311 }
312 _ => panic!("unrecognized external name"),
313 };
314 Relocation {
315 reloc: kind,
316 reloc_target,
317 offset,
318 addend,
319 }
320}
321
322fn libcall_cranelift_to_wasmtime(call: ir::LibCall) -> wasmtime_environ::obj::LibCall {
323 use wasmtime_environ::obj::LibCall as LC;
324 match call {
325 ir::LibCall::FloorF32 => LC::FloorF32,
326 ir::LibCall::FloorF64 => LC::FloorF64,
327 ir::LibCall::NearestF32 => LC::NearestF32,
328 ir::LibCall::NearestF64 => LC::NearestF64,
329 ir::LibCall::CeilF32 => LC::CeilF32,
330 ir::LibCall::CeilF64 => LC::CeilF64,
331 ir::LibCall::TruncF32 => LC::TruncF32,
332 ir::LibCall::TruncF64 => LC::TruncF64,
333 ir::LibCall::FmaF32 => LC::FmaF32,
334 ir::LibCall::FmaF64 => LC::FmaF64,
335 ir::LibCall::X86Pshufb => LC::X86Pshufb,
336 _ => panic!("cranelift emitted a libcall wasmtime does not support: {call:?}"),
337 }
338}
339
340struct BuiltinFunctionSignatures {
342 pointer_type: ir::Type,
343
344 host_call_conv: CallConv,
345 wasm_call_conv: CallConv,
346 argument_extension: ir::ArgumentExtension,
347}
348
349impl BuiltinFunctionSignatures {
350 fn new(compiler: &Compiler) -> Self {
351 Self {
352 pointer_type: compiler.isa().pointer_type(),
353 host_call_conv: CallConv::triple_default(compiler.isa().triple()),
354 wasm_call_conv: wasm_call_conv(compiler.isa(), compiler.tunables()),
355 argument_extension: compiler.isa().default_argument_extension(),
356 }
357 }
358
359 fn vmctx(&self) -> AbiParam {
360 AbiParam::special(self.pointer_type, ArgumentPurpose::VMContext)
361 }
362
363 fn pointer(&self) -> AbiParam {
364 AbiParam::new(self.pointer_type)
365 }
366
367 fn u32(&self) -> AbiParam {
368 AbiParam::new(ir::types::I32)
369 }
370
371 fn u64(&self) -> AbiParam {
372 AbiParam::new(ir::types::I64)
373 }
374
375 fn u8(&self) -> AbiParam {
376 AbiParam::new(ir::types::I8)
377 }
378
379 fn bool(&self) -> AbiParam {
380 AbiParam::new(ir::types::I8)
381 }
382
383 fn wasm_signature(&self, builtin: BuiltinFunctionIndex) -> Signature {
384 let mut _cur = 0;
385 macro_rules! iter {
386 (
387 $(
388 $( #[$attr:meta] )*
389 $name:ident( $( $pname:ident: $param:ident ),* ) $( -> $result:ident )?;
390 )*
391 ) => {
392 $(
393 $( #[$attr] )*
394 if _cur == builtin.index() {
395 return Signature {
396 params: vec![ $( self.$param() ),* ],
397 returns: vec![ $( self.$result() )? ],
398 call_conv: self.wasm_call_conv,
399 };
400 }
401 _cur += 1;
402 )*
403 };
404 }
405
406 wasmtime_environ::foreach_builtin_function!(iter);
407
408 unreachable!();
409 }
410
411 fn host_signature(&self, builtin: BuiltinFunctionIndex) -> Signature {
412 let mut sig = self.wasm_signature(builtin);
413 sig.call_conv = self.host_call_conv;
414
415 for arg in sig.params.iter_mut().chain(sig.returns.iter_mut()) {
419 if arg.value_type.is_int() {
420 arg.extension = self.argument_extension;
421 }
422 }
423
424 sig
425 }
426}
427
428const I31_REF_DISCRIMINANT: u32 = 1;