Reimplemented CheckStl::readingEmptyStlContainer() based on Libraries

This commit is contained in:
PKEuS 2015-11-20 14:15:00 +01:00
parent 53b2eca983
commit c0e33e20b4
3 changed files with 56 additions and 37 deletions

View File

@ -1386,7 +1386,7 @@ void CheckStl::dereferenceInvalidIteratorError(const Token* deref, const std::st
void CheckStl::readingEmptyStlContainer_parseUsage(const Token* tok, bool map, std::set<unsigned int>& empty, bool noerror) void CheckStl::readingEmptyStlContainer_parseUsage(const Token* tok, const Library::Container* container, std::map<unsigned int, const Library::Container*>& empty, bool noerror)
{ {
// Check for various conditions for the way stl containers and variables can be used // Check for various conditions for the way stl containers and variables can be used
if (tok->strAt(1) == "=" || (tok->strAt(1) == "[" && Token::simpleMatch(tok->linkAt(1), "] ="))) { if (tok->strAt(1) == "=" || (tok->strAt(1) == "[" && Token::simpleMatch(tok->linkAt(1), "] ="))) {
@ -1394,19 +1394,29 @@ void CheckStl::readingEmptyStlContainer_parseUsage(const Token* tok, bool map, s
empty.erase(tok->varId()); empty.erase(tok->varId());
} else if (Token::Match(tok, "%name% [")) { } else if (Token::Match(tok, "%name% [")) {
// Access through operator[] // Access through operator[]
if (map) { // operator[] inserts an element, if used on a std::map if (!container->arrayLike_indexOp) { // operator[] inserts an element if used on a std::map
if (!noerror && tok->strAt(-1) == "=") if (!noerror && tok->strAt(-1) == "=")
readingEmptyStlContainerError(tok); readingEmptyStlContainerError(tok);
empty.erase(tok->varId()); empty.erase(tok->varId());
} else if (!noerror) } else if (!noerror)
readingEmptyStlContainerError(tok); readingEmptyStlContainerError(tok);
} else if (Token::Match(tok, "%name% . %type% (")) { } else if (Token::Match(tok, "%name% . %type% (")) {
Library::Container::Yield yield = container->getYield(tok->strAt(2));
const Token* parent = tok->tokAt(3)->astParent();
// Member function call // Member function call
if (Token::Match(tok->tokAt(2), "find|at|data|c_str|back|front|empty|top|size|count")) { // These functions read from the container if (yield != Library::Container::NO_YIELD &&
((yield != Library::Container::START_ITERATOR &&
yield != Library::Container::END_ITERATOR) || !parent || Token::Match(parent, "%cop%|=|*"))) { // These functions read from the container
if (!noerror)
readingEmptyStlContainerError(tok);
} else {
Library::Container::Action action = container->getAction(tok->strAt(2));
if (action == Library::Container::FIND) {
if (!noerror) if (!noerror)
readingEmptyStlContainerError(tok); readingEmptyStlContainerError(tok);
} else } else
empty.erase(tok->varId()); empty.erase(tok->varId());
}
} else if (tok->strAt(-1) == "=") { } else if (tok->strAt(-1) == "=") {
// Assignment (RHS) // Assignment (RHS)
if (!noerror) if (!noerror)
@ -1417,13 +1427,6 @@ void CheckStl::readingEmptyStlContainer_parseUsage(const Token* tok, bool map, s
} }
} }
namespace {
static const std::set<std::string> MAP_STL_CONTAINERS = make_container< std::set<std::string> >() <<
"map" << "multimap" << "unordered_map" << "unordered_multimap" ;
static const std::set<std::string> NONMAP_STL_CONTAINERS = make_container< std::set<std::string> >() <<
"deque" << "forward_list" << "list" << "multiset" << "queue" << "set" << "stack" << "string" << "unordered_multiset" << "unordered_set" << "vector";
}
void CheckStl::readingEmptyStlContainer() void CheckStl::readingEmptyStlContainer()
{ {
if (!_settings->isEnabled("style")) if (!_settings->isEnabled("style"))
@ -1432,8 +1435,7 @@ void CheckStl::readingEmptyStlContainer()
if (!_settings->inconclusive) if (!_settings->inconclusive)
return; return;
std::set<unsigned int> empty_map; // empty std::map-like instances of STL containers std::map<unsigned int, const Library::Container*> emptyContainer;
std::set<unsigned int> empty_nonmap; // empty non-std::map-like instances of STL containers
const std::list<Scope>& scopeList = _tokenizer->getSymbolDatabase()->scopeList; const std::list<Scope>& scopeList = _tokenizer->getSymbolDatabase()->scopeList;
@ -1451,18 +1453,14 @@ void CheckStl::readingEmptyStlContainer()
if (!tok2->varId()) if (!tok2->varId())
continue; continue;
const bool map = empty_map.find(tok2->varId()) != empty_map.end(); std::map<unsigned int, const Library::Container*>::const_iterator container = emptyContainer.find(tok2->varId());
if (!map && empty_nonmap.find(tok2->varId()) == empty_nonmap.end()) if (container == emptyContainer.end())
continue; continue;
if (map) readingEmptyStlContainer_parseUsage(tok2, container->second, emptyContainer, true);
readingEmptyStlContainer_parseUsage(tok2, true, empty_map, true);
else
readingEmptyStlContainer_parseUsage(tok2, false, empty_nonmap, true);
} }
} else if (Token::Match(tok, "do|}|break|case")) { } else if (Token::Match(tok, "do|}|break|case")) {
empty_map.clear(); emptyContainer.clear();
empty_nonmap.clear();
} }
if (!tok->varId()) if (!tok->varId())
@ -1470,7 +1468,7 @@ void CheckStl::readingEmptyStlContainer()
// Check whether a variable should be marked as "empty" // Check whether a variable should be marked as "empty"
const Variable* var = tok->variable(); const Variable* var = tok->variable();
if (var) { if (var && !var->isArrayOrPointer() && !var->typeStartToken()->isStandardType()) {
bool insert = false; bool insert = false;
if (var->nameToken() == tok && var->isLocal() && !var->isStatic()) { // Local variable declared if (var->nameToken() == tok && var->isLocal() && !var->isStatic()) { // Local variable declared
insert = !Token::Match(tok->tokAt(1), "[(=]"); // Only if not initialized insert = !Token::Match(tok->tokAt(1), "[(=]"); // Only if not initialized
@ -1479,25 +1477,20 @@ void CheckStl::readingEmptyStlContainer()
} }
if (insert) { if (insert) {
if (var->isStlType(MAP_STL_CONTAINERS)) const Library::Container* container = _settings->library.detectContainer(var->typeStartToken());
empty_map.insert(var->declarationId()); if (container)
else if (var->isStlType(NONMAP_STL_CONTAINERS)) emptyContainer[var->declarationId()] = container;
empty_nonmap.insert(var->declarationId());
continue; continue;
} }
} }
const bool map = empty_map.find(tok->varId()) != empty_map.end(); std::map<unsigned int, const Library::Container*>::const_iterator container = emptyContainer.find(tok->varId());
if (!map && empty_nonmap.find(tok->varId()) == empty_nonmap.end()) if (container == emptyContainer.end())
continue; continue;
if (map) readingEmptyStlContainer_parseUsage(tok, container->second, emptyContainer, false);
readingEmptyStlContainer_parseUsage(tok, true, empty_map, false);
else
readingEmptyStlContainer_parseUsage(tok, false, empty_nonmap, false);
} }
empty_map.clear(); emptyContainer.clear();
empty_nonmap.clear();
} }
} }

