diff --git a/docs/control-structures.rst b/docs/control-structures.rst index a21de757d..0559e6546 100644 --- a/docs/control-structures.rst +++ b/docs/control-structures.rst @@ -44,6 +44,7 @@ this nonsensical example:: // SPDX-License-Identifier: GPL-3.0 pragma solidity >=0.4.22 <0.9.0; + // This will report a warning contract C { function g(uint a) public pure returns (uint ret) { return a + f(); } function f() internal pure returns (uint ret) { return g(7) + f(); } diff --git a/libsolidity/analysis/ControlFlowBuilder.cpp b/libsolidity/analysis/ControlFlowBuilder.cpp index bbd1c73bb..98dcbf069 100644 --- a/libsolidity/analysis/ControlFlowBuilder.cpp +++ b/libsolidity/analysis/ControlFlowBuilder.cpp @@ -260,6 +260,7 @@ bool ControlFlowBuilder::visit(PlaceholderStatement const&) bool ControlFlowBuilder::visit(FunctionCall const& _functionCall) { + solAssert(!!m_revertNode, ""); solAssert(!!m_currentNode, ""); solAssert(!!_functionCall.expression().annotation().type, ""); @@ -267,30 +268,43 @@ bool ControlFlowBuilder::visit(FunctionCall const& _functionCall) switch (functionType->kind()) { case FunctionType::Kind::Revert: - solAssert(!!m_revertNode, ""); visitNode(_functionCall); _functionCall.expression().accept(*this); ASTNode::listAccept(_functionCall.arguments(), *this); + connect(m_currentNode, m_revertNode); + m_currentNode = newLabel(); return false; case FunctionType::Kind::Require: case FunctionType::Kind::Assert: { - solAssert(!!m_revertNode, ""); visitNode(_functionCall); _functionCall.expression().accept(*this); ASTNode::listAccept(_functionCall.arguments(), *this); + connect(m_currentNode, m_revertNode); + auto nextNode = newLabel(); + connect(m_currentNode, nextNode); m_currentNode = nextNode; return false; } case FunctionType::Kind::Internal: { + visitNode(_functionCall); + _functionCall.expression().accept(*this); + ASTNode::listAccept(_functionCall.arguments(), *this); + m_currentNode->functionCalls.emplace_back(&_functionCall); - break; + + auto nextNode = newLabel(); + + connect(m_currentNode, nextNode); + m_currentNode = nextNode; + + return false; } default: break; diff --git a/libsolidity/analysis/ControlFlowRevertPruner.cpp b/libsolidity/analysis/ControlFlowRevertPruner.cpp index aedf5aed8..af301cc72 100644 --- a/libsolidity/analysis/ControlFlowRevertPruner.cpp +++ b/libsolidity/analysis/ControlFlowRevertPruner.cpp @@ -20,6 +20,8 @@ #include +#include + namespace solidity::frontend { @@ -192,7 +194,11 @@ void ControlFlowRevertPruner::modifyFunctionFlows() // change anymore, we treat all "unknown" states as // "reverting", since they can only be caused by // recursion. + for (CFGNode * node: _node->exits) + ranges::remove(node->entries, _node); + _node->exits = {functionFlow.revert}; + functionFlow.revert->entries.push_back(_node); return; default: break; diff --git a/test/libsolidity/smtCheckerTests/functions/function_call_does_not_clear_local_vars.sol b/test/libsolidity/smtCheckerTests/functions/function_call_does_not_clear_local_vars.sol index 8d21f58fa..7fd4fcff1 100644 --- a/test/libsolidity/smtCheckerTests/functions/function_call_does_not_clear_local_vars.sol +++ b/test/libsolidity/smtCheckerTests/functions/function_call_does_not_clear_local_vars.sol @@ -10,3 +10,4 @@ contract C { // ==== // SMTEngine: all // ---- +// Warning 5740: (122-136): Unreachable code. diff --git a/test/libsolidity/syntaxTests/multiVariableDeclaration/multiSingleVariableDeclaration.sol b/test/libsolidity/syntaxTests/multiVariableDeclaration/multiSingleVariableDeclaration.sol index 428428bee..b08ca8fc0 100644 --- a/test/libsolidity/syntaxTests/multiVariableDeclaration/multiSingleVariableDeclaration.sol +++ b/test/libsolidity/syntaxTests/multiVariableDeclaration/multiSingleVariableDeclaration.sol @@ -5,3 +5,4 @@ contract C { } } // ---- +// Warning 5740: (78-79): Unreachable code. diff --git a/test/libsolidity/syntaxTests/multiVariableDeclaration/multiVariableDeclarationSimple.sol b/test/libsolidity/syntaxTests/multiVariableDeclaration/multiVariableDeclarationSimple.sol index c29045702..742ae5a0d 100644 --- a/test/libsolidity/syntaxTests/multiVariableDeclaration/multiVariableDeclarationSimple.sol +++ b/test/libsolidity/syntaxTests/multiVariableDeclaration/multiVariableDeclarationSimple.sol @@ -10,4 +10,5 @@ contract C { function h() internal pure returns (bytes memory, string storage s) { s = s; } } // ---- +// Warning 5740: (111-115): Unreachable code. // Warning 6321: (250-262): Unnamed return variable can remain unassigned. Add an explicit return with value to all non-reverting code paths or name the variable. diff --git a/test/libsolidity/syntaxTests/nameAndTypeResolution/001_name_references.sol b/test/libsolidity/syntaxTests/nameAndTypeResolution/001_name_references.sol index 2bf964ff3..cfbf9365e 100644 --- a/test/libsolidity/syntaxTests/nameAndTypeResolution/001_name_references.sol +++ b/test/libsolidity/syntaxTests/nameAndTypeResolution/001_name_references.sol @@ -3,4 +3,5 @@ contract test { function f(uint256) public returns (uint out) { f(variable); test; out; } } // ---- +// Warning 5740: (103-112): Unreachable code. // Warning 6133: (103-107): Statement has no effect. diff --git a/test/libsolidity/syntaxTests/viewPureChecker/call_internal_functions_success.sol b/test/libsolidity/syntaxTests/viewPureChecker/call_internal_functions_success.sol index 3023967ca..858f5d7da 100644 --- a/test/libsolidity/syntaxTests/viewPureChecker/call_internal_functions_success.sol +++ b/test/libsolidity/syntaxTests/viewPureChecker/call_internal_functions_success.sol @@ -5,3 +5,9 @@ contract C { function i() payable public { i(); h(); g(); f(); } } // ---- +// Warning 5740: (102-105): Unreachable code. +// Warning 5740: (140-143): Unreachable code. +// Warning 5740: (145-148): Unreachable code. +// Warning 5740: (191-194): Unreachable code. +// Warning 5740: (196-199): Unreachable code. +// Warning 5740: (201-204): Unreachable code. diff --git a/test/libsolidity/syntaxTests/viewPureChecker/view_pure_abi_encode.sol b/test/libsolidity/syntaxTests/viewPureChecker/view_pure_abi_encode.sol index d7146f2b5..9e147e6b2 100644 --- a/test/libsolidity/syntaxTests/viewPureChecker/view_pure_abi_encode.sol +++ b/test/libsolidity/syntaxTests/viewPureChecker/view_pure_abi_encode.sol @@ -7,3 +7,4 @@ contract C { } } // ---- +// Warning 5740: (142-237): Unreachable code.