mirror of https://github.com/ConsenSys/mythril
blockchainethereumsmart-contractssoliditysecurityprogram-analysissecurity-analysissymbolic-execution
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
71 lines
2.5 KiB
71 lines
2.5 KiB
import pytest
|
|
|
|
from mythril.disassembler.disassembly import Disassembly
|
|
from mythril.laser.ethereum.instructions import Instruction
|
|
from mythril.laser.ethereum.state.calldata import ConcreteCalldata
|
|
from mythril.laser.ethereum.state.environment import Environment
|
|
from mythril.laser.ethereum.state.global_state import GlobalState
|
|
from mythril.laser.ethereum.state.machine_state import MachineState
|
|
from mythril.laser.ethereum.state.world_state import WorldState
|
|
from mythril.laser.ethereum.time_handler import time_handler
|
|
from mythril.laser.ethereum.transaction.transaction_models import (
|
|
MessageCallTransaction,
|
|
TransactionStartSignal,
|
|
)
|
|
from mythril.laser.smt import symbol_factory
|
|
from mythril.support.support_utils import get_code_hash
|
|
|
|
last_state = None
|
|
created_contract_account = None
|
|
|
|
|
|
def generate_salted_address(code_str, salt, caller):
|
|
addr = hex(caller.value)[2:]
|
|
addr = "0" * (40 - len(addr)) + addr
|
|
|
|
salt = hex(salt.value)[2:]
|
|
salt = "0" * (64 - len(salt)) + salt
|
|
|
|
contract_address = int(
|
|
get_code_hash("0xff" + addr + salt + get_code_hash(code_str)[2:])[26:], 16
|
|
)
|
|
return contract_address
|
|
|
|
|
|
def test_create2():
|
|
world_state = WorldState()
|
|
account = world_state.create_account(balance=10, address=101)
|
|
account.code = Disassembly("60606040")
|
|
environment = Environment(account, None, None, None, None, None, None)
|
|
og_state = GlobalState(
|
|
world_state, environment, None, MachineState(gas_limit=8000000)
|
|
)
|
|
code_raw = []
|
|
code = "606060606060"
|
|
for i in range(len(code) // 2):
|
|
code_raw.append(int(code[2 * i : 2 * (i + 1)], 16))
|
|
calldata = ConcreteCalldata("1", code_raw)
|
|
environment.calldata = calldata
|
|
og_state.transaction_stack.append(
|
|
(MessageCallTransaction(world_state=WorldState(), gas_limit=8000000), None)
|
|
)
|
|
value = symbol_factory.BitVecVal(3, 256)
|
|
salt = symbol_factory.BitVecVal(10, 256)
|
|
og_state.mstate.stack = [
|
|
salt,
|
|
symbol_factory.BitVecVal(6, 256),
|
|
symbol_factory.BitVecVal(0, 256),
|
|
value,
|
|
]
|
|
og_state.mstate.memory.extend(100)
|
|
og_state.mstate.memory[0:6] = [96] * 6
|
|
time_handler.start_execution(10)
|
|
instruction = Instruction("create2", dynamic_loader=None)
|
|
# Act + Assert
|
|
with pytest.raises(TransactionStartSignal) as t:
|
|
_ = instruction.evaluate(og_state)[0]
|
|
assert t.value.transaction.call_value == value
|
|
assert t.value.transaction.code.bytecode == code
|
|
assert t.value.transaction.callee_account.address == generate_salted_address(
|
|
code, salt, account.address
|
|
)
|
|
|