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:
PKEuS 2011-11-28 22:32:07 +02:00 committed by Reijo Tomperi
parent 39af75abb4
commit be0acad11f
2 changed files with 95 additions and 51 deletions

View File

@ -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;

View File

@ -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() {