2009-10-19 20:57:11 +02:00
/*
* Cppcheck - A tool for static C / C + + code analysis
2013-01-01 17:29:08 +01:00
* Copyright ( C ) 2007 - 2013 Daniel Marjamäki and Cppcheck team .
2009-10-19 20:57:11 +02: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 "checkexceptionsafety.h"
2011-12-09 19:53:00 +01:00
# include "symboldatabase.h"
2009-10-19 20:57:11 +02:00
# include "token.h"
//---------------------------------------------------------------------------
// Register CheckExceptionSafety..
2011-10-13 20:53:06 +02:00
namespace {
CheckExceptionSafety instance ;
2009-10-19 20:57:11 +02:00
}
//---------------------------------------------------------------------------
void CheckExceptionSafety : : destructors ( )
{
2011-12-09 19:53:00 +01:00
const SymbolDatabase * const symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
2009-10-19 20:57:11 +02:00
2011-12-09 19:53:00 +01:00
// Perform check..
2012-10-14 17:40:17 +02:00
const std : : size_t functions = symbolDatabase - > functionScopes . size ( ) ;
for ( std : : size_t i = 0 ; i < functions ; + + i ) {
const Scope * scope = symbolDatabase - > functionScopes [ i ] ;
const Function * j = scope - > function ;
if ( j ) {
2011-12-09 19:53:00 +01:00
// only looking for destructors
2012-10-14 17:40:17 +02:00
if ( j - > type = = Function : : eDestructor ) {
2011-12-09 19:53:00 +01:00
// Inspect this destructor..
2012-10-14 17:40:17 +02:00
for ( const Token * tok = scope - > classStart - > next ( ) ; tok ! = scope - > classEnd ; tok = tok - > next ( ) ) {
2012-06-17 14:33:18 +02:00
// Skip try blocks
if ( Token : : simpleMatch ( tok , " try { " ) ) {
tok = tok - > next ( ) - > link ( ) ;
}
2011-12-09 19:53:00 +01:00
// throw found within a destructor
if ( tok - > str ( ) = = " throw " ) {
destructorsError ( tok ) ;
break ;
}
}
2009-10-19 20:57:11 +02:00
}
}
}
}
2009-10-31 18:58:42 +01:00
2009-11-08 09:54:08 +01:00
void CheckExceptionSafety : : deallocThrow ( )
{
2012-01-21 19:11:06 +01:00
if ( ! _settings - > isEnabled ( " style " ) )
return ;
2011-12-09 19:53:00 +01:00
const SymbolDatabase * const symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
2010-12-31 13:58:17 +01:00
// Deallocate a global/member pointer and then throw exception
// the pointer will be a dead pointer
2012-10-14 17:40:17 +02:00
const std : : size_t functions = symbolDatabase - > functionScopes . size ( ) ;
for ( std : : size_t i = 0 ; i < functions ; + + i ) {
const Scope * scope = symbolDatabase - > functionScopes [ i ] ;
for ( const Token * tok = scope - > classStart - > next ( ) ; tok ! = scope - > classEnd ; tok = tok - > next ( ) ) {
// only looking for delete now
if ( tok - > str ( ) ! = " delete " )
continue ;
// Check if this is something similar with: "delete p;"
tok = tok - > next ( ) ;
if ( Token : : simpleMatch ( tok , " [ ] " ) )
tok = tok - > tokAt ( 2 ) ;
if ( ! tok )
break ;
if ( ! Token : : Match ( tok , " %var% ; " ) )
continue ;
const unsigned int varid ( tok - > varId ( ) ) ;
if ( varid = = 0 )
continue ;
// we only look for global variables
const Variable * var = symbolDatabase - > getVariableFromVarId ( varid ) ;
if ( ! var | | ! ( var - > isGlobal ( ) | | var - > isStatic ( ) ) )
continue ;
// Token where throw occurs
const Token * ThrowToken = 0 ;
// is there a throw after the deallocation?
const Token * const end2 = tok - > scope ( ) - > classEnd ;
for ( const Token * tok2 = tok ; tok2 ! = end2 ; tok2 = tok2 - > next ( ) ) {
// Throw after delete -> Dead pointer
if ( tok2 - > str ( ) = = " throw " ) {
if ( _settings - > inconclusive ) { // For inconclusive checking, throw directly.
deallocThrowError ( tok2 , tok - > str ( ) ) ;
break ;
}
ThrowToken = tok2 ;
2012-01-21 19:11:06 +01:00
}
2009-11-08 09:54:08 +01:00
2012-10-14 17:40:17 +02:00
// Variable is assigned -> Bail out
else if ( Token : : Match ( tok2 , " %varid% = " , varid ) ) {
if ( ThrowToken ) // For non-inconclusive checking, wait until we find an assignement to it. Otherwise we assume it is safe to leave a dead pointer.
deallocThrowError ( ThrowToken , tok2 - > str ( ) ) ;
break ;
}
// Variable passed to function. Assume it becomes assigned -> Bail out
else if ( Token : : Match ( tok2 , " [,(] &| %varid% [,)] " , varid ) ) // TODO: No bailout if passed by value or as const reference
break ;
2009-11-08 09:54:08 +01:00
}
}
}
}
2011-02-05 10:11:09 +01:00
//---------------------------------------------------------------------------
// catch(const exception & err)
// {
// throw err; // <- should be just "throw;"
// }
//---------------------------------------------------------------------------
void CheckExceptionSafety : : checkRethrowCopy ( )
{
2011-08-07 09:28:08 +02:00
if ( ! _settings - > isEnabled ( " style " ) )
2011-02-05 10:11:09 +01:00
return ;
2012-01-26 16:50:59 +01:00
const SymbolDatabase * const symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
2011-02-05 10:11:09 +01:00
2012-01-26 16:50:59 +01:00
for ( std : : list < Scope > : : const_iterator i = symbolDatabase - > scopeList . begin ( ) ; i ! = symbolDatabase - > scopeList . end ( ) ; + + i ) {
if ( i - > type ! = Scope : : eCatch )
continue ;
const unsigned int varid = i - > classStart - > tokAt ( - 2 ) - > varId ( ) ;
if ( varid ) {
2012-02-02 16:17:42 +01:00
for ( const Token * tok = i - > classStart - > next ( ) ; tok & & tok ! = i - > classEnd ; tok = tok - > next ( ) ) {
if ( Token : : simpleMatch ( tok , " catch ( " ) & & tok - > next ( ) - > link ( ) & & tok - > next ( ) - > link ( ) - > next ( ) ) // Don't check inner catch - it is handled in another iteration of outer loop.
tok = tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) ;
else if ( Token : : Match ( tok , " throw %varid% ; " , varid ) )
rethrowCopyError ( tok , tok - > strAt ( 1 ) ) ;
}
2012-01-26 16:50:59 +01:00
}
2011-02-05 10:11:09 +01:00
}
}
2012-02-02 16:17:42 +01:00
//---------------------------------------------------------------------------
// try {} catch (std::exception err) {} <- Should be "std::exception& err"
//---------------------------------------------------------------------------
void CheckExceptionSafety : : checkCatchExceptionByValue ( )
{
if ( ! _settings - > isEnabled ( " style " ) )
return ;
const SymbolDatabase * const symbolDatabase = _tokenizer - > getSymbolDatabase ( ) ;
for ( std : : list < Scope > : : const_iterator i = symbolDatabase - > scopeList . begin ( ) ; i ! = symbolDatabase - > scopeList . end ( ) ; + + i ) {
if ( i - > type ! = Scope : : eCatch )
continue ;
// Find a pass-by-value declaration in the catch(), excluding basic types
// e.g. catch (std::exception err)
const Variable * var = symbolDatabase - > getVariableFromVarId ( i - > classStart - > tokAt ( - 2 ) - > varId ( ) ) ;
if ( var & & var - > isClass ( ) & & ! var - > isPointer ( ) & & ! var - > isReference ( ) )
catchExceptionByValueError ( i - > classDef ) ;
}
}