mirror of
https://github.com/ethereum/solidity
synced 2023-10-03 13:03:40 +00:00
Merge pull request #4129 from ethereum/doWhileContinue
[BREAKING] Fix continue inside do-while.
This commit is contained in:
commit
22ff3b736a
@ -4,6 +4,7 @@
|
|||||||
Breaking Changes:
|
Breaking Changes:
|
||||||
* Disallow conversions between bytesX and uintY of different size.
|
* Disallow conversions between bytesX and uintY of different size.
|
||||||
* Commandline interface: Require ``-`` if standard input is used as source.
|
* Commandline interface: Require ``-`` if standard input is used as source.
|
||||||
|
* General: ``continue`` in a ``do...while`` loop jumps to the condition (it used to jump to the loop body). Warning: this may silently change the semantics of existing code.
|
||||||
* Type Checker: Disallow arithmetic operations for Boolean variables.
|
* Type Checker: Disallow arithmetic operations for Boolean variables.
|
||||||
|
|
||||||
Features:
|
Features:
|
||||||
|
@ -159,15 +159,14 @@ bool ControlFlowBuilder::visit(WhileStatement const& _whileStatement)
|
|||||||
{
|
{
|
||||||
auto afterWhile = newLabel();
|
auto afterWhile = newLabel();
|
||||||
auto whileBody = createLabelHere();
|
auto whileBody = createLabelHere();
|
||||||
|
auto condition = newLabel();
|
||||||
|
|
||||||
{
|
{
|
||||||
// Note that "continue" in this case currently indeed jumps to whileBody
|
BreakContinueScope scope(*this, afterWhile, condition);
|
||||||
// and not to the condition. This is inconsistent with JavaScript and C and
|
|
||||||
// therefore a bug. This will be fixed in the future (planned for 0.5.0)
|
|
||||||
// and the Control Flow Graph will have to be adjusted accordingly.
|
|
||||||
BreakContinueScope scope(*this, afterWhile, whileBody);
|
|
||||||
appendControlFlow(_whileStatement.body());
|
appendControlFlow(_whileStatement.body());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
placeAndConnectLabel(condition);
|
||||||
appendControlFlow(_whileStatement.condition());
|
appendControlFlow(_whileStatement.condition());
|
||||||
|
|
||||||
connect(m_currentNode, whileBody);
|
connect(m_currentNode, whileBody);
|
||||||
|
@ -666,32 +666,36 @@ bool ContractCompiler::visit(WhileStatement const& _whileStatement)
|
|||||||
{
|
{
|
||||||
StackHeightChecker checker(m_context);
|
StackHeightChecker checker(m_context);
|
||||||
CompilerContext::LocationSetter locationSetter(m_context, _whileStatement);
|
CompilerContext::LocationSetter locationSetter(m_context, _whileStatement);
|
||||||
|
|
||||||
eth::AssemblyItem loopStart = m_context.newTag();
|
eth::AssemblyItem loopStart = m_context.newTag();
|
||||||
eth::AssemblyItem loopEnd = m_context.newTag();
|
eth::AssemblyItem loopEnd = m_context.newTag();
|
||||||
m_continueTags.push_back(loopStart);
|
|
||||||
m_breakTags.push_back(loopEnd);
|
m_breakTags.push_back(loopEnd);
|
||||||
|
|
||||||
m_context << loopStart;
|
m_context << loopStart;
|
||||||
|
|
||||||
// While loops have the condition prepended
|
|
||||||
if (!_whileStatement.isDoWhile())
|
|
||||||
{
|
|
||||||
compileExpression(_whileStatement.condition());
|
|
||||||
m_context << Instruction::ISZERO;
|
|
||||||
m_context.appendConditionalJumpTo(loopEnd);
|
|
||||||
}
|
|
||||||
|
|
||||||
_whileStatement.body().accept(*this);
|
|
||||||
|
|
||||||
// Do-while loops have the condition appended
|
|
||||||
if (_whileStatement.isDoWhile())
|
if (_whileStatement.isDoWhile())
|
||||||
{
|
{
|
||||||
|
eth::AssemblyItem condition = m_context.newTag();
|
||||||
|
m_continueTags.push_back(condition);
|
||||||
|
|
||||||
|
_whileStatement.body().accept(*this);
|
||||||
|
|
||||||
|
m_context << condition;
|
||||||
|
compileExpression(_whileStatement.condition());
|
||||||
|
m_context << Instruction::ISZERO << Instruction::ISZERO;
|
||||||
|
m_context.appendConditionalJumpTo(loopStart);
|
||||||
|
}
|
||||||
|
else
|
||||||
|
{
|
||||||
|
m_continueTags.push_back(loopStart);
|
||||||
compileExpression(_whileStatement.condition());
|
compileExpression(_whileStatement.condition());
|
||||||
m_context << Instruction::ISZERO;
|
m_context << Instruction::ISZERO;
|
||||||
m_context.appendConditionalJumpTo(loopEnd);
|
m_context.appendConditionalJumpTo(loopEnd);
|
||||||
}
|
|
||||||
|
|
||||||
m_context.appendJumpTo(loopStart);
|
_whileStatement.body().accept(*this);
|
||||||
|
|
||||||
|
m_context.appendJumpTo(loopStart);
|
||||||
|
}
|
||||||
m_context << loopEnd;
|
m_context << loopEnd;
|
||||||
|
|
||||||
m_continueTags.pop_back();
|
m_continueTags.pop_back();
|
||||||
|
@ -482,6 +482,27 @@ BOOST_AUTO_TEST_CASE(do_while_loop)
|
|||||||
testContractAgainstCppOnRange("f(uint256)", do_while_loop_cpp, 0, 5);
|
testContractAgainstCppOnRange("f(uint256)", do_while_loop_cpp, 0, 5);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
BOOST_AUTO_TEST_CASE(do_while_loop_continue)
|
||||||
|
{
|
||||||
|
char const* sourceCode = R"(
|
||||||
|
contract test {
|
||||||
|
function f() public pure returns(uint r) {
|
||||||
|
uint i = 0;
|
||||||
|
do
|
||||||
|
{
|
||||||
|
if (i > 0) return 0;
|
||||||
|
i++;
|
||||||
|
continue;
|
||||||
|
} while (false);
|
||||||
|
return 42;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)";
|
||||||
|
compileAndRun(sourceCode);
|
||||||
|
|
||||||
|
ABI_CHECK(callContractFunction("f()"), encodeArgs(42));
|
||||||
|
}
|
||||||
|
|
||||||
BOOST_AUTO_TEST_CASE(nested_loops)
|
BOOST_AUTO_TEST_CASE(nested_loops)
|
||||||
{
|
{
|
||||||
// tests that break and continue statements in nested loops jump to the correct place
|
// tests that break and continue statements in nested loops jump to the correct place
|
||||||
|
@ -23,13 +23,8 @@ contract C {
|
|||||||
}
|
}
|
||||||
function k() internal view returns (S storage c) {
|
function k() internal view returns (S storage c) {
|
||||||
do {
|
do {
|
||||||
if (s.f) {
|
c = s;
|
||||||
continue;
|
continue;
|
||||||
break;
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
c = s;
|
|
||||||
}
|
|
||||||
} while(false);
|
} while(false);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -21,6 +21,15 @@ contract C {
|
|||||||
do {
|
do {
|
||||||
if (s.f) {
|
if (s.f) {
|
||||||
break;
|
break;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
c = s;
|
||||||
|
}
|
||||||
|
} while(false);
|
||||||
|
}
|
||||||
|
function i() internal view returns (S storage c) {
|
||||||
|
do {
|
||||||
|
if (s.f) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@ -28,8 +37,16 @@ contract C {
|
|||||||
}
|
}
|
||||||
} while(false);
|
} while(false);
|
||||||
}
|
}
|
||||||
|
function j() internal view returns (S storage c) {
|
||||||
|
do {
|
||||||
|
continue;
|
||||||
|
c = s;
|
||||||
|
} while(false);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
// ----
|
// ----
|
||||||
// Warning: (87-98): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
// Warning: (87-98): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
||||||
// Warning: (223-234): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
// Warning: (223-234): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
||||||
// Warning: (440-451): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
// Warning: (440-451): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
||||||
|
// Warning: (654-665): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
||||||
|
// Warning: (871-882): This variable is of storage pointer type and might be returned without assignment. This can cause storage corruption. Assign the variable (potentially from itself) to remove this warning.
|
||||||
|
Loading…
Reference in New Issue
Block a user