Refactorized iterator checking:

- Fixed false positive #5669
- Use symboldatabase in CheckStl::pushback()
- Improved support for erase on std::vector and find
This commit is contained in:
PKEuS 2014-04-12 19:44:37 +02:00
parent 1252c70449
commit e8ac355b39
2 changed files with 124 additions and 105 deletions

View File

@ -150,7 +150,7 @@ void CheckStl::iterators()
} }
// invalidate the iterator if it is erased // invalidate the iterator if it is erased
else if (tok2->strAt(2) == std::string("erase")) { else if (tok2->strAt(2) == "erase" && (tok2->strAt(4) != "*" || (container && tok2->varId() == container->declarationId()))) {
validIterator = false; validIterator = false;
eraseToken = tok2; eraseToken = tok2;
invalidationScope = tok2->scope(); invalidationScope = tok2->scope();
@ -593,53 +593,25 @@ void CheckStl::pushback()
} }
// Iterator becomes invalid after reserve, resize, insert, push_back or push_front.. // Iterator becomes invalid after reserve, resize, insert, push_back or push_front..
for (std::size_t i = 0; i < functions; ++i) { for (unsigned int iteratorId = 1; iteratorId < symbolDatabase->getVariableListSize(); iteratorId++) {
const Scope * scope = symbolDatabase->functionScopes[i]; const Variable* var = symbolDatabase->getVariableFromVarId(iteratorId);
for (const Token* tok = scope->classStart->next(); tok && tok != scope->classEnd; tok = tok->next()) {
if (!Token::simpleMatch(tok, "vector <")) // Check that its an iterator
if (!var || !var->isLocal() || !Token::Match(var->typeEndToken(), "iterator|const_iterator|reverse_iterator|const_reverse_iterator"))
continue; continue;
// if iterator declaration inside for() loop // ... on std::vector
bool iteratorDeclaredInsideLoop = false; if (!Token::Match(var->typeStartToken(), "std| ::| vector <"))
if ((tok->tokAt(-2) && Token::simpleMatch(tok->tokAt(-2), "for (")) ||
(tok->tokAt(-4) && Token::simpleMatch(tok->tokAt(-4), "for ( std ::"))) {
iteratorDeclaredInsideLoop = true;
}
while (tok && tok->str() != ">")
tok = tok->next();
if (!tok)
break;
if (!Token::Match(tok, "> :: iterator|const_iterator %var% =|;"))
continue; continue;
const unsigned int iteratorid(tok->tokAt(3)->varId());
if (iteratorid == 0)
continue;
if (iteratorDeclaredInsideLoop && tok->strAt(4) == "=") {
// skip "> :: iterator|const_iterator"
tok = tok->tokAt(3);
}
// the variable id for the vector // the variable id for the vector
unsigned int vectorid = 0; unsigned int vectorid = 0;
// count { , } and parentheses for tok2
int indent = 0;
const Token* validatingToken = 0; const Token* validatingToken = 0;
std::string invalidIterator; std::string invalidIterator;
for (const Token *tok2 = tok; indent >= 0 && tok2; tok2 = tok2->next()) { const Token* end2 = var->scope()->classEnd;
if (tok2->str() == "{" || tok2->str() == "(") for (const Token *tok2 = var->nameToken(); tok2 != end2; tok2 = tok2->next()) {
++indent;
else if (tok2->str() == "}" || tok2->str() == ")") {
if (indent == 0 && Token::simpleMatch(tok2, ") {"))
tok2 = tok2->next();
else
--indent;
}
if (validatingToken == tok2) { if (validatingToken == tok2) {
invalidIterator.clear(); invalidIterator.clear();
@ -649,10 +621,9 @@ void CheckStl::pushback()
// Using push_back or push_front inside a loop.. // Using push_back or push_front inside a loop..
if (Token::simpleMatch(tok2, "for (")) { if (Token::simpleMatch(tok2, "for (")) {
tok2 = tok2->tokAt(2); tok2 = tok2->tokAt(2);
++indent;
} }
if (Token::Match(tok2, "%varid% = %var% . begin|rbegin|cbegin|crbegin ( ) ; %varid% != %var% . end|rend|cend|crend ( ) ; ++| %varid% ++| ) {", iteratorid)) { if (Token::Match(tok2, "%varid% = %var% . begin|rbegin|cbegin|crbegin ( ) ; %varid% != %var% . end|rend|cend|crend ( ) ; ++| %varid% ++| ) {", iteratorId)) {
// variable id for the loop iterator // variable id for the loop iterator
const unsigned int varId(tok2->tokAt(2)->varId()); const unsigned int varId(tok2->tokAt(2)->varId());
if (varId == 0) if (varId == 0)
@ -666,7 +637,8 @@ void CheckStl::pushback()
if (tok3->str() == "break" || tok3->str() == "return") { if (tok3->str() == "break" || tok3->str() == "return") {
pushbackTok = 0; pushbackTok = 0;
break; break;
} else if (Token::Match(tok3, "%varid% . push_front|push_back|insert|reserve|resize|clear (", varId) && !tok3->previous()->isAssignmentOp()) { } else if (Token::Match(tok3, "%varid% . push_front|push_back|insert|reserve|resize|clear|erase (", varId) && !tok3->previous()->isAssignmentOp()) {
if (tok3->strAt(2) != "erase" || (tok3->tokAt(4)->varId() != iteratorId && tok3->tokAt(5)->varId() != iteratorId)) // This case is handled in: CheckStl::iterators()
pushbackTok = tok3->tokAt(2); pushbackTok = tok3->tokAt(2);
} }
} }
@ -676,10 +648,10 @@ void CheckStl::pushback()
} }
// Assigning iterator.. // Assigning iterator..
if (Token::Match(tok2, "%varid% =", iteratorid)) { if (Token::Match(tok2, "%varid% =", iteratorId)) {
if (Token::Match(tok2->tokAt(2), "%var% . begin|end|rbegin|rend|cbegin|cend|crbegin|crend|insert (")) { if (Token::Match(tok2->tokAt(2), "%var% . begin|end|rbegin|rend|cbegin|cend|crbegin|crend|insert|erase|find (")) {
if (!invalidIterator.empty() && Token::Match(tok2->tokAt(4), "insert ( %varid% ,", iteratorid)) { if (!invalidIterator.empty() && Token::Match(tok2->tokAt(4), "insert|erase ( *| %varid% )|,", iteratorId)) {
invalidIteratorError(tok2, invalidIterator, tok2->strAt(6)); invalidIteratorError(tok2, invalidIterator, var->name());
break; break;
} }
vectorid = tok2->tokAt(2)->varId(); vectorid = tok2->tokAt(2)->varId();
@ -691,12 +663,13 @@ void CheckStl::pushback()
} }
// push_back on vector.. // push_back on vector..
if (vectorid > 0 && Token::Match(tok2, "%varid% . push_front|push_back|insert|reserve|resize|clear (", vectorid)) { if (vectorid > 0 && Token::Match(tok2, "%varid% . push_front|push_back|insert|reserve|resize|clear|erase (", vectorid)) {
if (!invalidIterator.empty() && Token::Match(tok2->tokAt(2), "insert ( %varid% ,", iteratorid)) { if (!invalidIterator.empty() && Token::Match(tok2->tokAt(2), "insert|erase ( *| %varid% ,|)", iteratorId)) {
invalidIteratorError(tok2, invalidIterator, tok2->strAt(4)); invalidIteratorError(tok2, invalidIterator, var->name());
break; break;
} }
if (tok2->strAt(2) != "erase" || (tok2->tokAt(4)->varId() != iteratorId && tok2->tokAt(5)->varId() != iteratorId)) // This case is handled in: CheckStl::iterators()
invalidIterator = tok2->strAt(2); invalidIterator = tok2->strAt(2);
tok2 = tok2->linkAt(3); tok2 = tok2->linkAt(3);
} }
@ -710,14 +683,13 @@ void CheckStl::pushback()
// Using invalid iterator.. // Using invalid iterator..
if (!invalidIterator.empty()) { if (!invalidIterator.empty()) {
if (Token::Match(tok2, "++|--|*|+|-|(|,|=|!= %varid%", iteratorid)) if (Token::Match(tok2, "++|--|*|+|-|(|,|=|!= %varid%", iteratorId))
invalidIteratorError(tok2, invalidIterator, tok2->strAt(1)); invalidIteratorError(tok2, invalidIterator, tok2->strAt(1));
if (Token::Match(tok2, "%varid% ++|--|+|-|.", iteratorid)) if (Token::Match(tok2, "%varid% ++|--|+|-|.", iteratorId))
invalidIteratorError(tok2, invalidIterator, tok2->str()); invalidIteratorError(tok2, invalidIterator, tok2->str());
} }
} }
} }
}
} }

