From a8209e9899f1d1b5fa8edab0b5c4c9b6d81bb30d Mon Sep 17 00:00:00 2001 From: Leonardo Alt Date: Mon, 11 Mar 2019 11:56:08 +0100 Subject: [PATCH] [SMTChecker] Shortcut RationalNumber expressions --- Changelog.md | 1 + libsolidity/formal/SMTChecker.cpp | 31 +++++++++++++++++++ libsolidity/formal/SMTChecker.h | 5 +++ libsolidity/formal/SolverInterface.h | 1 + .../smtCheckerTests/simple/static_array.sol | 1 - .../types/rational_large_1.sol | 10 ++++++ 6 files changed, 48 insertions(+), 1 deletion(-) create mode 100644 test/libsolidity/smtCheckerTests/types/rational_large_1.sol diff --git a/Changelog.md b/Changelog.md index 30994ebc1..883e555b7 100644 --- a/Changelog.md +++ b/Changelog.md @@ -14,6 +14,7 @@ Compiler Features: Bugfixes: + * SMTChecker: Fix internal compiler error when contract contains too large rational number. * Type system: Detect if a contract's base uses types that require the experimental abi encoder while the contract still uses the old encoder * Yul Optimizer: Fix visitation order bug for the structural simplifier. diff --git a/libsolidity/formal/SMTChecker.cpp b/libsolidity/formal/SMTChecker.cpp index 5853eb97a..d9e234021 100644 --- a/libsolidity/formal/SMTChecker.cpp +++ b/libsolidity/formal/SMTChecker.cpp @@ -374,6 +374,9 @@ void SMTChecker::checkOverflow(OverflowTarget& _target) void SMTChecker::endVisit(UnaryOperation const& _op) { + if (_op.annotation().type->category() == Type::Category::RationalNumber) + return; + switch (_op.getOperator()) { case Token::Not: // ! @@ -431,8 +434,21 @@ void SMTChecker::endVisit(UnaryOperation const& _op) } } +bool SMTChecker::visit(UnaryOperation const& _op) +{ + return !shortcutRationalNumber(_op); +} + +bool SMTChecker::visit(BinaryOperation const& _op) +{ + return !shortcutRationalNumber(_op); +} + void SMTChecker::endVisit(BinaryOperation const& _op) { + if (_op.annotation().type->category() == Type::Category::RationalNumber) + return; + if (TokenTraits::isArithmeticOp(_op.getOperator())) arithmeticOperation(_op); else if (TokenTraits::isCompareOp(_op.getOperator())) @@ -908,6 +924,21 @@ void SMTChecker::defineGlobalFunction(string const& _name, Expression const& _ex } } +bool SMTChecker::shortcutRationalNumber(Expression const& _expr) +{ + if (_expr.annotation().type->category() == Type::Category::RationalNumber) + { + auto rationalType = dynamic_cast(_expr.annotation().type.get()); + solAssert(rationalType, ""); + if (rationalType->isNegative()) + defineExpr(_expr, smt::Expression(u2s(rationalType->literalValue(nullptr)))); + else + defineExpr(_expr, smt::Expression(rationalType->literalValue(nullptr))); + return true; + } + return false; +} + void SMTChecker::arithmeticOperation(BinaryOperation const& _op) { switch (_op.getOperator()) diff --git a/libsolidity/formal/SMTChecker.h b/libsolidity/formal/SMTChecker.h index f11f156e7..4039166d7 100644 --- a/libsolidity/formal/SMTChecker.h +++ b/libsolidity/formal/SMTChecker.h @@ -71,7 +71,9 @@ private: void endVisit(VariableDeclarationStatement const& _node) override; void endVisit(Assignment const& _node) override; void endVisit(TupleExpression const& _node) override; + bool visit(UnaryOperation const& _node) override; void endVisit(UnaryOperation const& _node) override; + bool visit(BinaryOperation const& _node) override; void endVisit(BinaryOperation const& _node) override; void endVisit(FunctionCall const& _node) override; void endVisit(Identifier const& _node) override; @@ -80,6 +82,9 @@ private: bool visit(MemberAccess const& _node) override; void endVisit(IndexAccess const& _node) override; + /// Do not visit subtree if node is a RationalNumber. + /// Symbolic _expr is the rational literal. + bool shortcutRationalNumber(Expression const& _expr); void arithmeticOperation(BinaryOperation const& _op); void compareOperation(BinaryOperation const& _op); void booleanOperation(BinaryOperation const& _op); diff --git a/libsolidity/formal/SolverInterface.h b/libsolidity/formal/SolverInterface.h index 80b48e0f6..042a16ab6 100644 --- a/libsolidity/formal/SolverInterface.h +++ b/libsolidity/formal/SolverInterface.h @@ -118,6 +118,7 @@ public: explicit Expression(bool _v): Expression(_v ? "true" : "false", Kind::Bool) {} Expression(size_t _number): Expression(std::to_string(_number), Kind::Int) {} Expression(u256 const& _number): Expression(_number.str(), Kind::Int) {} + Expression(s256 const& _number): Expression(_number.str(), Kind::Int) {} Expression(bigint const& _number): Expression(_number.str(), Kind::Int) {} Expression(Expression const&) = default; diff --git a/test/libsolidity/smtCheckerTests/simple/static_array.sol b/test/libsolidity/smtCheckerTests/simple/static_array.sol index be114fbb5..4147c0169 100644 --- a/test/libsolidity/smtCheckerTests/simple/static_array.sol +++ b/test/libsolidity/smtCheckerTests/simple/static_array.sol @@ -7,4 +7,3 @@ contract C int[3*1] x; } // ---- -// Warning: (153-156): Assertion checker does not yet implement this operator on non-integer types. diff --git a/test/libsolidity/smtCheckerTests/types/rational_large_1.sol b/test/libsolidity/smtCheckerTests/types/rational_large_1.sol new file mode 100644 index 000000000..152b29f09 --- /dev/null +++ b/test/libsolidity/smtCheckerTests/types/rational_large_1.sol @@ -0,0 +1,10 @@ +pragma experimental SMTChecker; +contract c { + function f() public pure returns (uint) { + uint x = 8e130%9; + assert(x == 8); + assert(x != 8); + } +} +// ---- +// Warning: (128-142): Assertion violation happens here