CheckCondition: Fix FPs for aliased variables (daca)
This commit is contained in:
parent
a02fd54a69
commit
9465dca583
|
@ -51,6 +51,13 @@ namespace {
|
||||||
CheckCondition instance;
|
CheckCondition instance;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
bool CheckCondition::isAliased(const std::set<unsigned int> &vars) const {
|
||||||
|
for (const Token *tok = _tokenizer->tokens(); tok; tok = tok->next()) {
|
||||||
|
if (Token::Match(tok, "= & %var% ;") && vars.find(tok->tokAt(2)->varId()) != vars.end())
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
void CheckCondition::assignIf()
|
void CheckCondition::assignIf()
|
||||||
{
|
{
|
||||||
|
@ -516,11 +523,15 @@ void CheckCondition::multiCondition2()
|
||||||
const Token *cond2 = tok->next()->astOperand2();
|
const Token *cond2 = tok->next()->astOperand2();
|
||||||
|
|
||||||
if (type == MULTICONDITIONTYPE::INNER) {
|
if (type == MULTICONDITIONTYPE::INNER) {
|
||||||
if (isOppositeCond(false, _tokenizer->isCPP(), cond1, cond2, _settings->library, true))
|
if (isOppositeCond(false, _tokenizer->isCPP(), cond1, cond2, _settings->library, true)) {
|
||||||
oppositeInnerConditionError(cond1, cond2);
|
if (!isAliased(vars))
|
||||||
|
oppositeInnerConditionError(cond1, cond2);
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
if (isSameExpression(_tokenizer->isCPP(), true, cond1, cond2, _settings->library, true))
|
if (isSameExpression(_tokenizer->isCPP(), true, cond1, cond2, _settings->library, true)) {
|
||||||
sameConditionAfterEarlyExitError(cond1, cond2);
|
if (!isAliased(vars))
|
||||||
|
sameConditionAfterEarlyExitError(cond1, cond2);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
|
@ -114,7 +114,7 @@ public:
|
||||||
void checkInvalidTestForOverflow();
|
void checkInvalidTestForOverflow();
|
||||||
|
|
||||||
private:
|
private:
|
||||||
|
bool isAliased(const std::set<unsigned int> &vars) const;
|
||||||
bool isOverlappingCond(const Token * const cond1, const Token * const cond2, bool pure) const;
|
bool isOverlappingCond(const Token * const cond1, const Token * const cond2, bool pure) const;
|
||||||
void assignIfError(const Token *tok1, const Token *tok2, const std::string &condition, bool result);
|
void assignIfError(const Token *tok1, const Token *tok2, const std::string &condition, bool result);
|
||||||
void mismatchingBitAndError(const Token *tok1, const MathLib::bigint num1, const Token *tok2, const MathLib::bigint num2);
|
void mismatchingBitAndError(const Token *tok1, const MathLib::bigint num1, const Token *tok2, const MathLib::bigint num2);
|
||||||
|
|
|
@ -76,6 +76,7 @@ private:
|
||||||
TEST_CASE(modulo);
|
TEST_CASE(modulo);
|
||||||
|
|
||||||
TEST_CASE(oppositeInnerCondition);
|
TEST_CASE(oppositeInnerCondition);
|
||||||
|
TEST_CASE(oppositeInnerConditionAlias);
|
||||||
TEST_CASE(oppositeInnerCondition2);
|
TEST_CASE(oppositeInnerCondition2);
|
||||||
TEST_CASE(oppositeInnerConditionAnd);
|
TEST_CASE(oppositeInnerConditionAnd);
|
||||||
|
|
||||||
|
@ -1597,6 +1598,20 @@ private:
|
||||||
ASSERT_EQUALS("[test.cpp:2] -> [test.cpp:3]: (warning) Opposite inner 'if' condition leads to a dead code block.\n", errout.str());
|
ASSERT_EQUALS("[test.cpp:2] -> [test.cpp:3]: (warning) Opposite inner 'if' condition leads to a dead code block.\n", errout.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void oppositeInnerConditionAlias() {
|
||||||
|
check("void f() {\n"
|
||||||
|
" struct S s;\n"
|
||||||
|
" bool hasFailed = false;\n"
|
||||||
|
" s.status = &hasFailed;\n"
|
||||||
|
"\n"
|
||||||
|
" if (! hasFailed) {\n"
|
||||||
|
" doStuff(&s);\n"
|
||||||
|
" if (hasFailed) {}\n"
|
||||||
|
" }\n"
|
||||||
|
"}");
|
||||||
|
ASSERT_EQUALS("", errout.str());
|
||||||
|
}
|
||||||
|
|
||||||
void oppositeInnerCondition2() {
|
void oppositeInnerCondition2() {
|
||||||
// first comparison: <
|
// first comparison: <
|
||||||
check("void f(int x) {\n"
|
check("void f(int x) {\n"
|
||||||
|
|
Loading…
Reference in New Issue