Update existing and add new test cases.

This commit is contained in:
Bhargava Shastry 2023-02-01 08:28:05 +01:00
parent f8880cad82
commit 5c6e12b2c0
10 changed files with 197 additions and 53 deletions

View File

@ -6,6 +6,8 @@
} }
// ---- // ----
// Trace: // Trace:
// CREATE(0, 0, 0)
// CREATE(0, 0, 0)
// Memory dump: // Memory dump:
// 0: 0000000000000000000000000000000000000000000000000000000000000001 // 0: 0000000000000000000000000000000000000000000000000000000000000001
// Storage dump: // Storage dump:

View File

@ -8,6 +8,8 @@
// EVMVersion: >=constantinople // EVMVersion: >=constantinople
// ---- // ----
// Trace: // Trace:
// CREATE2(0, 0, 0, 0)
// CREATE2(0, 0, 0, 0)
// Memory dump: // Memory dump:
// 0: 0000000000000000000000000000000000000000000000000000000000000001 // 0: 0000000000000000000000000000000000000000000000000000000000000001
// Storage dump: // Storage dump:

View File

@ -14,6 +14,7 @@
// ---- // ----
// Trace: // Trace:
// CALL(153, 0x11111111, 0, 64, 32, 256, 32) // CALL(153, 0x11111111, 0, 64, 32, 256, 32)
// RETURN(0, 0)
// Memory dump: // Memory dump:
// 40: 0000000000000000000000000000000000000000000000000000000000000042 // 40: 0000000000000000000000000000000000000000000000000000000000000042
// 100: 0000000000000000000000000000000000000000000000000000000000000042 // 100: 0000000000000000000000000000000000000000000000000000000000000042

View File

@ -7,6 +7,38 @@
} }
// ---- // ----
// Trace: // Trace:
// Interpreter execution step limit reached. // LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// LOG0(0, 0)
// Trace size limit reached.
// Memory dump: // Memory dump:
// Storage dump: // Storage dump:

View File

@ -5,5 +5,6 @@
// EVMVersion: >=constantinople // EVMVersion: >=constantinople
// ---- // ----
// Trace: // Trace:
// CALL(0, 0, 0, 0, 0, 0, 0)
// Memory dump: // Memory dump:
// Storage dump: // Storage dump:

View File

@ -0,0 +1,40 @@
{
returndatacopy(1, 1, 0)
calldatacopy(1, 1, 0)
extcodecopy(1, 1, 1, 0)
codecopy(1, 1, 0)
log0(1, 0)
log1(1, 0, 1)
log2(1, 0, 1, 1)
log3(1, 0, 1, 1, 1)
log4(1, 0, 1, 1, 1, 1)
pop(create(1, 1, 0))
pop(create2(1, 1, 0, 1))
pop(call(1, 1, 1, 1, 0, 1, 0))
pop(callcode(1, 1, 1, 1, 0, 1, 0))
pop(delegatecall(1, 1, 1, 0, 1, 0))
pop(staticcall(1, 1, 1, 0, 1, 0))
return(1, 0)
}
// ====
// EVMVersion: >=constantinople
// ----
// Trace:
// RETURNDATACOPY(0, 1, 0)
// CALLDATACOPY(0, 1, 0)
// EXTCODECOPY(1, 0, 1, 0)
// CODECOPY(0, 1, 0)
// LOG0(0, 0)
// LOG1(0, 0, 1)
// LOG2(0, 0, 1, 1)
// LOG3(0, 0, 1, 1, 1)
// LOG4(0, 0, 1, 1, 1, 1)
// CREATE(1, 0, 0)
// CREATE2(1, 0, 0, 1)
// CALL(1, 1, 1, 0, 0, 1, 0)
// CALLCODE(1, 1, 1, 0, 0, 1, 0)
// DELEGATECALL(1, 1, 0, 0, 1, 0)
// STATICCALL(1, 1, 0, 0, 1, 0)
// RETURN(0, 0)
// Memory dump:
// Storage dump:

View File

