mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
yulRun: Fix unexpected return value for call* instructions and add tests
This commit is contained in:
parent
b8699e7687
commit
d10d967008
@ -51,6 +51,7 @@ YulInterpreterTest::YulInterpreterTest(string const& _filename):
|
|||||||
{
|
{
|
||||||
m_source = m_reader.source();
|
m_source = m_reader.source();
|
||||||
m_expectation = m_reader.simpleExpectations();
|
m_expectation = m_reader.simpleExpectations();
|
||||||
|
m_simulateExternalCallsToSelf = m_reader.boolSetting("simulateExternalCall", false);
|
||||||
}
|
}
|
||||||
|
|
||||||
TestCase::TestResult YulInterpreterTest::run(ostream& _stream, string const& _linePrefix, bool const _formatted)
|
TestCase::TestResult YulInterpreterTest::run(ostream& _stream, string const& _linePrefix, bool const _formatted)
|
||||||
@ -98,8 +99,8 @@ string YulInterpreterTest::interpret()
|
|||||||
state,
|
state,
|
||||||
EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}),
|
EVMDialect::strictAssemblyForEVMObjects(langutil::EVMVersion{}),
|
||||||
*m_ast,
|
*m_ast,
|
||||||
/*disableExternalCalls=*/true,
|
/*disableExternalCalls=*/ !m_simulateExternalCallsToSelf,
|
||||||
/*disableMemoryTracing=*/false
|
/*disableMemoryTracing=*/ false
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
catch (InterpreterTerminatedGeneric const&)
|
catch (InterpreterTerminatedGeneric const&)
|
||||||
|
@ -47,6 +47,7 @@ private:
|
|||||||
|
|
||||||
std::shared_ptr<Block> m_ast;
|
std::shared_ptr<Block> m_ast;
|
||||||
std::shared_ptr<AsmAnalysisInfo> m_analysisInfo;
|
std::shared_ptr<AsmAnalysisInfo> m_analysisInfo;
|
||||||
|
bool m_simulateExternalCallsToSelf = false;
|
||||||
};
|
};
|
||||||
|
|
||||||
}
|
}
|
||||||
|
22
test/libyul/yulInterpreterTests/external_call_to_self.yul
Normal file
22
test/libyul/yulInterpreterTests/external_call_to_self.yul
Normal file
@ -0,0 +1,22 @@
|
|||||||
|
{
|
||||||
|
mstore(0x40, 0x42)
|
||||||
|
|
||||||
|
if iszero(calldatasize()) {
|
||||||
|
let x := call(gas(), address(), 0, 0x40, 0x20, 0x100, 0x20)
|
||||||
|
sstore(0x64, calldataload(0))
|
||||||
|
sstore(0x100, x)
|
||||||
|
return(0x0, 0)
|
||||||
|
}
|
||||||
|
return(0x40, 0x20)
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// simulateExternalCall: true
|
||||||
|
// ----
|
||||||
|
// Trace:
|
||||||
|
// CALL(153, 0x11111111, 0, 64, 32, 256, 32)
|
||||||
|
// RETURN(0, 0)
|
||||||
|
// Memory dump:
|
||||||
|
// 40: 0000000000000000000000000000000000000000000000000000000000000042
|
||||||
|
// 100: 0000000000000000000000000000000000000000000000000000000000000042
|
||||||
|
// Storage dump:
|
||||||
|
// 0000000000000000000000000000000000000000000000000000000000000100: 0000000000000000000000000000000000000000000000000000000000000001
|
@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
let x := call(gas(), 0x45, 0x5, 0, 0x20, 0x30, 0x20)
|
let x := call(gas(), 0x45, 0x5, 0, 0x20, 0x30, 0x20)
|
||||||
sstore(100, x)
|
sstore(0x64, x)
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Trace:
|
// Trace:
|
@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
let x := callcode(gas(), 0x45, 0x5, 0, 0x20, 0x30, 0x20)
|
||||||
|
sstore(100, x)
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Trace:
|
||||||
|
// CALLCODE(153, 69, 5, 0, 32, 48, 32)
|
||||||
|
// Memory dump:
|
||||||
|
// Storage dump:
|
||||||
|
// 0000000000000000000000000000000000000000000000000000000000000064: 0000000000000000000000000000000000000000000000000000000000000001
|
@ -0,0 +1,10 @@
|
|||||||
|
{
|
||||||
|
let x := delegatecall(gas(), 0x45, 0, 0x20, 0x30, 0x20)
|
||||||
|
sstore(100, x)
|
||||||
|
}
|
||||||
|
// ----
|
||||||
|
// Trace:
|
||||||
|
// DELEGATECALL(153, 69, 0, 32, 48, 32)
|
||||||
|
// Memory dump:
|
||||||
|
// Storage dump:
|
||||||
|
// 0000000000000000000000000000000000000000000000000000000000000064: 0000000000000000000000000000000000000000000000000000000000000001
|
@ -0,0 +1,12 @@
|
|||||||
|
{
|
||||||
|
let x := staticcall(gas(), 0x45, 0, 0x20, 0x30, 0x20)
|
||||||
|
sstore(0x64, x)
|
||||||
|
}
|
||||||
|
// ====
|
||||||
|
// EVMVersion: >=byzantium
|
||||||
|
// ----
|
||||||
|
// Trace:
|
||||||
|
// STATICCALL(153, 69, 0, 32, 48, 32)
|
||||||
|
// Memory dump:
|
||||||
|
// Storage dump:
|
||||||
|
// 0000000000000000000000000000000000000000000000000000000000000064: 0000000000000000000000000000000000000000000000000000000000000001
|
@ -329,13 +329,24 @@ u256 EVMInstructionInterpreter::eval(
|
|||||||
accessMemory(arg[3], arg[4]);
|
accessMemory(arg[3], arg[4]);
|
||||||
accessMemory(arg[5], arg[6]);
|
accessMemory(arg[5], arg[6]);
|
||||||
logTrace(_instruction, arg);
|
logTrace(_instruction, arg);
|
||||||
return arg[0] & 1;
|
// Randomly fail based on the called address if it isn't a call to self.
|
||||||
|
// Used for fuzzing.
|
||||||
|
return (
|
||||||
|
(arg[0] > 0) &&
|
||||||
|
(arg[1] == util::h160::Arith(m_state.address) || (arg[1] & 1))
|
||||||
|
) ? 1 : 0;
|
||||||
case Instruction::DELEGATECALL:
|
case Instruction::DELEGATECALL:
|
||||||
case Instruction::STATICCALL:
|
case Instruction::STATICCALL:
|
||||||
accessMemory(arg[2], arg[3]);
|
accessMemory(arg[2], arg[3]);
|
||||||
accessMemory(arg[4], arg[5]);
|
accessMemory(arg[4], arg[5]);
|
||||||
logTrace(_instruction, arg);
|
logTrace(_instruction, arg);
|
||||||
return 0;
|
|
||||||
|
// Randomly fail based on the called address if it isn't a call to self.
|
||||||
|
// Used for fuzzing.
|
||||||
|
return (
|
||||||
|
(arg[0] > 0) &&
|
||||||
|
(arg[1] == util::h160::Arith(m_state.address) || (arg[1] & 1))
|
||||||
|
) ? 1 : 0;
|
||||||
case Instruction::RETURN:
|
case Instruction::RETURN:
|
||||||
{
|
{
|
||||||
m_state.returndata = {};
|
m_state.returndata = {};
|
||||||
|
@ -500,5 +500,6 @@ void ExpressionEvaluator::runExternalCall(evmasm::Instruction _instruction)
|
|||||||
0,
|
0,
|
||||||
memOutSize.convert_to<size_t>()
|
memOutSize.convert_to<size_t>()
|
||||||
);
|
);
|
||||||
|
m_state.returndata = newInterpreter->returnData();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user