From 9eb961452feb8965a04f728220c2959a42b59946 Mon Sep 17 00:00:00 2001 From: DaniPopes <57450786+DaniPopes@users.noreply.github.com> Date: Thu, 8 Aug 2024 23:01:11 +0200 Subject: [PATCH] fix: add noinline to mstore builtin (#62) --- crates/revmc/src/compiler/translate.rs | 23 ++++++++++++++++++++--- crates/revmc/src/tests/mod.rs | 10 ++++++++++ 2 files changed, 30 insertions(+), 3 deletions(-) diff --git a/crates/revmc/src/compiler/translate.rs b/crates/revmc/src/compiler/translate.rs index 4b35253..963d306 100644 --- a/crates/revmc/src/compiler/translate.rs +++ b/crates/revmc/src/compiler/translate.rs @@ -55,7 +55,9 @@ type SwitchTargets = Vec<(u64, ::BasicBlock)>; #[derive(Clone, Copy, PartialEq, Eq)] enum ResumeKind { + /// Use `indirectbr`. Blocks, + /// Use a switch over `0..N`. Indexes, } @@ -486,7 +488,7 @@ impl<'a, B: Backend> FunctionCx<'a, B> { ensure!(is_eof_enabled, "EOF bytecode in non-EOF spec"); } - // self.call_printf(format_printf!("{}\n", data.to_op_in(self.bytecode)), &[]); + // self.call_printf(format_printf!("{}\n", self.op_block_name("")), &[]); let branch_to_next_opcode = |this: &mut Self| { debug_assert!( @@ -1692,7 +1694,10 @@ impl<'a, B: Backend> FunctionCx<'a, B> { #[must_use] fn call_builtin(&mut self, builtin: Builtin, args: &[B::Value]) -> Option { let function = self.builtin_function(builtin); - // self.call_printf(format_printf!("calling {}\n", builtin.name()), &[]); + // self.call_printf( + // format_printf!("{} - calling {}\n", self.op_block_name(""), builtin.name()), + // &[], + // ); self.bcx.call(function, args) } @@ -1934,8 +1939,20 @@ impl<'a, B: Backend> FunctionCx<'a, B> { self.build_check_instruction_result(ret); } + /// Builds: + /// - `Load` => `fn mload(offset: u256, out: ptr, ecx: ptr) -> InstructionResult` + /// - `Store` => `fn mstore(offset: u256, value: u256, ecx: ptr) -> InstructionResult` + /// - `Store8` => `fn mstore(offset: u256, value: u8, ecx: ptr) -> InstructionResult` fn build_mem_op(&mut self, kind: MemOpKind) { let is_load = matches!(kind, MemOpKind::Load); + // TODO: If `store` is inlined it can cause segfaults. https://github.com/paradigmxyz/revmc/issues/61 + if !is_load { + self.bcx.add_function_attribute( + None, + Attribute::NoInline, + FunctionAttributeLocation::Function, + ); + } let ptr_args = if is_load { &[1, 2][..] } else { &[2][..] }; for &ptr_arg in ptr_args { for attr in default_attrs::for_ref() { @@ -2224,7 +2241,7 @@ impl<'a, B: Backend> FunctionCx<'a, B> { let prefix = "__revmc_ir_builtin_"; let name = &format!("{prefix}{name}")[..]; - // self.call_printf(format_printf!("calling {name}\n"), &[]); + // self.call_printf(format_printf!("{} - calling {name}\n", self.op_block_name("")), &[]); debug_assert_eq!(args.len(), arg_types.len()); let linkage = revmc_backend::Linkage::Private; diff --git a/crates/revmc/src/tests/mod.rs b/crates/revmc/src/tests/mod.rs index a646719..63ffb10 100644 --- a/crates/revmc/src/tests/mod.rs +++ b/crates/revmc/src/tests/mod.rs @@ -167,6 +167,16 @@ tests! { expected_stack: &[2_U256, 1_U256], expected_gas: 3 + 3 + 3, }), + swap2(@raw { + bytecode: &[op::PUSH1, 1, op::PUSH1, 2, op::PUSH1, 3, op::SWAP2], + expected_stack: &[3_U256, 2_U256, 1_U256], + expected_gas: 3 + 3 + 3 + 3, + }), + swap3(@raw { + bytecode: &[op::PUSH1, 1, op::PUSH1, 2, op::PUSH1, 3, op::PUSH1, 4, op::SWAP3], + expected_stack: &[4_U256, 2_U256, 3_U256, 1_U256], + expected_gas: 3 + 3 + 3 + 3 + 3, + }), swapn(@raw { bytecode: &eof(&[op::PUSH1, 1, op::PUSH1, 2, op::SWAPN, 0, op::STOP]), spec_id: SpecId::PRAGUE_EOF,