Merge pull request #4129 from ethereum/doWhileContinue

[BREAKING] Fix continue inside do-while.
This commit is contained in:
chriseth 2018-05-15 15:04:19 +02:00
commit 22ff3b736a
6 changed files with 63 additions and 26 deletions

View File

@ -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:

View File

@ -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);

View File

@ -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();

View File

@ -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

View File

@ -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);
} }
} }

View File

@ -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.