From dc199169661deb3b6c04d29fb4465600e9a4a05a Mon Sep 17 00:00:00 2001 From: chrchr-github <78114321+chrchr-github@users.noreply.github.com> Date: Wed, 20 Sep 2023 15:49:13 +0200 Subject: [PATCH] Fix valueFlowConditionExpressions bailout for library function (refs #10045) (#5461) --- lib/symboldatabase.cpp | 10 ++++++++++ test/testsymboldatabase.cpp | 15 +++++++++++++++ 2 files changed, 25 insertions(+) diff --git a/lib/symboldatabase.cpp b/lib/symboldatabase.cpp index 0fc607663..99cede92b 100644 --- a/lib/symboldatabase.cpp +++ b/lib/symboldatabase.cpp @@ -1510,6 +1510,16 @@ void SymbolDatabase::createSymbolDatabaseIncompleteVars() continue; if (mSettings.standards.cpp >= Standards::CPP20 && cpp20keywords.count(tok->str()) > 0) continue; + std::string fstr = tok->str(); + const Token* ftok = tok->previous(); + while (Token::simpleMatch(ftok, "::")) { + if (!Token::Match(ftok->previous(), "%name%")) + break; + fstr.insert(0, ftok->previous()->str() + "::"); + ftok = ftok->tokAt(-2); + } + if (mSettings.library.functions.find(fstr) != mSettings.library.functions.end()) + continue; const_cast(tok)->isIncompleteVar(true); // TODO: avoid const_cast } } diff --git a/test/testsymboldatabase.cpp b/test/testsymboldatabase.cpp index 46167ec54..8fab8b21f 100644 --- a/test/testsymboldatabase.cpp +++ b/test/testsymboldatabase.cpp @@ -5464,6 +5464,21 @@ private: const Token* s4 = Token::findsimplematch(s3->next(), "string )"); ASSERT(s4 && !s4->isIncompleteVar()); } + { + GET_SYMBOL_DB("void destroy(int*, void (*cb_dealloc)(void *));\n" + "void f(int* p, int* q, int* r) {\n" + " destroy(p, free);\n" + " destroy(q, std::free);\n" + " destroy(r, N::free);\n" + "}\n"); + ASSERT(db && errout.str().empty()); + const Token* free1 = Token::findsimplematch(tokenizer.tokens(), "free"); + ASSERT(free1 && !free1->isIncompleteVar()); + const Token* free2 = Token::findsimplematch(free1->next(), "free"); + ASSERT(free2 && !free2->isIncompleteVar()); + const Token* free3 = Token::findsimplematch(free2->next(), "free"); + ASSERT(free3 && free3->isIncompleteVar()); + } } void enum1() {