View File

@ -72,6 +72,7 @@ private:
TEST_CASE(eraseAssignByFunctionCall); TEST_CASE(eraseAssignByFunctionCall);
TEST_CASE(eraseErase); TEST_CASE(eraseErase);
TEST_CASE(eraseByValue); TEST_CASE(eraseByValue);
TEST_CASE(eraseOnVector);
TEST_CASE(pushback1); TEST_CASE(pushback1);
TEST_CASE(pushback2); TEST_CASE(pushback2);
@ -85,7 +86,6 @@ private:
TEST_CASE(pushback10); TEST_CASE(pushback10);
TEST_CASE(pushback11); TEST_CASE(pushback11);
TEST_CASE(pushback12); TEST_CASE(pushback12);
TEST_CASE(insert1); TEST_CASE(insert1);
TEST_CASE(insert2); TEST_CASE(insert2);
@ -984,9 +984,56 @@ private:
" foo.erase(*it);\n" " foo.erase(*it);\n"
"}"); "}");
ASSERT_EQUALS("", errout.str()); ASSERT_EQUALS("", errout.str());
// #5669
check("void f() {\n"
" HashSet_Ref::iterator aIt = m_ImplementationMap.find( xEle );\n"
" m_SetLoadedFactories.erase(*aIt);\n"
" m_SetLoadedFactories.erase(aIt);\n"
"}");
ASSERT_EQUALS("", errout.str());
check("void f(const std::list<int>& m_ImplementationMap) {\n"
" std::list<int>::iterator aIt = m_ImplementationMap.find( xEle );\n"
" m_ImplementationMap.erase(*aIt);\n"
" m_ImplementationMap.erase(aIt);\n"
"}");
ASSERT_EQUALS("[test.cpp:4]: (error) Invalid iterator: aIt\n", errout.str());
check("void f(const std::list<int>& m_ImplementationMap) {\n"
" std::list<int>::iterator aIt = m_ImplementationMap.find( xEle1 );\n"
" std::list<int>::iterator bIt = m_ImplementationMap.find( xEle2 );\n"
" m_ImplementationMap.erase(*bIt);\n"
" m_ImplementationMap.erase(aIt);\n"
"}");
ASSERT_EQUALS("", errout.str());
} }
void eraseOnVector() {
check("void f(const std::vector<int>& m_ImplementationMap) {\n"
" std::vector<int>::iterator aIt = m_ImplementationMap.find( xEle );\n"
" m_ImplementationMap.erase(something(unknown));\n" // All iterators become invalidated when erasing from std::vector
" m_ImplementationMap.erase(aIt);\n"
"}");
ASSERT_EQUALS("[test.cpp:4]: (error) After erase(), the iterator 'aIt' may be invalid.\n", errout.str());
check("void f(const std::vector<int>& m_ImplementationMap) {\n"
" std::vector<int>::iterator aIt = m_ImplementationMap.find( xEle );\n"
" m_ImplementationMap.erase(*aIt);\n" // All iterators become invalidated when erasing from std::vector
" m_ImplementationMap.erase(aIt);\n"
"}");
ASSERT_EQUALS("[test.cpp:4]: (error) Invalid iterator: aIt\n", errout.str());
check("void f(const std::vector<int>& m_ImplementationMap) {\n"
" std::vector<int>::iterator aIt = m_ImplementationMap.find( xEle1 );\n"
" std::vector<int>::iterator bIt = m_ImplementationMap.find( xEle2 );\n"
" m_ImplementationMap.erase(*bIt);\n" // All iterators become invalidated when erasing from std::vector
" aIt = m_ImplementationMap.erase(aIt);\n"
"}");
ASSERT_EQUALS("[test.cpp:5]: (error) After erase(), the iterator 'aIt' may be invalid.\n", errout.str());
}
void pushback1() { void pushback1() {
check("void f(const std::vector<int> &foo)\n" check("void f(const std::vector<int> &foo)\n"
"{\n" "{\n"