Improvements to Nullpointer dereference on function call check:
printf format string parser improved (similar to my recent patch on the argument counter), frexp/modf supported (#1572), Code cleanup
This commit is contained in:
parent
39af75abb4
commit
be0acad11f
|
@ -53,73 +53,90 @@ bool CheckNullPointer::isUpper(const std::string &str)
|
||||||
void CheckNullPointer::parseFunctionCall(const Token &tok, std::list<const Token *> &var, unsigned char value)
|
void CheckNullPointer::parseFunctionCall(const Token &tok, std::list<const Token *> &var, unsigned char value)
|
||||||
{
|
{
|
||||||
// standard functions that dereference first parameter..
|
// standard functions that dereference first parameter..
|
||||||
// both uninitialized data and null pointers are invalid.
|
static std::set<std::string> functionNames1_all;
|
||||||
static std::set<std::string> functionNames1;
|
static std::set<std::string> functionNames1_nullptr;
|
||||||
if (functionNames1.empty()) {
|
if (functionNames1_all.empty()) {
|
||||||
functionNames1.insert("memchr");
|
functionNames1_all.insert("memchr");
|
||||||
functionNames1.insert("memcmp");
|
functionNames1_all.insert("memcmp");
|
||||||
functionNames1.insert("strcat");
|
functionNames1_all.insert("strcat");
|
||||||
functionNames1.insert("strncat");
|
functionNames1_all.insert("strncat");
|
||||||
functionNames1.insert("strchr");
|
functionNames1_all.insert("strchr");
|
||||||
functionNames1.insert("strrchr");
|
functionNames1_all.insert("strrchr");
|
||||||
functionNames1.insert("strcmp");
|
functionNames1_all.insert("strcmp");
|
||||||
functionNames1.insert("strncmp");
|
functionNames1_all.insert("strncmp");
|
||||||
functionNames1.insert("strdup");
|
functionNames1_all.insert("strdup");
|
||||||
functionNames1.insert("strndup");
|
functionNames1_all.insert("strndup");
|
||||||
functionNames1.insert("strlen");
|
functionNames1_all.insert("strlen");
|
||||||
functionNames1.insert("strstr");
|
functionNames1_all.insert("strstr");
|
||||||
functionNames1.insert("fclose");
|
functionNames1_all.insert("fclose");
|
||||||
functionNames1.insert("feof");
|
functionNames1_all.insert("feof");
|
||||||
functionNames1.insert("fwrite");
|
functionNames1_all.insert("fwrite");
|
||||||
functionNames1.insert("fseek");
|
functionNames1_all.insert("fseek");
|
||||||
functionNames1.insert("ftell");
|
functionNames1_all.insert("ftell");
|
||||||
functionNames1.insert("fgetpos");
|
functionNames1_all.insert("fgetpos");
|
||||||
functionNames1.insert("fsetpos");
|
functionNames1_all.insert("fsetpos");
|
||||||
functionNames1.insert("rewind");
|
functionNames1_all.insert("rewind");
|
||||||
functionNames1.insert("scanf");
|
functionNames1_all.insert("printf");
|
||||||
functionNames1.insert("fscanf");
|
functionNames1_all.insert("scanf");
|
||||||
functionNames1.insert("sscanf");
|
functionNames1_all.insert("fscanf");
|
||||||
|
functionNames1_all.insert("sscanf");
|
||||||
|
|
||||||
|
functionNames1_nullptr.insert("memcpy");
|
||||||
|
functionNames1_nullptr.insert("memmove");
|
||||||
|
functionNames1_nullptr.insert("memset");
|
||||||
|
functionNames1_nullptr.insert("strcpy");
|
||||||
|
functionNames1_nullptr.insert("sprintf");
|
||||||
|
functionNames1_nullptr.insert("vsprintf");
|
||||||
|
functionNames1_nullptr.insert("vprintf");
|
||||||
|
functionNames1_nullptr.insert("fprintf");
|
||||||
|
functionNames1_nullptr.insert("vfprintf");
|
||||||
}
|
}
|
||||||
|
|
||||||
// standard functions that dereference second parameter..
|
// standard functions that dereference second parameter..
|
||||||
// both uninitialized data and null pointers are invalid.
|
static std::set<std::string> functionNames2_all;
|
||||||
static std::set<std::string> functionNames2;
|
static std::set<std::string> functionNames2_nullptr;
|
||||||
if (functionNames2.empty()) {
|
if (functionNames2_all.empty()) {
|
||||||
functionNames2.insert("memcmp");
|
functionNames2_all.insert("memcmp");
|
||||||
functionNames2.insert("memcpy");
|
functionNames2_all.insert("memcpy");
|
||||||
functionNames2.insert("memmove");
|
functionNames2_all.insert("memmove");
|
||||||
functionNames2.insert("strcat");
|
functionNames2_all.insert("strcat");
|
||||||
functionNames2.insert("strncat");
|
functionNames2_all.insert("strncat");
|
||||||
functionNames2.insert("strcmp");
|
functionNames2_all.insert("strcmp");
|
||||||
functionNames2.insert("strncmp");
|
functionNames2_all.insert("strncmp");
|
||||||
functionNames2.insert("strcpy");
|
functionNames2_all.insert("strcpy");
|
||||||
functionNames2.insert("strncpy");
|
functionNames2_all.insert("strncpy");
|
||||||
functionNames2.insert("strstr");
|
functionNames2_all.insert("strstr");
|
||||||
functionNames2.insert("sprintf");
|
functionNames2_all.insert("sprintf");
|
||||||
functionNames2.insert("fprintf");
|
functionNames2_all.insert("fprintf");
|
||||||
functionNames2.insert("fscanf");
|
functionNames2_all.insert("fscanf");
|
||||||
functionNames2.insert("sscanf");
|
functionNames2_all.insert("sscanf");
|
||||||
|
|
||||||
|
functionNames2_nullptr.insert("frexp");
|
||||||
|
functionNames2_nullptr.insert("modf");
|
||||||
}
|
}
|
||||||
|
|
||||||
// 1st parameter..
|
// 1st parameter..
|
||||||
if ((Token::Match(&tok, "%var% ( %var% ,|)") && tok.tokAt(2)->varId() > 0) ||
|
if ((Token::Match(&tok, "%var% ( %var% ,|)") && tok.tokAt(2)->varId() > 0) ||
|
||||||
(value == 0 && Token::Match(&tok, "%var% ( 0 ,|)"))) {
|
(value == 0 && Token::Match(&tok, "%var% ( 0 ,|)"))) {
|
||||||
if (functionNames1.find(tok.str()) != functionNames1.end())
|
if (functionNames1_all.find(tok.str()) != functionNames1_all.end())
|
||||||
var.push_back(tok.tokAt(2));
|
var.push_back(tok.tokAt(2));
|
||||||
else if (value == 0 && Token::Match(&tok, "memcpy|memmove|memset|strcpy|printf|sprintf|vsprintf|vprintf|fprintf|vfprintf"))
|
else if (value == 0 && functionNames1_nullptr.find(tok.str()) != functionNames1_nullptr.end())
|
||||||
var.push_back(tok.tokAt(2));
|
|
||||||
else if (value == 0 && Token::Match(&tok, "snprintf|vsnprintf|fnprintf|vfnprintf") && tok.strAt(4) != "0")
|
|
||||||
var.push_back(tok.tokAt(2));
|
var.push_back(tok.tokAt(2));
|
||||||
else if (value != 0 && Token::simpleMatch(&tok, "fflush"))
|
else if (value != 0 && Token::simpleMatch(&tok, "fflush"))
|
||||||
var.push_back(tok.tokAt(2));
|
var.push_back(tok.tokAt(2));
|
||||||
|
else if (value == 0 && Token::Match(&tok, "snprintf|vsnprintf|fnprintf|vfnprintf") && tok.strAt(4) != "0") // Only if length is not zero
|
||||||
|
var.push_back(tok.tokAt(2));
|
||||||
}
|
}
|
||||||
|
|
||||||
// 2nd parameter..
|
// 2nd parameter..
|
||||||
if (Token::Match(&tok, "%var% ( %any%")) {
|
if (Token::Match(&tok, "%var% ( %any%")) {
|
||||||
const Token* secondParameter = tok.tokAt(2)->nextArgument();
|
const Token* secondParameter = tok.tokAt(2)->nextArgument();
|
||||||
if (secondParameter && ((value == 0 && secondParameter->str() == "0") || (Token::Match(secondParameter, "%var%") && secondParameter->varId() > 0)))
|
if (secondParameter && ((value == 0 && secondParameter->str() == "0") || (Token::Match(secondParameter, "%var%") && secondParameter->varId() > 0))) {
|
||||||
if (functionNames2.find(tok.str()) != functionNames2.end())
|
if (functionNames2_all.find(tok.str()) != functionNames2_all.end())
|
||||||
var.push_back(secondParameter);
|
var.push_back(secondParameter);
|
||||||
|
else if (value == 0 && functionNames2_nullptr.find(tok.str()) != functionNames2_nullptr.end())
|
||||||
|
var.push_back(secondParameter);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (Token::Match(&tok, "printf|sprintf|snprintf|fprintf|fnprintf|scanf|sscanf|fscanf")) {
|
if (Token::Match(&tok, "printf|sprintf|snprintf|fprintf|fnprintf|scanf|sscanf|fscanf")) {
|
||||||
|
@ -155,13 +172,20 @@ void CheckNullPointer::parseFunctionCall(const Token &tok, std::list<const Token
|
||||||
for (std::string::iterator i = formatString.begin(); i != formatString.end(); ++i) {
|
for (std::string::iterator i = formatString.begin(); i != formatString.end(); ++i) {
|
||||||
if (*i == '%') {
|
if (*i == '%') {
|
||||||
percent = !percent;
|
percent = !percent;
|
||||||
} else if (percent && std::isalpha(*i)) {
|
} else if (percent) {
|
||||||
|
while (!std::isalpha(*i)) {
|
||||||
|
if (*i == '*')
|
||||||
|
argListTok = argListTok->nextArgument();
|
||||||
|
++i;
|
||||||
|
}
|
||||||
|
|
||||||
if ((*i == 'n' || *i == 's' || scan) && (!scan || value == 0)) {
|
if ((*i == 'n' || *i == 's' || scan) && (!scan || value == 0)) {
|
||||||
if ((value == 0 && argListTok->str() == "0") || (Token::Match(argListTok, "%var%") && argListTok->varId() > 0)) {
|
if ((value == 0 && argListTok->str() == "0") || (Token::Match(argListTok, "%var%") && argListTok->varId() > 0)) {
|
||||||
var.push_back(argListTok);
|
var.push_back(argListTok);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (*i != 'm') // %m is a non-standard glibc extension that requires no parameter
|
||||||
argListTok = argListTok->nextArgument(); // Find next argument
|
argListTok = argListTok->nextArgument(); // Find next argument
|
||||||
if (!argListTok)
|
if (!argListTok)
|
||||||
break;
|
break;
|
||||||
|
|
|
@ -1124,6 +1124,15 @@ private:
|
||||||
" memcmp(bar(xyz()), 0, 123);\n"
|
" memcmp(bar(xyz()), 0, 123);\n"
|
||||||
"}\n");
|
"}\n");
|
||||||
ASSERT_EQUALS("[test.cpp:3]: (error) Null pointer dereference\n", errout.str());
|
ASSERT_EQUALS("[test.cpp:3]: (error) Null pointer dereference\n", errout.str());
|
||||||
|
|
||||||
|
check("void foo(const char *s)\n"
|
||||||
|
"{\n"
|
||||||
|
" char *p = malloc(100);\n"
|
||||||
|
" frexp(1.0, p);\n"
|
||||||
|
" char *q = 0;\n"
|
||||||
|
" frexp(1.0, q);\n"
|
||||||
|
"}\n");
|
||||||
|
ASSERT_EQUALS("[test.cpp:6]: (error) Possible null pointer dereference: q\n", errout.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
// Check if pointer is null and the dereference it
|
// Check if pointer is null and the dereference it
|
||||||
|
@ -1506,6 +1515,17 @@ private:
|
||||||
" printf(\"%s\", s);\n"
|
" printf(\"%s\", s);\n"
|
||||||
"}");
|
"}");
|
||||||
ASSERT_EQUALS("", errout.str());
|
ASSERT_EQUALS("", errout.str());
|
||||||
|
|
||||||
|
|
||||||
|
check("void f(char* s) {\n"
|
||||||
|
" printf(\"text: %m%s, %s\", s, 0);\n"
|
||||||
|
"}");
|
||||||
|
ASSERT_EQUALS("[test.cpp:2]: (error) Null pointer dereference\n", errout.str());
|
||||||
|
|
||||||
|
check("void f(char* s) {\n"
|
||||||
|
" printf(\"text: %*s, %s\", s, 0);\n"
|
||||||
|
"}");
|
||||||
|
ASSERT_EQUALS("[test.cpp:2]: (error) Null pointer dereference\n", errout.str());
|
||||||
}
|
}
|
||||||
|
|
||||||
void scanf_with_invalid_va_argument() {
|
void scanf_with_invalid_va_argument() {
|
||||||
|
|
Loading…
Reference in New Issue