[SMTChecker] Fix ice in unsupported functions with multi return values

This commit is contained in:
Leonardo Alt 2019-05-14 23:10:56 +02:00
parent 54ce3df321
commit 60a4f03d3d
10 changed files with 69 additions and 27 deletions

View File

@ -335,7 +335,9 @@ void SMTChecker::endVisit(VariableDeclarationStatement const& _varDecl)
solAssert(symbTuple, ""); solAssert(symbTuple, "");
auto const& components = symbTuple->components(); auto const& components = symbTuple->components();
auto const& declarations = _varDecl.declarations(); auto const& declarations = _varDecl.declarations();
if (components.size() == declarations.size()) if (!components.empty())
{
solAssert(components.size() == declarations.size(), "");
for (unsigned i = 0; i < declarations.size(); ++i) for (unsigned i = 0; i < declarations.size(); ++i)
if ( if (
components.at(i) && components.at(i) &&
@ -343,6 +345,7 @@ void SMTChecker::endVisit(VariableDeclarationStatement const& _varDecl)
m_context.knownVariable(*declarations.at(i)) m_context.knownVariable(*declarations.at(i))
) )
assignment(*declarations.at(i), components.at(i)->currentValue(), declarations.at(i)->location()); assignment(*declarations.at(i), components.at(i)->currentValue(), declarations.at(i)->location());
}
} }
} }
else if (m_context.knownVariable(*_varDecl.declarations().front())) else if (m_context.knownVariable(*_varDecl.declarations().front()))
@ -633,6 +636,7 @@ void SMTChecker::endVisit(BinaryOperation const& _op)
void SMTChecker::endVisit(FunctionCall const& _funCall) void SMTChecker::endVisit(FunctionCall const& _funCall)
{ {
solAssert(_funCall.annotation().kind != FunctionCallKind::Unset, ""); solAssert(_funCall.annotation().kind != FunctionCallKind::Unset, "");
createExpr(_funCall);
if (_funCall.annotation().kind == FunctionCallKind::StructConstructorCall) if (_funCall.annotation().kind == FunctionCallKind::StructConstructorCall)
{ {
m_errorReporter.warning( m_errorReporter.warning(
@ -694,7 +698,6 @@ void SMTChecker::endVisit(FunctionCall const& _funCall)
checkCondition(thisBalance < expr(*value), _funCall.location(), "Insufficient funds", "address(this).balance", &thisBalance); checkCondition(thisBalance < expr(*value), _funCall.location(), "Insufficient funds", "address(this).balance", &thisBalance);
m_context.transfer(m_context.thisAddress(), expr(address), expr(*value)); m_context.transfer(m_context.thisAddress(), expr(address), expr(*value));
createExpr(_funCall);
break; break;
} }
default: default:
@ -741,14 +744,8 @@ void SMTChecker::visitGasLeft(FunctionCall const& _funCall)
void SMTChecker::inlineFunctionCall(FunctionCall const& _funCall) void SMTChecker::inlineFunctionCall(FunctionCall const& _funCall)
{ {
FunctionDefinition const* funDef = inlinedFunctionCallToDefinition(_funCall); FunctionDefinition const* funDef = inlinedFunctionCallToDefinition(_funCall);
if (!funDef) solAssert(funDef, "");
{ if (visitedFunction(funDef))
m_errorReporter.warning(
_funCall.location(),
"Assertion checker does not yet implement this type of function call."
);
}
else if (visitedFunction(funDef))
m_errorReporter.warning( m_errorReporter.warning(
_funCall.location(), _funCall.location(),
"Assertion checker does not support recursive function calls.", "Assertion checker does not support recursive function calls.",
@ -779,7 +776,6 @@ void SMTChecker::inlineFunctionCall(FunctionCall const& _funCall)
// The callstack entry is popped only in endVisit(FunctionDefinition) instead of here // The callstack entry is popped only in endVisit(FunctionDefinition) instead of here
// as well to avoid code duplication (not all entries are from inlined function calls). // as well to avoid code duplication (not all entries are from inlined function calls).
createExpr(_funCall);
auto const& returnParams = funDef->returnParameters(); auto const& returnParams = funDef->returnParameters();
if (returnParams.size() > 1) if (returnParams.size() > 1)
{ {
@ -866,7 +862,6 @@ void SMTChecker::visitTypeConversion(FunctionCall const& _funCall)
defineExpr(_funCall, expr(*argument)); defineExpr(_funCall, expr(*argument));
else else
{ {
createExpr(_funCall);
m_context.setUnknownValue(*m_context.expression(_funCall)); m_context.setUnknownValue(*m_context.expression(_funCall));
auto const& funCallCategory = _funCall.annotation().type->category(); auto const& funCallCategory = _funCall.annotation().type->category();
// TODO: truncating and bytesX needs a different approach because of right padding. // TODO: truncating and bytesX needs a different approach because of right padding.
@ -1365,10 +1360,13 @@ void SMTChecker::assignment(
else if (auto tuple = dynamic_cast<TupleExpression const*>(&_left)) else if (auto tuple = dynamic_cast<TupleExpression const*>(&_left))
{ {
auto const& components = tuple->components(); auto const& components = tuple->components();
solAssert(_right.size() == components.size(), ""); if (!_right.empty())
for (unsigned i = 0; i < _right.size(); ++i) {
if (auto component = components.at(i)) solAssert(_right.size() == components.size(), "");
assignment(*component, {_right.at(i)}, component->annotation().type, component->location()); for (unsigned i = 0; i < _right.size(); ++i)
if (auto component = components.at(i))
assignment(*component, {_right.at(i)}, component->annotation().type, component->location());
}
} }
else else
m_errorReporter.warning( m_errorReporter.warning(

View File

@ -36,8 +36,6 @@ library MerkleProof {
// ---- // ----
// Warning: (755-767): Assertion checker does not yet support this expression. // Warning: (755-767): Assertion checker does not yet support this expression.
// Warning: (988-1032): Assertion checker does not yet implement this type of function call. // Warning: (988-1032): Assertion checker does not yet implement this type of function call.
// Warning: (988-1032): Internal error: Expression undefined for SMT solver.
// Warning: (1175-1219): Assertion checker does not yet implement this type of function call. // Warning: (1175-1219): Assertion checker does not yet implement this type of function call.
// Warning: (1175-1219): Internal error: Expression undefined for SMT solver.
// Warning: (755-767): Assertion checker does not yet support this expression. // Warning: (755-767): Assertion checker does not yet support this expression.
// Warning: (769-772): Overflow (resulting value larger than 2**256 - 1) happens here // Warning: (769-772): Overflow (resulting value larger than 2**256 - 1) happens here

View File

@ -83,12 +83,10 @@ contract InternalCall {
// Warning: (1144-1206): Function state mutability can be restricted to pure // Warning: (1144-1206): Function state mutability can be restricted to pure
// Warning: (1212-1274): Function state mutability can be restricted to pure // Warning: (1212-1274): Function state mutability can be restricted to pure
// Warning: (1280-1342): Function state mutability can be restricted to pure // Warning: (1280-1342): Function state mutability can be restricted to pure
// Warning: (714-749): Internal error: Expression undefined for SMT solver.
// Warning: (799-811): Assertion checker does not yet support the type of this literal (literal_string "helloTwo()"). // Warning: (799-811): Assertion checker does not yet support the type of this literal (literal_string "helloTwo()").
// Warning: (782-813): Type conversion is not yet fully supported and might yield false positives. // Warning: (782-813): Type conversion is not yet fully supported and might yield false positives.
// Warning: (771-814): Assertion checker does not yet implement this type of function call. // Warning: (771-814): Assertion checker does not yet implement this type of function call.
// Warning: (825-830): Assertion checker does not yet support the type of this variable. // Warning: (825-830): Assertion checker does not yet support the type of this variable.
// Warning: (887-919): Internal error: Expression undefined for SMT solver.
// Warning: (690-750): Underflow (resulting value less than 0) happens here // Warning: (690-750): Underflow (resulting value less than 0) happens here
// Warning: (690-750): Overflow (resulting value larger than 2**160 - 1) happens here // Warning: (690-750): Overflow (resulting value larger than 2**160 - 1) happens here
// Warning: (1057-1068): Assertion checker does not yet implement type function () returns (uint256) // Warning: (1057-1068): Assertion checker does not yet implement type function () returns (uint256)

View File

@ -9,6 +9,5 @@ contract C {
// ---- // ----
// Warning: (133-143): Unused local variable. // Warning: (133-143): Unused local variable.
// Warning: (133-143): Assertion checker does not yet support the type of this variable. // Warning: (133-143): Assertion checker does not yet support the type of this variable.
// Warning: (146-163): Assertion checker does not yet implement this expression.
// Warning: (146-163): Internal error: Expression undefined for SMT solver.
// Warning: (146-163): Assertion checker does not yet implement this type. // Warning: (146-163): Assertion checker does not yet implement this type.
// Warning: (146-163): Assertion checker does not yet implement this expression.

View File

@ -0,0 +1,12 @@
pragma experimental SMTChecker;pragma experimental ABIEncoderV2;
contract C {
function f() public pure returns (bytes memory, bytes memory) {
return (abi.encode(""), abi.encodePacked( "7?8r"));
}
}
// ----
// Warning: (31-64): Experimental features are turned on. Do not use experimental features on live deployments.
// Warning: (173-175): Assertion checker does not yet support the type of this literal (literal_string "").
// Warning: (162-176): Assertion checker does not yet implement this type of function call.
// Warning: (196-202): Assertion checker does not yet support the type of this literal (literal_string "7?8r").
// Warning: (178-203): Assertion checker does not yet implement this type of function call.

View File

@ -16,5 +16,4 @@ contract D
} }
} }
// ---- // ----
// Warning: (180-187): Internal error: Expression undefined for SMT solver.
// Warning: (191-206): Assertion violation happens here // Warning: (191-206): Assertion violation happens here

View File

@ -0,0 +1,12 @@
pragma experimental SMTChecker;
contract C {
function g() public pure returns (uint, uint) {
uint a;
uint b;
(a, b) = g();
}
}
//
// ----
// Warning: (126-129): Assertion checker does not support recursive function calls.

View File

@ -0,0 +1,28 @@
pragma experimental SMTChecker;
contract c { function f() public pure {22237625-86535-0+1;
555565-3*51;
}
}contract C {
function g() public pure returns (
uint,
uint,
uint,
uint,
uint,
uint,
uint,
uint,
uint,
uint,
int, uint, bytes14) {
uint
a;
uint b;
(,,,,,,,a,b,,,,) = g();
}
}
// ----
// Warning: (72-90): Statement has no effect.
// Warning: (96-107): Statement has no effect.
// Warning: (304-307): Assertion checker does not support recursive function calls.

View File

@ -18,7 +18,5 @@ contract C
} }
// ---- // ----
// Warning: (86-93): Assertion checker does not support recursive function calls. // Warning: (86-93): Assertion checker does not support recursive function calls.
// Warning: (86-93): Internal error: Expression undefined for SMT solver.
// Warning: (86-93): Assertion checker does not support recursive function calls. // Warning: (86-93): Assertion checker does not support recursive function calls.
// Warning: (86-93): Internal error: Expression undefined for SMT solver.
// Warning: (253-266): Assertion violation happens here // Warning: (253-266): Assertion violation happens here

View File

@ -17,8 +17,8 @@ contract C
// Warning: (157-170): Unused local variable. // Warning: (157-170): Unused local variable.
// Warning: (157-170): Assertion checker does not yet support the type of this variable. // Warning: (157-170): Assertion checker does not yet support the type of this variable.
// Warning: (139-146): Assertion checker does not yet implement this type. // Warning: (139-146): Assertion checker does not yet implement this type.
// Warning: (149-153): Assertion checker does not yet implement this type.
// Warning: (149-153): Assertion checker does not yet implement this expression. // Warning: (149-153): Assertion checker does not yet implement this expression.
// Warning: (139-153): Assertion checker does not yet implement type struct C.S storage ref // Warning: (139-153): Assertion checker does not yet implement type struct C.S storage ref
// Warning: (173-177): Assertion checker does not yet implement this expression.
// Warning: (173-177): Internal error: Expression undefined for SMT solver.
// Warning: (173-177): Assertion checker does not yet implement this type. // Warning: (173-177): Assertion checker does not yet implement this type.
// Warning: (173-177): Assertion checker does not yet implement this expression.