Fix 10331: wrong conditional value after assignment+return (#3461)

This commit is contained in:
Paul Fultz II 2021-09-21 12:25:41 -05:00 committed by GitHub
parent 47ea670eb3
commit 255dc0484a
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
4 changed files with 69 additions and 5 deletions

View File

@ -12,6 +12,14 @@
#include <tuple> #include <tuple>
#include <utility> #include <utility>
struct OnExit {
std::function<void()> f;
~OnExit() {
f();
}
};
struct ForwardTraversal { struct ForwardTraversal {
enum class Progress { Continue, Break, Skip }; enum class Progress { Continue, Break, Skip };
enum class Terminate { None, Bail, Escape, Modified, Inconclusive, Conditional }; enum class Terminate { None, Bail, Escape, Modified, Inconclusive, Conditional };
@ -25,6 +33,7 @@ struct ForwardTraversal {
bool analyzeTerminate; bool analyzeTerminate;
Analyzer::Terminate terminate = Analyzer::Terminate::None; Analyzer::Terminate terminate = Analyzer::Terminate::None;
bool forked = false; bool forked = false;
std::vector<Token*> loopEnds = {};
Progress Break(Analyzer::Terminate t = Analyzer::Terminate::None) { Progress Break(Analyzer::Terminate t = Analyzer::Terminate::None) {
if ((!analyzeOnly || analyzeTerminate) && t != Analyzer::Terminate::None) if ((!analyzeOnly || analyzeTerminate) && t != Analyzer::Terminate::None)
@ -85,9 +94,15 @@ struct ForwardTraversal {
template<class T, REQUIRES("T must be a Token class", std::is_convertible<T*, const Token*> )> template<class T, REQUIRES("T must be a Token class", std::is_convertible<T*, const Token*> )>
Progress traverseTok(T* tok, std::function<Progress(T*)> f, bool traverseUnknown, T** out = nullptr) { Progress traverseTok(T* tok, std::function<Progress(T*)> f, bool traverseUnknown, T** out = nullptr) {
if (Token::Match(tok, "asm|goto|continue|setjmp|longjmp")) if (Token::Match(tok, "asm|goto|setjmp|longjmp"))
return Break(); return Break(Analyzer::Terminate::Bail);
else if (Token::Match(tok, "return|throw") || isEscapeFunction(tok, &settings->library)) { else if (Token::simpleMatch(tok, "continue")) {
if (loopEnds.empty())
return Break(Analyzer::Terminate::Escape);
// If we are in a loop then jump to the end
if (out)
*out = loopEnds.back();
} else if (Token::Match(tok, "return|throw") || isEscapeFunction(tok, &settings->library)) {
traverseRecursive(tok->astOperand1(), f, traverseUnknown); traverseRecursive(tok->astOperand1(), f, traverseUnknown);
traverseRecursive(tok->astOperand2(), f, traverseUnknown); traverseRecursive(tok->astOperand2(), f, traverseUnknown);
return Break(Analyzer::Terminate::Escape); return Break(Analyzer::Terminate::Escape);
@ -361,6 +376,10 @@ struct ForwardTraversal {
} }
Progress updateInnerLoop(Token* endBlock, Token* stepTok, Token* condTok) { Progress updateInnerLoop(Token* endBlock, Token* stepTok, Token* condTok) {
loopEnds.push_back(endBlock);
OnExit oe{[&] {
loopEnds.pop_back();
}};
if (endBlock && updateScope(endBlock) == Progress::Break) if (endBlock && updateScope(endBlock) == Progress::Break)
return Break(); return Break();
if (stepTok && updateRecursive(stepTok) == Progress::Break) if (stepTok && updateRecursive(stepTok) == Progress::Break)
@ -632,7 +651,7 @@ struct ForwardTraversal {
} }
actions |= (thenBranch.action | elseBranch.action); actions |= (thenBranch.action | elseBranch.action);
if (bail) if (bail)
return Break(); return Break(Analyzer::Terminate::Bail);
if (thenBranch.isDead() && elseBranch.isDead()) { if (thenBranch.isDead() && elseBranch.isDead()) {
if (thenBranch.isModified() && elseBranch.isModified()) if (thenBranch.isModified() && elseBranch.isModified())
return Break(Analyzer::Terminate::Modified); return Break(Analyzer::Terminate::Modified);

View File

@ -5313,6 +5313,7 @@ struct ConditionHandler {
startTokens[1] = top->link()->linkAt(1)->tokAt(2); startTokens[1] = top->link()->linkAt(1)->tokAt(2);
int changeBlock = -1; int changeBlock = -1;
int bailBlock = -1;
for (int i = 0; i < 2; i++) { for (int i = 0; i < 2; i++) {
const Token* const startToken = startTokens[i]; const Token* const startToken = startTokens[i];
@ -5326,6 +5327,9 @@ struct ConditionHandler {
deadBranch[i] = r.terminate == Analyzer::Terminate::Escape; deadBranch[i] = r.terminate == Analyzer::Terminate::Escape;
if (r.action.isModified() && !deadBranch[i]) if (r.action.isModified() && !deadBranch[i])
changeBlock = i; changeBlock = i;
if (r.terminate != Analyzer::Terminate::None && r.terminate != Analyzer::Terminate::Escape &&
r.terminate != Analyzer::Terminate::Modified)
bailBlock = i;
changeKnownToPossible(values); changeKnownToPossible(values);
} }
if (changeBlock >= 0 && !Token::simpleMatch(top->previous(), "while (")) { if (changeBlock >= 0 && !Token::simpleMatch(top->previous(), "while (")) {
@ -5336,6 +5340,13 @@ struct ConditionHandler {
"valueFlowAfterCondition: " + cond.vartok->expressionString() + "valueFlowAfterCondition: " + cond.vartok->expressionString() +
" is changed in conditional block"); " is changed in conditional block");
return; return;
} else if (bailBlock >= 0) {
if (settings->debugwarnings)
bailout(tokenlist,
errorLogger,
startTokens[bailBlock]->link(),
"valueFlowAfterCondition: bailing in conditional block");
return;
} }
// After conditional code.. // After conditional code..

View File

@ -121,6 +121,7 @@ private:
TEST_CASE(nullpointer79); // #10400 TEST_CASE(nullpointer79); // #10400
TEST_CASE(nullpointer80); // #10410 TEST_CASE(nullpointer80); // #10410
TEST_CASE(nullpointer81); // #8724 TEST_CASE(nullpointer81); // #8724
TEST_CASE(nullpointer82); // #10331
TEST_CASE(nullpointer_addressOf); // address of TEST_CASE(nullpointer_addressOf); // address of
TEST_CASE(nullpointerSwitch); // #2626 TEST_CASE(nullpointerSwitch); // #2626
TEST_CASE(nullpointer_cast); // #4692 TEST_CASE(nullpointer_cast); // #4692
@ -2474,6 +2475,26 @@ private:
ASSERT_EQUALS("", errout.str()); ASSERT_EQUALS("", errout.str());
} }
void nullpointer82() // #10331
{
check("bool g();\n"
"int* h();\n"
"void f(int* ptr) {\n"
" if (!ptr) {\n"
" if (g())\n"
" goto done;\n"
" ptr = h();\n"
" if (!ptr)\n"
" return;\n"
" }\n"
" if (*ptr == 1)\n"
" return;\n"
"\n"
"done:\n"
"}\n");
ASSERT_EQUALS("", errout.str());
}
void nullpointer_addressOf() { // address of void nullpointer_addressOf() { // address of
check("void f() {\n" check("void f() {\n"
" struct X *x = 0;\n" " struct X *x = 0;\n"

View File

@ -1703,7 +1703,8 @@ private:
" if (x==123){}\n" " if (x==123){}\n"
"}"); "}");
ASSERT_EQUALS_WITHOUT_LINENUMBERS( ASSERT_EQUALS_WITHOUT_LINENUMBERS(
"[test.cpp:3]: (debug) valueflow.cpp::valueFlowConditionExpressions bailout: Skipping function due to incomplete variable a\n", "[test.cpp:3]: (debug) valueflow.cpp::valueFlowConditionExpressions bailout: Skipping function due to incomplete variable a\n"
"[test.cpp:2]: (debug) valueflow.cpp::(valueFlow) bailout: valueFlowAfterCondition: bailing in conditional block\n",
errout.str()); errout.str());
// #5721 - FP // #5721 - FP
@ -2952,6 +2953,18 @@ private:
" return 0;\n" " return 0;\n"
"}\n"; "}\n";
ASSERT_EQUALS(true, testValueOfXKnown(code, 5U, 1)); ASSERT_EQUALS(true, testValueOfXKnown(code, 5U, 1));
code = "int f(int x) {\n"
" if (x == 1) {\n"
" for(int i=0;i<1;i++) {\n"
" if (x == 1)\n"
" continue;\n"
" }\n"
" }\n"
" return x;\n"
"}\n";
ASSERT_EQUALS(true, testValueOfX(code, 8U, 1));
ASSERT_EQUALS(false, testValueOfXImpossible(code, 8U, 1));
} }
void valueFlowAfterConditionExpr() { void valueFlowAfterConditionExpr() {