2010-10-31 11:51:25 +01:00
/*
* Cppcheck - A tool for static C / C + + code analysis
2011-01-09 20:33:36 +01:00
* Copyright ( C ) 2007 - 2011 Daniel Marjamäki and Cppcheck team .
2010-10-31 11:51:25 +01:00
*
* This program is free software : you can redistribute it and / or modify
* it under the terms of the GNU General Public License as published by
* the Free Software Foundation , either version 3 of the License , or
* ( at your option ) any later version .
*
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program . If not , see < http : //www.gnu.org/licenses/>.
*/
//---------------------------------------------------------------------------
# include "checknullpointer.h"
# include "executionpath.h"
# include "mathlib.h"
2011-03-06 21:23:33 +01:00
# include "symboldatabase.h"
2011-10-12 22:11:27 +02:00
# include <cctype>
2010-10-31 11:51:25 +01:00
//---------------------------------------------------------------------------
// Register this check class (by creating a static instance of it)
2011-10-13 20:53:06 +02:00
namespace {
CheckNullPointer instance ;
2010-10-31 11:51:25 +01:00
}
//---------------------------------------------------------------------------
2011-01-17 19:23:00 +01:00
/** Is string uppercase? */
bool CheckNullPointer : : isUpper ( const std : : string & str )
{
2011-10-13 20:53:06 +02:00
for ( unsigned int i = 0 ; i < str . length ( ) ; + + i ) {
2011-01-17 19:23:00 +01:00
if ( str [ i ] > = ' a ' & & str [ i ] < = ' z ' )
return false ;
}
return true ;
}
2010-10-31 11:51:25 +01:00
/**
* @ brief parse a function call and extract information about variable usage
* @ param tok first token
* @ param var variables that the function read / write .
* @ param value 0 = > invalid with null pointers as parameter .
* 1 - . . = > invalid with uninitialized data .
*/
void CheckNullPointer : : parseFunctionCall ( const Token & tok , std : : list < const Token * > & var , unsigned char value )
{
// standard functions that dereference first parameter..
// both uninitialized data and null pointers are invalid.
static std : : set < std : : string > functionNames1 ;
2011-10-13 20:53:06 +02:00
if ( functionNames1 . empty ( ) ) {
2010-10-31 11:51:25 +01:00
functionNames1 . insert ( " memchr " ) ;
functionNames1 . insert ( " memcmp " ) ;
functionNames1 . insert ( " strcat " ) ;
functionNames1 . insert ( " strncat " ) ;
functionNames1 . insert ( " strchr " ) ;
functionNames1 . insert ( " strrchr " ) ;
functionNames1 . insert ( " strcmp " ) ;
functionNames1 . insert ( " strncmp " ) ;
functionNames1 . insert ( " strdup " ) ;
functionNames1 . insert ( " strndup " ) ;
functionNames1 . insert ( " strlen " ) ;
functionNames1 . insert ( " strstr " ) ;
2011-01-30 09:34:58 +01:00
functionNames1 . insert ( " fclose " ) ;
functionNames1 . insert ( " feof " ) ;
functionNames1 . insert ( " fwrite " ) ;
functionNames1 . insert ( " fseek " ) ;
functionNames1 . insert ( " ftell " ) ;
functionNames1 . insert ( " fgetpos " ) ;
functionNames1 . insert ( " fsetpos " ) ;
functionNames1 . insert ( " rewind " ) ;
2011-10-12 22:11:27 +02:00
functionNames1 . insert ( " scanf " ) ;
functionNames1 . insert ( " fscanf " ) ;
functionNames1 . insert ( " sscanf " ) ;
2010-10-31 11:51:25 +01:00
}
// standard functions that dereference second parameter..
// both uninitialized data and null pointers are invalid.
static std : : set < std : : string > functionNames2 ;
2011-10-13 20:53:06 +02:00
if ( functionNames2 . empty ( ) ) {
2010-10-31 11:51:25 +01:00
functionNames2 . insert ( " memcmp " ) ;
functionNames2 . insert ( " memcpy " ) ;
functionNames2 . insert ( " memmove " ) ;
functionNames2 . insert ( " strcat " ) ;
functionNames2 . insert ( " strncat " ) ;
functionNames2 . insert ( " strcmp " ) ;
functionNames2 . insert ( " strncmp " ) ;
functionNames2 . insert ( " strcpy " ) ;
functionNames2 . insert ( " strncpy " ) ;
functionNames2 . insert ( " strstr " ) ;
2011-10-12 22:11:27 +02:00
functionNames2 . insert ( " sprintf " ) ;
functionNames2 . insert ( " fprintf " ) ;
functionNames2 . insert ( " fscanf " ) ;
functionNames2 . insert ( " sscanf " ) ;
2010-10-31 11:51:25 +01:00
}
// 1st parameter..
2010-12-26 19:29:58 +01:00
if ( ( Token : : Match ( & tok , " %var% ( %var% ,|) " ) & & tok . tokAt ( 2 ) - > varId ( ) > 0 ) | |
2011-10-13 20:53:06 +02:00
( value = = 0 & & Token : : Match ( & tok , " %var% ( 0 ,|) " ) ) ) {
2010-10-31 11:51:25 +01:00
if ( functionNames1 . find ( tok . str ( ) ) ! = functionNames1 . end ( ) )
var . push_back ( tok . tokAt ( 2 ) ) ;
2011-10-12 22:11:27 +02:00
else if ( value = = 0 & & Token : : Match ( & tok , " memcpy|memmove|memset|strcpy|printf|sprintf|vsprintf|vprintf|fprintf|vfprintf " ) )
2011-06-26 21:46:33 +02:00
var . push_back ( tok . tokAt ( 2 ) ) ;
2011-10-12 22:11:27 +02:00
else if ( value = = 0 & & Token : : Match ( & tok , " snprintf|vsnprintf|fnprintf|vfnprintf " ) & & tok . strAt ( 4 ) ! = " 0 " )
2010-10-31 11:51:25 +01:00
var . push_back ( tok . tokAt ( 2 ) ) ;
2011-01-06 08:12:34 +01:00
else if ( value ! = 0 & & Token : : simpleMatch ( & tok , " fflush " ) )
2010-10-31 11:51:25 +01:00
var . push_back ( tok . tokAt ( 2 ) ) ;
}
// 2nd parameter..
2011-10-23 11:23:48 +02:00
if ( Token : : Match ( & tok , " %var% ( %any% " ) ) {
const Token * secondParameter = tok . tokAt ( 2 ) - > nextArgument ( ) ;
if ( secondParameter & & ( ( value = = 0 & & secondParameter - > str ( ) = = " 0 " ) | | ( Token : : Match ( secondParameter , " %var% " ) & & secondParameter - > varId ( ) > 0 ) ) )
if ( functionNames2 . find ( tok . str ( ) ) ! = functionNames2 . end ( ) )
2011-10-23 15:23:36 +02:00
var . push_back ( secondParameter ) ;
2010-10-31 11:51:25 +01:00
}
2011-09-05 20:18:58 +02:00
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( & tok , " printf|sprintf|snprintf|fprintf|fnprintf|scanf|sscanf|fscanf " ) ) {
2011-10-12 22:11:27 +02:00
const Token * argListTok = 0 ; // Points to first va_list argument
std : : string formatString ;
bool scan = Token : : Match ( & tok , " scanf|sscanf|fscanf " ) ;
2011-10-23 11:23:48 +02:00
if ( Token : : Match ( & tok , " printf|scanf ( %str% " ) ) {
2011-10-12 22:11:27 +02:00
formatString = tok . strAt ( 2 ) ;
2011-10-23 11:23:48 +02:00
if ( tok . strAt ( 3 ) = = " , " )
argListTok = tok . tokAt ( 4 ) ;
else
argListTok = 0 ;
} else if ( Token : : Match ( & tok , " sprintf|fprintf|sscanf|fscanf ( %any% " ) ) {
const Token * formatStringTok = tok . tokAt ( 2 ) - > nextArgument ( ) ; // Find second parameter (format string)
if ( formatStringTok & & Token : : Match ( formatStringTok , " %str% " ) ) {
argListTok = formatStringTok - > nextArgument ( ) ; // Find third parameter (first argument of va_args)
formatString = formatStringTok - > str ( ) ;
}
} else if ( Token : : Match ( & tok , " snprintf|fnprintf ( %any% " ) ) {
const Token * formatStringTok = tok . tokAt ( 2 ) ;
for ( int i = 0 ; i < 2 & & formatStringTok ; i + + ) {
formatStringTok = formatStringTok - > nextArgument ( ) ; // Find third parameter (format string)
}
if ( formatStringTok & & Token : : Match ( formatStringTok , " %str% " ) ) {
argListTok = formatStringTok - > nextArgument ( ) ; // Find fourth parameter (first argument of va_args)
formatString = formatStringTok - > str ( ) ;
}
2011-10-12 22:11:27 +02:00
}
2011-10-23 11:23:48 +02:00
2011-10-13 20:53:06 +02:00
if ( argListTok ) {
2011-10-12 22:11:27 +02:00
bool percent = false ;
2011-10-13 20:53:06 +02:00
for ( std : : string : : iterator i = formatString . begin ( ) ; i ! = formatString . end ( ) ; + + i ) {
if ( * i = = ' % ' ) {
2011-10-12 22:11:27 +02:00
percent = ! percent ;
2011-10-13 20:53:06 +02:00
} else if ( percent & & std : : isalpha ( * i ) ) {
2011-10-16 07:06:18 +02:00
if ( ( * i = = ' n ' | | * i = = ' s ' | | scan ) & & ( ! scan | | value = = 0 ) ) {
2011-10-13 20:53:06 +02:00
if ( ( value = = 0 & & argListTok - > str ( ) = = " 0 " ) | | ( Token : : Match ( argListTok , " %var% " ) & & argListTok - > varId ( ) > 0 ) ) {
2011-10-12 22:11:27 +02:00
var . push_back ( argListTok ) ;
}
}
2011-10-23 11:23:48 +02:00
argListTok = argListTok - > nextArgument ( ) ; // Find next argument
2011-10-16 07:50:34 +02:00
if ( ! argListTok )
2011-10-16 07:06:18 +02:00
break ;
2011-10-23 11:23:48 +02:00
2011-10-12 22:11:27 +02:00
percent = false ;
}
}
}
2011-09-05 20:18:58 +02:00
}
2010-10-31 11:51:25 +01:00
}
2010-10-31 15:32:19 +01:00
/**
* Is there a pointer dereference ? Everything that should result in
* a nullpointer dereference error message will result in a true
* return value . If it ' s unknown if the pointer is dereferenced false
* is returned .
* @ param tok token for the pointer
* @ param unknown it is not known if there is a pointer dereference ( could be reported as a debug message )
* @ return true = > there is a dereference
*/
bool CheckNullPointer : : isPointerDeRef ( const Token * tok , bool & unknown )
{
unknown = false ;
// Dereferencing pointer..
2011-04-03 21:06:42 +02:00
if ( Token : : Match ( tok - > tokAt ( - 3 ) , " !!sizeof [;{}=+-/(,] * %var% " ) & & Token : : Match ( tok - > tokAt ( - 3 ) , " !!decltype [;{}=+-/(,] * %var% " ) )
2010-10-31 15:32:19 +01:00
return true ;
2011-10-18 19:34:14 +02:00
// read/write member variable
if ( ! Token : : simpleMatch ( tok - > tokAt ( - 2 ) , " & ( " ) & & ! Token : : Match ( tok - > tokAt ( - 2 ) , " sizeof|decltype ( " ) & & tok - > strAt ( - 1 ) ! = " & " & & tok - > strAt ( - 1 ) ! = " && " & & Token : : Match ( tok - > next ( ) , " . %var% " ) ) {
if ( tok - > strAt ( 3 ) ! = " ( " )
return true ;
unknown = true ;
return false ;
}
2010-10-31 15:32:19 +01:00
2010-10-31 15:40:34 +01:00
if ( Token : : Match ( tok - > previous ( ) , " [;{}=+-/(,] %var% [ " ) )
2010-10-31 15:32:19 +01:00
return true ;
2010-10-31 15:40:34 +01:00
if ( Token : : Match ( tok - > previous ( ) , " return %var% [ " ) )
2010-10-31 15:32:19 +01:00
return true ;
if ( Token : : Match ( tok , " %var% ( " ) )
return true ;
2010-10-31 15:46:08 +01:00
if ( Token : : Match ( tok , " %var% = %var% . " ) & &
tok - > varId ( ) > 0 & &
tok - > varId ( ) = = tok - > tokAt ( 2 ) - > varId ( ) )
return true ;
2010-10-31 15:32:19 +01:00
// Not a dereference..
if ( Token : : Match ( tok - > previous ( ) , " [;{}] %var% = " ) )
return false ;
// unknown if it's a dereference
unknown = true ;
// assume that it's not a dereference (no false positives)
return false ;
}
2010-12-31 18:30:04 +01:00
bool CheckNullPointer : : isPointer ( const unsigned int varid )
{
// Check if given variable is a pointer
2011-03-06 21:23:33 +01:00
const SymbolDatabase * symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
const Variable * variableInfo = symbolDatabase - > getVariableFromVarId ( varid ) ;
const Token * tok = variableInfo ? variableInfo - > typeStartToken ( ) : NULL ;
if ( Token : : Match ( tok , " %type% %type% * %varid% [;)=] " , varid ) )
return true ;
2010-10-31 15:32:19 +01:00
2010-12-31 18:30:04 +01:00
// maybe not a pointer
if ( ! Token : : Match ( tok , " %type% * %varid% [;)=] " , varid ) )
return false ;
// it is a pointer
if ( ! tok - > previous ( ) | |
Token : : Match ( tok - > previous ( ) , " [({};] " ) | |
2011-10-13 20:53:06 +02:00
tok - > previous ( ) - > isName ( ) ) {
2010-12-31 18:30:04 +01:00
return true ;
}
// it is not a pointer
return false ;
}
2010-10-31 11:51:25 +01:00
void CheckNullPointer : : nullPointerAfterLoop ( )
{
// Locate insufficient null-pointer handling after loop
2011-10-13 20:53:06 +02:00
for ( const Token * tok = _tokenizer - > tokens ( ) ; tok ; tok = tok - > next ( ) ) {
2010-12-31 14:17:10 +01:00
// only interested in while ( %var% )
// TODO: Aren't there false negatives. Shouldn't other loops be handled such as:
// - while ( ! %var% )
// - while ( %var% && .. )
2011-07-25 15:45:00 +02:00
if ( ! Token : : Match ( tok , " while ( %var% )|&& " ) )
2010-10-31 11:51:25 +01:00
continue ;
2010-12-31 14:17:10 +01:00
// Get variable id for the loop variable
2010-10-31 11:51:25 +01:00
const unsigned int varid ( tok - > tokAt ( 2 ) - > varId ( ) ) ;
if ( varid = = 0 )
continue ;
2011-03-28 18:44:25 +02:00
// Is variable a pointer?
if ( ! isPointer ( varid ) )
continue ;
2010-12-31 14:17:10 +01:00
// Get variable name for the loop variable
2010-10-31 11:51:25 +01:00
const std : : string varname ( tok - > strAt ( 2 ) ) ;
2010-12-31 14:17:10 +01:00
// Locate the end of the while loop body..
2011-07-25 15:45:00 +02:00
const Token * tok2 = tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) ;
2010-10-31 11:51:25 +01:00
2011-10-09 15:09:02 +02:00
// Is this checking inconclusive?
bool inconclusive = false ;
2010-12-31 14:17:10 +01:00
// Check if the variable is dereferenced after the while loop
2011-10-13 20:53:06 +02:00
while ( 0 ! = ( tok2 = tok2 ? tok2 - > next ( ) : 0 ) ) {
2011-10-09 15:09:02 +02:00
// inner and outer scopes
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " { " | | tok2 - > str ( ) = = " } " ) {
2011-10-09 15:09:02 +02:00
// Not inconclusive: bail out
if ( ! _settings - > inconclusive )
break ;
inconclusive = true ;
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " } " ) {
2011-10-09 15:09:02 +02:00
// "}" => leaving function? then break.
const Token * tok3 = tok2 - > link ( ) - > previous ( ) ;
if ( ! tok3 | | ! Token : : Match ( tok3 , " [);] " ) )
break ;
if ( tok3 - > str ( ) = = " ) " & & ! Token : : Match ( tok3 - > link ( ) - > previous ( ) , " if|for|while " ) )
break ;
}
}
// Stop checking if "break" is found
else if ( tok2 - > str ( ) = = " break " )
2010-10-31 11:51:25 +01:00
break ;
2010-12-31 14:17:10 +01:00
// loop variable is found..
2011-10-13 20:53:06 +02:00
else if ( tok2 - > varId ( ) = = varid ) {
2010-12-31 14:17:10 +01:00
// dummy variable.. is it unknown if pointer is dereferenced or not?
2010-10-31 15:46:08 +01:00
bool unknown = false ;
2010-12-31 14:17:10 +01:00
// Is the loop variable dereferenced?
2011-10-13 20:53:06 +02:00
if ( CheckNullPointer : : isPointerDeRef ( tok2 , unknown ) ) {
2011-10-09 15:09:02 +02:00
nullPointerError ( tok2 , varname , tok - > linenr ( ) , inconclusive ) ;
2010-10-31 11:51:25 +01:00
}
2011-10-18 19:34:14 +02:00
else if ( unknown & & _settings - > inconclusive ) {
nullPointerError ( tok2 , varname , tok - > linenr ( ) , true ) ;
}
2010-10-31 11:51:25 +01:00
break ;
}
}
}
}
void CheckNullPointer : : nullPointerLinkedList ( )
{
2010-12-31 14:37:34 +01:00
// looping through items in a linked list in a inner loop.
// Here is an example:
// for (const Token *tok = tokens; tok; tok = tok->next) {
// if (tok->str() == "hello")
// tok = tok->next; // <- tok might become a null pointer!
// }
2011-10-13 20:53:06 +02:00
for ( const Token * tok1 = _tokenizer - > tokens ( ) ; tok1 ; tok1 = tok1 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
// search for a "for" token..
if ( ! Token : : simpleMatch ( tok1 , " for ( " ) )
continue ;
2010-12-31 14:37:34 +01:00
// is there any dereferencing occurring in the for statement
2011-03-30 16:45:31 +02:00
// parlevel2 counts the parentheses when using tok2.
2010-10-31 11:51:25 +01:00
unsigned int parlevel2 = 1 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok2 = tok1 - > tokAt ( 2 ) ; tok2 ; tok2 = tok2 - > next ( ) ) {
2010-12-15 18:45:53 +01:00
// Parentheses..
2010-10-31 11:51:25 +01:00
if ( tok2 - > str ( ) = = " ( " )
+ + parlevel2 ;
2011-10-13 20:53:06 +02:00
else if ( tok2 - > str ( ) = = " ) " ) {
2010-10-31 11:51:25 +01:00
if ( parlevel2 < = 1 )
break ;
- - parlevel2 ;
}
2010-12-15 18:45:53 +01:00
// Dereferencing a variable inside the "for" parentheses..
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok2 , " %var% . %var% " ) ) {
2010-12-31 14:37:34 +01:00
// Variable id for dereferenced variable
2010-10-31 11:51:25 +01:00
const unsigned int varid ( tok2 - > varId ( ) ) ;
if ( varid = = 0 )
continue ;
if ( Token : : Match ( tok2 - > tokAt ( - 2 ) , " %varid% ? " , varid ) )
continue ;
2010-12-31 14:37:34 +01:00
// Variable name of dereferenced variable
2010-10-31 11:51:25 +01:00
const std : : string varname ( tok2 - > str ( ) ) ;
// Check usage of dereferenced variable in the loop..
unsigned int indentlevel3 = 0 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok3 = tok1 - > next ( ) - > link ( ) ; tok3 ; tok3 = tok3 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok3 - > str ( ) = = " { " )
+ + indentlevel3 ;
2011-10-13 20:53:06 +02:00
else if ( tok3 - > str ( ) = = " } " ) {
2010-10-31 11:51:25 +01:00
if ( indentlevel3 < = 1 )
break ;
- - indentlevel3 ;
}
2010-12-31 14:37:34 +01:00
// TODO: are there false negatives for "while ( %varid% ||"
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok3 , " while ( %varid% &&|) " , varid)) {
2010-12-31 14:37:34 +01:00
// Make sure there is a "break" or "return" inside the loop.
// Without the "break" a null pointer could be dereferenced in the
// for statement.
// indentlevel4 is a counter for { and }. When scanning the code with tok4
2010-10-31 11:51:25 +01:00
unsigned int indentlevel4 = indentlevel3 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok4 = tok3 - > next ( ) - > link ( ) ; tok4 ; tok4 = tok4 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok4 - > str ( ) = = " { " )
+ + indentlevel4 ;
2011-10-13 20:53:06 +02:00
else if ( tok4 - > str ( ) = = " } " ) {
if ( indentlevel4 < = 1 ) {
2010-10-31 11:51:25 +01:00
// Is this variable a pointer?
2010-12-31 18:30:04 +01:00
if ( isPointer ( varid ) )
2010-10-31 11:51:25 +01:00
nullPointerError ( tok1 , varname , tok3 - > linenr ( ) ) ;
break ;
}
- - indentlevel4 ;
}
2010-12-31 14:37:34 +01:00
// There is a "break" or "return" inside the loop.
// TODO: there can be false negatives. There could still be
// execution paths that are not properly terminated
2010-10-31 11:51:25 +01:00
else if ( tok4 - > str ( ) = = " break " | | tok4 - > str ( ) = = " return " )
break ;
}
}
}
}
}
}
}
void CheckNullPointer : : nullPointerStructByDeRefAndChec ( )
{
2010-12-31 15:56:41 +01:00
// Dereferencing a struct pointer and then checking if it's NULL..
2010-12-31 15:57:08 +01:00
2010-12-31 15:56:41 +01:00
// skipvar: don't check vars that has been tested against null already
2010-10-31 11:51:25 +01:00
std : : set < unsigned int > skipvar ;
skipvar . insert ( 0 ) ;
2010-12-31 15:56:41 +01:00
// Scan through all tokens
2011-10-13 20:53:06 +02:00
for ( const Token * tok1 = _tokenizer - > tokens ( ) ; tok1 ; tok1 = tok1 - > next ( ) ) {
2010-12-31 15:56:41 +01:00
// Checking if some pointer is null.
// then add the pointer to skipvar => is it known that it isn't NULL
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok1 , " if|while ( !| %var% ) " ) ) {
2010-10-31 11:51:25 +01:00
tok1 = tok1 - > tokAt ( 2 ) ;
if ( tok1 - > str ( ) = = " ! " )
tok1 = tok1 - > next ( ) ;
skipvar . insert ( tok1 - > varId ( ) ) ;
continue ;
2011-10-13 20:53:06 +02:00
} else if ( Token : : Match ( tok1 , " ( ! %var% || " ) | |
Token : : Match ( tok1 , " ( %var% && " ) ) {
2011-10-08 07:39:20 +02:00
// TODO: there are false negatives caused by this. The
// variable should be removed from skipvar after the
// condition
2011-10-07 21:08:21 +02:00
tok1 = tok1 - > next ( ) ;
if ( tok1 - > str ( ) = = " ! " )
tok1 = tok1 - > next ( ) ;
skipvar . insert ( tok1 - > varId ( ) ) ;
continue ;
}
2010-10-31 11:51:25 +01:00
2011-02-19 21:01:38 +01:00
/**
* @ todo There are lots of false negatives here . A dereference
* is only investigated if a few specific conditions are met .
*/
// dereference in assignment
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok1 , " [;{}] %var% . %var% " ) ) {
2011-02-19 21:01:38 +01:00
tok1 = tok1 - > next ( ) ;
}
2010-10-31 11:51:25 +01:00
// dereference in assignment
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok1 , " [{};] %var% = %var% . %var% " ) ) {
2010-10-31 11:51:25 +01:00
if ( std : : string ( tok1 - > strAt ( 1 ) ) = = tok1 - > strAt ( 3 ) )
continue ;
tok1 = tok1 - > tokAt ( 3 ) ;
}
2011-07-30 19:51:06 +02:00
// dereference in condition
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok1 , " if ( !| %var% . " ) ) {
2011-07-30 19:51:06 +02:00
tok1 = tok1 - > tokAt ( 2 ) ;
2011-08-01 21:57:23 +02:00
if ( tok1 - > str ( ) = = " ! " )
tok1 = tok1 - > next ( ) ;
2011-07-30 19:51:06 +02:00
}
2011-04-03 21:06:42 +02:00
// dereference in function call (but not sizeof|decltype)
else if ( ( Token : : Match ( tok1 - > tokAt ( - 2 ) , " %var% ( %var% . %var% " ) & & ! Token : : Match ( tok1 - > tokAt ( - 2 ) , " sizeof|decltype ( %var% . %var% " ) ) | |
2011-10-13 20:53:06 +02:00
Token : : Match ( tok1 - > previous ( ) , " , %var% . %var% " ) ) {
2010-12-17 21:09:12 +01:00
// Is the function return value taken by the pointer?
bool assignment = false ;
const unsigned int varid1 ( tok1 - > varId ( ) ) ;
2010-12-19 21:27:28 +01:00
if ( varid1 = = 0 )
continue ;
2010-12-17 21:09:12 +01:00
const Token * tok2 = tok1 - > previous ( ) ;
2011-10-13 20:53:06 +02:00
while ( tok2 & & ! Token : : Match ( tok2 , " [;{}] " ) ) {
if ( Token : : Match ( tok2 , " %varid% = " , varid1 ) ) {
2010-12-17 21:09:12 +01:00
assignment = true ;
break ;
}
tok2 = tok2 - > previous ( ) ;
}
if ( assignment )
continue ;
2010-10-31 11:51:25 +01:00
}
// Goto next token
2011-10-13 20:53:06 +02:00
else {
2010-10-31 11:51:25 +01:00
continue ;
}
// struct dereference was found - investigate if it is later
// checked that it is not NULL
const unsigned int varid1 ( tok1 - > varId ( ) ) ;
if ( skipvar . find ( varid1 ) ! = skipvar . end ( ) )
continue ;
2010-12-31 15:56:41 +01:00
// name of struct pointer
2010-10-31 11:51:25 +01:00
const std : : string varname ( tok1 - > str ( ) ) ;
2011-03-12 17:42:58 +01:00
// is pointer local?
bool isLocal = false ;
const Variable * var = _tokenizer - > getSymbolDatabase ( ) - > getVariableFromVarId ( tok1 - > varId ( ) ) ;
2011-03-12 18:27:19 +01:00
if ( var & & ( var - > isLocal ( ) | | var - > isArgument ( ) ) )
2011-03-12 17:42:58 +01:00
isLocal = true ;
2011-03-12 20:57:19 +01:00
// member function may or may not nullify the pointer if it's global (#2647)
2011-10-13 20:53:06 +02:00
if ( ! isLocal ) {
2011-03-12 20:57:19 +01:00
const Token * tok2 = tok1 ;
while ( Token : : Match ( tok2 , " %var% . " ) )
tok2 = tok2 - > tokAt ( 2 ) ;
if ( Token : : Match ( tok2 , " %var% ( " ) )
continue ;
}
2010-12-31 15:56:41 +01:00
// count { and } using tok2
2010-10-31 11:51:25 +01:00
unsigned int indentlevel2 = 0 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok2 = tok1 - > tokAt ( 3 ) ; tok2 ; tok2 = tok2 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok2 - > str ( ) = = " { " )
+ + indentlevel2 ;
2011-10-13 20:53:06 +02:00
else if ( tok2 - > str ( ) = = " } " ) {
2011-07-30 19:51:06 +02:00
if ( indentlevel2 = = 0 )
2010-10-31 11:51:25 +01:00
break ;
- - indentlevel2 ;
}
2011-03-29 18:45:32 +02:00
// label / ?:
else if ( tok2 - > str ( ) = = " : " )
2010-10-31 11:51:25 +01:00
break ;
// Reassignment of the struct
2011-10-13 20:53:06 +02:00
else if ( tok2 - > varId ( ) = = varid1 ) {
if ( tok2 - > next ( ) - > str ( ) = = " = " ) {
2011-07-30 19:51:06 +02:00
// Avoid false positives when there is 'else if'
// TODO: can this be handled better?
if ( tok1 - > strAt ( - 2 ) = = " if " )
skipvar . insert ( varid1 ) ;
2010-10-31 11:51:25 +01:00
break ;
2011-07-30 19:51:06 +02:00
}
2010-10-31 11:51:25 +01:00
if ( Token : : Match ( tok2 - > tokAt ( - 2 ) , " [,(] & " ) )
break ;
}
// Loop..
/** @todo don't bail out if the variable is not used in the loop */
else if ( tok2 - > str ( ) = = " do " )
break ;
2011-03-28 17:28:21 +02:00
// return/break at base level => stop checking
else if ( indentlevel2 = = 0 & & ( tok2 - > str ( ) = = " return " | | tok2 - > str ( ) = = " break " ) )
2010-10-31 11:51:25 +01:00
break ;
2011-03-12 17:42:58 +01:00
// Function call: If the pointer is not a local variable it
2011-03-12 15:02:06 +01:00
// might be changed by the call.
else if ( Token : : Match ( tok2 , " [;{}] %var% ( " ) & &
2011-10-13 20:53:06 +02:00
Token : : simpleMatch ( tok2 - > tokAt ( 2 ) - > link ( ) , " ) ; " ) & & ! isLocal ) {
2011-03-12 15:02:06 +01:00
break ;
}
2010-12-31 15:56:41 +01:00
// Check if pointer is null.
2011-02-19 21:10:31 +01:00
// TODO: false negatives for "if (!p || .."
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok2 , " if ( !| %varid% ) | & & " , varid1)) {
2010-10-31 11:51:25 +01:00
// Is this variable a pointer?
2010-12-31 18:30:04 +01:00
if ( isPointer ( varid1 ) )
2010-10-31 11:51:25 +01:00
nullPointerError ( tok1 , varname , tok2 - > linenr ( ) ) ;
break ;
}
}
}
}
void CheckNullPointer : : nullPointerByDeRefAndChec ( )
{
2011-08-14 16:53:09 +02:00
const SymbolDatabase * symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
2010-10-31 11:51:25 +01:00
// Dereferencing a pointer and then checking if it's NULL..
2011-01-06 13:18:49 +01:00
// This check will first scan for the check. And then scan backwards
// from the check, searching for dereferencing.
2011-10-13 20:53:06 +02:00
for ( const Token * tok = _tokenizer - > tokens ( ) ; tok ; tok = tok - > next ( ) ) {
2011-01-06 13:18:49 +01:00
// TODO: false negatives.
// - logical operators
// - while
2011-10-13 20:53:06 +02:00
if ( tok - > str ( ) = = " if " & & Token : : Match ( tok - > previous ( ) , " ; if ( !| %var% )|%oror%|&& " ) ) {
2011-08-02 11:20:09 +02:00
const Token * vartok = tok - > tokAt ( 2 ) ;
if ( vartok - > str ( ) = = " ! " )
vartok = vartok - > next ( ) ;
2011-01-06 13:18:49 +01:00
// Variable id for pointer
2011-08-02 11:20:09 +02:00
const unsigned int varid ( vartok - > varId ( ) ) ;
2010-10-31 11:51:25 +01:00
if ( varid = = 0 )
continue ;
2011-01-06 13:18:49 +01:00
// Name of pointer
2011-08-02 11:20:09 +02:00
const std : : string varname ( vartok - > str ( ) ) ;
2010-10-31 11:51:25 +01:00
// Check that variable is a pointer..
2010-12-31 18:30:04 +01:00
if ( ! isPointer ( varid ) )
2010-10-31 11:51:25 +01:00
continue ;
2011-01-06 13:18:49 +01:00
// Token where pointer is declared
2011-08-14 16:53:09 +02:00
const Variable * var = symbolDatabase - > getVariableFromVarId ( varid ) ;
if ( ! var )
continue ;
const Token * const decltok = var - > nameToken ( ) ;
2010-12-31 18:30:04 +01:00
2011-10-13 20:53:06 +02:00
for ( const Token * tok1 = tok - > previous ( ) ; tok1 & & tok1 ! = decltok ; tok1 = tok1 - > previous ( ) ) {
if ( tok1 - > str ( ) = = " ) " & & Token : : Match ( tok1 - > link ( ) - > previous ( ) , " %var% ( " ) ) {
2011-06-21 18:45:30 +02:00
const Token * tok2 = tok1 - > link ( ) ;
2011-08-02 11:20:09 +02:00
while ( tok2 & & ! Token : : Match ( tok2 , " [;{}?:] " ) )
2011-06-21 18:45:30 +02:00
tok2 = tok2 - > previous ( ) ;
2011-08-02 11:20:09 +02:00
if ( Token : : Match ( tok2 , " [?:] " ) )
break ;
2011-06-21 18:45:30 +02:00
if ( Token : : Match ( tok2 , " [;{}] %varid% = %var% " , varid ) )
break ;
2010-12-17 21:09:12 +01:00
2011-07-31 14:07:35 +02:00
if ( Token : : Match ( tok1 - > link ( ) - > previous ( ) , " while ( %varid% " , varid ) )
break ;
2011-01-10 19:57:26 +01:00
2011-10-07 21:08:21 +02:00
if ( Token : : Match ( tok1 - > link ( ) , " ( ! %varid% || " , varid ) | |
2011-10-13 20:53:06 +02:00
Token : : Match ( tok1 - > link ( ) , " ( %varid% && " , varid ) ) {
2011-10-08 12:58:10 +02:00
tok1 = tok1 - > link ( ) ;
continue ;
}
2011-10-07 21:08:21 +02:00
2011-10-13 20:53:06 +02:00
if ( Token : : simpleMatch ( tok1 - > link ( ) - > previous ( ) , " sizeof ( " ) ) {
2011-07-31 14:07:35 +02:00
tok1 = tok1 - > link ( ) - > previous ( ) ;
continue ;
}
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok2 , " [;{}] %var% ( %varid% , " , varid ) ) {
2011-08-14 16:53:09 +02:00
std : : list < const Token * > varlist ;
parseFunctionCall ( * ( tok2 - > next ( ) ) , varlist , 0 ) ;
2011-10-13 20:53:06 +02:00
if ( ! varlist . empty ( ) & & varlist . front ( ) = = tok2 - > tokAt ( 3 ) ) {
2011-07-31 14:07:35 +02:00
nullPointerError ( tok2 - > tokAt ( 3 ) , varname , tok - > linenr ( ) ) ;
break ;
}
}
2011-08-02 11:20:09 +02:00
// calling unknown function => it might initialize the pointer
2011-08-14 16:53:09 +02:00
if ( ! ( var - > isLocal ( ) | | var - > isArgument ( ) ) )
2011-08-02 11:20:09 +02:00
break ;
2011-01-31 17:30:27 +01:00
}
2011-01-15 12:09:36 +01:00
if ( tok1 - > str ( ) = = " break " )
break ;
2011-10-13 20:53:06 +02:00
if ( tok1 - > varId ( ) = = varid ) {
2011-10-02 19:27:18 +02:00
// Don't write warning if the dereferencing is
// guarded by ?:
const Token * tok2 = tok1 - > previous ( ) ;
2011-10-13 20:53:06 +02:00
if ( tok2 & & ( tok2 - > isArithmeticalOp ( ) | | tok2 - > str ( ) = = " ( " ) ) {
while ( tok2 & & ! Token : : Match ( tok2 , " [;{}?:] " ) ) {
2011-10-02 19:27:18 +02:00
if ( tok2 - > str ( ) = = " ) " )
tok2 = tok2 - > link ( ) ;
tok2 = tok2 - > previous ( ) ;
}
}
if ( Token : : Match ( tok2 , " [?:] " ) )
continue ;
2011-01-06 13:18:49 +01:00
// unknown : this is set by isPointerDeRef if it is
// uncertain
2010-10-31 16:07:50 +01:00
bool unknown = false ;
2011-01-06 13:18:49 +01:00
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok1 - > tokAt ( - 2 ) , " %varid% = %varid% . " , varid ) ) {
2010-11-04 18:18:19 +01:00
break ;
2011-10-13 20:53:06 +02:00
} else if ( Token : : Match ( tok1 - > previous ( ) , " &&|%oror% " ) ) {
2011-08-20 09:47:55 +02:00
break ;
2011-10-13 20:53:06 +02:00
} else if ( Token : : Match ( tok1 - > tokAt ( - 2 ) , " &&|%oror% ! " ) ) {
2011-08-20 09:47:55 +02:00
break ;
2011-10-13 20:53:06 +02:00
} else if ( CheckNullPointer : : isPointerDeRef ( tok1 , unknown ) ) {
2010-10-31 11:51:25 +01:00
nullPointerError ( tok1 , varname , tok - > linenr ( ) ) ;
break ;
2011-10-13 20:53:06 +02:00
} else if ( Token : : simpleMatch ( tok1 - > previous ( ) , " & " ) ) {
2010-10-31 11:51:25 +01:00
break ;
2011-10-13 20:53:06 +02:00
} else if ( Token : : simpleMatch ( tok1 - > next ( ) , " = " ) ) {
2010-10-31 11:51:25 +01:00
break ;
}
}
else if ( tok1 - > str ( ) = = " { " ||
tok1 - > str ( ) = = " } " )
break ;
// label..
else if ( Token : : Match ( tok1 , " %type% : " ) )
break ;
}
}
}
}
void CheckNullPointer : : nullPointerByCheckAndDeRef ( )
{
// Check if pointer is NULL and then dereference it..
2011-01-06 13:18:49 +01:00
// used to check if a variable is a pointer.
// TODO: Use isPointer?
2010-11-04 21:09:32 +01:00
std : : set < unsigned int > pointerVariables ;
2011-01-06 13:18:49 +01:00
2011-10-13 20:53:06 +02:00
for ( const Token * tok = _tokenizer - > tokens ( ) ; tok ; tok = tok - > next ( ) ) {
2010-11-04 21:09:32 +01:00
if ( Token : : Match ( tok , " * %var% [;,)=] " ) )
pointerVariables . insert ( tok - > next ( ) - > varId ( ) ) ;
2011-10-13 20:53:06 +02:00
else if ( Token : : simpleMatch ( tok , " if ( " ) ) {
2011-01-06 13:18:49 +01:00
// TODO: investigate false negatives:
// - handle "while"?
// - if there are logical operators
// - if (x) { } else { ... }
2011-01-17 19:23:00 +01:00
// If the if-body ends with a unknown macro then bailout
{
2011-03-30 16:45:31 +02:00
// goto the end parenthesis
2011-01-17 19:23:00 +01:00
const Token * endpar = tok - > next ( ) - > link ( ) ;
2011-02-18 20:05:44 +01:00
const Token * endbody = Token : : simpleMatch ( endpar , " ) { " ) ? endpar - > next ( ) - > link ( ) : 0 ;
2011-01-17 19:23:00 +01:00
if ( endbody & &
Token : : Match ( endbody - > tokAt ( - 3 ) , " [;{}] %var% ; " ) & &
isUpper ( endbody - > tokAt ( - 2 ) - > str ( ) ) )
continue ;
}
2011-01-06 13:18:49 +01:00
// vartok : token for the variable
2010-11-04 21:22:29 +01:00
const Token * vartok = 0 ;
2011-02-19 21:28:18 +01:00
if ( Token : : Match ( tok , " if ( ! %var% )|&& " ) )
2010-11-04 21:22:29 +01:00
vartok = tok - > tokAt ( 3 ) ;
2011-02-20 14:38:49 +01:00
else if ( Token : : Match ( tok , " if|while ( %var% )|&& " ) )
2011-01-09 20:16:16 +01:00
vartok = tok - > tokAt ( 2 ) ;
2011-08-01 07:51:32 +02:00
else if ( Token : : Match ( tok , " if ( ! ( %var% = " ) )
vartok = tok - > tokAt ( 4 ) ;
2010-11-04 21:22:29 +01:00
else
continue ;
2011-01-06 13:18:49 +01:00
// variable id for pointer
2010-11-04 21:22:29 +01:00
const unsigned int varid ( vartok - > varId ( ) ) ;
2010-10-31 11:51:25 +01:00
if ( varid = = 0 )
continue ;
2011-01-06 13:18:49 +01:00
2011-03-28 18:48:27 +02:00
const unsigned int linenr = vartok - > linenr ( ) ;
2011-01-06 13:18:49 +01:00
// Check if variable is a pointer. TODO: Use isPointer?
2010-11-04 21:09:32 +01:00
if ( pointerVariables . find ( varid ) = = pointerVariables . end ( ) )
continue ;
2010-11-04 21:22:29 +01:00
2011-08-07 17:06:25 +02:00
if ( Token : : Match ( vartok - > next ( ) , " && ( %varid% = " , varid ) )
continue ;
2011-01-06 13:18:49 +01:00
// if this is true then it is known that the pointer is null
bool null = true ;
2011-01-09 20:16:16 +01:00
// start token = inside the if-body
const Token * tok1 = tok - > next ( ) - > link ( ) - > tokAt ( 2 ) ;
2011-03-27 11:04:58 +02:00
// indentlevel inside the if-body is 1
unsigned int indentlevel = 1 ;
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok , " if|while ( %var% )|&& " ) ) {
2011-01-09 20:16:16 +01:00
// pointer might be null
null = false ;
// start token = first token after the if/while body
2011-03-13 10:34:54 +01:00
tok1 = tok1 - > previous ( ) - > link ( ) ;
tok1 = tok1 ? tok1 - > next ( ) : NULL ;
if ( ! tok1 )
continue ;
2011-03-27 11:04:58 +02:00
// indentlevel at the base level is 0
indentlevel = 0 ;
2011-01-09 20:16:16 +01:00
}
2010-11-04 21:22:29 +01:00
// Name of the pointer
const std : : string & pointerName = vartok - > str ( ) ;
2011-10-29 14:55:52 +02:00
// Set to true if we would normally bail out the check.
bool inconclusive = false ;
2011-01-06 13:18:49 +01:00
// Count { and } for tok2
2011-10-13 20:53:06 +02:00
for ( const Token * tok2 = tok1 ; tok2 ; tok2 = tok2 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok2 - > str ( ) = = " { " )
+ + indentlevel ;
2011-10-13 20:53:06 +02:00
else if ( tok2 - > str ( ) = = " } " ) {
2010-10-31 11:51:25 +01:00
if ( indentlevel = = 0 )
break ;
- - indentlevel ;
2011-03-27 21:29:49 +02:00
2011-10-13 20:53:06 +02:00
if ( null & & indentlevel = = 0 ) {
2011-03-27 21:29:49 +02:00
// skip all "else" blocks because they are not executed in this execution path
while ( Token : : simpleMatch ( tok2 , " } else { " ) )
tok2 = tok2 - > tokAt ( 2 ) - > link ( ) ;
null = false ;
}
2010-10-31 11:51:25 +01:00
}
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok2 , " goto|return|continue|break|throw|if|switch " ) ) {
2011-07-25 21:58:00 +02:00
bool dummy = false ;
2010-11-01 19:21:08 +01:00
if ( Token : : Match ( tok2 , " return * %varid% " , varid ) )
2011-10-29 14:55:52 +02:00
nullPointerError ( tok2 , pointerName , linenr , inconclusive ) ;
2011-07-25 21:58:00 +02:00
else if ( Token : : Match ( tok2 , " return %varid% " , varid ) & &
CheckNullPointer : : isPointerDeRef ( tok2 - > next ( ) , dummy ) )
2011-10-29 14:55:52 +02:00
nullPointerError ( tok2 , pointerName , linenr , inconclusive ) ;
2010-10-31 11:51:25 +01:00
break ;
}
2010-11-29 20:19:31 +01:00
// parameters to sizeof are not dereferenced
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok2 , " decltype|sizeof ( " ) ) {
2010-11-29 20:19:31 +01:00
tok2 = tok2 - > next ( ) - > link ( ) ;
continue ;
}
2011-07-31 17:19:23 +02:00
// function call, check if pointer is dereferenced
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok2 , " %var% ( " ) ) {
2011-07-31 17:19:23 +02:00
std : : list < const Token * > var ;
parseFunctionCall ( * tok2 , var , 0 ) ;
2011-10-13 20:53:06 +02:00
for ( std : : list < const Token * > : : const_iterator it = var . begin ( ) ; it ! = var . end ( ) ; + + it ) {
if ( ( * it ) - > varId ( ) = = varid ) {
2011-10-29 14:55:52 +02:00
nullPointerError ( * it , pointerName , linenr , inconclusive ) ;
2011-07-31 17:19:23 +02:00
break ;
}
}
}
2010-12-16 21:41:50 +01:00
// calling unknown function (abort/init)..
if ( Token : : simpleMatch ( tok2 , " ) ; " ) & &
2011-09-03 19:43:27 +02:00
( Token : : Match ( tok2 - > link ( ) - > tokAt ( - 2 ) , " [;{}.] %var% ( " ) | |
2011-10-13 20:53:06 +02:00
Token : : Match ( tok2 - > link ( ) - > tokAt ( - 5 ) , " [;{}] ( * %var% ) ( " ) ) ) {
2011-10-29 14:55:52 +02:00
// noreturn function?
// If inside null pointer check we unknown function call, we must
// assume that it can terminate the program and possible null pointer
// error wont ever happen.
if ( tok2 - > strAt ( 2 ) = = " } " ) {
if ( ! _settings - > inconclusive ) {
2011-10-29 10:35:31 +02:00
break ;
2011-10-29 14:55:52 +02:00
}
inconclusive = true ;
2011-10-29 10:35:31 +02:00
}
2011-07-31 17:32:25 +02:00
// init function (global variables)
const SymbolDatabase * symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
const Variable * var = symbolDatabase - > getVariableFromVarId ( varid ) ;
if ( ! var | | ! ( var - > isLocal ( ) | | var - > isArgument ( ) ) )
break ;
2010-10-31 11:51:25 +01:00
}
2011-10-13 20:53:06 +02:00
if ( tok2 - > varId ( ) = = varid ) {
2011-01-06 13:18:49 +01:00
// unknown: this is set to true by isPointerDeRef if
// the function fails to determine if there
// is a dereference or not
2010-10-31 15:32:19 +01:00
bool unknown = false ;
2010-10-31 11:51:25 +01:00
if ( Token : : Match ( tok2 - > previous ( ) , " [;{}=] %var% = 0 ; " ) )
;
2010-10-31 15:32:19 +01:00
else if ( CheckNullPointer : : isPointerDeRef ( tok2 , unknown ) )
2011-10-29 14:55:52 +02:00
nullPointerError ( tok2 , pointerName , linenr , inconclusive ) ;
2010-10-31 11:51:25 +01:00
2011-10-29 15:48:54 +02:00
else if ( unknown & & _settings - > inconclusive )
nullPointerError ( tok2 , pointerName , linenr , true ) ;
2010-10-31 11:51:25 +01:00
else
break ;
}
}
}
}
}
void CheckNullPointer : : nullPointer ( )
{
nullPointerAfterLoop ( ) ;
nullPointerLinkedList ( ) ;
nullPointerStructByDeRefAndChec ( ) ;
nullPointerByDeRefAndChec ( ) ;
nullPointerByCheckAndDeRef ( ) ;
}
2010-12-15 18:45:53 +01:00
/** Dereferencing null constant (simplified token list) */
2010-10-31 11:51:25 +01:00
void CheckNullPointer : : nullConstantDereference ( )
{
// this is kept at 0 for all scopes that are not executing
unsigned int indentlevel = 0 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok = _tokenizer - > tokens ( ) ; tok ; tok = tok - > next ( ) ) {
2010-10-31 11:51:25 +01:00
// start of executable scope..
if ( indentlevel = = 0 & & Token : : Match ( tok , " ) const| { " ) )
indentlevel = 1 ;
2011-10-13 20:53:06 +02:00
else if ( indentlevel > = 1 ) {
2010-10-31 11:51:25 +01:00
if ( tok - > str ( ) = = " { " )
+ + indentlevel ;
2011-10-13 20:53:06 +02:00
else if ( tok - > str ( ) = = " } " ) {
2010-10-31 11:51:25 +01:00
if ( indentlevel < = 2 )
indentlevel = 0 ;
else
- - indentlevel ;
}
2010-11-29 20:19:31 +01:00
if ( tok - > str ( ) = = " ( " & & Token : : Match ( tok - > previous ( ) , " sizeof|decltype " ) )
2010-10-31 11:51:25 +01:00
tok = tok - > link ( ) ;
2011-10-13 20:53:06 +02:00
else if ( Token : : simpleMatch ( tok , " exit ( ) " )) {
2010-10-31 11:51:25 +01:00
// Goto end of scope
2011-10-13 20:53:06 +02:00
while ( tok & & tok - > str ( ) ! = " } " ) {
2010-10-31 11:51:25 +01:00
if ( tok - > str ( ) = = " { " )
tok = tok - > link ( ) ;
tok = tok - > next ( ) ;
}
if ( ! tok )
break ;
}
2011-10-13 20:53:06 +02:00
else if ( Token : : simpleMatch ( tok , " * 0 " ) ) {
if ( Token : : Match ( tok - > previous ( ) , " return|;|{|}|=|(|,|%op% " ) ) {
2010-10-31 11:51:25 +01:00
nullPointerError ( tok ) ;
}
}
2010-12-26 19:29:58 +01:00
2011-10-13 20:53:06 +02:00
else if ( indentlevel > 0 & & Token : : Match ( tok - > previous ( ) , " [= { } ; ] % var % ( " )) {
2010-12-26 19:29:58 +01:00
std : : list < const Token * > var ;
parseFunctionCall ( * tok , var , 0 ) ;
// is one of the var items a NULL pointer?
2011-10-13 20:53:06 +02:00
for ( std : : list < const Token * > : : const_iterator it = var . begin ( ) ; it ! = var . end ( ) ; + + it ) {
if ( ( * it ) - > str ( ) = = " 0 " ) {
2010-12-26 19:29:58 +01:00
nullPointerError ( * it ) ;
}
}
}
2010-10-31 11:51:25 +01:00
}
}
}
/// @addtogroup Checks
/// @{
/**
* @ brief % Check for null pointer usage ( using ExecutionPath )
*/
2011-10-13 20:53:06 +02:00
class Nullpointer : public ExecutionPath {
2010-10-31 11:51:25 +01:00
public :
/** Startup constructor */
2011-10-13 20:53:06 +02:00
Nullpointer ( Check * c ) : ExecutionPath ( c , 0 ) , null ( false ) {
2010-10-31 11:51:25 +01:00
}
private :
/** Create checking of specific variable: */
Nullpointer ( Check * c , const unsigned int id , const std : : string & name )
: ExecutionPath ( c , id ) ,
varname ( name ) ,
2011-10-13 20:53:06 +02:00
null ( false ) {
2010-10-31 11:51:25 +01:00
}
/** Copy this check */
2011-10-13 20:53:06 +02:00
ExecutionPath * copy ( ) {
2010-10-31 11:51:25 +01:00
return new Nullpointer ( * this ) ;
}
/** no implementation => compiler error if used by accident */
void operator = ( const Nullpointer & ) ;
/** is other execution path equal? */
2011-10-13 20:53:06 +02:00
bool is_equal ( const ExecutionPath * e ) const {
2010-10-31 11:51:25 +01:00
const Nullpointer * c = static_cast < const Nullpointer * > ( e ) ;
return ( varname = = c - > varname & & null = = c - > null ) ;
}
/** variable name for this check (empty => dummy check) */
const std : : string varname ;
/** is this variable null? */
bool null ;
/** variable is set to null */
2011-10-13 20:53:06 +02:00
static void setnull ( std : : list < ExecutionPath * > & checks , const unsigned int varid ) {
2010-10-31 11:51:25 +01:00
std : : list < ExecutionPath * > : : iterator it ;
2011-10-13 20:53:06 +02:00
for ( it = checks . begin ( ) ; it ! = checks . end ( ) ; + + it ) {
2010-10-31 11:51:25 +01:00
Nullpointer * c = dynamic_cast < Nullpointer * > ( * it ) ;
if ( c & & c - > varId = = varid )
c - > null = true ;
}
}
/**
* Dereferencing variable . Check if it is safe ( if the variable is null there ' s an error )
* @ param checks Checks
* @ param tok token where dereferencing happens
*/
2011-10-13 20:53:06 +02:00
static void dereference ( std : : list < ExecutionPath * > & checks , const Token * tok ) {
2010-10-31 11:51:25 +01:00
const unsigned int varid ( tok - > varId ( ) ) ;
std : : list < ExecutionPath * > : : iterator it ;
2011-10-13 20:53:06 +02:00
for ( it = checks . begin ( ) ; it ! = checks . end ( ) ; + + it ) {
2010-10-31 11:51:25 +01:00
Nullpointer * c = dynamic_cast < Nullpointer * > ( * it ) ;
2011-10-13 20:53:06 +02:00
if ( c & & c - > varId = = varid & & c - > null ) {
2010-10-31 11:51:25 +01:00
CheckNullPointer * checkNullPointer = dynamic_cast < CheckNullPointer * > ( c - > owner ) ;
2011-10-13 20:53:06 +02:00
if ( checkNullPointer ) {
2010-10-31 11:51:25 +01:00
checkNullPointer - > nullPointerError ( tok , c - > varname ) ;
return ;
}
}
}
}
/** parse tokens */
2011-10-13 20:53:06 +02:00
const Token * parse ( const Token & tok , std : : list < ExecutionPath * > & checks ) const {
if ( Token : : Match ( tok . previous ( ) , " [;{}] const| struct| %type% * %var% ; " ) ) {
2010-10-31 11:51:25 +01:00
const Token * vartok = tok . tokAt ( 2 ) ;
if ( tok . str ( ) = = " const " )
vartok = vartok - > next ( ) ;
2011-05-28 05:30:19 +02:00
if ( tok . str ( ) = = " struct " )
vartok = vartok - > next ( ) ;
2010-10-31 11:51:25 +01:00
if ( vartok - > varId ( ) ! = 0 )
checks . push_back ( new Nullpointer ( owner , vartok - > varId ( ) , vartok - > str ( ) ) ) ;
return vartok - > next ( ) ;
}
// Template pointer variable..
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( tok . previous ( ) , " [;{}] %type% ::|< " ) ) {
2010-10-31 11:51:25 +01:00
const Token * vartok = & tok ;
while ( Token : : Match ( vartok , " %type% :: " ) )
vartok = vartok - > tokAt ( 2 ) ;
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( vartok , " %type% < %type% " ) ) {
2010-10-31 11:51:25 +01:00
vartok = vartok - > tokAt ( 3 ) ;
while ( vartok & & ( vartok - > str ( ) = = " * " | | vartok - > isName ( ) ) )
vartok = vartok - > next ( ) ;
}
if ( vartok
& & ( vartok - > str ( ) = = " > " | | vartok - > isName ( ) )
2011-10-13 20:53:06 +02:00
& & Token : : Match ( vartok - > next ( ) , " * %var% ;|= " ) ) {
2010-10-31 11:51:25 +01:00
vartok = vartok - > tokAt ( 2 ) ;
checks . push_back ( new Nullpointer ( owner , vartok - > varId ( ) , vartok - > str ( ) ) ) ;
if ( Token : : simpleMatch ( vartok - > next ( ) , " = 0 ; " ) )
setnull ( checks , vartok - > varId ( ) ) ;
return vartok - > next ( ) ;
}
}
2011-10-13 20:53:06 +02:00
if ( Token : : simpleMatch ( & tok , " try { " ) ) {
2010-10-31 11:51:25 +01:00
// Bail out all used variables
unsigned int indentlevel = 0 ;
2011-10-13 20:53:06 +02:00
for ( const Token * tok2 = & tok ; tok2 ; tok2 = tok2 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok2 - > str ( ) = = " { " )
+ + indentlevel ;
2011-10-13 20:53:06 +02:00
else if ( tok2 - > str ( ) = = " } " ) {
2010-10-31 11:51:25 +01:00
if ( indentlevel = = 0 )
break ;
if ( indentlevel = = 1 & & ! Token : : simpleMatch ( tok2 , " } catch ( " ) )
return tok2 ;
- - indentlevel ;
2011-10-13 20:53:06 +02:00
} else if ( tok2 - > varId ( ) )
2010-10-31 11:51:25 +01:00
bailOutVar ( checks , tok2 - > varId ( ) ) ;
}
}
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( & tok , " %var% ( " ) ) {
2010-10-31 11:51:25 +01:00
if ( tok . str ( ) = = " sizeof " )
return tok . next ( ) - > link ( ) ;
// parse usage..
std : : list < const Token * > var ;
CheckNullPointer : : parseFunctionCall ( tok , var , 0 ) ;
for ( std : : list < const Token * > : : const_iterator it = var . begin ( ) ; it ! = var . end ( ) ; + + it )
dereference ( checks , * it ) ;
}
2010-12-09 18:17:56 +01:00
else if ( Token : : simpleMatch ( & tok , " ( 0 && " ) )
return tok . link ( ) ;
2011-10-13 20:53:06 +02:00
if ( tok . varId ( ) ! = 0 ) {
2011-01-06 13:18:49 +01:00
// unknown : not really used. it is passed to isPointerDeRef.
// if isPointerDeRef fails to determine if there
// is a dereference the this will be set to true.
2010-10-31 15:32:19 +01:00
bool unknown = false ;
2011-01-06 13:18:49 +01:00
2010-10-31 11:51:25 +01:00
if ( Token : : Match ( tok . previous ( ) , " [;{}=] %var% = 0 ; " ) )
setnull ( checks , tok . varId ( ) ) ;
2010-10-31 15:32:19 +01:00
else if ( CheckNullPointer : : isPointerDeRef ( & tok , unknown ) )
2010-10-31 11:51:25 +01:00
dereference ( checks , & tok ) ;
2011-10-18 19:34:14 +02:00
else if ( unknown & & owner - > inconclusiveFlag ( ) )
dereference ( checks , & tok ) ;
2010-10-31 11:51:25 +01:00
else
2010-10-31 15:32:19 +01:00
// TODO: Report debug warning that it's unknown if a
// pointer is dereferenced
2010-10-31 11:51:25 +01:00
bailOutVar ( checks , tok . varId ( ) ) ;
}
2011-10-13 20:53:06 +02:00
else if ( tok . str ( ) = = " delete " ) {
2010-10-31 11:51:25 +01:00
const Token * ret = tok . next ( ) ;
if ( Token : : simpleMatch ( ret , " [ ] " ) )
ret = ret - > tokAt ( 2 ) ;
if ( Token : : Match ( ret , " %var% ; " ) )
return ret - > next ( ) ;
}
2011-07-25 21:40:32 +02:00
2011-10-13 20:53:06 +02:00
else if ( tok . str ( ) = = " return " ) {
2011-07-25 21:40:32 +02:00
bool unknown = false ;
const Token * vartok = tok . next ( ) ;
if ( vartok - > str ( ) = = " * " )
vartok = vartok - > next ( ) ;
2011-10-13 20:53:06 +02:00
if ( vartok - > varId ( ) & & CheckNullPointer : : isPointerDeRef ( vartok , unknown ) ) {
2011-07-25 21:40:32 +02:00
dereference ( checks , vartok ) ;
}
}
2010-10-31 11:51:25 +01:00
return & tok ;
}
/** parse condition. @sa ExecutionPath::parseCondition */
2011-10-13 20:53:06 +02:00
bool parseCondition ( const Token & tok , std : : list < ExecutionPath * > & checks ) {
for ( const Token * tok2 = & tok ; tok2 ; tok2 = tok2 - > next ( ) ) {
2010-10-31 11:51:25 +01:00
if ( tok2 - > str ( ) = = " ( " | | tok2 - > str ( ) = = " ) " )
break ;
if ( Token : : Match ( tok2 , " [<>=] * %var% " ) )
dereference ( checks , tok2 - > tokAt ( 2 ) ) ;
}
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( & tok , " !| %var% ( " ) ) {
2010-10-31 11:51:25 +01:00
std : : list < const Token * > var ;
CheckNullPointer : : parseFunctionCall ( tok . str ( ) = = " ! " ? * tok . next ( ) : tok , var , 0 ) ;
for ( std : : list < const Token * > : : const_iterator it = var . begin ( ) ; it ! = var . end ( ) ; + + it )
dereference ( checks , * it ) ;
}
return ExecutionPath : : parseCondition ( tok , checks ) ;
}
2011-10-13 20:53:06 +02:00
void parseLoopBody ( const Token * tok , std : : list < ExecutionPath * > & checks ) const {
while ( tok ) {
2010-10-31 11:51:25 +01:00
if ( Token : : Match ( tok , " {|}|return|goto|break|if " ) )
return ;
const Token * next = parse ( * tok , checks ) ;
if ( next )
tok = tok - > next ( ) ;
}
}
} ;
/// @}
void CheckNullPointer : : executionPaths ( )
{
// Check for null pointer errors..
Nullpointer c ( this ) ;
checkExecutionPaths ( _tokenizer - > tokens ( ) , & c ) ;
}
void CheckNullPointer : : nullPointerError ( const Token * tok )
{
reportError ( tok , Severity : : error , " nullPointer " , " Null pointer dereference " ) ;
}
void CheckNullPointer : : nullPointerError ( const Token * tok , const std : : string & varname )
{
reportError ( tok , Severity : : error , " nullPointer " , " Possible null pointer dereference: " + varname ) ;
}
2011-10-09 15:09:02 +02:00
void CheckNullPointer : : nullPointerError ( const Token * tok , const std : : string & varname , const unsigned int line , bool inconclusive )
2010-10-31 11:51:25 +01:00
{
2011-10-09 15:09:02 +02:00
const std : : string errmsg ( " Possible null pointer dereference: " + varname + " - otherwise it is redundant to check if " + varname + " is null at line " + MathLib : : toString < unsigned int > ( line ) ) ;
if ( inconclusive )
reportInconclusiveError ( tok , Severity : : error , " nullPointer " , errmsg ) ;
else
reportError ( tok , Severity : : error , " nullPointer " , errmsg ) ;
2010-10-31 11:51:25 +01:00
}