From d81ebe97c3c89ce3f333a267b75b21ccbda8ae42 Mon Sep 17 00:00:00 2001 From: Leo Alt Date: Mon, 27 Sep 2021 16:55:59 +0200 Subject: [PATCH] Fix magic access --- Changelog.md | 1 + libsolidity/formal/SMTEncoder.cpp | 64 +++++++++++-------- .../smtCheckerTests/special/msg_parens_1.sol | 12 ++++ 3 files changed, 50 insertions(+), 27 deletions(-) create mode 100644 test/libsolidity/smtCheckerTests/special/msg_parens_1.sol diff --git a/Changelog.md b/Changelog.md index 96c6dd12e..70e948b88 100644 --- a/Changelog.md +++ b/Changelog.md @@ -7,6 +7,7 @@ Compiler Features: Bugfixes: + * SMTChecker: Fix internal error in magic type access (``block``, ``msg``, ``tx``). diff --git a/libsolidity/formal/SMTEncoder.cpp b/libsolidity/formal/SMTEncoder.cpp index e011b1756..11ac5ab7a 100644 --- a/libsolidity/formal/SMTEncoder.cpp +++ b/libsolidity/formal/SMTEncoder.cpp @@ -1316,32 +1316,42 @@ bool SMTEncoder::visit(MemberAccess const& _memberAccess) solAssert(name == "block" || name == "msg" || name == "tx", ""); defineExpr(_memberAccess, state().txMember(name + "." + _memberAccess.memberName())); } - else if (auto magicType = dynamic_cast(exprType); magicType->kind() == MagicType::Kind::MetaType) + else if (auto magicType = dynamic_cast(exprType)) { - auto const& memberName = _memberAccess.memberName(); - if (memberName == "min" || memberName == "max") + if (magicType->kind() == MagicType::Kind::Block) + defineExpr(_memberAccess, state().txMember("block." + _memberAccess.memberName())); + else if (magicType->kind() == MagicType::Kind::Message) + defineExpr(_memberAccess, state().txMember("msg." + _memberAccess.memberName())); + else if (magicType->kind() == MagicType::Kind::Transaction) + defineExpr(_memberAccess, state().txMember("tx." + _memberAccess.memberName())); + else if (magicType->kind() == MagicType::Kind::MetaType) { - if (IntegerType const* integerType = dynamic_cast(magicType->typeArgument())) - defineExpr(_memberAccess, memberName == "min" ? integerType->minValue() : integerType->maxValue()); - else if (EnumType const* enumType = dynamic_cast(magicType->typeArgument())) - defineExpr(_memberAccess, memberName == "min" ? enumType->minValue() : enumType->maxValue()); + auto const& memberName = _memberAccess.memberName(); + if (memberName == "min" || memberName == "max") + { + if (IntegerType const* integerType = dynamic_cast(magicType->typeArgument())) + defineExpr(_memberAccess, memberName == "min" ? integerType->minValue() : integerType->maxValue()); + else if (EnumType const* enumType = dynamic_cast(magicType->typeArgument())) + defineExpr(_memberAccess, memberName == "min" ? enumType->minValue() : enumType->maxValue()); + } + else if (memberName == "interfaceId") + { + ContractDefinition const& contract = dynamic_cast(*magicType->typeArgument()).contractDefinition(); + defineExpr(_memberAccess, contract.interfaceId()); + } + else + // NOTE: supporting name, creationCode, runtimeCode would be easy enough, but the bytes/string they return are not + // at all usable in the SMT checker currently + m_errorReporter.warning( + 7507_error, + _memberAccess.location(), + "Assertion checker does not yet support this expression." + ); + } - else if (memberName == "interfaceId") - { - ContractDefinition const& contract = dynamic_cast(*magicType->typeArgument()).contractDefinition(); - defineExpr(_memberAccess, contract.interfaceId()); - } - else - // NOTE: supporting name, creationCode, runtimeCode would be easy enough, but the bytes/string they return are not - // at all usable in the SMT checker currently - m_errorReporter.warning( - 7507_error, - _memberAccess.location(), - "Assertion checker does not yet support this expression." - ); } else - solUnimplementedAssert(false, ""); + solAssert(false, ""); return false; } @@ -1419,12 +1429,12 @@ bool SMTEncoder::visit(MemberAccess const& _memberAccess) return false; } } - else - m_errorReporter.warning( - 7650_error, - _memberAccess.location(), - "Assertion checker does not yet support this expression." - ); + + m_errorReporter.warning( + 7650_error, + _memberAccess.location(), + "Assertion checker does not yet support this expression." + ); return true; } diff --git a/test/libsolidity/smtCheckerTests/special/msg_parens_1.sol b/test/libsolidity/smtCheckerTests/special/msg_parens_1.sol new file mode 100644 index 000000000..9a1df3724 --- /dev/null +++ b/test/libsolidity/smtCheckerTests/special/msg_parens_1.sol @@ -0,0 +1,12 @@ +contract C { + function f() public payable { + assert((msg).value == 10); + assert((true ? msg : msg).value == 12); + } +} +// ==== +// SMTEngine: all +// SMTIgnoreOS: macos +// ---- +// Warning 6328: (46-71): CHC: Assertion violation happens here.\nCounterexample:\n\n\nTransaction trace:\nC.constructor()\nC.f(){ value: 11 } +// Warning 6328: (75-113): CHC: Assertion violation happens here.\nCounterexample:\n\n\nTransaction trace:\nC.constructor()\nC.f()