//! Helpers used to print a message and abort in case of certain panics and some detected UB. use crate::prelude::*; fn codegen_print(fx: &mut FunctionCx<'_, '_, impl cranelift_module::Backend>, msg: &str) { let puts = fx .cx .module .declare_function( "puts", Linkage::Import, &Signature { call_conv: CallConv::triple_default(fx.triple()), params: vec![AbiParam::new(pointer_ty(fx.tcx))], returns: vec![AbiParam::new(types::I32)], }, ) .unwrap(); let puts = fx.cx.module.declare_func_in_func(puts, &mut fx.bcx.func); #[cfg(debug_assertions)] { fx.add_comment(puts, "puts"); } let symbol_name = fx.tcx.symbol_name(fx.instance); let real_msg = format!("trap at {:?} ({}): {}\0", fx.instance, symbol_name, msg); let msg_ptr = fx.anonymous_str("trap", &real_msg); fx.bcx.ins().call(puts, &[msg_ptr]); } /// Trap code: user1 pub(crate) fn trap_abort( fx: &mut FunctionCx<'_, '_, impl cranelift_module::Backend>, msg: impl AsRef, ) { codegen_print(fx, msg.as_ref()); fx.bcx.ins().trap(TrapCode::User(1)); } /// Use this for example when a function call should never return. This will fill the current block, /// so you can **not** add instructions to it afterwards. /// /// Trap code: user65535 pub(crate) fn trap_unreachable( fx: &mut FunctionCx<'_, '_, impl cranelift_module::Backend>, msg: impl AsRef, ) { codegen_print(fx, msg.as_ref()); fx.bcx.ins().trap(TrapCode::UnreachableCodeReached); } /// Like `trap_unreachable` but returns a fake value of the specified type. /// /// Trap code: user65535 pub(crate) fn trap_unreachable_ret_value<'tcx>( fx: &mut FunctionCx<'_, 'tcx, impl cranelift_module::Backend>, dest_layout: TyAndLayout<'tcx>, msg: impl AsRef, ) -> CValue<'tcx> { codegen_print(fx, msg.as_ref()); let true_ = fx.bcx.ins().iconst(types::I32, 1); fx.bcx.ins().trapnz(true_, TrapCode::UnreachableCodeReached); CValue::by_ref(Pointer::const_addr(fx, 0), dest_layout) } /// Use this when something is unimplemented, but `libcore` or `libstd` requires it to codegen. /// Unlike `trap_unreachable` this will not fill the current block, so you **must** add instructions /// to it afterwards. /// /// Trap code: user65535 pub(crate) fn trap_unimplemented( fx: &mut FunctionCx<'_, '_, impl cranelift_module::Backend>, msg: impl AsRef, ) { codegen_print(fx, msg.as_ref()); let true_ = fx.bcx.ins().iconst(types::I32, 1); fx.bcx.ins().trapnz(true_, TrapCode::User(!0)); } /// Like `trap_unimplemented` but returns a fake value of the specified type. /// /// Trap code: user65535 pub(crate) fn trap_unimplemented_ret_value<'tcx>( fx: &mut FunctionCx<'_, 'tcx, impl cranelift_module::Backend>, dest_layout: TyAndLayout<'tcx>, msg: impl AsRef, ) -> CValue<'tcx> { trap_unimplemented(fx, msg); CValue::by_ref(Pointer::const_addr(fx, 0), dest_layout) }