View File

@ -151,7 +151,7 @@ public:
void readingEmptyStlContainer(); void readingEmptyStlContainer();
private: private:
void readingEmptyStlContainer_parseUsage(const Token* tok, bool map, std::set<unsigned int>& empty, bool noerror); void readingEmptyStlContainer_parseUsage(const Token* tok, const Library::Container* container, std::map<unsigned int, const Library::Container*>& empty, bool noerror);
void missingComparisonError(const Token *incrementToken1, const Token *incrementToken2); void missingComparisonError(const Token *incrementToken1, const Token *incrementToken2);
void string_c_strThrowError(const Token *tok); void string_c_strThrowError(const Token *tok);

View File

@ -2786,12 +2786,38 @@ private:
"}", true); "}", true);
ASSERT_EQUALS("", errout.str()); ASSERT_EQUALS("", errout.str());
check("void f(std::vector<int> v) {\n" check("void f(std::set<int> v) {\n"
" v.clear();\n" " v.clear();\n"
" int i = v.find(foobar);\n" " int i = v.find(foobar);\n"
"}", true); "}", true);
ASSERT_EQUALS("[test.cpp:3]: (style, inconclusive) Reading from empty STL container 'v'\n", errout.str()); ASSERT_EQUALS("[test.cpp:3]: (style, inconclusive) Reading from empty STL container 'v'\n", errout.str());
check("void f(std::set<int> v) {\n"
" v.clear();\n"
" v.begin();\n"
"}", true);
ASSERT_EQUALS("[test.cpp:3]: (style, inconclusive) Reading from empty STL container 'v'\n", errout.str());
check("void f(std::set<int> v) {\n"
" v.clear();\n"
" *v.begin();\n"
"}", true);
ASSERT_EQUALS("[test.cpp:3]: (style, inconclusive) Reading from empty STL container 'v'\n", errout.str());
check("void f(std::set<int> v) {\n"
" v.clear();\n"
" for(auto i = v.cbegin();\n"
" i != v.cend(); ++i) {}\n"
"}", true);
ASSERT_EQUALS("[test.cpp:3]: (style, inconclusive) Reading from empty STL container 'v'\n"
"[test.cpp:4]: (style, inconclusive) Reading from empty STL container 'v'\n", errout.str());
check("void f(std::set<int> v) {\n"
" v.clear();\n"
" foo(v.begin());\n"
"}", true);
ASSERT_EQUALS("", errout.str());
check("void f() {\n" check("void f() {\n"
" std::map<int, std::string> CMap;\n" " std::map<int, std::string> CMap;\n"
" std::string strValue = CMap[1];\n" " std::string strValue = CMap[1];\n"