@ -0,0 +1,40 @@
{
returndatacopy(1, 1, 0)
calldatacopy(1, 1, 0)
extcodecopy(1, 1, 1, 0)
codecopy(1, 1, 0)
log0(1, 0)
log1(1, 0, 1)
log2(1, 0, 1, 1)
log3(1, 0, 1, 1, 1)
log4(1, 0, 1, 1, 1, 1)
pop(create(1, 1, 0))
pop(create2(1, 1, 0, 1))
pop(call(1, 1, 1, 1, 0, 1, 0))
pop(callcode(1, 1, 1, 1, 0, 1, 0))
pop(delegatecall(1, 1, 1, 0, 1, 0))
pop(staticcall(1, 1, 1, 0, 1, 0))
revert(1, 0)
}
// ====
// EVMVersion: >=constantinople
// ----
// Trace:
// RETURNDATACOPY(0, 1, 0)
// CALLDATACOPY(0, 1, 0)
// EXTCODECOPY(1, 0, 1, 0)
// CODECOPY(0, 1, 0)
// LOG0(0, 0)
// LOG1(0, 0, 1)
// LOG2(0, 0, 1, 1)
// LOG3(0, 0, 1, 1, 1)
// LOG4(0, 0, 1, 1, 1, 1)
// CREATE(1, 0, 0)
// CREATE2(1, 0, 0, 1)
// CALL(1, 1, 1, 0, 0, 1, 0)
// CALLCODE(1, 1, 1, 0, 0, 1, 0)
// DELEGATECALL(1, 1, 0, 0, 1, 0)
// STATICCALL(1, 1, 0, 0, 1, 0)
// REVERT(0, 0)
// Memory dump:
// Storage dump:

View File

@ -5,5 +5,6 @@
} }
// ---- // ----
// Trace: // Trace:
// CALLDATACOPY(0, 0, 0)
// Memory dump: // Memory dump:
// Storage dump: // Storage dump:

View File

@ -0,0 +1,45 @@
{
returndatacopy(1, 1, 0)
calldatacopy(1, 1, 0)
extcodecopy(1, 1, 1, 0)
codecopy(1, 1, 0)
log0(1, 0)
log1(1, 0, 1)
log2(1, 0, 1, 1)
log3(1, 0, 1, 1, 1)
log4(1, 0, 1, 1, 1, 1)
pop(create(1, 1, 0))
pop(create2(1, 1, 0, 1))
pop(call(1, 1, 1, 1, 0, 1, 0))
pop(callcode(1, 1, 1, 1, 0, 1, 0))
pop(delegatecall(1, 1, 1, 0, 1, 0))
pop(staticcall(1, 1, 1, 0, 1, 0))
return(1, 0)
}
// ====
// EVMVersion: >=constantinople
// ----
// step: loadResolver
//
// {
// {
// let _1 := 0
// let _2 := 1
// returndatacopy(0, _2, _1)
// calldatacopy(0, _2, _1)
// extcodecopy(_2, 0, _2, _1)
// codecopy(0, _2, _1)
// log0(0, _1)
// log1(0, _1, _2)
// log2(0, _1, _2, _2)
// log3(0, _1, _2, _2, _2)
// log4(0, _1, _2, _2, _2, _2)
// pop(create(_2, 0, _1))
// pop(create2(_2, 0, _1, _2))
// pop(call(_2, _2, _2, 0, _1, _2, _1))
// pop(callcode(_2, _2, _2, 0, _1, _2, _1))
// pop(delegatecall(_2, _2, 0, _1, _2, _1))
// pop(staticcall(_2, _2, 0, _1, _2, _1))
// return(0, _1)
// }
// }

View File

