Reimplemented CheckStl::readingEmptyStlContainer() based on Libraries
This commit is contained in:
parent
53b2eca983
commit
c0e33e20b4
|
@ -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();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -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);
|
||||||
|
|
|
@ -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"
|
||||||
|
|
Loading…
Reference in New Issue