From 6a7de73161568d5c457588768ac9dea66218514a Mon Sep 17 00:00:00 2001 From: Guruprasad Kamath Date: Thu, 29 Aug 2024 12:48:47 +0200 Subject: [PATCH] refactor eof-call opcodes --- src/ethereum/prague/vm/instructions/system.py | 337 +++++++----------- 1 file changed, 131 insertions(+), 206 deletions(-) diff --git a/src/ethereum/prague/vm/instructions/system.py b/src/ethereum/prague/vm/instructions/system.py index e53e0df752..3bc40dcd73 100644 --- a/src/ethereum/prague/vm/instructions/system.py +++ b/src/ethereum/prague/vm/instructions/system.py @@ -65,8 +65,6 @@ from ..memory import memory_read_bytes, memory_write from ..stack import pop, push -# TODO: Integrate EOF CALL* and CREATE in generic - def generic_create( evm: Evm, @@ -741,56 +739,57 @@ def invalid(evm: Evm) -> None: pass -def ext_call(evm: Evm) -> None: +def generic_eof_call( + evm: Evm, + input_offset: U256, + input_size: U256, + caller: Address, + target: Address, + current_target: Address, + code_address: Address, + value: U256, + should_transfer_value: bool, + is_static: bool, + allow_legacy_code: bool, +) -> None: """ - Message-call into an account for EOF. + Core logic used by the `EXT_CALL*` family of opcodes. Parameters ---------- evm : The current EVM frame. + input_offset : + The offset in memory where the call data starts. + input_size : + The size of the call data. + caller : + The address of the account that initiated the call. + target : + The address of the account to call. + current_target : + The address of the account making the call. + code_address : + The address of the account containing the code. + value : + The value to transfer. + should_transfer_value : + Whether the value should be transferred. + is_static : + Whether the call is static. + allow_legacy_code : + Whether to allow legacy code. + + Raises + ------ + WriteInStaticContext : + If a write operation is attempted in a static context. """ from ...vm.interpreter import STACK_DEPTH_LIMIT, process_message from ..eof import Eof, EofVersion, get_eof_version - # STACK - try: - target_address = to_address_without_mask(pop(evm.stack)) - except ValueError: - raise ExceptionalHalt - input_offset = pop(evm.stack) - input_size = pop(evm.stack) - value = pop(evm.stack) - - # GAS - extend_memory = calculate_gas_extend_memory( - evm.memory, [(input_offset, input_size)] - ) - - if target_address in evm.accessed_addresses: - access_gas_cost = GAS_WARM_ACCESS - else: - evm.accessed_addresses.add(target_address) - access_gas_cost = GAS_COLD_ACCOUNT_ACCESS - - create_gas_cost = ( - Uint(0) - if is_account_alive(evm.env.state, target_address) or value == 0 - else GAS_NEW_ACCOUNT - ) - transfer_gas_cost = Uint(0) if value == 0 else GAS_CALL_VALUE - charge_gas( - evm, - extend_memory.cost - + access_gas_cost - + create_gas_cost - + transfer_gas_cost, - ) - - # OPERATION if evm.gas_left < EOF_CALL_MIN_RETAINED_GAS: push(evm.stack, U256(1)) - evm.pc += 1 return message_call_gas = evm.gas_left - max( evm.gas_left // 64, EOF_CALL_MIN_RETAINED_GAS @@ -799,19 +798,28 @@ def ext_call(evm: Evm) -> None: if evm.message.is_static and value != U256(0): raise WriteInStaticContext - evm.memory += b"\x00" * extend_memory.expand_by + evm.return_data = b"" + call_data = memory_read_bytes(evm.memory, input_offset, input_size) + container = get_account(evm.env.state, code_address).code + eof_version = get_eof_version(container) sender_balance = get_account( evm.env.state, evm.message.current_target ).balance + if ( + message_call_gas < EOF_CALL_MIN_CALLEE_GAS + or sender_balance < value + or evm.message.depth + 1 > STACK_DEPTH_LIMIT + ): + push(evm.stack, U256(1)) + return - evm.return_data = b"" - - call_data = memory_read_bytes(evm.memory, input_offset, input_size) + if not allow_legacy_code and eof_version == EofVersion.LEGACY: + push(evm.stack, U256(1)) + return - container = get_account(evm.env.state, target_address).code + evm.gas_left -= message_call_gas - eof_version = get_eof_version(container) if eof_version == EofVersion.LEGACY: eof = None else: @@ -831,29 +839,18 @@ def ext_call(evm: Evm) -> None: is_deploy_container=is_deploy_container, ) - if ( - message_call_gas < EOF_CALL_MIN_CALLEE_GAS - or sender_balance < value - or evm.message.depth + 1 > STACK_DEPTH_LIMIT - ): - push(evm.stack, U256(1)) - evm.pc += 1 - return - - evm.gas_left -= message_call_gas - child_message = Message( - caller=evm.message.current_target, - target=target_address, + caller=caller, + target=target, gas=Uint(message_call_gas), value=value, data=call_data, code=container, - current_target=target_address, + current_target=current_target, depth=evm.message.depth + 1, - code_address=target_address, - should_transfer_value=True, - is_static=False, + code_address=code_address, + should_transfer_value=should_transfer_value, + is_static=is_static, accessed_addresses=evm.accessed_addresses.copy(), accessed_storage_keys=evm.accessed_storage_keys.copy(), parent_evm=evm, @@ -873,23 +870,16 @@ def ext_call(evm: Evm) -> None: incorporate_child_on_success(evm, child_evm) push(evm.stack, U256(0)) - # PROGRAM COUNTER - evm.pc += 1 - -def ext_delegatecall(evm: Evm) -> None: +def ext_call(evm: Evm) -> None: """ - Message-call into an account with the callee's context - for EOF. + Message-call into an account for EOF. Parameters ---------- evm : The current EVM frame. """ - from ...vm.interpreter import STACK_DEPTH_LIMIT, process_message - from ..eof import Eof, EofVersion, get_eof_version - # STACK try: target_address = to_address_without_mask(pop(evm.stack)) @@ -897,6 +887,7 @@ def ext_delegatecall(evm: Evm) -> None: raise ExceptionalHalt input_offset = pop(evm.stack) input_size = pop(evm.stack) + value = pop(evm.stack) # GAS extend_memory = calculate_gas_extend_memory( @@ -909,86 +900,86 @@ def ext_delegatecall(evm: Evm) -> None: evm.accessed_addresses.add(target_address) access_gas_cost = GAS_COLD_ACCOUNT_ACCESS - charge_gas(evm, extend_memory.cost + access_gas_cost) - - # OPERATION - if evm.gas_left < EOF_CALL_MIN_RETAINED_GAS: - push(evm.stack, U256(1)) - evm.pc += 1 - return - message_call_gas = evm.gas_left - max( - evm.gas_left // 64, EOF_CALL_MIN_RETAINED_GAS + create_gas_cost = ( + Uint(0) + if is_account_alive(evm.env.state, target_address) or value == 0 + else GAS_NEW_ACCOUNT + ) + transfer_gas_cost = Uint(0) if value == 0 else GAS_CALL_VALUE + charge_gas( + evm, + extend_memory.cost + + access_gas_cost + + create_gas_cost + + transfer_gas_cost, ) + # OPERATION evm.memory += b"\x00" * extend_memory.expand_by + generic_eof_call( + evm=evm, + input_offset=input_offset, + input_size=input_size, + caller=evm.message.current_target, + target=target_address, + current_target=target_address, + code_address=target_address, + value=value, + should_transfer_value=True, + is_static=False, + allow_legacy_code=True, + ) - evm.return_data = b"" + # PROGRAM COUNTER + evm.pc += 1 - call_data = memory_read_bytes(evm.memory, input_offset, input_size) - container = get_account(evm.env.state, target_address).code +def ext_delegatecall(evm: Evm) -> None: + """ + Message-call into an account with the callee's context + for EOF. - if ( - message_call_gas < EOF_CALL_MIN_CALLEE_GAS - or evm.message.depth + 1 > STACK_DEPTH_LIMIT - or get_eof_version(container) == EofVersion.LEGACY - ): - push(evm.stack, U256(1)) - evm.pc += 1 - return + Parameters + ---------- + evm : + The current EVM frame. + """ + # STACK + try: + target_address = to_address_without_mask(pop(evm.stack)) + except ValueError: + raise ExceptionalHalt + input_offset = pop(evm.stack) + input_size = pop(evm.stack) - evm.gas_left -= message_call_gas + # GAS + extend_memory = calculate_gas_extend_memory( + evm.memory, [(input_offset, input_size)] + ) - eof_version = get_eof_version(container) - if eof_version == EofVersion.LEGACY: - eof = None + if target_address in evm.accessed_addresses: + access_gas_cost = GAS_WARM_ACCESS else: - is_init_container = False - is_deploy_container = False - metadata = metadata_from_container( - container, - validate=False, - is_init_container=is_init_container, - is_deploy_container=is_deploy_container, - ) - eof = Eof( - version=eof_version, - container=container, - metadata=metadata, - is_init_container=is_init_container, - is_deploy_container=is_deploy_container, - ) + evm.accessed_addresses.add(target_address) + access_gas_cost = GAS_COLD_ACCOUNT_ACCESS - child_message = Message( + charge_gas(evm, extend_memory.cost + access_gas_cost) + + # OPERATION + evm.memory += b"\x00" * extend_memory.expand_by + generic_eof_call( + evm=evm, + input_offset=input_offset, + input_size=input_size, caller=evm.message.caller, target=evm.message.current_target, - gas=Uint(message_call_gas), - value=U256(0), - data=call_data, - code=container, current_target=evm.message.current_target, - depth=evm.message.depth + 1, code_address=target_address, + value=U256(0), should_transfer_value=False, is_static=False, - accessed_addresses=evm.accessed_addresses.copy(), - accessed_storage_keys=evm.accessed_storage_keys.copy(), - parent_evm=evm, - eof=eof, + allow_legacy_code=False, ) - child_evm = process_message(child_message, evm.env) - - evm.return_data = child_evm.output - - if child_evm.error: - incorporate_child_on_error(evm, child_evm) - if isinstance(child_evm.error, Revert): - push(evm.stack, U256(1)) - else: - push(evm.stack, U256(2)) - else: - incorporate_child_on_success(evm, child_evm) - push(evm.stack, U256(0)) # PROGRAM COUNTER evm.pc += 1 @@ -997,16 +988,12 @@ def ext_delegatecall(evm: Evm) -> None: def ext_staticcall(evm: Evm) -> None: """ Static message-call into an account for EOF. - # TODO: Refactor this function to use generic_call Parameters ---------- evm : The current EVM frame. """ - from ...vm.interpreter import STACK_DEPTH_LIMIT, process_message - from ..eof import Eof, EofVersion, get_eof_version - # STACK try: target_address = to_address_without_mask(pop(evm.stack)) @@ -1029,82 +1016,20 @@ def ext_staticcall(evm: Evm) -> None: charge_gas(evm, extend_memory.cost + access_gas_cost) # OPERATION - if evm.gas_left < EOF_CALL_MIN_RETAINED_GAS: - push(evm.stack, U256(1)) - evm.pc += 1 - return - message_call_gas = evm.gas_left - max( - evm.gas_left // 64, EOF_CALL_MIN_RETAINED_GAS - ) - evm.memory += b"\x00" * extend_memory.expand_by - - evm.return_data = b"" - - call_data = memory_read_bytes(evm.memory, input_offset, input_size) - - container = get_account(evm.env.state, target_address).code - - if ( - message_call_gas < EOF_CALL_MIN_CALLEE_GAS - or evm.message.depth + 1 > STACK_DEPTH_LIMIT - ): - push(evm.stack, U256(1)) - evm.pc += 1 - return - - evm.gas_left -= message_call_gas - - eof_version = get_eof_version(container) - if eof_version == EofVersion.LEGACY: - eof = None - else: - is_init_container = False - is_deploy_container = False - metadata = metadata_from_container( - container, - validate=False, - is_init_container=is_init_container, - is_deploy_container=is_deploy_container, - ) - eof = Eof( - version=eof_version, - container=container, - metadata=metadata, - is_init_container=is_init_container, - is_deploy_container=is_deploy_container, - ) - - child_message = Message( + generic_eof_call( + evm=evm, + input_offset=input_offset, + input_size=input_size, caller=evm.message.current_target, target=target_address, - gas=Uint(message_call_gas), - value=U256(0), - data=call_data, - code=container, current_target=target_address, - depth=evm.message.depth + 1, code_address=target_address, + value=U256(0), should_transfer_value=False, is_static=True, - accessed_addresses=evm.accessed_addresses.copy(), - accessed_storage_keys=evm.accessed_storage_keys.copy(), - parent_evm=evm, - eof=eof, + allow_legacy_code=True, ) - child_evm = process_message(child_message, evm.env) - - evm.return_data = child_evm.output - - if child_evm.error: - incorporate_child_on_error(evm, child_evm) - if isinstance(child_evm.error, Revert): - push(evm.stack, U256(1)) - else: - push(evm.stack, U256(2)) - else: - incorporate_child_on_success(evm, child_evm) - push(evm.stack, U256(0)) # PROGRAM COUNTER evm.pc += 1