@ -216,7 +216,6 @@ u256 EVMInstructionInterpreter::eval(
m_state.memory, m_state.calldata, m_state.memory, m_state.calldata,
size_t(arg[0]), size_t(arg[1]), size_t(arg[2]) size_t(arg[0]), size_t(arg[1]), size_t(arg[2])
); );
if (arg[2] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::CODESIZE: case Instruction::CODESIZE:
@ -227,7 +226,6 @@ u256 EVMInstructionInterpreter::eval(
m_state.memory, m_state.code, m_state.memory, m_state.code,
size_t(arg[0]), size_t(arg[1]), size_t(arg[2]) size_t(arg[0]), size_t(arg[1]), size_t(arg[2])
); );
if (arg[2] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::GASPRICE: case Instruction::GASPRICE:
@ -247,7 +245,6 @@ u256 EVMInstructionInterpreter::eval(
m_state.memory, m_state.code, m_state.memory, m_state.code,
size_t(arg[1]), size_t(arg[2]), size_t(arg[3]) size_t(arg[1]), size_t(arg[2]), size_t(arg[3])
); );
if (arg[3] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::RETURNDATASIZE: case Instruction::RETURNDATASIZE:
@ -258,7 +255,6 @@ u256 EVMInstructionInterpreter::eval(
m_state.memory, m_state.returndata, m_state.memory, m_state.returndata,
size_t(arg[0]), size_t(arg[1]), size_t(arg[2]) size_t(arg[0]), size_t(arg[1]), size_t(arg[2])
); );
if (arg[2] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::BLOCKHASH: case Instruction::BLOCKHASH:
@ -301,53 +297,43 @@ u256 EVMInstructionInterpreter::eval(
return 0x99; return 0x99;
case Instruction::LOG0: case Instruction::LOG0:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::LOG1: case Instruction::LOG1:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::LOG2: case Instruction::LOG2:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::LOG3: case Instruction::LOG3:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
case Instruction::LOG4: case Instruction::LOG4:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
return 0; return 0;
// --------------- calls --------------- // --------------- calls ---------------
case Instruction::CREATE: case Instruction::CREATE:
accessMemory(arg[1], arg[2]); accessMemory(arg[1], arg[2]);
if (arg[2] != 0)
{
logTrace(_instruction, arg); logTrace(_instruction, arg);
if (arg[2] != 0)
return (0xcccccc + arg[1]) & u256("0xffffffffffffffffffffffffffffffffffffffff"); return (0xcccccc + arg[1]) & u256("0xffffffffffffffffffffffffffffffffffffffff");
} else
return 0xcccccc; return 0xcccccc;
case Instruction::CREATE2: case Instruction::CREATE2:
accessMemory(arg[1], arg[2]); accessMemory(arg[1], arg[2]);
if (arg[2] != 0)
{
logTrace(_instruction, arg); logTrace(_instruction, arg);
if (arg[2] != 0)
return (0xdddddd + arg[1]) & u256("0xffffffffffffffffffffffffffffffffffffffff"); return (0xdddddd + arg[1]) & u256("0xffffffffffffffffffffffffffffffffffffffff");
} else
return 0xdddddd; return 0xdddddd;
case Instruction::CALL: case Instruction::CALL:
case Instruction::CALLCODE: case Instruction::CALLCODE:
if (arg[4] != 0)
accessMemory(arg[3], arg[4]); accessMemory(arg[3], arg[4]);
if (arg[6] != 0)
accessMemory(arg[5], arg[6]); accessMemory(arg[5], arg[6]);
if (arg[4] != 0 && arg[6] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
// Randomly fail based on the called address if it isn't a call to self. // Randomly fail based on the called address if it isn't a call to self.
// Used for fuzzing. // Used for fuzzing.
@ -357,11 +343,8 @@ u256 EVMInstructionInterpreter::eval(
) ? 1 : 0; ) ? 1 : 0;
case Instruction::DELEGATECALL: case Instruction::DELEGATECALL:
case Instruction::STATICCALL: case Instruction::STATICCALL:
if (arg[3] != 0)
accessMemory(arg[2], arg[3]); accessMemory(arg[2], arg[3]);
if (arg[5] != 0)
accessMemory(arg[4], arg[5]); accessMemory(arg[4], arg[5]);
if (arg[3] != 0 && arg[5] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
// Randomly fail based on the called address if it isn't a call to self. // Randomly fail based on the called address if it isn't a call to self.
// Used for fuzzing. // Used for fuzzing.
@ -374,16 +357,13 @@ u256 EVMInstructionInterpreter::eval(
m_state.returndata = {}; m_state.returndata = {};
if (accessMemory(arg[0], arg[1])) if (accessMemory(arg[0], arg[1]))
m_state.returndata = m_state.readMemory(arg[0], arg[1]); m_state.returndata = m_state.readMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg, m_state.returndata); logTrace(_instruction, arg, m_state.returndata);
BOOST_THROW_EXCEPTION(ExplicitlyTerminatedWithReturn()); BOOST_THROW_EXCEPTION(ExplicitlyTerminatedWithReturn());
} }
case Instruction::REVERT: case Instruction::REVERT:
accessMemory(arg[0], arg[1]); accessMemory(arg[0], arg[1]);
if (arg[1] != 0)
logTrace(_instruction, arg); logTrace(_instruction, arg);
m_state.storage.clear(); m_state.storage.clear();
m_state.trace.clear();
BOOST_THROW_EXCEPTION(ExplicitlyTerminated()); BOOST_THROW_EXCEPTION(ExplicitlyTerminated());
case Instruction::INVALID: case Instruction::INVALID:
logTrace(_instruction); logTrace(_instruction);
@ -611,7 +591,7 @@ std::pair<bool, size_t> EVMInstructionInterpreter::isInputMemoryPtrModified(
std::vector<u256> const& _arguments std::vector<u256> const& _arguments
) )
{ {
if (_pseudoInstruction == "return" || _pseudoInstruction == "revert") if (_pseudoInstruction == "RETURN" || _pseudoInstruction == "REVERT")
{ {
if (_arguments[1] == 0) if (_arguments[1] == 0)
return {true, 0}; return {true, 0};
@ -619,15 +599,15 @@ std::pair<bool, size_t> EVMInstructionInterpreter::isInputMemoryPtrModified(
return {false, 0}; return {false, 0};
} }
else if ( else if (
_pseudoInstruction == "returndatacopy" || _pseudoInstruction == "calldatacopy" _pseudoInstruction == "RETURNDATACOPY" || _pseudoInstruction == "CALLDATACOPY"
|| _pseudoInstruction == "codecopy") || _pseudoInstruction == "CODECOPY")
{ {
if (_arguments[2] == 0) if (_arguments[2] == 0)
return {true, 0}; return {true, 0};
else else
return {false, 0}; return {false, 0};
} }
else if (_pseudoInstruction == "extcodedatacopy") else if (_pseudoInstruction == "EXTCODECOPY")
{ {
if (_arguments[3] == 0) if (_arguments[3] == 0)
return {true, 1}; return {true, 1};
@ -635,29 +615,29 @@ std::pair<bool, size_t> EVMInstructionInterpreter::isInputMemoryPtrModified(
return {false, 0}; return {false, 0};
} }
else if ( else if (
_pseudoInstruction == "log0" || _pseudoInstruction == "log1" || _pseudoInstruction == "log2" _pseudoInstruction == "LOG0" || _pseudoInstruction == "LOG1" || _pseudoInstruction == "LOG2"
|| _pseudoInstruction == "log3" || _pseudoInstruction == "log4") || _pseudoInstruction == "LOG3" || _pseudoInstruction == "LOG4")
{ {
if (_arguments[1] == 0) if (_arguments[1] == 0)
return {true, 0}; return {true, 0};
else else
return {false, 0}; return {false, 0};
} }
if (_pseudoInstruction == "create" || _pseudoInstruction == "create2") if (_pseudoInstruction == "CREATE" || _pseudoInstruction == "CREATE2")
{ {
if (_arguments[2] == 0) if (_arguments[2] == 0)
return {true, 1}; return {true, 1};
else else
return {false, 0}; return {false, 0};
} }
if (_pseudoInstruction == "call" || _pseudoInstruction == "callcode") if (_pseudoInstruction == "CALL" || _pseudoInstruction == "CALLCODE")
{ {
if (_arguments[4] == 0) if (_arguments[4] == 0)
return {true, 3}; return {true, 3};
else else
return {false, 0}; return {false, 0};
} }
else if (_pseudoInstruction == "delegatecall" || _pseudoInstruction == "staticcall") else if (_pseudoInstruction == "DELEGATECALL" || _pseudoInstruction == "STATICCALL")
{ {
if (_arguments[3] == 0) if (_arguments[3] == 0)
return {true, 2}; return {true, 2};