2010-11-13 08:08:45 +01:00
/*
* Cppcheck - A tool for static C / C + + code analysis
2016-01-01 14:34:45 +01:00
* Copyright ( C ) 2007 - 2016 Cppcheck team .
2010-11-13 08:08:45 +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 "symboldatabase.h"
# include "tokenize.h"
# include "token.h"
# include "settings.h"
# include "errorlogger.h"
2015-11-29 10:49:10 +01:00
# include "utils.h"
2010-11-13 08:08:45 +01:00
# include <string>
2014-05-23 20:58:28 +02:00
# include <ostream>
2011-06-24 04:35:15 +02:00
# include <climits>
2014-05-23 14:17:39 +02:00
# include <iostream>
2015-01-09 21:34:57 +01:00
# include <iomanip>
2016-01-04 14:20:07 +01:00
# include <cctype>
2010-11-13 08:08:45 +01:00
//---------------------------------------------------------------------------
SymbolDatabase : : SymbolDatabase ( const Tokenizer * tokenizer , const Settings * settings , ErrorLogger * errorLogger )
: _tokenizer ( tokenizer ) , _settings ( settings ) , _errorLogger ( errorLogger )
{
2011-03-11 01:43:29 +01:00
// create global scope
2014-02-16 10:32:10 +01:00
scopeList . push_back ( Scope ( this , nullptr , nullptr ) ) ;
2011-03-11 01:43:29 +01:00
// pointer to current scope
Scope * scope = & scopeList . back ( ) ;
2012-04-18 13:00:34 +02:00
// Store current access in each scope (depends on evaluation progress)
std : : map < const Scope * , AccessControl > access ;
2015-04-10 14:18:52 +02:00
const bool printDebug = _settings - > debugwarnings ;
2011-03-11 01:43:29 +01:00
// find all scopes
2014-02-16 10:32:10 +01:00
for ( const Token * tok = _tokenizer - > tokens ( ) ; tok ; tok = tok ? tok - > next ( ) : nullptr ) {
2014-04-18 20:42:22 +02:00
// #5593 suggested to add here:
2014-04-18 21:19:49 +02:00
if ( _errorLogger )
2014-06-04 18:00:22 +02:00
_errorLogger - > reportProgress ( _tokenizer - > list . getSourceFilePath ( ) ,
2014-04-18 21:19:49 +02:00
" SymbolDatabase " ,
tok - > progressValue ( ) ) ;
2010-11-13 08:08:45 +01:00
// Locate next class
2015-06-29 19:56:02 +02:00
if ( ( _tokenizer - > isCPP ( ) & & Token : : Match ( tok , " class|struct|union|namespace ::| %name% {|:|::|< " ) & & tok - > strAt ( - 1 ) ! = " friend " )
2015-06-30 08:46:30 +02:00
| | ( _tokenizer - > isC ( ) & & Token : : Match ( tok , " struct|union %name% { " ) ) ) {
2012-11-30 06:03:58 +01:00
const Token * tok2 = tok - > tokAt ( 2 ) ;
2012-10-16 06:11:28 +02:00
2012-11-30 06:03:58 +01:00
if ( tok - > strAt ( 1 ) = = " :: " )
tok2 = tok2 - > next ( ) ;
2012-04-18 13:00:34 +02:00
2012-11-30 06:03:58 +01:00
while ( tok2 & & tok2 - > str ( ) = = " :: " )
tok2 = tok2 - > tokAt ( 2 ) ;
2015-01-19 16:15:11 +01:00
// skip over template args
if ( tok2 & & tok2 - > str ( ) = = " < " & & tok2 - > link ( ) )
tok2 = tok2 - > link ( ) - > next ( ) ;
2012-11-30 06:03:58 +01:00
// make sure we have valid code
2013-03-28 06:36:49 +01:00
if ( ! tok2 | | ! Token : : Match ( tok2 , " {|: " ) ) {
// check for qualified variable
if ( tok2 & & tok2 - > next ( ) ) {
if ( tok2 - > next ( ) - > str ( ) = = " ; " )
tok = tok2 - > next ( ) ;
2014-03-14 16:26:37 +01:00
else if ( Token : : simpleMatch ( tok2 - > next ( ) , " = { " ) &&
2015-05-23 23:18:59 +02:00
Token : : simpleMatch ( tok2 - > linkAt ( 2 ) , " } ; " ) )
2013-03-28 06:36:49 +01:00
tok = tok2 - > linkAt ( 2 ) - > next ( ) ;
2014-03-23 20:08:16 +01:00
else if ( Token : : Match ( tok2 - > next ( ) , " (| { " ) &&
tok2 - > next ( ) - > link ( ) - > strAt ( 1 ) = = " ; " )
2013-03-28 06:36:49 +01:00
tok = tok2 - > next ( ) - > link ( ) - > next ( ) ;
2015-01-19 06:38:54 +01:00
// skip variable declaration
else if ( Token : : Match ( tok2 , " *|& " ) )
continue ;
2013-03-28 06:36:49 +01:00
else
break ; // bail
continue ;
}
break ; // bail
}
2012-11-30 06:03:58 +01:00
Scope * new_scope = findScope ( tok - > next ( ) , scope ) ;
2012-10-16 19:12:27 +02:00
2012-11-30 06:03:58 +01:00
if ( new_scope ) {
// only create base list for classes and structures
if ( new_scope - > isClassOrStruct ( ) ) {
// goto initial '{'
2015-07-01 16:31:49 +02:00
if ( ! new_scope - > definedType ) {
_tokenizer - > syntaxError ( nullptr ) ; // #6808
}
2013-03-05 15:28:40 +01:00
tok2 = new_scope - > definedType - > initBaseInfo ( tok , tok2 ) ;
2012-11-30 06:03:58 +01:00
// make sure we have valid code
if ( ! tok2 ) {
break ;
}
}
2013-02-01 06:31:02 +01:00
// definition may be different than declaration
2015-06-29 19:56:02 +02:00
if ( _tokenizer - > isCPP ( ) & & tok - > str ( ) = = " class " ) {
2013-02-01 06:31:02 +01:00
access [ new_scope ] = Private ;
new_scope - > type = Scope : : eClass ;
} else if ( tok - > str ( ) = = " struct " ) {
access [ new_scope ] = Public ;
new_scope - > type = Scope : : eStruct ;
}
2012-11-30 06:03:58 +01:00
new_scope - > classDef = tok ;
new_scope - > classStart = tok2 ;
new_scope - > classEnd = tok2 - > link ( ) ;
scope = new_scope ;
tok = tok2 ;
} else {
scopeList . push_back ( Scope ( this , tok , scope ) ) ;
new_scope = & scopeList . back ( ) ;
if ( tok - > str ( ) = = " class " )
access [ new_scope ] = Private ;
else if ( tok - > str ( ) = = " struct " )
access [ new_scope ] = Public ;
2013-03-05 15:28:40 +01:00
// fill typeList...
if ( new_scope - > isClassOrStruct ( ) | | new_scope - > type = = Scope : : eUnion ) {
Type * new_type = findType ( tok - > next ( ) , scope ) ;
if ( ! new_type ) {
typeList . push_back ( Type ( new_scope - > classDef , new_scope , scope ) ) ;
new_type = & typeList . back ( ) ;
scope - > definedTypes . push_back ( new_type ) ;
} else
new_type - > classScope = new_scope ;
new_scope - > definedType = new_type ;
}
2012-11-30 06:03:58 +01:00
// only create base list for classes and structures
if ( new_scope - > isClassOrStruct ( ) ) {
// goto initial '{'
2013-03-05 15:28:40 +01:00
tok2 = new_scope - > definedType - > initBaseInfo ( tok , tok2 ) ;
2012-11-30 06:03:58 +01:00
// make sure we have valid code
if ( ! tok2 ) {
2014-03-27 19:06:13 +01:00
_tokenizer - > syntaxError ( tok ) ;
2012-11-30 06:03:58 +01:00
}
}
new_scope - > classStart = tok2 ;
new_scope - > classEnd = tok2 - > link ( ) ;
2011-02-19 20:38:00 +01:00
// make sure we have valid code
2012-11-30 06:03:58 +01:00
if ( ! new_scope - > classEnd ) {
2014-03-27 19:06:13 +01:00
_tokenizer - > syntaxError ( tok ) ;
2011-02-19 20:38:00 +01:00
}
2010-11-13 08:08:45 +01:00
2012-11-30 06:03:58 +01:00
// make the new scope the current scope
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( new_scope ) ;
2013-03-05 13:33:38 +01:00
scope = new_scope ;
2010-12-16 19:04:47 +01:00
2012-11-30 06:03:58 +01:00
tok = tok2 ;
}
2012-06-05 06:37:55 +02:00
}
// Namespace and unknown macro (#3854)
2014-08-31 11:27:12 +02:00
else if ( _tokenizer - > isCPP ( ) & &
2015-01-31 10:50:39 +01:00
Token : : Match ( tok , " namespace %name% %type% ( " ) & &
2012-06-21 19:00:53 +02:00
tok - > tokAt ( 2 ) - > isUpperCaseName ( ) & &
2012-06-05 06:37:55 +02:00
Token : : simpleMatch ( tok - > linkAt ( 3 ) , " ) { " ) ) {
scopeList . push_back ( Scope ( this , tok , scope ) ) ;
Scope * new_scope = & scopeList . back ( ) ;
access [ new_scope ] = Public ;
const Token * tok2 = tok - > linkAt ( 3 ) - > next ( ) ;
new_scope - > classStart = tok2 ;
new_scope - > classEnd = tok2 - > link ( ) ;
// make sure we have valid code
if ( ! new_scope - > classEnd ) {
scopeList . pop_back ( ) ;
break ;
}
// make the new scope the current scope
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( new_scope ) ;
2012-06-05 06:37:55 +02:00
scope = & scopeList . back ( ) ;
2010-11-13 08:08:45 +01:00
tok = tok2 ;
}
2011-01-20 18:02:52 +01:00
// forward declaration
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( tok , " class|struct|union %name% ; " ) & &
2012-11-30 06:03:58 +01:00
tok - > strAt ( - 1 ) ! = " friend " ) {
2013-03-05 13:33:38 +01:00
if ( ! findType ( tok - > next ( ) , scope ) ) {
// fill typeList..
typeList . push_back ( Type ( tok , 0 , scope ) ) ;
scope - > definedTypes . push_back ( & typeList . back ( ) ) ;
2012-11-30 06:03:58 +01:00
}
2011-01-20 18:02:52 +01:00
tok = tok - > tokAt ( 2 ) ;
}
2011-03-14 02:01:33 +01:00
// using namespace
2015-06-14 22:18:28 +02:00
else if ( _tokenizer - > isCPP ( ) & & Token : : Match ( tok , " using namespace ::| %type% ; | : : " )) {
2012-12-20 06:53:04 +01:00
Scope : : UsingInfo using_info ;
using_info . start = tok ; // save location
2014-04-10 16:11:11 +02:00
using_info . scope = findNamespace ( tok - > tokAt ( 2 ) , scope ) ;
2012-12-20 06:53:04 +01:00
scope - > usingList . push_back ( using_info ) ;
2011-03-14 02:01:33 +01:00
2013-02-14 06:31:41 +01:00
// check for global namespace
if ( tok - > strAt ( 2 ) = = " :: " )
tok = tok - > tokAt ( 4 ) ;
else
tok = tok - > tokAt ( 3 ) ;
// skip over qualification
while ( tok & & Token : : Match ( tok , " %type% :: " ) )
tok = tok - > tokAt ( 2 ) ;
2011-03-14 02:01:33 +01:00
}
2011-03-30 01:48:12 +02:00
// unnamed struct and union
2011-03-29 13:08:59 +02:00
else if ( Token : : Match ( tok , " struct|union { " ) & &
2015-01-31 10:50:39 +01:00
Token : : Match ( tok - > next ( ) - > link ( ) , " } *|&| %name% ;|[ " ) ) {
2011-03-27 19:48:41 +02:00
scopeList . push_back ( Scope ( this , tok , scope ) ) ;
Scope * new_scope = & scopeList . back ( ) ;
2012-04-18 13:00:34 +02:00
access [ new_scope ] = Public ;
2011-03-27 19:48:41 +02:00
2011-12-17 19:04:03 +01:00
const Token * varNameTok = tok - > next ( ) - > link ( ) - > next ( ) ;
if ( varNameTok - > str ( ) = = " * " ) {
varNameTok = varNameTok - > next ( ) ;
2012-01-26 17:04:25 +01:00
} else if ( varNameTok - > str ( ) = = " & " ) {
varNameTok = varNameTok - > next ( ) ;
2011-12-17 19:04:03 +01:00
}
2013-03-05 13:33:38 +01:00
typeList . push_back ( Type ( tok , new_scope , scope ) ) ;
new_scope - > definedType = & typeList . back ( ) ;
scope - > definedTypes . push_back ( & typeList . back ( ) ) ;
2015-01-30 21:56:27 +01:00
scope - > addVariable ( varNameTok , tok , tok , access [ scope ] , new_scope - > definedType , scope , & settings - > library ) ;
2011-03-27 19:48:41 +02:00
const Token * tok2 = tok - > next ( ) ;
new_scope - > classStart = tok2 ;
new_scope - > classEnd = tok2 - > link ( ) ;
// make sure we have valid code
2011-10-13 20:53:06 +02:00
if ( ! new_scope - > classEnd ) {
2011-03-27 19:48:41 +02:00
scopeList . pop_back ( ) ;
break ;
}
// make the new scope the current scope
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( new_scope ) ;
2013-03-05 13:33:38 +01:00
scope = new_scope ;
2011-03-27 19:48:41 +02:00
tok = tok2 ;
}
2014-09-24 06:09:08 +02:00
// anonymous struct, union and namespace
else if ( ( Token : : Match ( tok , " struct|union { " ) & &
Token : : simpleMatch ( tok - > next ( ) - > link ( ) , " } ; " ) ) | |
Token : : simpleMatch ( tok , " namespace { " ) ) {
2011-04-01 02:54:23 +02:00
scopeList . push_back ( Scope ( this , tok , scope ) ) ;
Scope * new_scope = & scopeList . back ( ) ;
2012-04-18 13:00:34 +02:00
access [ new_scope ] = Public ;
2011-04-01 02:54:23 +02:00
const Token * tok2 = tok - > next ( ) ;
new_scope - > classStart = tok2 ;
new_scope - > classEnd = tok2 - > link ( ) ;
2013-03-05 13:33:38 +01:00
typeList . push_back ( Type ( tok , new_scope , scope ) ) ;
new_scope - > definedType = & typeList . back ( ) ;
scope - > definedTypes . push_back ( & typeList . back ( ) ) ;
2011-04-01 02:54:23 +02:00
// make sure we have valid code
2011-10-13 20:53:06 +02:00
if ( ! new_scope - > classEnd ) {
2011-04-01 02:54:23 +02:00
scopeList . pop_back ( ) ;
break ;
}
// make the new scope the current scope
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( new_scope ) ;
2013-03-05 13:33:38 +01:00
scope = new_scope ;
2011-04-01 02:54:23 +02:00
tok = tok2 ;
}
2011-10-13 20:53:06 +02:00
else {
2011-04-01 02:54:23 +02:00
// check for end of scope
2011-10-13 20:53:06 +02:00
if ( tok = = scope - > classEnd ) {
2014-08-31 11:27:12 +02:00
access . erase ( scope ) ;
scope = const_cast < Scope * > ( scope - > nestedIn ) ;
2010-11-13 08:08:45 +01:00
continue ;
}
// check if in class or structure
2011-10-13 20:53:06 +02:00
else if ( scope - > type = = Scope : : eClass | | scope - > type = = Scope : : eStruct ) {
2014-02-16 11:47:52 +01:00
const Token * funcStart = nullptr ;
const Token * argStart = nullptr ;
2010-11-13 08:08:45 +01:00
// What section are we in..
if ( tok - > str ( ) = = " private: " )
2012-04-18 13:00:34 +02:00
access [ scope ] = Private ;
2010-11-13 08:08:45 +01:00
else if ( tok - > str ( ) = = " protected: " )
2012-04-18 13:00:34 +02:00
access [ scope ] = Protected ;
2012-04-09 11:19:19 +02:00
else if ( tok - > str ( ) = = " public: " | | tok - > str ( ) = = " __published: " )
2012-04-18 13:00:34 +02:00
access [ scope ] = Public ;
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( tok , " public|protected|private %name% : " ) ) {
2010-11-13 08:08:45 +01:00
if ( tok - > str ( ) = = " private " )
2012-04-18 13:00:34 +02:00
access [ scope ] = Private ;
2010-11-13 08:08:45 +01:00
else if ( tok - > str ( ) = = " protected " )
2012-04-18 13:00:34 +02:00
access [ scope ] = Protected ;
2012-02-11 12:26:48 +01:00
else
2012-04-18 13:00:34 +02:00
access [ scope ] = Public ;
2010-11-13 08:08:45 +01:00
tok = tok - > tokAt ( 2 ) ;
}
// class function?
2014-03-30 11:06:44 +02:00
else if ( isFunction ( tok , scope , & funcStart , & argStart ) ) {
if ( tok - > previous ( ) - > str ( ) ! = " :: " ) {
Function function ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// save the function definition argument start '('
function . argDef = argStart ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// save the access type
function . access = access [ scope ] ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// save the function name location
function . tokenDef = funcStart ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// save the function parent scope
function . nestedIn = scope ;
2012-05-24 17:40:43 +02:00
2014-03-30 11:06:44 +02:00
// operator function
2015-07-22 13:57:14 +02:00
if ( function . tokenDef - > isOperatorKeyword ( ) ) {
2015-01-08 05:45:31 +01:00
function . isOperator ( true ) ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// 'operator =' is special
if ( function . tokenDef - > str ( ) = = " operator= " )
function . type = Function : : eOperatorEqual ;
}
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// class constructor/destructor
else if ( function . tokenDef - > str ( ) = = scope - > className ) {
// destructor
if ( function . tokenDef - > previous ( ) - > str ( ) = = " ~ " )
function . type = Function : : eDestructor ;
// copy/move constructor?
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( function . tokenDef , " %name% ( const| %name% &|&& &| %name%| ) " ) | |
Token : : Match ( function . tokenDef , " %name% ( const| %name% < " ) ) {
2015-12-18 13:46:12 +01:00
const Token * typeTok = function . tokenDef - > tokAt ( 2 ) ;
if ( typeTok - > str ( ) = = " const " )
typeTok = typeTok - > next ( ) ;
if ( typeTok - > strAt ( 1 ) = = " < " ) { // TODO: Remove this branch (#4710)
if ( Token : : Match ( typeTok - > linkAt ( 1 ) , " > & %name%| ) " ) )
2014-03-30 11:06:44 +02:00
function . type = Function : : eCopyConstructor ;
2015-12-18 13:46:12 +01:00
else if ( Token : : Match ( typeTok - > linkAt ( 1 ) , " > &&|& & %name%| ) " ) )
2014-03-30 11:06:44 +02:00
function . type = Function : : eMoveConstructor ;
else
function . type = Function : : eConstructor ;
2015-12-18 13:46:12 +01:00
} else if ( typeTok - > strAt ( 1 ) = = " && " | | typeTok - > strAt ( 2 ) = = " & " )
2013-04-04 19:53:55 +02:00
function . type = Function : : eMoveConstructor ;
else
2014-03-30 11:06:44 +02:00
function . type = Function : : eCopyConstructor ;
2015-12-18 13:46:12 +01:00
if ( typeTok - > str ( ) ! = function . tokenDef - > str ( ) )
2014-03-30 11:06:44 +02:00
function . type = Function : : eConstructor ; // Overwrite, if types are not identical
}
// regular constructor
2013-04-04 19:53:55 +02:00
else
2014-03-30 11:06:44 +02:00
function . type = Function : : eConstructor ;
2011-03-11 01:43:29 +01:00
2014-03-30 11:06:44 +02:00
if ( function . tokenDef - > previous ( ) - > str ( ) = = " explicit " )
2015-01-08 05:45:31 +01:00
function . isExplicit ( true ) ;
2013-04-04 19:53:55 +02:00
}
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
const Token * tok1 = tok ;
// look for end of previous statement
while ( tok1 - > previous ( ) & & ! Token : : Match ( tok1 - > previous ( ) , " ;|}|{|public:|protected:|private: " ) ) {
// virtual function
if ( tok1 - > previous ( ) - > str ( ) = = " virtual " ) {
2015-01-08 05:45:31 +01:00
function . isVirtual ( true ) ;
2014-03-30 11:06:44 +02:00
break ;
}
// static function
else if ( tok1 - > previous ( ) - > str ( ) = = " static " ) {
2015-01-08 05:45:31 +01:00
function . isStatic ( true ) ;
2014-03-30 11:06:44 +02:00
break ;
}
// friend function
else if ( tok1 - > previous ( ) - > str ( ) = = " friend " ) {
2015-01-08 05:45:31 +01:00
function . isFriend ( true ) ;
2014-03-30 11:06:44 +02:00
break ;
}
tok1 = tok1 - > previous ( ) ;
}
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// find the return type
if ( ! function . isConstructor ( ) & & ! function . isDestructor ( ) ) {
while ( tok1 & & Token : : Match ( tok1 - > next ( ) , " virtual|static|friend|const|struct|union " ) )
tok1 = tok1 - > next ( ) ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
if ( tok1 )
function . retDef = tok1 ;
2010-11-13 08:08:45 +01:00
}
2014-04-12 16:06:31 +02:00
const Token * end = function . argDef - > link ( ) ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// const function
if ( end - > next ( ) - > str ( ) = = " const " )
2015-01-08 05:45:31 +01:00
function . isConst ( true ) ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
// count the number of constructors
if ( function . isConstructor ( ) )
scope - > numConstructors + + ;
if ( function . type = = Function : : eCopyConstructor | |
function . type = = Function : : eMoveConstructor )
scope - > numCopyOrMoveConstructors + + ;
2013-08-24 07:25:50 +02:00
2014-03-30 11:06:44 +02:00
// assume implementation is inline (definition and implementation same)
function . token = function . tokenDef ;
function . arg = function . argDef ;
2013-08-24 07:25:50 +02:00
2014-03-30 11:06:44 +02:00
// out of line function
2016-01-02 23:49:06 +01:00
if ( _tokenizer - > isFunctionHead ( end , " ; " ) ) {
2014-03-30 11:06:44 +02:00
// find the function implementation later
tok = end - > next ( ) ;
2016-01-02 23:49:06 +01:00
2016-01-02 18:53:51 +01:00
if ( tok - > str ( ) = = " const " )
tok = tok - > next ( ) ;
2016-01-02 23:49:06 +01:00
2016-01-02 18:53:51 +01:00
if ( tok - > str ( ) = = " & " ) {
function . hasLvalRefQualifier ( true ) ;
tok = tok - > next ( ) ;
} else if ( tok - > str ( ) = = " && " ) {
function . hasRvalRefQualifier ( true ) ;
tok = tok - > next ( ) ;
2016-01-03 08:36:00 +01:00
} else if ( tok - > str ( ) = = " noexcept " ) {
function . isNoExcept ( ! Token : : simpleMatch ( tok - > next ( ) , " ( false ) " ) ) ;
tok = tok - > next ( ) ;
if ( tok - > str ( ) = = " ( " )
tok = tok - > link ( ) - > next ( ) ;
2016-01-02 23:49:06 +01:00
} else if ( Token : : simpleMatch ( tok , " throw ( " ) ) {
function . isThrow ( true ) ;
if ( tok - > strAt ( 2 ) ! = " ) " )
function . throwArg = end - > tokAt ( 2 ) ;
tok = tok - > linkAt ( 1 ) - > next ( ) ;
}
2016-01-03 00:01:29 +01:00
if ( Token : : Match ( tok , " = %any% ; " ) ) {
2016-01-02 23:49:06 +01:00
function . isPure ( tok - > strAt ( 1 ) = = " 0 " ) ;
2016-01-03 00:01:29 +01:00
function . isDefault ( tok - > strAt ( 1 ) = = " default " ) ;
function . isDelete ( tok - > strAt ( 1 ) = = " delete " ) ;
2016-01-02 23:49:06 +01:00
tok = tok - > tokAt ( 2 ) ;
2016-01-02 18:53:51 +01:00
}
2010-11-13 08:08:45 +01:00
2015-01-02 21:38:19 +01:00
scope - > addFunction ( function ) ;
2014-03-30 11:06:44 +02:00
}
2013-02-15 06:44:07 +01:00
2014-03-30 11:06:44 +02:00
// 'const' or unknown macro (#5197)
else if ( Token : : Match ( end , " ) % any % ; " )) {
tok = end - > tokAt ( 2 ) ;
2015-01-02 21:38:19 +01:00
scope - > addFunction ( function ) ;
2014-03-30 11:06:44 +02:00
}
2013-11-25 03:43:39 +01:00
2014-03-30 11:06:44 +02:00
// inline function
else {
2015-01-08 05:45:31 +01:00
function . isInline ( true ) ;
function . hasBody ( true ) ;
2013-11-25 03:43:39 +01:00
2014-04-10 16:17:10 +02:00
if ( Token : : Match ( end , " ) const| noexcept " ) ) {
int arg = 2 ;
if ( end - > strAt ( 1 ) = = " const " )
arg + + ;
if ( end - > strAt ( arg ) = = " ( " )
function . noexceptArg = end - > tokAt ( arg + 1 ) ;
2015-01-08 05:45:31 +01:00
function . isNoExcept ( true ) ;
2014-04-10 16:17:10 +02:00
} else if ( Token : : Match ( end , " ) const | throw ( " )) {
int arg = 3 ;
if ( end - > strAt ( 1 ) = = " const " )
arg + + ;
if ( end - > strAt ( arg ) ! = " ) " )
function . throwArg = end - > tokAt ( arg ) ;
2015-01-08 05:45:31 +01:00
function . isThrow ( true ) ;
2016-01-02 20:00:18 +01:00
} else if ( Token : : Match ( end , " ) const | & | & & | [ ; { ] " )) {
2016-01-02 18:53:51 +01:00
int arg = 1 ;
if ( end - > strAt ( arg ) = = " const " )
arg + + ;
if ( end - > strAt ( arg ) = = " & " )
function . hasLvalRefQualifier ( true ) ;
else if ( end - > strAt ( arg ) = = " && " )
function . hasRvalRefQualifier ( true ) ;
2014-04-10 16:17:10 +02:00
}
2014-03-30 11:06:44 +02:00
// find start of function '{'
2015-04-06 19:47:21 +02:00
bool foundInitList = false ;
while ( end & & end - > str ( ) ! = " { " & & end - > str ( ) ! = " ; " ) {
if ( end - > link ( ) & & Token : : Match ( end , " (|< " ) ) {
end = end - > link ( ) ;
} else if ( foundInitList & &
Token : : Match ( end , " %name%|> { " ) & &
Token : : Match ( end - > linkAt ( 1 ) , " } ,|{ " ) ) {
end = end - > linkAt ( 1 ) ;
} else {
if ( end - > str ( ) = = " : " )
foundInitList = true ;
end = end - > next ( ) ;
}
}
2014-03-31 21:04:01 +02:00
if ( ! end | | end - > str ( ) = = " ; " )
2014-03-30 11:06:44 +02:00
continue ;
2010-11-13 08:08:45 +01:00
2015-01-02 21:38:19 +01:00
scope - > addFunction ( function ) ;
2011-02-20 14:25:42 +01:00
2014-03-30 11:06:44 +02:00
Function * funcptr = & scope - > functionList . back ( ) ;
const Token * tok2 = funcStart ;
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
addNewFunction ( & scope , & tok2 ) ;
if ( scope ) {
scope - > functionOf = function . nestedIn ;
scope - > function = funcptr ;
scope - > function - > functionScope = scope ;
}
2010-11-13 08:08:45 +01:00
2014-03-30 11:06:44 +02:00
tok = tok2 ;
2011-03-14 02:18:49 +01:00
}
2010-11-13 08:08:45 +01:00
}
2011-02-03 07:57:10 +01:00
2014-03-30 11:06:44 +02:00
// nested class or friend function?
2011-10-13 20:53:06 +02:00
else {
2014-03-30 11:06:44 +02:00
/** @todo check entire qualification for match */
Scope * nested = scope - > findInNestedListRecursive ( tok - > strAt ( - 2 ) ) ;
if ( nested )
addClassFunction ( & scope , & tok , argStart ) ;
else {
/** @todo handle friend functions */
}
2011-02-03 07:57:10 +01:00
}
}
2010-11-13 08:08:45 +01:00
// friend class declaration?
2015-06-14 22:18:28 +02:00
else if ( _tokenizer - > isCPP ( ) & & Token : : Match ( tok , " friend class| ::| %any% ; | : : " )) {
2013-03-05 15:28:40 +01:00
Type : : FriendInfo friendInfo ;
2010-11-13 08:08:45 +01:00
2012-12-05 20:31:17 +01:00
// save the name start
2013-03-05 19:53:24 +01:00
friendInfo . nameStart = tok - > strAt ( 1 ) = = " class " ? tok - > tokAt ( 2 ) : tok - > next ( ) ;
2012-12-05 20:31:17 +01:00
friendInfo . nameEnd = friendInfo . nameStart ;
// skip leading "::"
if ( friendInfo . nameEnd - > str ( ) = = " :: " )
friendInfo . nameEnd = friendInfo . nameEnd - > next ( ) ;
// skip qualification "name ::"
while ( friendInfo . nameEnd & & friendInfo . nameEnd - > strAt ( 1 ) = = " :: " )
friendInfo . nameEnd = friendInfo . nameEnd - > tokAt ( 2 ) ;
// save the name
if ( friendInfo . nameEnd )
friendInfo . name = friendInfo . nameEnd - > str ( ) ;
2012-02-24 20:45:56 +01:00
// fill this in after parsing is complete
2013-03-05 15:28:40 +01:00
friendInfo . type = 0 ;
2010-11-13 08:08:45 +01:00
2015-08-19 19:11:53 +02:00
if ( ! scope - > definedType )
2015-06-14 22:18:28 +02:00
_tokenizer - > syntaxError ( tok ) ;
2015-08-19 19:11:53 +02:00
2013-03-05 15:28:40 +01:00
scope - > definedType - > friendList . push_back ( friendInfo ) ;
2010-11-13 08:08:45 +01:00
}
2011-10-13 20:53:06 +02:00
} else if ( scope - > type = = Scope : : eNamespace | | scope - > type = = Scope : : eGlobal ) {
2014-02-16 11:47:52 +01:00
const Token * funcStart = nullptr ;
const Token * argStart = nullptr ;
2010-11-13 08:08:45 +01:00
2010-11-20 07:26:50 +01:00
// function?
2012-02-11 12:26:48 +01:00
if ( isFunction ( tok , scope , & funcStart , & argStart ) ) {
2014-03-31 20:33:01 +02:00
const Token * scopeBegin = argStart - > link ( ) - > next ( ) ;
if ( scopeBegin - > isName ( ) ) { // Jump behind 'const' or unknown Macro
scopeBegin = scopeBegin - > next ( ) ;
2015-06-28 16:49:16 +02:00
if ( _tokenizer - > isCPP ( ) & & scopeBegin - > str ( ) = = " throw " )
2014-04-10 16:17:10 +02:00
scopeBegin = scopeBegin - > next ( ) ;
2014-03-31 20:33:01 +02:00
if ( scopeBegin - > link ( ) & & scopeBegin - > str ( ) = = " ( " ) // Jump behind unknown macro of type THROW(...)
scopeBegin = scopeBegin - > link ( ) - > next ( ) ;
2010-11-20 07:26:50 +01:00
}
2014-03-31 20:33:01 +02:00
// has body?
2016-01-02 18:53:51 +01:00
if ( Token : : Match ( scopeBegin , " &|&&| {|: " ) ) {
2012-02-11 12:26:48 +01:00
tok = funcStart ;
2010-11-20 07:26:50 +01:00
// class function
2014-04-13 13:05:30 +02:00
if ( tok - > previous ( ) & & tok - > previous ( ) - > str ( ) = = " :: " )
2012-02-11 12:26:48 +01:00
addClassFunction ( & scope , & tok , argStart ) ;
2010-11-20 07:26:50 +01:00
2014-03-31 20:33:01 +02:00
// class destructor
2014-04-13 13:05:30 +02:00
else if ( tok - > previous ( ) & &
tok - > previous ( ) - > str ( ) = = " ~ " & &
2014-03-31 20:33:01 +02:00
tok - > strAt ( - 2 ) = = " :: " )
addClassFunction ( & scope , & tok , argStart ) ;
2010-11-20 07:26:50 +01:00
// regular function
2011-10-13 20:53:06 +02:00
else {
2012-02-11 12:26:48 +01:00
Function * function = addGlobalFunction ( scope , tok , argStart , funcStart ) ;
2014-04-20 20:40:55 +02:00
2015-08-19 19:11:53 +02:00
if ( ! function )
2014-04-10 20:11:03 +02:00
_tokenizer - > syntaxError ( tok ) ;
2014-04-10 16:17:10 +02:00
// global functions can't be const but we have tests that are
if ( Token : : Match ( argStart - > link ( ) , " ) const| noexcept " ) ) {
int arg = 2 ;
if ( argStart - > link ( ) - > strAt ( 1 ) = = " const " )
arg + + ;
if ( argStart - > link ( ) - > strAt ( arg ) = = " ( " )
function - > noexceptArg = argStart - > link ( ) - > tokAt ( arg + 1 ) ;
2015-01-08 05:45:31 +01:00
function - > isNoExcept ( true ) ;
2014-04-10 16:17:10 +02:00
} else if ( Token : : Match ( argStart - > link ( ) , " ) const| throw ( " ) ) {
int arg = 3 ;
if ( argStart - > link ( ) - > strAt ( 1 ) = = " const " )
arg + + ;
if ( argStart - > link ( ) - > strAt ( arg ) ! = " ) " )
function - > throwArg = argStart - > link ( ) - > tokAt ( arg ) ;
2015-01-08 05:45:31 +01:00
function - > isThrow ( true ) ;
}
const Token * tok1 = tok - > previous ( ) ;
// look for end of previous statement
while ( tok1 & & ! Token : : Match ( tok1 , " ;|}|{ " ) ) {
// static function
if ( tok1 - > str ( ) = = " static " ) {
function - > isStaticLocal ( true ) ;
break ;
}
// extern function
else if ( tok1 - > str ( ) = = " extern " ) {
function - > isExtern ( true ) ;
break ;
}
tok1 = tok1 - > previous ( ) ;
2014-04-10 16:17:10 +02:00
}
2011-01-23 22:31:35 +01:00
}
2010-12-16 19:04:47 +01:00
// syntax error?
2014-04-10 20:11:03 +02:00
if ( ! scope )
_tokenizer - > syntaxError ( tok ) ;
2010-11-20 07:26:50 +01:00
}
2014-03-31 20:33:01 +02:00
// function prototype?
else if ( scopeBegin - > str ( ) = = " ; " ) {
2012-03-23 17:59:51 +01:00
bool newFunc = true ; // Is this function already in the database?
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : const_iterator i = scope - > functionMap . find ( tok - > str ( ) ) ; i ! = scope - > functionMap . end ( ) & & i - > first = = tok - > str ( ) ; + + i ) {
if ( Function : : argsMatch ( scope , i - > second - > argDef - > next ( ) , argStart - > next ( ) , " " , 0 ) ) {
2012-03-23 17:59:51 +01:00
newFunc = false ;
2012-10-30 15:48:06 +01:00
break ;
}
2012-03-23 17:59:51 +01:00
}
2013-08-31 18:58:55 +02:00
2012-03-23 17:59:51 +01:00
// save function prototype in database
if ( newFunc ) {
2013-08-31 18:58:55 +02:00
Function * func = addGlobalFunctionDecl ( scope , tok , argStart , funcStart ) ;
2014-04-10 16:17:10 +02:00
if ( Token : : Match ( argStart - > link ( ) , " ) const| noexcept " ) ) {
int arg = 2 ;
if ( argStart - > link ( ) - > strAt ( 1 ) = = " const " )
arg + + ;
if ( argStart - > link ( ) - > strAt ( arg ) = = " ( " )
func - > noexceptArg = argStart - > link ( ) - > tokAt ( arg + 1 ) ;
2015-01-08 05:45:31 +01:00
func - > isNoExcept ( true ) ;
2014-04-10 16:17:10 +02:00
} else if ( Token : : Match ( argStart - > link ( ) , " ) const| throw ( " ) ) {
int arg = 3 ;
if ( argStart - > link ( ) - > strAt ( 1 ) = = " const " )
arg + + ;
if ( argStart - > link ( ) - > strAt ( arg ) ! = " ) " )
func - > throwArg = argStart - > link ( ) - > tokAt ( arg ) ;
2015-01-08 05:45:31 +01:00
func - > isThrow ( true ) ;
}
const Token * tok1 = tok - > previous ( ) ;
// look for end of previous statement
while ( tok1 & & ! Token : : Match ( tok1 , " ;|}|{ " ) ) {
// extern function
if ( tok1 - > str ( ) = = " extern " ) {
func - > isExtern ( true ) ;
break ;
}
tok1 = tok1 - > previous ( ) ;
2014-04-10 16:17:10 +02:00
}
2012-03-23 17:59:51 +01:00
}
2014-03-31 20:33:01 +02:00
tok = scopeBegin ;
2011-03-14 03:59:25 +01:00
continue ;
}
2010-11-20 07:26:50 +01:00
}
2012-09-11 18:03:47 +02:00
} else if ( scope - > isExecutable ( ) ) {
2013-03-05 19:53:24 +01:00
if ( Token : : Match ( tok , " else|try|do { " ) ) {
const Token * tok1 = tok - > next ( ) ;
if ( tok - > str ( ) = = " else " )
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eElse , tok1 ) ) ;
if ( tok - > str ( ) = = " do " )
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eDo , tok1 ) ) ;
2014-08-31 11:27:12 +02:00
else //if (tok->str() == "try")
2013-03-05 19:53:24 +01:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eTry , tok1 ) ) ;
2011-03-11 01:43:29 +01:00
tok = tok1 ;
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( & scopeList . back ( ) ) ;
2011-03-11 01:43:29 +01:00
scope = & scopeList . back ( ) ;
2013-03-05 19:53:24 +01:00
} else if ( Token : : Match ( tok , " if|for|while|catch|switch ( " ) & & Token : : simpleMatch ( tok - > next ( ) - > link ( ) , " ) { " )) {
2015-09-04 10:38:46 +02:00
const Token * scopeStartTok = tok - > next ( ) - > link ( ) - > next ( ) ;
2014-07-02 16:16:19 +02:00
if ( tok - > str ( ) = = " if " )
2015-09-04 10:38:46 +02:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eIf , scopeStartTok ) ) ;
2013-03-05 19:53:24 +01:00
else if ( tok - > str ( ) = = " for " ) {
2015-09-04 10:38:46 +02:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eFor , scopeStartTok ) ) ;
2013-03-05 19:53:24 +01:00
} else if ( tok - > str ( ) = = " while " )
2015-09-04 10:38:46 +02:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eWhile , scopeStartTok ) ) ;
2013-03-05 19:53:24 +01:00
else if ( tok - > str ( ) = = " catch " ) {
2015-09-04 10:38:46 +02:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eCatch , scopeStartTok ) ) ;
2014-08-31 10:21:10 +02:00
} else // if (tok->str() == "switch")
2015-09-04 10:38:46 +02:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eSwitch , scopeStartTok ) ) ;
2013-03-05 19:53:24 +01:00
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( & scopeList . back ( ) ) ;
2012-01-26 04:05:29 +01:00
scope = & scopeList . back ( ) ;
2013-03-05 19:53:24 +01:00
if ( scope - > type = = Scope : : eFor )
2015-01-30 21:56:27 +01:00
scope - > checkVariable ( tok - > tokAt ( 2 ) , Local , & settings - > library ) ; // check for variable declaration and add it to new scope if found
2013-03-05 19:53:24 +01:00
else if ( scope - > type = = Scope : : eCatch )
2015-01-30 21:56:27 +01:00
scope - > checkVariable ( tok - > tokAt ( 2 ) , Throw , & settings - > library ) ; // check for variable declaration and add it to new scope if found
2015-09-04 10:38:46 +02:00
tok = scopeStartTok ;
2015-11-06 10:16:44 +01:00
} else if ( tok - > str ( ) = = " { " ) {
if ( tok - > previous ( ) - > varId ( ) )
tok = tok - > link ( ) ;
else if ( tok - > strAt ( - 1 ) = = " ) " & & tok - > linkAt ( - 1 ) - > strAt ( - 1 ) = = " ] " ) {
2014-08-31 10:21:10 +02:00
scopeList . push_back ( Scope ( this , tok - > linkAt ( - 1 ) - > linkAt ( - 1 ) , scope , Scope : : eLambda , tok ) ) ;
scope - > nestedList . push_back ( & scopeList . back ( ) ) ;
scope = & scopeList . back ( ) ;
2014-09-03 11:15:05 +02:00
} else if ( ! Token : : Match ( tok - > previous ( ) , " =|,|(|return " ) & & ! ( tok - > strAt ( - 1 ) = = " ) " & & Token : : Match ( tok - > linkAt ( - 1 ) - > previous ( ) , " =|,|(|return " ) ) ) {
2011-03-11 01:43:29 +01:00
scopeList . push_back ( Scope ( this , tok , scope , Scope : : eUnconditional , tok ) ) ;
2013-03-05 18:42:42 +01:00
scope - > nestedList . push_back ( & scopeList . back ( ) ) ;
2011-03-11 01:43:29 +01:00
scope = & scopeList . back ( ) ;
2011-10-13 20:53:06 +02:00
} else {
2011-02-26 21:53:57 +01:00
tok = tok - > link ( ) ;
}
}
}
2010-11-20 07:26:50 +01:00
}
2010-11-13 08:08:45 +01:00
}
2014-09-01 17:01:05 +02:00
if ( ! _tokenizer - > isC ( ) ) {
// fill in base class info
for ( std : : list < Type > : : iterator it = typeList . begin ( ) ; it ! = typeList . end ( ) ; + + it ) {
// finish filling in base class info
2015-07-01 00:04:01 +02:00
for ( unsigned int i = 0 ; i < it - > derivedFrom . size ( ) ; + + i ) {
2015-07-01 07:50:13 +02:00
const Type * found = findType ( it - > derivedFrom [ i ] . nameTok , it - > enclosingScope ) ;
if ( found & & found - > findDependency ( & ( * it ) ) ) {
// circular dependency
//_tokenizer->syntaxError(nullptr);
} else {
it - > derivedFrom [ i ] . type = found ;
}
}
2014-09-01 17:01:05 +02:00
}
2010-11-13 08:08:45 +01:00
2014-09-01 17:01:05 +02:00
// fill in friend info
for ( std : : list < Type > : : iterator it = typeList . begin ( ) ; it ! = typeList . end ( ) ; + + it ) {
for ( std : : list < Type : : FriendInfo > : : iterator i = it - > friendList . begin ( ) ; i ! = it - > friendList . end ( ) ; + + i ) {
i - > type = findType ( i - > nameStart , it - > enclosingScope ) ;
}
2012-02-24 20:45:56 +01:00
}
2014-09-01 17:01:05 +02:00
// fill in using info
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
for ( std : : list < Scope : : UsingInfo > : : iterator i = it - > usingList . begin ( ) ; i ! = it - > usingList . end ( ) ; + + i ) {
// only find if not already found
if ( i - > scope = = nullptr ) {
// check scope for match
scope = findScope ( i - > start - > tokAt ( 2 ) , & ( * it ) ) ;
if ( scope ) {
// set found scope
i - > scope = scope ;
break ;
}
2014-04-10 16:11:11 +02:00
}
2012-12-20 06:53:04 +01:00
}
}
}
2010-12-02 07:35:01 +01:00
// fill in variable info
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2011-02-26 21:57:16 +01:00
// find variables
2015-01-30 21:56:27 +01:00
it - > getVariableList ( & settings - > library ) ;
2010-12-02 07:35:01 +01:00
}
2011-02-26 21:51:12 +01:00
// fill in function arguments
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2011-02-26 21:51:12 +01:00
std : : list < Function > : : iterator func ;
2012-03-23 17:59:51 +01:00
for ( func = it - > functionList . begin ( ) ; func ! = it - > functionList . end ( ) ; + + func ) {
2011-02-26 21:51:12 +01:00
// add arguments
2012-04-27 21:51:13 +02:00
func - > addArguments ( this , scope ) ;
2011-02-26 21:51:12 +01:00
}
}
2012-10-10 20:42:07 +02:00
// fill in function scopes
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2012-10-08 16:15:07 +02:00
if ( it - > type = = Scope : : eFunction )
2012-10-10 20:42:07 +02:00
functionScopes . push_back ( & * it ) ;
}
// fill in class and struct scopes
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2012-10-10 20:42:07 +02:00
if ( it - > isClassOrStruct ( ) )
classAndStructScopes . push_back ( & * it ) ;
2012-10-08 16:15:07 +02:00
}
2013-08-26 06:03:26 +02:00
// fill in function return types
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
std : : list < Function > : : iterator func ;
for ( func = it - > functionList . begin ( ) ; func ! = it - > functionList . end ( ) ; + + func ) {
// add return types
if ( func - > retDef ) {
const Token * type = func - > retDef ;
while ( Token : : Match ( type , " static|const|struct|union " ) )
type = type - > next ( ) ;
if ( type )
func - > retType = findTypeInNested ( type , func - > nestedIn ) ;
}
}
}
2015-01-21 10:34:58 +01:00
if ( tokenizer - > isC ( ) ) {
// For C code it is easy, as there are no constructors and no default values
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2011-03-11 01:43:29 +01:00
scope = & ( * it ) ;
2015-01-21 10:34:58 +01:00
if ( scope - > definedType )
scope - > definedType - > needInitialization = Type : : True ;
}
} else {
// For C++, it is more difficult: Determine if user defined type needs initialization...
unsigned int unknowns = 0 ; // stop checking when there are no unknowns
unsigned int retry = 0 ; // bail if we don't resolve all the variable types for some reason
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
do {
unknowns = 0 ;
2013-07-08 11:45:26 +02:00
2015-01-21 10:34:58 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
scope = & ( * it ) ;
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
if ( ! scope - > definedType ) {
_blankTypes . push_back ( Type ( ) ) ;
scope - > definedType = & _blankTypes . back ( ) ;
}
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
if ( scope - > isClassOrStruct ( ) & & scope - > definedType - > needInitialization = = Type : : Unknown ) {
// check for default constructor
bool hasDefaultConstructor = false ;
2011-01-15 07:59:37 +01:00
2015-01-21 10:34:58 +01:00
std : : list < Function > : : const_iterator func ;
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
for ( func = scope - > functionList . begin ( ) ; func ! = scope - > functionList . end ( ) ; + + func ) {
if ( func - > type = = Function : : eConstructor ) {
// check for no arguments: func ( )
if ( func - > argCount ( ) = = 0 ) {
hasDefaultConstructor = true ;
break ;
2010-12-02 07:35:01 +01:00
}
2015-01-21 10:34:58 +01:00
/** check for arguments with default values */
else if ( func - > argCount ( ) = = func - > initializedArgCount ( ) ) {
hasDefaultConstructor = true ;
break ;
}
}
2010-12-02 07:35:01 +01:00
}
2015-01-21 10:34:58 +01:00
// User defined types with user defined default constructor doesn't need initialization.
// We assume the default constructor initializes everything.
// Another check will figure out if the constructor actually initializes everything.
if ( hasDefaultConstructor )
scope - > definedType - > needInitialization = Type : : False ;
// check each member variable to see if it needs initialization
else {
bool needInitialization = false ;
bool unknown = false ;
std : : list < Variable > : : const_iterator var ;
for ( var = scope - > varlist . begin ( ) ; var ! = scope - > varlist . end ( ) & & ! needInitialization ; + + var ) {
if ( var - > isClass ( ) ) {
if ( var - > type ( ) ) {
// does this type need initialization?
if ( var - > type ( ) - > needInitialization = = Type : : True )
needInitialization = true ;
else if ( var - > type ( ) - > needInitialization = = Type : : Unknown )
unknown = true ;
}
} else if ( ! var - > hasDefault ( ) )
needInitialization = true ;
}
2010-12-02 07:35:01 +01:00
if ( needInitialization )
2013-03-05 13:33:38 +01:00
scope - > definedType - > needInitialization = Type : : True ;
2015-01-21 10:34:58 +01:00
else if ( ! unknown )
2013-03-05 13:33:38 +01:00
scope - > definedType - > needInitialization = Type : : False ;
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
if ( scope - > definedType - > needInitialization = = Type : : Unknown )
unknowns + + ;
}
} else if ( scope - > type = = Scope : : eUnion & & scope - > definedType - > needInitialization = = Type : : Unknown )
scope - > definedType - > needInitialization = Type : : True ;
}
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
retry + + ;
} while ( unknowns & & retry < 100 ) ;
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
// this shouldn't happen so output a debug warning
2015-04-10 14:18:52 +02:00
if ( retry = = 100 & & printDebug ) {
2015-01-21 10:34:58 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
scope = & ( * it ) ;
2010-12-02 07:35:01 +01:00
2015-01-21 10:34:58 +01:00
if ( scope - > isClassOrStruct ( ) & & scope - > definedType - > needInitialization = = Type : : Unknown )
debugMessage ( scope - > classDef , " SymbolDatabase::SymbolDatabase couldn't resolve all user defined types. " ) ;
}
2010-12-02 07:35:01 +01:00
}
2010-11-13 08:08:45 +01:00
}
2011-02-26 21:57:16 +01:00
// create variable symbol table
_variableList . resize ( _tokenizer - > varIdCount ( ) + 1 ) ;
2014-02-16 10:32:10 +01:00
std : : fill_n ( _variableList . begin ( ) , _variableList . size ( ) , ( const Variable * ) nullptr ) ;
2011-02-26 21:57:16 +01:00
// check all scopes for variables
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
2011-03-11 01:43:29 +01:00
scope = & ( * it ) ;
2011-02-26 21:57:16 +01:00
// add all variables
2014-08-31 11:27:12 +02:00
for ( std : : list < Variable > : : iterator var = scope - > varlist . begin ( ) ; var ! = scope - > varlist . end ( ) ; + + var ) {
2013-07-20 12:31:04 +02:00
unsigned int varId = var - > declarationId ( ) ;
2011-02-26 21:57:16 +01:00
if ( varId )
2011-03-11 01:43:29 +01:00
_variableList [ varId ] = & ( * var ) ;
2013-08-12 06:21:03 +02:00
// fix up variables without type
2013-08-16 05:42:12 +02:00
if ( ! var - > type ( ) & & ! var - > typeStartToken ( ) - > isStandardType ( ) ) {
2013-08-12 06:21:03 +02:00
const Type * type = findType ( var - > typeStartToken ( ) , scope ) ;
if ( type )
var - > type ( type ) ;
}
2011-02-26 21:57:16 +01:00
}
2013-01-16 15:37:07 +01:00
// add all function parameters
2014-08-31 11:27:12 +02:00
for ( std : : list < Function > : : iterator func = scope - > functionList . begin ( ) ; func ! = scope - > functionList . end ( ) ; + + func ) {
for ( std : : list < Variable > : : iterator arg = func - > argumentList . begin ( ) ; arg ! = func - > argumentList . end ( ) ; + + arg ) {
2011-02-26 21:57:16 +01:00
// check for named parameters
2013-07-20 12:31:04 +02:00
if ( arg - > nameToken ( ) & & arg - > declarationId ( ) ) {
const unsigned int declarationId = arg - > declarationId ( ) ;
if ( declarationId > 0U )
_variableList [ declarationId ] = & ( * arg ) ;
2013-08-12 06:21:03 +02:00
// fix up parameters without type
2013-08-16 05:42:12 +02:00
if ( ! arg - > type ( ) & & ! arg - > typeStartToken ( ) - > isStandardType ( ) ) {
2013-08-17 18:43:15 +02:00
const Type * type = findTypeInNested ( arg - > typeStartToken ( ) , scope ) ;
2013-08-12 06:21:03 +02:00
if ( type )
arg - > type ( type ) ;
}
2011-02-26 21:57:16 +01:00
}
}
}
}
2011-06-24 04:35:15 +02:00
2013-01-25 06:49:04 +01:00
// fill in missing variables if possible
2013-02-20 07:46:06 +01:00
const std : : size_t functions = functionScopes . size ( ) ;
for ( std : : size_t i = 0 ; i < functions ; + + i ) {
const Scope * func = functionScopes [ i ] ;
2014-08-12 08:02:54 +02:00
for ( const Token * tok = func - > classStart - > next ( ) ; tok & & tok ! = func - > classEnd ; tok = tok - > next ( ) ) {
2013-02-20 07:46:06 +01:00
// check for member variable
2014-08-12 08:02:54 +02:00
if ( tok - > varId ( ) & & tok - > next ( ) & &
2013-07-20 17:51:13 +02:00
( tok - > next ( ) - > str ( ) = = " . " | |
( tok - > next ( ) - > str ( ) = = " [ " & & tok - > linkAt ( 1 ) - > strAt ( 1 ) = = " . " ) ) ) {
const Token * tok1 = tok - > next ( ) - > str ( ) = = " . " ? tok - > tokAt ( 2 ) : tok - > linkAt ( 1 ) - > tokAt ( 2 ) ;
2013-02-20 07:46:06 +01:00
if ( tok1 & & tok1 - > varId ( ) & & _variableList [ tok1 - > varId ( ) ] = = 0 ) {
const Variable * var = _variableList [ tok - > varId ( ) ] ;
2013-03-05 13:33:38 +01:00
if ( var & & var - > typeScope ( ) ) {
2013-02-20 07:46:06 +01:00
// find the member variable of this variable
2013-03-05 13:33:38 +01:00
const Variable * var1 = var - > typeScope ( ) - > getVariable ( tok1 - > str ( ) ) ;
2013-02-20 07:46:06 +01:00
if ( var1 ) {
// add this variable to the look up table
_variableList [ tok1 - > varId ( ) ] = var1 ;
2013-01-25 06:49:04 +01:00
}
}
}
}
}
}
2011-06-24 04:35:15 +02:00
/* set all unknown array dimensions that are set by a variable to the maximum size of that variable type */
2012-07-08 23:39:46 +02:00
for ( std : : size_t i = 1 ; i < = _tokenizer - > varIdCount ( ) ; i + + ) {
2011-06-24 04:35:15 +02:00
// check each array variable
2011-10-13 20:53:06 +02:00
if ( _variableList [ i ] & & _variableList [ i ] - > isArray ( ) ) {
2011-06-24 04:35:15 +02:00
// check each array dimension
2014-07-23 16:51:23 +02:00
const std : : vector < Dimension > & dimensions = _variableList [ i ] - > dimensions ( ) ;
for ( std : : size_t j = 0 ; j < dimensions . size ( ) ; j + + ) {
Dimension & dimension = const_cast < Dimension & > ( dimensions [ j ] ) ;
2011-06-24 04:35:15 +02:00
// check for a single token dimension that is a variable
2012-09-10 16:14:24 +02:00
if ( dimension . num = = 0 ) {
dimension . known = false ;
if ( ! dimension . start | | ( dimension . start ! = dimension . end ) | | ! dimension . start - > varId ( ) )
continue ;
2011-06-24 04:35:15 +02:00
// get maximum size from type
// find where this type is defined
const Variable * var = getVariableFromVarId ( dimension . start - > varId ( ) ) ;
2011-06-24 14:02:41 +02:00
2011-06-24 04:35:15 +02:00
// make sure it is in the database
if ( ! var )
break ;
// get type token
const Token * index_type = var - > typeEndToken ( ) ;
2011-10-13 20:53:06 +02:00
if ( index_type - > str ( ) = = " char " ) {
2011-06-24 04:35:15 +02:00
if ( index_type - > isUnsigned ( ) )
dimension . num = UCHAR_MAX + 1 ;
else if ( index_type - > isSigned ( ) )
dimension . num = SCHAR_MAX + 1 ;
else
dimension . num = CHAR_MAX + 1 ;
2011-10-13 20:53:06 +02:00
} else if ( index_type - > str ( ) = = " short " ) {
2011-06-24 04:35:15 +02:00
if ( index_type - > isUnsigned ( ) )
dimension . num = USHRT_MAX + 1 ;
else
dimension . num = SHRT_MAX + 1 ;
}
2011-06-24 14:02:41 +02:00
2011-06-24 04:35:15 +02:00
// checkScope assumes size is signed int so we limit the following sizes to INT_MAX
2011-10-13 20:53:06 +02:00
else if ( index_type - > str ( ) = = " int " ) {
2011-06-24 04:35:15 +02:00
if ( index_type - > isUnsigned ( ) )
dimension . num = UINT_MAX + 1ULL ;
else
dimension . num = INT_MAX + 1ULL ;
2011-10-13 20:53:06 +02:00
} else if ( index_type - > str ( ) = = " long " ) {
if ( index_type - > isUnsigned ( ) ) {
2011-06-24 04:35:15 +02:00
if ( index_type - > isLong ( ) )
2011-07-01 22:59:17 +02:00
dimension . num = ULLONG_MAX ; // should be ULLONG_MAX + 1ULL
2011-06-24 04:35:15 +02:00
else
2011-07-01 22:59:17 +02:00
dimension . num = ULONG_MAX ; // should be ULONG_MAX + 1ULL
2011-10-13 20:53:06 +02:00
} else {
2011-06-24 04:35:15 +02:00
if ( index_type - > isLong ( ) )
2011-07-01 22:59:17 +02:00
dimension . num = LLONG_MAX ; // should be LLONG_MAX + 1LL
2011-06-24 04:35:15 +02:00
else
2011-07-01 22:59:17 +02:00
dimension . num = LONG_MAX ; // should be LONG_MAX + 1LL
2011-06-24 04:35:15 +02:00
}
}
}
}
}
}
2014-11-01 14:36:17 +01:00
// Set scope pointers
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
Token * start = const_cast < Token * > ( it - > classStart ) ;
Token * end = const_cast < Token * > ( it - > classEnd ) ;
if ( it - > type = = Scope : : eGlobal ) {
start = const_cast < Token * > ( _tokenizer - > list . front ( ) ) ;
end = const_cast < Token * > ( _tokenizer - > list . back ( ) ) ;
}
if ( start & & end ) {
start - > scope ( & * it ) ;
end - > scope ( & * it ) ;
}
if ( start ! = end & & start - > next ( ) ! = end ) {
for ( Token * tok = start - > next ( ) ; tok ! = end ; tok = tok - > next ( ) ) {
if ( tok - > str ( ) = = " { " ) {
bool break2 = false ;
for ( std : : list < Scope * > : : const_iterator innerScope = it - > nestedList . begin ( ) ; innerScope ! = it - > nestedList . end ( ) ; + + innerScope ) {
if ( tok = = ( * innerScope ) - > classStart ) { // Is begin of inner scope
tok = tok - > link ( ) ;
if ( ! tok | | tok - > next ( ) = = end | | ! tok - > next ( ) ) {
break2 = true ;
break ;
}
tok = tok - > next ( ) ;
break ;
}
}
if ( break2 )
break ;
}
tok - > scope ( & * it ) ;
}
}
}
// Set function definition and declaration pointers
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
for ( std : : list < Function > : : const_iterator func = it - > functionList . begin ( ) ; func ! = it - > functionList . end ( ) ; + + func ) {
if ( func - > tokenDef )
const_cast < Token * > ( func - > tokenDef ) - > function ( & * func ) ;
if ( func - > token )
const_cast < Token * > ( func - > token ) - > function ( & * func ) ;
}
}
// Set function call pointers
for ( const Token * tok = _tokenizer - > list . front ( ) ; tok ! = _tokenizer - > list . back ( ) ; tok = tok - > next ( ) ) {
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( tok , " %name% ( " ) ) {
2015-01-02 16:11:21 +01:00
if ( ! tok - > function ( ) & & tok - > varId ( ) = = 0 )
2014-11-01 14:36:17 +01:00
const_cast < Token * > ( tok ) - > function ( findFunction ( tok ) ) ;
}
}
2014-11-13 06:29:51 +01:00
// Set C++ 11 delegate constructor function call pointers
for ( std : : list < Scope > : : iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
for ( std : : list < Function > : : const_iterator func = it - > functionList . begin ( ) ; func ! = it - > functionList . end ( ) ; + + func ) {
// look for initializer list
if ( func - > type = = Function : : eConstructor & & func - > functionScope & &
func - > functionScope - > functionOf & & func - > arg & & func - > arg - > link ( ) - > strAt ( 1 ) = = " : " ) {
const Token * tok = func - > arg - > link ( ) - > tokAt ( 2 ) ;
while ( tok & & tok ! = func - > functionScope - > classStart ) {
2015-05-10 11:27:47 +02:00
if ( Token : : Match ( tok , " %name% {|( " ) ) {
2014-11-13 06:29:51 +01:00
if ( tok - > str ( ) = = func - > tokenDef - > str ( ) ) {
const_cast < Token * > ( tok ) - > function ( func - > functionScope - > functionOf - > findFunction ( tok ) ) ;
break ;
}
tok = tok - > linkAt ( 1 ) ;
}
tok = tok - > next ( ) ;
}
}
}
}
2015-08-14 20:46:13 +02:00
// Set type pointers
for ( const Token * tok = _tokenizer - > list . front ( ) ; tok ! = _tokenizer - > list . back ( ) ; tok = tok - > next ( ) ) {
if ( ! tok - > isName ( ) | | tok - > varId ( ) | | tok - > function ( ) )
continue ;
const_cast < Token * > ( tok ) - > type ( findVariableType ( tok - > scope ( ) , tok ) ) ;
}
2014-11-01 14:36:17 +01:00
// Set variable pointers
for ( const Token * tok = _tokenizer - > list . front ( ) ; tok ! = _tokenizer - > list . back ( ) ; tok = tok - > next ( ) ) {
if ( tok - > varId ( ) )
const_cast < Token * > ( tok ) - > variable ( getVariableFromVarId ( tok - > varId ( ) ) ) ;
// Set Token::variable pointer for array member variable
// Since it doesn't point at a fixed location it doesn't have varid
if ( tok - > variable ( ) ! = nullptr & &
tok - > variable ( ) - > typeScope ( ) & &
2015-01-31 10:50:39 +01:00
Token : : Match ( tok , " %name% [|. " ) ) {
2014-11-01 14:36:17 +01:00
Token * tok2 = tok - > next ( ) ;
// Locate "]"
2015-08-16 14:22:46 +02:00
while ( tok2 & & tok2 - > str ( ) = = " [ " )
tok2 = tok2 - > link ( ) - > next ( ) ;
2014-11-01 14:36:17 +01:00
Token * membertok = nullptr ;
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( tok2 , " . %name% " ) )
2014-11-01 14:36:17 +01:00
membertok = tok2 - > next ( ) ;
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( tok2 , " ) . %name% " ) & & tok - > strAt ( - 1 ) = = " ( " )
2014-11-01 14:36:17 +01:00
membertok = tok2 - > tokAt ( 2 ) ;
if ( membertok ) {
const Variable * var = tok - > variable ( ) ;
if ( var & & var - > typeScope ( ) ) {
const Variable * membervar = var - > typeScope ( ) - > getVariable ( membertok - > str ( ) ) ;
if ( membervar )
membertok - > variable ( membervar ) ;
}
}
}
// check for function returning record type
// func(...).var
// func(...)[...].var
else if ( tok - > function ( ) & & tok - > next ( ) - > str ( ) = = " ( " & &
2015-01-31 10:50:39 +01:00
( Token : : Match ( tok - > next ( ) - > link ( ) , " ) . %name% !!( " ) | |
( Token : : Match ( tok - > next ( ) - > link ( ) , " ) [ " ) & & Token : : Match ( tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) , " ] . %name% !!( " ) ) ) ) {
2014-11-01 14:36:17 +01:00
const Type * type = tok - > function ( ) - > retType ;
if ( type ) {
Token * membertok ;
if ( tok - > next ( ) - > link ( ) - > next ( ) - > str ( ) = = " . " )
membertok = tok - > next ( ) - > link ( ) - > next ( ) - > next ( ) ;
else
membertok = tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) - > next ( ) - > next ( ) ;
const Variable * membervar = membertok - > variable ( ) ;
if ( ! membervar ) {
if ( type - > classScope ) {
membervar = type - > classScope - > getVariable ( membertok - > str ( ) ) ;
if ( membervar )
membertok - > variable ( membervar ) ;
}
}
}
}
}
2010-11-13 08:08:45 +01:00
}
2014-11-02 10:36:52 +01:00
SymbolDatabase : : ~ SymbolDatabase ( )
{
2015-08-14 20:46:13 +02:00
// Clear scope, type, function and variable pointers
2014-11-02 10:36:52 +01:00
for ( const Token * tok = _tokenizer - > list . front ( ) ; tok ! = _tokenizer - > list . back ( ) ; tok = tok - > next ( ) ) {
const_cast < Token * > ( tok ) - > scope ( 0 ) ;
2015-08-14 20:46:13 +02:00
const_cast < Token * > ( tok ) - > type ( 0 ) ;
2014-11-02 10:36:52 +01:00
const_cast < Token * > ( tok ) - > function ( 0 ) ;
const_cast < Token * > ( tok ) - > variable ( 0 ) ;
}
}
2015-11-14 18:43:07 +01:00
bool SymbolDatabase : : isFunction ( const Token * tok , const Scope * outerScope , const Token * * funcStart , const Token * * argStart ) const
2010-11-13 08:08:45 +01:00
{
2014-09-29 13:05:51 +02:00
if ( tok - > varId ( ) )
return false ;
2015-01-31 10:50:39 +01:00
// function returning function pointer? '... ( ... %name% ( ... ))( ... ) {'
2010-11-13 08:08:45 +01:00
if ( tok - > str ( ) = = " ( " & &
2014-04-01 10:13:00 +02:00
tok - > link ( ) - > previous ( ) - > str ( ) = = " ) " ) {
const Token * tok2 = tok - > link ( ) - > next ( ) ;
if ( tok2 & & tok2 - > str ( ) = = " ( " & & Token : : Match ( tok2 - > link ( ) - > next ( ) , " {|;|const|= " ) ) {
2014-11-30 13:00:27 +01:00
const Token * argStartTok = tok - > link ( ) - > previous ( ) - > link ( ) ;
* funcStart = argStartTok - > previous ( ) ;
* argStart = argStartTok ;
2014-04-01 10:13:00 +02:00
return true ;
}
2010-11-13 08:08:45 +01:00
}
// regular function?
2015-05-26 00:28:08 +02:00
else if ( Token : : Match ( tok , " %name% ( " ) & & ! isReservedName ( tok - > str ( ) ) & & tok - > previous ( ) & &
2012-08-22 12:37:50 +02:00
( tok - > previous ( ) - > isName ( ) | | tok - > strAt ( - 1 ) = = " > " | | tok - > strAt ( - 1 ) = = " & " | | tok - > strAt ( - 1 ) = = " * " | | // Either a return type in front of tok
tok - > strAt ( - 1 ) = = " :: " | | tok - > strAt ( - 1 ) = = " ~ " | | // or a scope qualifier in front of tok
2014-03-31 20:33:01 +02:00
outerScope - > isClassOrStruct ( ) ) ) { // or a ctor/dtor
2015-02-24 06:11:31 +01:00
const Token * tok1 = tok - > previous ( ) ;
// skip over destructor "~"
if ( tok1 - > str ( ) = = " ~ " )
tok1 = tok1 - > previous ( ) ;
// skip over qualification
while ( Token : : simpleMatch ( tok1 , " :: " ) ) {
if ( Token : : Match ( tok1 - > tokAt ( - 1 ) , " %name% " ) )
tok1 = tok1 - > tokAt ( - 2 ) ;
2015-04-09 21:01:47 +02:00
else if ( tok1 - > strAt ( - 1 ) = = " > " & & tok1 - > linkAt ( - 1 ) & & Token : : Match ( tok1 - > linkAt ( - 1 ) - > previous ( ) , " %name% " ) )
tok1 = tok1 - > linkAt ( - 1 ) - > tokAt ( - 2 ) ;
2015-02-24 06:11:31 +01:00
else
tok1 = tok1 - > tokAt ( - 1 ) ;
}
2015-04-04 11:33:25 +02:00
// done if constructor or destructor
if ( ! Token : : Match ( tok1 , " {|}|;|public:|protected:|private: " ) & & tok1 ) {
// skip over pointers and references
while ( Token : : Match ( tok1 , " [*&] " ) )
tok1 = tok1 - > tokAt ( - 1 ) ;
2015-02-24 06:11:31 +01:00
2015-04-04 11:33:25 +02:00
// skip over template
if ( tok1 & & tok1 - > str ( ) = = " > " ) {
if ( tok1 - > link ( ) )
tok1 = tok1 - > link ( ) - > previous ( ) ;
else
return false ;
}
2015-02-24 06:11:31 +01:00
2015-04-04 11:33:25 +02:00
// function can't have number or variable as return type
if ( tok1 & & ( tok1 - > isNumber ( ) | | tok1 - > varId ( ) ) )
2015-02-28 16:07:43 +01:00
return false ;
2015-02-24 06:11:31 +01:00
2015-04-04 11:33:25 +02:00
// skip over return type
if ( Token : : Match ( tok1 , " %name% " ) ) {
if ( tok1 - > str ( ) = = " return " )
return false ;
tok1 = tok1 - > previous ( ) ;
}
2015-02-24 06:11:31 +01:00
2015-04-04 11:33:25 +02:00
// skip over qualification
while ( Token : : simpleMatch ( tok1 , " :: " ) ) {
if ( Token : : Match ( tok1 - > tokAt ( - 1 ) , " %name% " ) )
tok1 = tok1 - > tokAt ( - 2 ) ;
else
tok1 = tok1 - > tokAt ( - 1 ) ;
}
// skip over modifiers and other stuff
while ( Token : : Match ( tok1 , " const|static|extern|template|virtual|struct|class " ) )
tok1 = tok1 - > previous ( ) ;
2015-02-24 06:11:31 +01:00
2015-04-04 11:33:25 +02:00
// should be at a sequence point if this is a function
if ( ! Token : : Match ( tok1 , " >|{|}|;|public:|protected:|private: " ) & & tok1 )
return false ;
}
2015-02-24 06:11:31 +01:00
2014-03-31 20:33:01 +02:00
const Token * tok2 = tok - > next ( ) - > link ( ) - > next ( ) ;
2014-04-10 16:17:10 +02:00
if ( tok2 & &
( Token : : Match ( tok2 , " const| ;|{|= " ) | |
2015-01-31 10:50:39 +01:00
( tok2 - > isUpperCaseName ( ) & & Token : : Match ( tok2 , " %name% ;|{ " ) ) | |
( tok2 - > isUpperCaseName ( ) & & Token : : Match ( tok2 , " %name% ( " ) & & tok2 - > next ( ) - > link ( ) - > strAt ( 1 ) = = " { " ) | |
Token : : Match ( tok2 , " : ::| %name% (|::|<|{ " ) | |
2016-01-02 18:53:51 +01:00
Token : : Match ( tok2 , " const| &|&&| ;|{ " ) | |
2016-01-22 19:22:24 +01:00
Token : : Match ( tok2 , " const| override ;|{ " ) | |
2014-04-10 16:17:10 +02:00
Token : : Match ( tok2 , " = delete|default ; " ) | |
2014-05-20 06:10:34 +02:00
Token : : Match ( tok2 , " const| noexcept {|:|;|= " ) | |
2014-04-10 16:17:10 +02:00
( Token : : Match ( tok2 , " const| noexcept|throw ( " ) & &
2014-05-20 06:10:34 +02:00
tok2 - > str ( ) = = " const " ? ( tok2 - > tokAt ( 2 ) & & Token : : Match ( tok2 - > linkAt ( 2 ) , " ) const| {|:|;|= " ) ) :
( tok2 - > next ( ) & & Token : : Match ( tok2 - > next ( ) - > link ( ) , " ) {|:|;|= " ) ) ) ) ) {
2014-03-31 20:33:01 +02:00
* funcStart = tok ;
* argStart = tok - > next ( ) ;
return true ;
}
2010-11-13 08:08:45 +01:00
}
2014-04-13 13:05:30 +02:00
// UNKNOWN_MACRO(a,b) { ... }
else if ( outerScope - > type = = Scope : : eGlobal & &
2015-01-31 10:50:39 +01:00
Token : : Match ( tok , " %name% ( " ) & &
2014-04-13 13:05:30 +02:00
tok - > isUpperCaseName ( ) & &
2014-05-18 20:39:52 +02:00
Token : : simpleMatch ( tok - > linkAt ( 1 ) , " ) { " ) & &
2014-04-13 13:05:30 +02:00
( ! tok - > previous ( ) | | Token : : Match ( tok - > previous ( ) , " [;{}] " ) ) ) {
* funcStart = tok ;
* argStart = tok - > next ( ) ;
return true ;
}
2013-01-04 10:35:24 +01:00
// template constructor?
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( tok , " %name% < " ) & & Token : : simpleMatch ( tok - > next ( ) - > link ( ) , " > ( " ) ) {
2014-04-01 10:13:00 +02:00
const Token * tok2 = tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) ;
if ( Token : : Match ( tok2 , " ) const| ;|{|= " ) | |
2015-01-31 10:50:39 +01:00
Token : : Match ( tok2 , " ) : ::| %name% (|::|<|{ " ) | |
2014-04-10 16:17:10 +02:00
Token : : Match ( tok - > next ( ) - > link ( ) - > next ( ) - > link ( ) , " ) const| noexcept {|;|( " ) ) {
2014-04-01 10:13:00 +02:00
* funcStart = tok ;
* argStart = tok2 - > link ( ) ;
return true ;
}
2013-01-04 10:35:24 +01:00
}
2010-11-13 08:08:45 +01:00
return false ;
}
2016-02-03 17:08:46 +01:00
void SymbolDatabase : : validateExecutableScopes ( ) const
2015-12-05 20:55:26 +01:00
{
2016-02-03 17:08:46 +01:00
const std : : size_t functions = functionScopes . size ( ) ;
for ( std : : size_t i = 0 ; i < functions ; + + i ) {
const Scope * const scope = functionScopes [ i ] ;
const Function * const function = scope - > function ;
if ( scope - > isExecutable ( ) & & ! function ) {
const std : : list < const Token * > callstack ( 1 , scope - > classDef ) ;
const std : : string msg = std : : string ( " Executable scope ' " ) + scope - > classDef - > str ( ) + " ' with unknown function. " ;
const ErrorLogger : : ErrorMessage errmsg ( callstack , & _tokenizer - > list , Severity : : debug ,
" symbolDatabaseWarning " ,
msg ,
false ) ;
_errorLogger - > reportErr ( errmsg ) ;
}
}
}
2015-12-06 08:14:04 +01:00
2016-02-03 21:52:02 +01:00
namespace {
const Function * getFunctionForArgumentvariable ( const Variable * const var , const std : : vector < const Scope * > & functionScopes )
{
const std : : size_t functions = functionScopes . size ( ) ;
for ( std : : size_t i = 0 ; i < functions ; + + i ) {
const Scope * const scope = functionScopes [ i ] ;
const Function * const function = scope - > function ;
if ( function ) {
for ( std : : size_t arg = 0 ; arg < function - > argCount ( ) ; + + arg ) {
if ( var = = function - > getArgumentVar ( arg ) )
return function ;
}
}
}
return nullptr ;
}
}
2016-02-03 17:08:46 +01:00
void SymbolDatabase : : validateVariables ( ) const
{
for ( std : : vector < const Variable * > : : const_iterator iter = _variableList . begin ( ) ; iter ! = _variableList . end ( ) ; + + iter ) {
2016-02-03 21:52:02 +01:00
if ( * iter ) {
const Variable * const var = * iter ;
if ( ! var - > scope ( ) ) {
const Function * function = getFunctionForArgumentvariable ( var , functionScopes ) ;
if ( ! var - > isArgument ( ) | | ( function & & function - > hasBody ( ) ) ) {
throw InternalError ( var - > nameToken ( ) , " Analysis failed (variable without scope). If the code is valid then please report this failure. " , InternalError : : INTERNAL ) ;
//std::cout << "!!!Variable found without scope: " << var->nameToken()->str() << std::endl;
}
}
2015-12-05 20:55:26 +01:00
}
}
}
2016-02-03 17:08:46 +01:00
void SymbolDatabase : : validate ( ) const
{
if ( _settings - > debugwarnings ) {
validateExecutableScopes ( ) ;
}
//validateVariables();
}
2015-08-29 00:09:33 +02:00
bool Variable : : isPointerArray ( ) const
{
2015-08-28 23:06:39 +02:00
return isArray ( ) & & nameToken ( ) & & nameToken ( ) - > previous ( ) & & ( nameToken ( ) - > previous ( ) - > str ( ) = = " * " ) ;
}
2014-10-16 09:11:09 +02:00
const Token * Variable : : declEndToken ( ) const
{
Token const * declEnd = typeStartToken ( ) ;
while ( declEnd & & ! Token : : Match ( declEnd , " [;,)={] " ) ) {
if ( declEnd - > link ( ) & & Token : : Match ( declEnd , " (|[ " ) )
declEnd = declEnd - > link ( ) ;
declEnd = declEnd - > next ( ) ;
}
return declEnd ;
}
2015-01-30 21:56:27 +01:00
void Variable : : evaluate ( const Library * lib )
2012-05-11 17:56:47 +02:00
{
2015-08-25 21:19:19 +02:00
if ( _name )
setFlag ( fIsArray , arrayDimensions ( lib ) ) ;
2012-05-11 17:56:47 +02:00
const Token * tok = _start ;
2012-05-15 21:03:43 +02:00
while ( tok & & tok - > previous ( ) & & tok - > previous ( ) - > isName ( ) )
2012-05-11 17:56:47 +02:00
tok = tok - > previous ( ) ;
2012-05-14 21:47:02 +02:00
for ( const Token * const end = _name ? _name : _end ; tok ! = end ; ) {
2012-05-11 17:56:47 +02:00
if ( tok - > str ( ) = = " static " )
setFlag ( fIsStatic , true ) ;
2012-06-22 11:23:50 +02:00
else if ( tok - > str ( ) = = " extern " )
setFlag ( fIsExtern , true ) ;
2012-05-11 17:56:47 +02:00
else if ( tok - > str ( ) = = " mutable " )
setFlag ( fIsMutable , true ) ;
else if ( tok - > str ( ) = = " const " )
setFlag ( fIsConst , true ) ;
else if ( tok - > str ( ) = = " * " ) {
2015-08-25 21:19:19 +02:00
setFlag ( fIsPointer , ! isArray ( ) | | Token : : Match ( tok - > previous ( ) , " ( * %name% ) " ) ) ;
2012-05-11 17:56:47 +02:00
setFlag ( fIsConst , false ) ; // Points to const, isn't necessarily const itself
2013-04-04 18:47:44 +02:00
} else if ( tok - > str ( ) = = " & " ) {
if ( isReference ( ) )
setFlag ( fIsRValueRef , true ) ;
2012-05-11 17:56:47 +02:00
setFlag ( fIsReference , true ) ;
2013-04-04 18:47:44 +02:00
} else if ( tok - > str ( ) = = " && " ) { // Before simplification, && isn't split up
setFlag ( fIsRValueRef , true ) ;
setFlag ( fIsReference , true ) ; // Set also fIsReference
}
2012-05-14 21:47:02 +02:00
2013-04-15 22:48:28 +02:00
if ( tok - > str ( ) = = " < " & & tok - > link ( ) )
tok = tok - > link ( ) ;
2012-05-14 21:47:02 +02:00
else
tok = tok - > next ( ) ;
2012-05-11 17:56:47 +02:00
}
2014-09-14 11:26:16 +02:00
while ( Token : : Match ( _start , " static|const %any% " ) )
2012-06-08 18:05:02 +02:00
_start = _start - > next ( ) ;
while ( _end & & _end - > previous ( ) & & _end - > str ( ) = = " const " )
_end = _end - > previous ( ) ;
2014-01-28 15:44:56 +01:00
if ( _start ) {
2015-11-15 17:51:27 +01:00
std : : string strtype = _start - > str ( ) ;
for ( const Token * typeToken = _start ; Token : : Match ( typeToken , " %type% :: %type% " ) ; typeToken = typeToken - > tokAt ( 2 ) )
strtype + = " :: " + typeToken - > strAt ( 2 ) ;
setFlag ( fIsClass , ! lib - > podtype ( strtype ) & & ! _start - > isStandardType ( ) & & ! isPointer ( ) & & ! isReference ( ) ) ;
2014-05-07 15:59:21 +02:00
setFlag ( fIsStlType , Token : : simpleMatch ( _start , " std :: " ) ) ;
2014-09-05 12:03:08 +02:00
setFlag ( fIsStlString , isStlType ( ) & & ( Token : : Match ( _start - > tokAt ( 2 ) , " string|wstring|u16string|u32string !!:: " ) | | ( Token : : simpleMatch ( _start - > tokAt ( 2 ) , " basic_string < " ) & & ! Token : : simpleMatch ( _start - > linkAt ( 3 ) , " > :: " ) ) ) ) ;
2014-01-28 15:44:56 +01:00
}
2012-09-06 20:50:46 +02:00
if ( _access = = Argument ) {
tok = _name ;
2012-12-29 08:32:43 +01:00
if ( ! tok ) {
// Argument without name
tok = _end ;
// back up to start of array dimensions
while ( tok & & tok - > str ( ) = = " ] " )
tok = tok - > link ( ) - > previous ( ) ;
// add array dimensions if present
2013-03-12 15:42:00 +01:00
if ( tok & & tok - > next ( ) - > str ( ) = = " [ " )
2015-01-30 21:56:27 +01:00
setFlag ( fIsArray , arrayDimensions ( lib ) ) ;
2012-12-29 08:32:43 +01:00
}
2012-09-06 20:50:46 +02:00
if ( ! tok )
return ;
tok = tok - > next ( ) ;
2012-05-11 17:56:47 +02:00
while ( tok - > str ( ) = = " [ " )
tok = tok - > link ( ) ;
setFlag ( fHasDefault , tok - > str ( ) = = " = " ) ;
}
2013-02-18 06:33:53 +01:00
// check for C++11 member initialization
if ( _scope & & _scope - > isClassOrStruct ( ) ) {
2014-10-12 15:06:07 +02:00
// type var = x or
// type var = {x}
2013-02-18 06:33:53 +01:00
// type var = x; gets simplified to: type var ; var = x ;
2014-10-16 09:11:09 +02:00
Token const * declEnd = declEndToken ( ) ;
2015-01-31 10:50:39 +01:00
if ( ( Token : : Match ( declEnd , " ; %name% = " ) & & declEnd - > strAt ( 1 ) = = _name - > str ( ) ) | |
2014-10-16 09:11:09 +02:00
Token : : Match ( declEnd , " =|{ " ) )
2013-02-18 06:33:53 +01:00
setFlag ( fHasDefault , true ) ;
}
2014-08-09 11:44:55 +02:00
if ( _start ) {
2014-09-14 11:26:16 +02:00
if ( Token : : Match ( _start , " bool|char|short|int|long " ) )
2014-08-09 11:44:55 +02:00
setFlag ( fIsIntType , true ) ;
2014-09-14 11:26:16 +02:00
else if ( Token : : Match ( _start , " float|double " ) )
2014-08-09 11:44:55 +02:00
setFlag ( fIsFloatType , true ) ;
}
2012-05-11 17:56:47 +02:00
}
2012-04-18 13:00:34 +02:00
bool Function : : argsMatch ( const Scope * scope , const Token * first , const Token * second , const std : : string & path , unsigned int depth )
2010-11-13 08:08:45 +01:00
{
2012-11-26 16:34:44 +01:00
const bool isCPP = scope - > check - > isCPP ( ) ;
2014-09-01 17:01:05 +02:00
if ( ! isCPP ) // C does not support overloads
return true ;
2012-11-26 16:34:44 +01:00
2014-09-01 17:01:05 +02:00
// skip "struct"
if ( first - > str ( ) = = " struct " )
first = first - > next ( ) ;
if ( second - > str ( ) = = " struct " )
second = second - > next ( ) ;
2012-11-26 16:34:44 +01:00
2013-01-27 02:46:00 +01:00
// skip const on type passed by value
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( first , " const %type% %name%|,|) " ) )
2013-01-27 02:46:00 +01:00
first = first - > next ( ) ;
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( second , " const %type% %name%|,|) " ) )
2013-01-27 02:46:00 +01:00
second = second - > next ( ) ;
2013-03-14 19:11:29 +01:00
while ( first - > str ( ) = = second - > str ( ) & &
first - > isLong ( ) = = second - > isLong ( ) & &
first - > isUnsigned ( ) = = second - > isUnsigned ( ) ) {
2010-11-13 08:08:45 +01:00
// at end of argument list
2011-10-13 20:53:06 +02:00
if ( first - > str ( ) = = " ) " ) {
2012-04-18 13:00:34 +02:00
return true ;
2010-11-13 08:08:45 +01:00
}
// skip default value assignment
2011-10-13 20:53:06 +02:00
else if ( first - > next ( ) - > str ( ) = = " = " ) {
2012-02-11 12:26:48 +01:00
first = first - > nextArgument ( ) ;
2013-05-18 18:33:24 +02:00
if ( first )
first = first - > tokAt ( - 2 ) ;
2012-02-11 12:26:48 +01:00
if ( second - > next ( ) - > str ( ) = = " = " ) {
second = second - > nextArgument ( ) ;
2013-05-18 18:33:24 +02:00
if ( second )
second = second - > tokAt ( - 2 ) ;
2012-02-11 12:26:48 +01:00
if ( ! first | | ! second ) { // End of argument list (first or second)
2012-04-18 13:00:34 +02:00
return ! first & & ! second ;
2012-02-11 12:26:48 +01:00
}
} else if ( ! first ) { // End of argument list (first)
2012-04-18 13:00:34 +02:00
return second - > next ( ) & & second - > next ( ) - > str ( ) = = " ) " ;
2012-02-11 12:26:48 +01:00
}
2013-05-18 18:33:24 +02:00
} else if ( second - > next ( ) - > str ( ) = = " = " ) {
second = second - > nextArgument ( ) ;
if ( second )
second = second - > tokAt ( - 2 ) ;
2015-02-19 15:16:38 +01:00
if ( ! second ) { // End of argument list (second)
return false ;
2013-05-18 18:33:24 +02:00
}
2011-08-26 01:13:53 +02:00
}
2010-11-13 08:08:45 +01:00
// definition missing variable name
2015-08-16 14:22:46 +02:00
else if ( ( first - > next ( ) - > str ( ) = = " , " & & second - > next ( ) - > str ( ) ! = " , " ) | |
( first - > next ( ) - > str ( ) = = " ) " & & second - > next ( ) - > str ( ) ! = " ) " ) ) {
2010-11-13 08:08:45 +01:00
second = second - > next ( ) ;
2013-05-18 18:33:24 +02:00
// skip default value assignment
if ( second - > next ( ) - > str ( ) = = " = " ) {
2015-08-16 14:22:46 +02:00
do {
2013-05-18 18:33:24 +02:00
second = second - > next ( ) ;
2015-08-16 14:22:46 +02:00
} while ( ! Token : : Match ( second - > next ( ) , " ,|) " ) ) ;
2013-05-18 18:33:24 +02:00
}
} else if ( first - > next ( ) - > str ( ) = = " [ " & & second - > next ( ) - > str ( ) ! = " [ " )
2012-12-28 16:28:29 +01:00
second = second - > next ( ) ;
2010-11-13 08:08:45 +01:00
// function missing variable name
2015-08-16 14:22:46 +02:00
else if ( ( second - > next ( ) - > str ( ) = = " , " & & first - > next ( ) - > str ( ) ! = " , " ) | |
( second - > next ( ) - > str ( ) = = " ) " & & first - > next ( ) - > str ( ) ! = " ) " ) ) {
2010-11-13 08:08:45 +01:00
first = first - > next ( ) ;
2013-05-18 18:33:24 +02:00
// skip default value assignment
if ( first - > next ( ) - > str ( ) = = " = " ) {
2015-08-16 14:22:46 +02:00
do {
2013-05-18 18:33:24 +02:00
first = first - > next ( ) ;
2015-08-16 14:22:46 +02:00
} while ( ! Token : : Match ( first - > next ( ) , " ,|) " ) ) ;
2013-05-18 18:33:24 +02:00
}
} else if ( second - > next ( ) - > str ( ) = = " [ " & & first - > next ( ) - > str ( ) ! = " [ " )
2012-12-28 16:28:29 +01:00
first = first - > next ( ) ;
2010-11-13 08:08:45 +01:00
// argument list has different number of arguments
else if ( second - > str ( ) = = " ) " )
break ;
// variable names are different
2015-01-31 10:50:39 +01:00
else if ( ( Token : : Match ( first - > next ( ) , " %name% ,|)|= " ) & &
Token : : Match ( second - > next ( ) , " %name% ,|) " ) ) & &
2011-10-13 20:53:06 +02:00
( first - > next ( ) - > str ( ) ! = second - > next ( ) - > str ( ) ) ) {
2010-11-13 08:08:45 +01:00
// skip variable names
first = first - > next ( ) ;
second = second - > next ( ) ;
}
// variable with class path
2015-01-31 10:50:39 +01:00
else if ( depth & & Token : : Match ( first - > next ( ) , " %name% " ) ) {
2010-11-13 08:08:45 +01:00
std : : string param = path + first - > next ( ) - > str ( ) ;
2015-12-03 13:04:55 +01:00
if ( Token : : simpleMatch ( second - > next ( ) , param . c_str ( ) ) ) {
2010-11-13 08:08:45 +01:00
second = second - > tokAt ( int ( depth ) * 2 ) ;
2011-10-13 20:53:06 +02:00
} else if ( depth > 1 ) {
2012-04-18 13:00:34 +02:00
std : : string short_path = path ;
2010-11-13 08:08:45 +01:00
// remove last " :: "
short_path . resize ( short_path . size ( ) - 4 ) ;
// remove last name
2015-08-16 14:22:46 +02:00
std : : string : : size_type lastSpace = short_path . find_last_of ( ' ' ) ;
if ( lastSpace ! = std : : string : : npos )
short_path . resize ( lastSpace + 1 ) ;
2010-11-13 08:08:45 +01:00
param = short_path + first - > next ( ) - > str ( ) ;
2015-12-03 13:04:55 +01:00
if ( Token : : simpleMatch ( second - > next ( ) , param . c_str ( ) ) ) {
2010-11-13 08:08:45 +01:00
second = second - > tokAt ( ( int ( depth ) - 1 ) * 2 ) ;
}
}
}
2010-12-07 19:42:30 +01:00
// nested class variable
2015-01-31 10:50:39 +01:00
else if ( depth = = 0 & & Token : : Match ( first - > next ( ) , " %name% " ) & &
2011-01-21 07:42:41 +01:00
second - > next ( ) - > str ( ) = = scope - > className & & second - > strAt ( 2 ) = = " :: " & &
2011-10-13 20:53:06 +02:00
first - > next ( ) - > str ( ) = = second - > strAt ( 3 ) ) {
2010-12-07 19:42:30 +01:00
second = second - > tokAt ( 2 ) ;
}
2010-11-13 08:08:45 +01:00
first = first - > next ( ) ;
second = second - > next ( ) ;
2012-11-26 16:34:44 +01:00
2014-09-01 17:01:05 +02:00
// skip "struct"
if ( first - > str ( ) = = " struct " )
first = first - > next ( ) ;
if ( second - > str ( ) = = " struct " )
second = second - > next ( ) ;
2013-01-27 02:46:00 +01:00
// skip const on type passed by value
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( first , " const %type% %name%|,|) " ) )
2013-01-27 02:46:00 +01:00
first = first - > next ( ) ;
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( second , " const %type% %name%|,|) " ) )
2013-01-27 02:46:00 +01:00
second = second - > next ( ) ;
2010-11-13 08:08:45 +01:00
}
2012-04-18 13:00:34 +02:00
return false ;
2010-11-13 08:08:45 +01:00
}
2012-02-11 12:26:48 +01:00
Function * SymbolDatabase : : addGlobalFunction ( Scope * & scope , const Token * & tok , const Token * argStart , const Token * funcStart )
2012-03-23 17:59:51 +01:00
{
2015-12-06 08:14:04 +01:00
Function * function = nullptr ;
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : iterator i = scope - > functionMap . find ( tok - > str ( ) ) ; i ! = scope - > functionMap . end ( ) & & i - > first = = tok - > str ( ) ; + + i ) {
if ( Function : : argsMatch ( scope , i - > second - > argDef - > next ( ) , argStart - > next ( ) , " " , 0 ) ) {
function = const_cast < Function * > ( i - > second ) ;
2014-03-14 05:40:17 +01:00
break ;
}
2012-03-23 17:59:51 +01:00
}
2012-10-14 17:30:37 +02:00
2012-03-23 17:59:51 +01:00
if ( ! function )
2013-08-31 18:58:55 +02:00
function = addGlobalFunctionDecl ( scope , tok , argStart , funcStart ) ;
2012-03-23 17:59:51 +01:00
function - > arg = argStart ;
function - > token = funcStart ;
2015-01-08 05:45:31 +01:00
function - > hasBody ( true ) ;
2012-03-23 17:59:51 +01:00
addNewFunction ( & scope , & tok ) ;
if ( scope ) {
2012-04-25 20:52:59 +02:00
scope - > function = function ;
2012-04-27 21:51:13 +02:00
function - > functionScope = scope ;
2012-03-23 17:59:51 +01:00
return function ;
}
2015-12-06 08:14:04 +01:00
return nullptr ;
2012-03-23 17:59:51 +01:00
}
2013-08-31 18:58:55 +02:00
Function * SymbolDatabase : : addGlobalFunctionDecl ( Scope * & scope , const Token * tok , const Token * argStart , const Token * funcStart )
2012-02-11 12:26:48 +01:00
{
Function function ;
// save the function definition argument start '('
function . argDef = argStart ;
// save the access type
function . access = Public ;
// save the function name location
function . tokenDef = funcStart ;
2015-01-08 05:45:31 +01:00
function . isInline ( false ) ;
function . hasBody ( false ) ;
2012-02-11 12:26:48 +01:00
function . type = Function : : eFunction ;
2012-05-24 17:40:43 +02:00
function . nestedIn = scope ;
2012-02-11 12:26:48 +01:00
2013-08-31 18:58:55 +02:00
const Token * tok1 = tok ;
// look for end of previous statement
while ( tok1 - > previous ( ) & & ! Token : : Match ( tok1 - > previous ( ) , " ;|}|{ " ) )
tok1 = tok1 - > previous ( ) ;
// find the return type
2015-01-09 16:08:34 +01:00
while ( tok1 & & Token : : Match ( tok1 - > next ( ) , " static|extern|const " ) ) {
2015-01-09 00:34:48 +01:00
if ( tok1 - > str ( ) = = " static " )
function . isStaticLocal ( true ) ;
else if ( tok1 - > str ( ) = = " extern " )
function . isExtern ( true ) ;
2013-08-31 18:58:55 +02:00
tok1 = tok1 - > next ( ) ;
2015-01-09 00:34:48 +01:00
}
2013-08-31 18:58:55 +02:00
if ( tok1 )
function . retDef = tok1 ;
2015-01-02 21:38:19 +01:00
scope - > addFunction ( function ) ;
2012-03-23 17:59:51 +01:00
return & scope - > functionList . back ( ) ;
2012-02-11 12:26:48 +01:00
}
void SymbolDatabase : : addClassFunction ( Scope * * scope , const Token * * tok , const Token * argStart )
2010-11-13 08:08:45 +01:00
{
2012-01-22 19:34:53 +01:00
const bool destructor ( ( * tok ) - > previous ( ) - > str ( ) = = " ~ " ) ;
2016-01-02 18:53:51 +01:00
const bool has_const ( argStart - > link ( ) - > strAt ( 1 ) = = " const " ) ;
const bool lval ( argStart - > link ( ) - > strAt ( has_const ? 2 : 1 ) = = " & " ) ;
const bool rval ( argStart - > link ( ) - > strAt ( has_const ? 2 : 1 ) = = " && " ) ;
2014-07-24 16:34:13 +02:00
const Token * tok1 ;
2010-11-23 18:41:07 +01:00
// skip class/struct name
2012-01-22 19:34:53 +01:00
if ( destructor )
2010-11-23 18:41:07 +01:00
tok1 = ( * tok ) - > tokAt ( - 3 ) ;
2015-04-09 21:01:47 +02:00
else if ( ( * tok ) - > strAt ( - 2 ) = = " > " & & ( * tok ) - > linkAt ( - 2 ) )
tok1 = ( * tok ) - > linkAt ( - 2 ) - > previous ( ) ;
2010-11-23 18:41:07 +01:00
else
tok1 = ( * tok ) - > tokAt ( - 2 ) ;
2010-11-13 08:08:45 +01:00
2011-08-11 23:57:54 +02:00
// syntax error?
if ( ! tok1 )
return ;
2014-07-24 16:34:13 +02:00
int count = 0 ;
std : : string path ;
unsigned int path_length = 0 ;
2010-11-13 08:08:45 +01:00
// back up to head of path
2012-11-30 06:03:58 +01:00
while ( tok1 & & tok1 - > previous ( ) & & tok1 - > previous ( ) - > str ( ) = = " :: " & &
2015-04-09 21:01:47 +02:00
tok1 - > tokAt ( - 2 ) & & ( tok1 - > tokAt ( - 2 ) - > isName ( ) | | ( tok1 - > strAt ( - 2 ) = = " > " & & tok1 - > linkAt ( - 2 ) ) ) ) {
if ( tok1 - > strAt ( - 2 ) = = " > " ) {
tok1 = tok1 - > tokAt ( - 2 ) ;
const Token * tok2 = tok1 - > previous ( ) ;
path = " :: " + path ;
if ( tok2 ) {
do {
path = tok1 - > str ( ) + " " + path ;
tok1 = tok1 - > previous ( ) ;
count + + ;
path_length + + ;
} while ( tok1 ! = tok2 ) ;
}
} else {
path = tok1 - > str ( ) + " :: " + path ;
tok1 = tok1 - > tokAt ( - 2 ) ;
count + + ;
path_length + + ;
}
2010-11-13 08:08:45 +01:00
}
2011-12-13 00:24:34 +01:00
if ( tok1 & & count ) {
2010-11-13 08:08:45 +01:00
path = tok1 - > str ( ) + " :: " + path ;
path_length + + ;
}
2011-03-11 01:43:29 +01:00
std : : list < Scope > : : iterator it1 ;
2010-11-13 08:08:45 +01:00
// search for match
2011-10-13 20:53:06 +02:00
for ( it1 = scopeList . begin ( ) ; it1 ! = scopeList . end ( ) ; + + it1 ) {
2011-03-11 01:43:29 +01:00
Scope * scope1 = & ( * it1 ) ;
2010-11-13 08:08:45 +01:00
bool match = false ;
2014-04-10 16:11:11 +02:00
// check in namespace if using found
if ( * scope = = scope1 & & ! scope1 - > usingList . empty ( ) ) {
std : : list < Scope : : UsingInfo > : : const_iterator it2 ;
for ( it2 = scope1 - > usingList . begin ( ) ; it2 ! = scope1 - > usingList . end ( ) ; + + it2 ) {
if ( it2 - > scope ) {
Function * func = findFunctionInScope ( tok1 , it2 - > scope ) ;
if ( func ) {
2015-01-08 05:45:31 +01:00
if ( ! func - > hasBody ( ) ) {
func - > hasBody ( true ) ;
2014-04-10 16:11:11 +02:00
func - > token = * tok ;
func - > arg = argStart ;
addNewFunction ( scope , tok ) ;
if ( * scope ) {
( * scope ) - > functionOf = func - > nestedIn ;
2015-01-17 16:28:39 +01:00
( * scope ) - > function = func ;
2014-04-10 16:11:11 +02:00
( * scope ) - > function - > functionScope = * scope ;
}
return ;
}
}
}
}
}
2011-10-13 20:53:06 +02:00
if ( scope1 - > className = = tok1 - > str ( ) & & ( scope1 - > type ! = Scope : : eFunction ) ) {
2011-04-01 02:54:23 +02:00
// do the scopes match (same scope) or do their names match (multiple namespaces)
2013-08-21 16:32:10 +02:00
if ( ( * scope = = scope1 - > nestedIn ) | | ( * scope & &
2011-01-21 07:42:41 +01:00
( * scope ) - > className = = scope1 - > nestedIn - > className & &
! ( * scope ) - > className . empty ( ) & &
2011-10-13 20:53:06 +02:00
( * scope ) - > type = = scope1 - > nestedIn - > type ) ) {
2012-01-07 09:28:26 +01:00
// nested scopes => check that they match
{
const Scope * s1 = * scope ;
const Scope * s2 = scope1 - > nestedIn ;
while ( s1 & & s2 ) {
if ( s1 - > className ! = s2 - > className )
break ;
s1 = s1 - > nestedIn ;
s2 = s2 - > nestedIn ;
}
// Not matching scopes
if ( s1 | | s2 )
continue ;
}
2011-01-21 07:42:41 +01:00
Scope * scope2 = scope1 ;
2010-11-13 08:08:45 +01:00
2011-10-13 20:53:06 +02:00
while ( scope2 & & count > 0 ) {
2010-11-20 07:26:50 +01:00
count - - ;
tok1 = tok1 - > tokAt ( 2 ) ;
2011-01-21 07:42:41 +01:00
scope2 = scope2 - > findInNestedList ( tok1 - > str ( ) ) ;
2010-11-20 07:26:50 +01:00
}
2011-10-13 20:53:06 +02:00
if ( count = = 0 & & scope2 ) {
2010-11-20 07:26:50 +01:00
match = true ;
2011-01-21 07:42:41 +01:00
scope1 = scope2 ;
2010-11-20 07:26:50 +01:00
}
2010-11-13 08:08:45 +01:00
}
}
2011-10-13 20:53:06 +02:00
if ( match ) {
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : iterator it = scope1 - > functionMap . find ( ( * tok ) - > str ( ) ) ; it ! = scope1 - > functionMap . end ( ) & & it - > first = = ( * tok ) - > str ( ) ; + + it ) {
Function * func = const_cast < Function * > ( it - > second ) ;
2015-01-08 05:45:31 +01:00
if ( ! func - > hasBody ( ) ) {
2012-05-22 21:58:46 +02:00
if ( Function : : argsMatch ( scope1 , func - > argDef , ( * tok ) - > next ( ) , path , path_length ) ) {
if ( func - > type = = Function : : eDestructor & & destructor ) {
2015-01-08 05:45:31 +01:00
func - > hasBody ( true ) ;
2012-05-22 21:58:46 +02:00
} else if ( func - > type ! = Function : : eDestructor & & ! destructor ) {
2010-11-13 08:08:45 +01:00
// normal function?
2014-04-12 16:06:31 +02:00
if ( ( * tok ) - > next ( ) - > link ( ) ) {
2014-09-19 09:29:31 +02:00
const bool hasConstKeyword = ( * tok ) - > next ( ) - > link ( ) - > next ( ) - > str ( ) = = " const " ;
2016-01-02 18:53:51 +01:00
if ( ( func - > isConst ( ) = = hasConstKeyword ) & &
( func - > hasLvalRefQualifier ( ) = = lval ) & &
( func - > hasRvalRefQualifier ( ) = = rval ) ) {
2015-01-08 05:45:31 +01:00
func - > hasBody ( true ) ;
2010-11-13 08:08:45 +01:00
}
}
}
2015-01-08 05:45:31 +01:00
if ( func - > hasBody ( ) ) {
2012-05-22 21:58:46 +02:00
func - > token = * tok ;
func - > arg = argStart ;
addNewFunction ( scope , tok ) ;
if ( * scope ) {
( * scope ) - > functionOf = scope1 ;
2015-01-17 16:28:39 +01:00
( * scope ) - > function = func ;
2012-05-22 21:58:46 +02:00
( * scope ) - > function - > functionScope = * scope ;
}
return ;
2010-12-16 19:04:47 +01:00
}
2010-11-13 08:08:45 +01:00
}
}
}
}
}
2012-04-27 21:51:13 +02:00
// class function of unknown class
addNewFunction ( scope , tok ) ;
2010-11-13 08:08:45 +01:00
}
2011-01-21 07:42:41 +01:00
void SymbolDatabase : : addNewFunction ( Scope * * scope , const Token * * tok )
2010-11-13 08:08:45 +01:00
{
const Token * tok1 = * tok ;
2011-03-11 01:43:29 +01:00
scopeList . push_back ( Scope ( this , tok1 , * scope ) ) ;
2015-09-04 16:00:44 +02:00
Scope * newScope = & scopeList . back ( ) ;
2010-11-13 08:08:45 +01:00
2015-04-06 19:47:21 +02:00
// find start of function '{'
bool foundInitList = false ;
while ( tok1 & & tok1 - > str ( ) ! = " { " & & tok1 - > str ( ) ! = " ; " ) {
if ( tok1 - > link ( ) & & Token : : Match ( tok1 , " (|< " ) ) {
2012-07-10 15:15:11 +02:00
tok1 = tok1 - > link ( ) ;
2015-04-06 19:47:21 +02:00
} else if ( foundInitList & &
Token : : Match ( tok1 , " %name%|> { " ) & &
Token : : Match ( tok1 - > linkAt ( 1 ) , " } ,|{ " ) ) {
tok1 = tok1 - > linkAt ( 1 ) ;
} else {
if ( tok1 - > str ( ) = = " : " )
foundInitList = true ;
tok1 = tok1 - > next ( ) ;
}
2012-07-10 15:15:11 +02:00
}
2010-11-13 08:08:45 +01:00
2015-04-06 19:47:21 +02:00
if ( tok1 & & tok1 - > str ( ) = = " { " ) {
2015-09-04 16:00:44 +02:00
newScope - > classStart = tok1 ;
newScope - > classEnd = tok1 - > link ( ) ;
2010-11-13 08:08:45 +01:00
2010-12-16 19:04:47 +01:00
// syntax error?
2015-09-04 16:00:44 +02:00
if ( ! newScope - > classEnd ) {
2011-03-11 01:43:29 +01:00
scopeList . pop_back ( ) ;
2010-12-16 19:04:47 +01:00
while ( tok1 - > next ( ) )
tok1 = tok1 - > next ( ) ;
2014-02-16 10:32:10 +01:00
* scope = nullptr ;
2010-12-16 19:04:47 +01:00
* tok = tok1 ;
return ;
}
2015-09-04 16:00:44 +02:00
( * scope ) - > nestedList . push_back ( newScope ) ;
* scope = newScope ;
2010-11-13 08:08:45 +01:00
* tok = tok1 ;
2011-10-13 20:53:06 +02:00
} else {
2011-03-11 01:43:29 +01:00
scopeList . pop_back ( ) ;
2014-02-16 10:32:10 +01:00
* scope = nullptr ;
* tok = nullptr ;
2011-01-22 08:36:47 +01:00
}
2010-11-13 08:08:45 +01:00
}
2013-03-05 15:28:40 +01:00
const Token * Type : : initBaseInfo ( const Token * tok , const Token * tok1 )
2010-11-13 08:08:45 +01:00
{
// goto initial '{'
2012-11-30 06:03:58 +01:00
const Token * tok2 = tok1 ;
2011-10-13 20:53:06 +02:00
while ( tok2 & & tok2 - > str ( ) ! = " { " ) {
2010-11-13 08:08:45 +01:00
// skip unsupported templates
if ( tok2 - > str ( ) = = " < " )
2013-04-15 22:48:28 +02:00
tok2 = tok2 - > link ( ) ;
2010-11-13 08:08:45 +01:00
// check for base classes
2012-04-27 21:51:13 +02:00
else if ( Token : : Match ( tok2 , " :|, " ) ) {
2010-11-13 08:08:45 +01:00
tok2 = tok2 - > next ( ) ;
2011-02-19 20:38:00 +01:00
// check for invalid code
2011-02-21 20:25:35 +01:00
if ( ! tok2 | | ! tok2 - > next ( ) )
2014-02-16 10:32:10 +01:00
return nullptr ;
2011-02-19 20:38:00 +01:00
2014-09-19 09:29:31 +02:00
Type : : BaseInfo base ;
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " virtual " ) {
2011-03-22 01:03:41 +01:00
base . isVirtual = true ;
tok2 = tok2 - > next ( ) ;
}
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " public " ) {
2010-11-13 08:08:45 +01:00
base . access = Public ;
tok2 = tok2 - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( tok2 - > str ( ) = = " protected " ) {
2010-11-13 08:08:45 +01:00
base . access = Protected ;
tok2 = tok2 - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( tok2 - > str ( ) = = " private " ) {
2010-11-13 08:08:45 +01:00
base . access = Private ;
tok2 = tok2 - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else {
2010-11-13 08:08:45 +01:00
if ( tok - > str ( ) = = " class " )
base . access = Private ;
else if ( tok - > str ( ) = = " struct " )
base . access = Public ;
}
2015-06-02 18:28:43 +02:00
if ( ! tok2 )
return nullptr ;
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " virtual " ) {
2011-03-22 01:03:41 +01:00
base . isVirtual = true ;
tok2 = tok2 - > next ( ) ;
2011-06-30 01:43:39 +02:00
}
2015-06-02 20:15:21 +02:00
if ( ! tok2 )
return nullptr ;
2015-09-04 15:06:20 +02:00
base . nameTok = tok2 ;
2011-06-30 01:43:39 +02:00
// handle global namespace
2011-10-13 20:53:06 +02:00
if ( tok2 - > str ( ) = = " :: " ) {
2011-06-30 01:43:39 +02:00
tok2 = tok2 - > next ( ) ;
2011-03-22 01:03:41 +01:00
}
2010-11-13 08:08:45 +01:00
// handle derived base classes
2015-01-31 10:50:39 +01:00
while ( Token : : Match ( tok2 , " %name% :: " ) ) {
2010-11-13 08:08:45 +01:00
tok2 = tok2 - > tokAt ( 2 ) ;
}
2015-06-02 20:15:21 +02:00
if ( ! tok2 )
return nullptr ;
2010-11-13 08:08:45 +01:00
2013-03-05 15:28:40 +01:00
base . name = tok2 - > str ( ) ;
2010-11-13 08:08:45 +01:00
2015-01-09 22:03:43 +01:00
tok2 = tok2 - > next ( ) ;
2011-02-15 01:50:13 +01:00
// add unhandled templates
2015-01-09 22:03:43 +01:00
if ( tok2 & & tok2 - > link ( ) & & tok2 - > str ( ) = = " < " ) {
for ( const Token * const end = tok2 - > link ( ) - > next ( ) ; tok2 ! = end ; tok2 = tok2 - > next ( ) ) {
base . name + = tok2 - > str ( ) ;
2010-11-30 19:52:42 +01:00
}
2010-11-25 07:43:39 +01:00
}
2010-11-13 08:08:45 +01:00
// save pattern for base class name
2012-05-17 10:33:24 +02:00
derivedFrom . push_back ( base ) ;
2015-01-09 22:03:43 +01:00
} else
2013-06-16 21:46:01 +02:00
tok2 = tok2 - > next ( ) ;
2010-11-13 08:08:45 +01:00
}
return tok2 ;
}
2011-01-15 07:59:37 +01:00
2011-03-03 03:08:27 +01:00
void SymbolDatabase : : debugMessage ( const Token * tok , const std : : string & msg ) const
{
2011-10-13 20:53:06 +02:00
if ( tok & & _settings - > debugwarnings ) {
2012-05-06 13:01:56 +02:00
const std : : list < const Token * > locationList ( 1 , tok ) ;
const ErrorLogger : : ErrorMessage errmsg ( locationList , & _tokenizer - > list ,
2011-03-03 03:08:27 +01:00
Severity : : debug ,
2011-04-14 18:02:01 +02:00
" debug " ,
2012-08-12 11:11:54 +02:00
msg ,
2011-04-14 18:02:01 +02:00
false ) ;
2011-03-03 03:08:27 +01:00
if ( _errorLogger )
_errorLogger - > reportErr ( errmsg ) ;
}
}
2013-03-14 17:00:22 +01:00
const Function * Type : : getFunction ( const std : : string & funcName ) const
{
if ( classScope ) {
2015-01-02 21:38:19 +01:00
std : : multimap < std : : string , const Function * > : : const_iterator it = classScope - > functionMap . find ( funcName ) ;
if ( it ! = classScope - > functionMap . end ( ) )
return it - > second ;
2013-03-14 17:00:22 +01:00
}
for ( std : : size_t i = 0 ; i < derivedFrom . size ( ) ; i + + ) {
if ( derivedFrom [ i ] . type ) {
2015-12-06 08:14:04 +01:00
const Function * const func = derivedFrom [ i ] . type - > getFunction ( funcName ) ;
2013-03-14 17:00:22 +01:00
if ( func )
return func ;
}
}
return 0 ;
}
2015-09-04 15:12:40 +02:00
bool Type : : hasCircularDependencies ( std : : set < BaseInfo > * ancestors ) const
2014-03-29 12:21:35 +01:00
{
2015-09-04 15:12:40 +02:00
std : : set < BaseInfo > knownAncestors ;
if ( ! ancestors ) {
ancestors = & knownAncestors ;
2014-03-29 12:21:35 +01:00
}
for ( std : : vector < BaseInfo > : : const_iterator parent = derivedFrom . begin ( ) ; parent ! = derivedFrom . end ( ) ; + + parent ) {
if ( ! parent - > type )
continue ;
else if ( this = = parent - > type )
return true ;
2015-09-04 15:12:40 +02:00
else if ( ancestors - > find ( * parent ) ! = ancestors - > end ( ) )
2014-03-29 12:21:35 +01:00
return true ;
else {
2015-09-04 15:12:40 +02:00
ancestors - > insert ( * parent ) ;
if ( parent - > type - > hasCircularDependencies ( ancestors ) )
2014-03-29 12:21:35 +01:00
return true ;
}
}
return false ;
}
2015-09-04 15:12:40 +02:00
bool Type : : findDependency ( const Type * ancestor ) const
2015-07-01 07:50:13 +02:00
{
2015-09-04 15:12:40 +02:00
if ( this = = ancestor )
2015-07-01 07:50:13 +02:00
return true ;
for ( std : : vector < BaseInfo > : : const_iterator parent = derivedFrom . begin ( ) ; parent ! = derivedFrom . end ( ) ; + + parent ) {
2015-09-04 15:12:40 +02:00
if ( parent - > type & & parent - > type - > findDependency ( ancestor ) )
2015-07-01 07:50:13 +02:00
return true ;
}
return false ;
2015-07-01 00:04:01 +02:00
}
2015-01-30 21:56:27 +01:00
bool Variable : : arrayDimensions ( const Library * lib )
2011-06-23 04:41:11 +02:00
{
2015-01-30 21:56:27 +01:00
const Library : : Container * container = lib - > detectContainer ( _start ) ;
if ( container & & container - > arrayLike_indexOp & & container - > size_templateArgNo > 0 ) {
const Token * tok = Token : : findsimplematch ( _start , " < " ) ;
if ( tok ) {
2015-08-26 13:48:19 +02:00
Dimension dimension_ ;
2015-01-30 21:56:27 +01:00
tok = tok - > next ( ) ;
for ( int i = 0 ; i < container - > size_templateArgNo & & tok ; i + + ) {
tok = tok - > nextTemplateArgument ( ) ;
}
if ( tok ) {
2015-01-30 22:04:26 +01:00
dimension_ . start = tok ;
dimension_ . end = Token : : findmatch ( tok , " ,|> " ) ;
if ( dimension_ . end )
dimension_ . end = dimension_ . end - > previous ( ) ;
if ( dimension_ . start = = dimension_ . end )
dimension_ . num = MathLib : : toLongNumber ( dimension_ . start - > str ( ) ) ;
2015-01-30 21:56:27 +01:00
}
2015-01-30 22:04:26 +01:00
_dimensions . push_back ( dimension_ ) ;
2015-01-30 21:56:27 +01:00
return true ;
}
}
2011-06-23 04:41:11 +02:00
2015-01-30 21:56:27 +01:00
const Token * dim = _name ;
if ( ! dim ) {
// Argument without name
dim = _end ;
// back up to start of array dimensions
while ( dim & & dim - > str ( ) = = " ] " )
dim = dim - > link ( ) - > previous ( ) ;
}
if ( dim )
dim = dim - > next ( ) ;
2015-01-31 13:12:06 +01:00
if ( dim & & dim - > str ( ) = = " ) " )
dim = dim - > next ( ) ;
2011-06-23 04:41:11 +02:00
2015-01-30 22:04:26 +01:00
bool arr = false ;
2011-10-13 20:53:06 +02:00
while ( dim & & dim - > next ( ) & & dim - > str ( ) = = " [ " ) {
2015-01-30 22:04:26 +01:00
Dimension dimension_ ;
2011-08-28 19:32:42 +02:00
// check for empty array dimension []
2011-10-13 20:53:06 +02:00
if ( dim - > next ( ) - > str ( ) ! = " ] " ) {
2015-01-30 22:04:26 +01:00
dimension_ . start = dim - > next ( ) ;
dimension_ . end = dim - > link ( ) - > previous ( ) ;
if ( dimension_ . start = = dimension_ . end & & dimension_ . start - > isNumber ( ) )
dimension_ . num = MathLib : : toLongNumber ( dimension_ . start - > str ( ) ) ;
2011-08-28 19:32:42 +02:00
}
2015-01-30 22:04:26 +01:00
_dimensions . push_back ( dimension_ ) ;
2011-06-23 04:41:11 +02:00
dim = dim - > link ( ) - > next ( ) ;
2015-01-30 22:04:26 +01:00
arr = true ;
2011-06-23 04:41:11 +02:00
}
2015-01-30 22:04:26 +01:00
return arr ;
2011-06-23 04:41:11 +02:00
}
2012-01-05 18:22:54 +01:00
static std : : ostream & operator < < ( std : : ostream & s , Scope : : ScopeType type )
{
s < < ( type = = Scope : : eGlobal ? " Global " :
type = = Scope : : eClass ? " Class " :
type = = Scope : : eStruct ? " Struct " :
type = = Scope : : eUnion ? " Union " :
type = = Scope : : eNamespace ? " Namespace " :
type = = Scope : : eFunction ? " Function " :
type = = Scope : : eIf ? " If " :
type = = Scope : : eElse ? " Else " :
type = = Scope : : eFor ? " For " :
type = = Scope : : eWhile ? " While " :
type = = Scope : : eDo ? " Do " :
type = = Scope : : eSwitch ? " Switch " :
2012-01-26 04:05:29 +01:00
type = = Scope : : eTry ? " Try " :
type = = Scope : : eCatch ? " Catch " :
2012-01-05 18:22:54 +01:00
type = = Scope : : eUnconditional ? " Unconditional " :
2014-08-31 10:21:10 +02:00
type = = Scope : : eLambda ? " Lambda " :
2012-01-05 18:22:54 +01:00
" Unknown " ) ;
return s ;
}
2015-01-09 21:40:50 +01:00
static std : : string tokenToString ( const Token * tok , const Tokenizer * tokenizer )
2015-01-09 21:34:57 +01:00
{
std : : ostringstream oss ;
if ( tok ) {
oss < < tok - > str ( ) < < " " ;
2015-01-09 21:40:50 +01:00
oss < < tokenizer - > list . fileLine ( tok ) < < " " ;
2015-01-09 21:34:57 +01:00
}
oss < < tok ;
return oss . str ( ) ;
}
2015-01-09 21:40:50 +01:00
static std : : string scopeToString ( const Scope * scope , const Tokenizer * tokenizer )
2015-01-09 21:34:57 +01:00
{
std : : ostringstream oss ;
if ( scope ) {
oss < < scope - > type < < " " ;
if ( scope - > classDef )
2015-01-09 21:40:50 +01:00
oss < < tokenizer - > list . fileLine ( scope - > classDef ) < < " " ;
2015-01-09 21:34:57 +01:00
}
oss < < scope ;
return oss . str ( ) ;
}
2012-01-05 18:22:54 +01:00
void SymbolDatabase : : printVariable ( const Variable * var , const char * indent ) const
{
2015-01-09 21:34:57 +01:00
std : : cout < < indent < < " _name: " < < tokenToString ( var - > nameToken ( ) , _tokenizer ) < < std : : endl ;
2012-01-05 18:24:27 +01:00
if ( var - > nameToken ( ) ) {
2013-07-20 12:31:04 +02:00
std : : cout < < indent < < " declarationId: " < < var - > declarationId ( ) < < std : : endl ;
2015-01-09 21:34:57 +01:00
}
std : : cout < < indent < < " _start: " < < tokenToString ( var - > typeStartToken ( ) , _tokenizer ) < < std : : endl ;
std : : cout < < indent < < " _end: " < < tokenToString ( var - > typeEndToken ( ) , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < indent < < " _index: " < < var - > index ( ) < < std : : endl ;
2012-01-05 18:24:27 +01:00
std : : cout < < indent < < " _access: " < <
( var - > isPublic ( ) ? " Public " :
var - > isProtected ( ) ? " Protected " :
var - > isPrivate ( ) ? " Private " :
var - > isGlobal ( ) ? " Global " :
var - > isNamespace ( ) ? " Namespace " :
var - > isArgument ( ) ? " Argument " :
var - > isLocal ( ) ? " Local " :
2012-01-26 04:05:29 +01:00
var - > isThrow ( ) ? " Throw " :
2015-01-09 21:34:57 +01:00
" Unknown " ) < < std : : endl ;
2012-01-05 18:24:27 +01:00
std : : cout < < indent < < " _flags: " < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < indent < < " isMutable: " < < var - > isMutable ( ) < < std : : endl ;
std : : cout < < indent < < " isStatic: " < < var - > isStatic ( ) < < std : : endl ;
std : : cout < < indent < < " isExtern: " < < var - > isExtern ( ) < < std : : endl ;
std : : cout < < indent < < " isLocal: " < < var - > isLocal ( ) < < std : : endl ;
std : : cout < < indent < < " isConst: " < < var - > isConst ( ) < < std : : endl ;
std : : cout < < indent < < " isClass: " < < var - > isClass ( ) < < std : : endl ;
std : : cout < < indent < < " isArray: " < < var - > isArray ( ) < < std : : endl ;
std : : cout < < indent < < " isPointer: " < < var - > isPointer ( ) < < std : : endl ;
std : : cout < < indent < < " isReference: " < < var - > isReference ( ) < < std : : endl ;
std : : cout < < indent < < " isRValueRef: " < < var - > isRValueReference ( ) < < std : : endl ;
std : : cout < < indent < < " hasDefault: " < < var - > hasDefault ( ) < < std : : endl ;
std : : cout < < indent < < " isStlType: " < < var - > isStlType ( ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < indent < < " _type: " ;
2012-01-05 18:24:27 +01:00
if ( var - > type ( ) ) {
2013-03-05 13:33:38 +01:00
std : : cout < < var - > type ( ) - > name ( ) ;
2015-02-04 06:08:41 +01:00
std : : cout < < " " < < _tokenizer - > list . fileLine ( var - > type ( ) - > classDef ) ;
std : : cout < < " " < < var - > type ( ) < < std : : endl ;
2012-01-05 18:24:27 +01:00
} else
2012-01-05 18:22:54 +01:00
std : : cout < < " none " < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < indent < < " _scope: " < < scopeToString ( var - > scope ( ) , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < indent < < " _dimensions: " ;
2012-07-08 23:39:46 +02:00
for ( std : : size_t i = 0 ; i < var - > dimensions ( ) . size ( ) ; i + + ) {
2012-01-05 18:24:27 +01:00
std : : cout < < " " < < var - > dimension ( i ) ;
2012-09-10 16:14:24 +02:00
if ( ! var - > dimensions ( ) [ i ] . known )
std : : cout < < " ? " ;
2012-01-05 18:22:54 +01:00
}
std : : cout < < std : : endl ;
}
void SymbolDatabase : : printOut ( const char * title ) const
{
2015-01-09 21:34:57 +01:00
std : : cout < < std : : setiosflags ( std : : ios : : boolalpha ) ;
2012-01-05 18:22:54 +01:00
if ( title )
std : : cout < < " \n ### " < < title < < " ### \n " ;
2013-03-05 15:28:40 +01:00
for ( std : : list < Scope > : : const_iterator scope = scopeList . begin ( ) ; scope ! = scopeList . end ( ) ; + + scope ) {
2015-01-09 21:34:57 +01:00
std : : cout < < " Scope: " < < & * scope < < " " < < scope - > type < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < " className: " < < scope - > className < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < " classDef: " < < tokenToString ( scope - > classDef , _tokenizer ) < < std : : endl ;
std : : cout < < " classStart: " < < tokenToString ( scope - > classStart , _tokenizer ) < < std : : endl ;
std : : cout < < " classEnd: " < < tokenToString ( scope - > classEnd , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : list < Function > : : const_iterator func ;
// find the function body if not implemented inline
2012-01-05 18:24:27 +01:00
for ( func = scope - > functionList . begin ( ) ; func ! = scope - > functionList . end ( ) ; + + func ) {
2012-01-05 18:22:54 +01:00
std : : cout < < " Function: " < < & * func < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < " name: " < < tokenToString ( func - > tokenDef , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < " type: " < < ( func - > type = = Function : : eConstructor ? " Constructor " :
func - > type = = Function : : eCopyConstructor ? " CopyConstructor " :
2013-04-04 19:53:55 +02:00
func - > type = = Function : : eMoveConstructor ? " MoveConstructor " :
2012-01-05 18:22:54 +01:00
func - > type = = Function : : eOperatorEqual ? " OperatorEqual " :
func - > type = = Function : : eDestructor ? " Destructor " :
func - > type = = Function : : eFunction ? " Function " :
2015-01-09 21:34:57 +01:00
" Unknown " ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < " access: " < < ( func - > access = = Public ? " Public " :
func - > access = = Protected ? " Protected " :
func - > access = = Private ? " Private " :
2015-01-09 21:34:57 +01:00
" Unknown " ) < < std : : endl ;
std : : cout < < " hasBody: " < < func - > hasBody ( ) < < std : : endl ;
std : : cout < < " isInline: " < < func - > isInline ( ) < < std : : endl ;
std : : cout < < " isConst: " < < func - > isConst ( ) < < std : : endl ;
std : : cout < < " isVirtual: " < < func - > isVirtual ( ) < < std : : endl ;
std : : cout < < " isPure: " < < func - > isPure ( ) < < std : : endl ;
std : : cout < < " isStatic: " < < func - > isStatic ( ) < < std : : endl ;
std : : cout < < " isStaticLocal: " < < func - > isStaticLocal ( ) < < std : : endl ;
std : : cout < < " isExtern: " < < func - > isExtern ( ) < < std : : endl ;
std : : cout < < " isFriend: " < < func - > isFriend ( ) < < std : : endl ;
std : : cout < < " isExplicit: " < < func - > isExplicit ( ) < < std : : endl ;
std : : cout < < " isDefault: " < < func - > isDefault ( ) < < std : : endl ;
std : : cout < < " isDelete: " < < func - > isDelete ( ) < < std : : endl ;
std : : cout < < " isNoExcept: " < < func - > isNoExcept ( ) < < std : : endl ;
std : : cout < < " isThrow: " < < func - > isThrow ( ) < < std : : endl ;
std : : cout < < " isOperator: " < < func - > isOperator ( ) < < std : : endl ;
2016-01-02 18:53:51 +01:00
std : : cout < < " hasLvalRefQual: " < < func - > hasLvalRefQualifier ( ) < < std : : endl ;
std : : cout < < " hasRvalRefQual: " < < func - > hasRvalRefQualifier ( ) < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < " attributes: " ;
if ( func - > isAttributeConst ( ) )
std : : cout < < " const " ;
if ( func - > isAttributePure ( ) )
std : : cout < < " pure " ;
if ( func - > isAttributeNoreturn ( ) )
std : : cout < < " noreturn " ;
if ( func - > isAttributeNothrow ( ) )
std : : cout < < " nothrow " ;
if ( func - > isAttributeConstructor ( ) )
std : : cout < < " constructor " ;
if ( func - > isAttributeDestructor ( ) )
std : : cout < < " destructor " ;
std : : cout < < std : : endl ;
2014-04-10 16:17:10 +02:00
std : : cout < < " noexceptArg: " < < ( func - > noexceptArg ? func - > noexceptArg - > str ( ) : " none " ) < < std : : endl ;
std : : cout < < " throwArg: " < < ( func - > throwArg ? func - > throwArg - > str ( ) : " none " ) < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < " tokenDef: " < < tokenToString ( func - > tokenDef , _tokenizer ) < < std : : endl ;
std : : cout < < " argDef: " < < tokenToString ( func - > argDef , _tokenizer ) < < std : : endl ;
2013-09-03 10:48:01 +02:00
if ( ! func - > isConstructor ( ) & & ! func - > isDestructor ( ) )
2015-01-09 21:34:57 +01:00
std : : cout < < " retDef: " < < tokenToString ( func - > retDef , _tokenizer ) < < std : : endl ;
2013-08-26 06:03:26 +02:00
std : : cout < < " retType: " < < func - > retType < < std : : endl ;
2015-01-08 05:45:31 +01:00
if ( func - > hasBody ( ) ) {
2015-01-09 21:34:57 +01:00
std : : cout < < " token: " < < tokenToString ( func - > token , _tokenizer ) < < std : : endl ;
std : : cout < < " arg: " < < tokenToString ( func - > arg , _tokenizer ) < < std : : endl ;
2012-03-23 17:59:51 +01:00
}
2015-01-09 21:34:57 +01:00
std : : cout < < " nestedIn: " < < scopeToString ( func - > nestedIn , _tokenizer ) < < std : : endl ;
std : : cout < < " functionScope: " < < scopeToString ( func - > functionScope , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : list < Variable > : : const_iterator var ;
2012-01-05 18:24:27 +01:00
for ( var = func - > argumentList . begin ( ) ; var ! = func - > argumentList . end ( ) ; + + var ) {
2012-01-05 18:22:54 +01:00
std : : cout < < " Variable: " < < & * var < < std : : endl ;
printVariable ( & * var , " " ) ;
}
}
std : : list < Variable > : : const_iterator var ;
2012-01-05 18:24:27 +01:00
for ( var = scope - > varlist . begin ( ) ; var ! = scope - > varlist . end ( ) ; + + var ) {
2012-01-05 18:22:54 +01:00
std : : cout < < " Variable: " < < & * var < < std : : endl ;
printVariable ( & * var , " " ) ;
}
std : : cout < < " nestedIn: " < < scope - > nestedIn ;
2012-01-05 18:24:27 +01:00
if ( scope - > nestedIn ) {
std : : cout < < " " < < scope - > nestedIn - > type < < " "
2012-01-23 04:32:53 +01:00
< < scope - > nestedIn - > className ;
2012-01-05 18:22:54 +01:00
}
2012-01-23 04:32:53 +01:00
std : : cout < < std : : endl ;
2012-01-05 18:22:54 +01:00
2013-03-14 06:34:12 +01:00
std : : cout < < " definedType: " < < scope - > definedType < < std : : endl ;
2013-03-05 15:28:40 +01:00
2012-01-05 18:22:54 +01:00
std : : cout < < " nestedList[ " < < scope - > nestedList . size ( ) < < " ] = ( " ;
std : : list < Scope * > : : const_iterator nsi ;
2013-03-05 15:28:40 +01:00
std : : size_t count = scope - > nestedList . size ( ) ;
2012-01-05 18:24:27 +01:00
for ( nsi = scope - > nestedList . begin ( ) ; nsi ! = scope - > nestedList . end ( ) ; + + nsi ) {
2012-11-30 06:03:58 +01:00
std : : cout < < " " < < ( * nsi ) < < " " < < ( * nsi ) - > type < < " " < < ( * nsi ) - > className ;
2012-01-05 18:22:54 +01:00
if ( count - - > 1 )
std : : cout < < " , " ;
}
std : : cout < < " ) " < < std : : endl ;
2012-12-20 06:53:04 +01:00
std : : list < Scope : : UsingInfo > : : const_iterator use ;
2012-01-05 18:22:54 +01:00
2012-01-05 18:24:27 +01:00
for ( use = scope - > usingList . begin ( ) ; use ! = scope - > usingList . end ( ) ; + + use ) {
2012-12-20 06:53:04 +01:00
std : : cout < < " using: " < < use - > scope < < " " < < use - > start - > strAt ( 2 ) ;
const Token * tok1 = use - > start - > tokAt ( 3 ) ;
2012-01-05 18:24:27 +01:00
while ( tok1 & & tok1 - > str ( ) = = " :: " ) {
2012-01-05 18:22:54 +01:00
std : : cout < < " :: " < < tok1 - > strAt ( 1 ) ;
tok1 = tok1 - > tokAt ( 2 ) ;
}
2012-12-20 06:53:04 +01:00
std : : cout < < " " < < _tokenizer - > list . fileLine ( use - > start ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
}
2015-01-09 21:34:57 +01:00
std : : cout < < " functionOf: " < < scopeToString ( scope - > functionOf , _tokenizer ) < < std : : endl ;
2012-01-05 18:22:54 +01:00
std : : cout < < " function: " < < scope - > function ;
2015-01-09 21:34:57 +01:00
if ( scope - > function )
std : : cout < < " " < < scope - > function - > name ( ) ;
2012-01-05 18:22:54 +01:00
std : : cout < < std : : endl ;
}
2013-03-05 15:28:40 +01:00
for ( std : : list < Type > : : const_iterator type = typeList . begin ( ) ; type ! = typeList . end ( ) ; + + type ) {
2013-08-19 06:06:53 +02:00
std : : cout < < " Type: " < < & ( * type ) < < std : : endl ;
std : : cout < < " name: " < < type - > name ( ) < < std : : endl ;
2015-01-09 21:34:57 +01:00
std : : cout < < " classDef: " < < tokenToString ( type - > classDef , _tokenizer ) < < std : : endl ;
2013-03-05 15:28:40 +01:00
std : : cout < < " classScope: " < < type - > classScope < < std : : endl ;
std : : cout < < " enclosingScope: " < < type - > enclosingScope < < std : : endl ;
std : : cout < < " needInitialization: " < < ( type - > needInitialization = = Type : : Unknown ? " Unknown " :
type - > needInitialization = = Type : : True ? " True " :
type - > needInitialization = = Type : : False ? " False " :
" Invalid " ) < < std : : endl ;
std : : cout < < " derivedFrom[ " < < type - > derivedFrom . size ( ) < < " ] = ( " ;
std : : size_t count = type - > derivedFrom . size ( ) ;
for ( std : : size_t i = 0 ; i < type - > derivedFrom . size ( ) ; + + i ) {
if ( type - > derivedFrom [ i ] . isVirtual )
std : : cout < < " Virtual " ;
2013-08-19 06:06:53 +02:00
std : : cout < < ( type - > derivedFrom [ i ] . access = = Public ? " Public " :
type - > derivedFrom [ i ] . access = = Protected ? " Protected " :
type - > derivedFrom [ i ] . access = = Private ? " Private " :
2013-03-05 15:28:40 +01:00
" Unknown " ) ;
if ( type - > derivedFrom [ i ] . type )
2013-08-19 06:06:53 +02:00
std : : cout < < " " < < type - > derivedFrom [ i ] . type ;
2013-03-05 15:28:40 +01:00
else
std : : cout < < " Unknown " ;
std : : cout < < " " < < type - > derivedFrom [ i ] . name ;
if ( count - - > 1 )
std : : cout < < " , " ;
}
std : : cout < < " ) " < < std : : endl ;
std : : cout < < " friendList[ " < < type - > friendList . size ( ) < < " ] = ( " ;
std : : list < Type : : FriendInfo > : : const_iterator fii ;
count = type - > friendList . size ( ) ;
for ( fii = type - > friendList . begin ( ) ; fii ! = type - > friendList . end ( ) ; + + fii ) {
if ( fii - > type )
std : : cout < < fii - > type ;
else
std : : cout < < " Unknown " ;
std : : cout < < " " < < fii - > name ;
if ( count - - > 1 )
std : : cout < < " , " ;
}
std : : cout < < " ) " < < std : : endl ;
}
2013-03-13 06:31:13 +01:00
for ( std : : size_t i = 1 ; i < _variableList . size ( ) ; i + + ) {
std : : cout < < " _variableList[ " < < i < < " ]: " < < _variableList [ i ] ;
if ( _variableList [ i ] ) {
std : : cout < < " " < < _variableList [ i ] - > name ( ) < < " "
< < _tokenizer - > list . fileLine ( _variableList [ i ] - > nameToken ( ) ) ;
}
std : : cout < < std : : endl ;
2012-01-05 18:22:54 +01:00
}
2015-01-09 21:34:57 +01:00
std : : cout < < std : : resetiosflags ( std : : ios : : boolalpha ) ;
2012-01-05 18:22:54 +01:00
}
2014-07-14 15:51:45 +02:00
void SymbolDatabase : : printXml ( std : : ostream & out ) const
2014-07-13 17:21:45 +02:00
{
2015-01-09 21:34:57 +01:00
out < < std : : setiosflags ( std : : ios : : boolalpha ) ;
2014-07-13 17:21:45 +02:00
// Scopes..
2014-07-14 15:51:45 +02:00
out < < " <scopes> " < < std : : endl ;
2014-07-13 17:21:45 +02:00
for ( std : : list < Scope > : : const_iterator scope = scopeList . begin ( ) ; scope ! = scopeList . end ( ) ; + + scope ) {
2014-07-14 15:51:45 +02:00
out < < " <scope " ;
out < < " id= \" " < < & * scope < < " \" " ;
out < < " type= \" " < < scope - > type < < " \" " ;
2014-07-13 17:21:45 +02:00
if ( ! scope - > className . empty ( ) )
2015-12-07 18:11:13 +01:00
out < < " className= \" " < < ErrorLogger : : toxml ( scope - > className ) < < " \" " ;
2014-07-16 15:27:13 +02:00
if ( scope - > classStart )
out < < " classStart= \" " < < scope - > classStart < < ' \" ' ;
if ( scope - > classEnd )
out < < " classEnd= \" " < < scope - > classEnd < < ' \" ' ;
2014-07-14 15:51:45 +02:00
if ( scope - > nestedIn )
out < < " nestedIn= \" " < < scope - > nestedIn < < " \" " ;
2014-07-20 11:44:01 +02:00
if ( scope - > function )
out < < " function= \" " < < scope - > function < < " \" " ;
if ( scope - > functionList . empty ( ) & & scope - > varlist . empty ( ) )
2014-07-14 15:51:45 +02:00
out < < " /> " < < std : : endl ;
2014-07-20 11:44:01 +02:00
else {
2014-07-14 15:51:45 +02:00
out < < ' > ' < < std : : endl ;
if ( ! scope - > functionList . empty ( ) ) {
out < < " <functionList> " < < std : : endl ;
for ( std : : list < Function > : : const_iterator function = scope - > functionList . begin ( ) ; function ! = scope - > functionList . end ( ) ; + + function ) {
2015-12-07 18:11:13 +01:00
out < < " <function id= \" " < < & * function < < " \" tokenDef= \" " < < function - > tokenDef < < " \" name= \" " < < ErrorLogger : : toxml ( function - > name ( ) ) < < ' \" ' ;
2014-07-14 15:51:45 +02:00
if ( function - > argCount ( ) = = 0U )
out < < " /> " < < std : : endl ;
else {
out < < " > " < < std : : endl ;
for ( unsigned int argnr = 0 ; argnr < function - > argCount ( ) ; + + argnr ) {
const Variable * arg = function - > getArgumentVar ( argnr ) ;
2014-07-14 18:35:15 +02:00
out < < " <arg nr= \" " < < argnr + 1 < < " \" variable= \" " < < arg < < " \" /> " < < std : : endl ;
2014-07-14 15:51:45 +02:00
}
out < < " </function> " < < std : : endl ;
}
}
out < < " </functionList> " < < std : : endl ;
}
2014-07-14 19:25:30 +02:00
if ( ! scope - > varlist . empty ( ) ) {
out < < " <varlist> " < < std : : endl ;
for ( std : : list < Variable > : : const_iterator var = scope - > varlist . begin ( ) ; var ! = scope - > varlist . end ( ) ; + + var )
out < < " <var id= \" " < < & * var < < " \" /> " < < std : : endl ;
out < < " </varlist> " < < std : : endl ;
}
out < < " </scope> " < < std : : endl ;
2014-07-14 15:51:45 +02:00
}
2014-07-13 17:21:45 +02:00
}
2014-07-14 15:51:45 +02:00
out < < " </scopes> " < < std : : endl ;
2014-07-14 18:35:15 +02:00
// Variables..
out < < " <variables> " < < std : : endl ;
for ( unsigned int i = 1U ; i < _variableList . size ( ) ; i + + ) {
const Variable * var = _variableList [ i ] ;
2014-07-14 19:17:39 +02:00
if ( ! var )
continue ;
2014-07-14 18:35:15 +02:00
out < < " <var id= \" " < < var < < ' \" ' ;
out < < " nameToken= \" " < < var - > nameToken ( ) < < ' \" ' ;
out < < " typeStartToken= \" " < < var - > typeStartToken ( ) < < ' \" ' ;
out < < " typeEndToken= \" " < < var - > typeEndToken ( ) < < ' \" ' ;
2015-01-09 21:34:57 +01:00
out < < " isArgument= \" " < < var - > isArgument ( ) < < ' \" ' ;
out < < " isArray= \" " < < var - > isArray ( ) < < ' \" ' ;
out < < " isClass= \" " < < var - > isClass ( ) < < ' \" ' ;
out < < " isLocal= \" " < < var - > isLocal ( ) < < ' \" ' ;
out < < " isPointer= \" " < < var - > isPointer ( ) < < ' \" ' ;
out < < " isReference= \" " < < var - > isReference ( ) < < ' \" ' ;
out < < " isStatic= \" " < < var - > isStatic ( ) < < ' \" ' ;
2014-07-14 18:35:15 +02:00
out < < " /> " < < std : : endl ;
}
out < < " </variables> " < < std : : endl ;
2015-01-09 21:34:57 +01:00
out < < std : : resetiosflags ( std : : ios : : boolalpha ) ;
2014-07-13 17:21:45 +02:00
}
2011-01-15 07:59:37 +01:00
//---------------------------------------------------------------------------
2015-12-18 08:24:02 +01:00
static const Type * findVariableTypeIncludingUsedNamespaces ( const SymbolDatabase * symbolDatabase , const Scope * scope , const Token * typeTok )
{
const Type * argType = symbolDatabase - > findVariableType ( scope , typeTok ) ;
if ( argType )
return argType ;
// look for variable type in any using namespace in this scope or above
while ( scope ) {
for ( std : : list < Scope : : UsingInfo > : : const_iterator ui = scope - > usingList . begin ( ) ;
2015-12-18 18:13:13 +01:00
ui ! = scope - > usingList . end ( ) ; + + ui ) {
2015-12-18 08:24:02 +01:00
if ( ui - > scope ) {
argType = symbolDatabase - > findVariableType ( ui - > scope , typeTok ) ;
if ( argType )
return argType ;
}
}
scope = scope - > nestedIn ;
}
return nullptr ;
}
//---------------------------------------------------------------------------
2012-04-27 21:51:13 +02:00
void Function : : addArguments ( const SymbolDatabase * symbolDatabase , const Scope * scope )
2011-02-26 21:51:12 +01:00
{
// check for non-empty argument list "( ... )"
2012-10-30 15:48:06 +01:00
const Token * start = arg ? arg : argDef ;
if ( start & & start - > link ( ) ! = start - > next ( ) & & ! Token : : simpleMatch ( start , " ( void ) " ) ) {
2011-02-26 21:51:12 +01:00
unsigned int count = 0 ;
2011-12-17 19:04:03 +01:00
2012-10-30 15:48:06 +01:00
for ( const Token * tok = start - > next ( ) ; tok ; tok = tok - > next ( ) ) {
2014-09-19 09:29:31 +02:00
if ( Token : : Match ( tok , " ,|) " ) )
return ; // Syntax error
2011-12-17 19:04:03 +01:00
const Token * startTok = tok ;
2014-02-16 10:32:10 +01:00
const Token * endTok = nullptr ;
const Token * nameTok = nullptr ;
2011-02-26 21:51:12 +01:00
2012-05-22 10:29:33 +02:00
do {
2011-10-13 20:53:06 +02:00
if ( tok - > varId ( ) ! = 0 ) {
2011-02-26 21:51:12 +01:00
nameTok = tok ;
endTok = tok - > previous ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( tok - > str ( ) = = " [ " ) {
2011-08-28 19:32:42 +02:00
// skip array dimension(s)
tok = tok - > link ( ) ;
while ( tok - > next ( ) - > str ( ) = = " [ " )
tok = tok - > next ( ) - > link ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( tok - > str ( ) = = " < " ) {
2013-04-15 22:48:28 +02:00
tok = tok - > link ( ) ;
if ( ! tok ) // something is wrong so just bail out
2011-12-13 00:24:34 +01:00
return ;
2011-04-29 02:53:31 +02:00
}
2011-03-03 03:08:27 +01:00
2011-02-26 21:51:12 +01:00
tok = tok - > next ( ) ;
2011-04-29 02:53:31 +02:00
if ( ! tok ) // something is wrong so just bail
return ;
2012-05-22 10:29:33 +02:00
} while ( tok - > str ( ) ! = " , " & & tok - > str ( ) ! = " ) " & & tok - > str ( ) ! = " = " ) ;
2011-02-26 21:51:12 +01:00
2012-05-11 17:56:47 +02:00
const Token * typeTok = startTok - > tokAt ( startTok - > str ( ) = = " const " ? 1 : 0 ) ;
2012-04-27 21:51:13 +02:00
if ( typeTok - > str ( ) = = " struct " )
typeTok = typeTok - > next ( ) ;
2014-08-19 11:55:00 +02:00
if ( Token : : Match ( typeTok , " %type% :: " ) )
typeTok = typeTok - > tokAt ( 2 ) ;
2012-04-18 13:00:34 +02:00
2011-03-03 03:08:27 +01:00
// check for argument with no name or missing varid
2011-10-13 20:53:06 +02:00
if ( ! endTok ) {
2014-06-26 09:16:55 +02:00
if ( tok - > previous ( ) - > isName ( ) & & tok - > strAt ( - 1 ) ! = " const " ) {
2012-04-18 13:00:34 +02:00
if ( tok - > previous ( ) ! = typeTok ) {
2011-03-03 03:08:27 +01:00
nameTok = tok - > previous ( ) ;
endTok = nameTok - > previous ( ) ;
2015-01-08 05:45:31 +01:00
if ( hasBody ( ) )
2011-04-23 03:41:55 +02:00
symbolDatabase - > debugMessage ( nameTok , " Function::addArguments found argument \' " + nameTok - > str ( ) + " \' with varid 0. " ) ;
2011-10-13 20:53:06 +02:00
} else
2016-01-23 09:48:21 +01:00
endTok = typeTok ;
2011-10-13 20:53:06 +02:00
} else
2011-03-03 03:08:27 +01:00
endTok = tok - > previous ( ) ;
}
2011-02-26 21:51:12 +01:00
2014-02-16 10:32:10 +01:00
const : : Type * argType = nullptr ;
2012-12-20 06:53:04 +01:00
if ( ! typeTok - > isStandardType ( ) ) {
2015-12-18 08:24:02 +01:00
argType = findVariableTypeIncludingUsedNamespaces ( symbolDatabase , scope , typeTok ) ;
2012-12-20 06:53:04 +01:00
}
2011-02-26 21:51:12 +01:00
2011-03-30 05:16:13 +02:00
// skip default values
2011-10-13 20:53:06 +02:00
if ( tok - > str ( ) = = " = " ) {
2015-08-16 14:22:46 +02:00
do {
2012-09-14 15:32:35 +02:00
if ( tok - > link ( ) & & Token : : Match ( tok , " [{[(<] " ) )
2012-09-06 20:02:53 +02:00
tok = tok - > link ( ) ;
2011-03-30 05:16:13 +02:00
tok = tok - > next ( ) ;
2015-08-16 14:22:46 +02:00
} while ( tok - > str ( ) ! = " , " & & tok - > str ( ) ! = " ) " ) ;
2011-03-30 05:16:13 +02:00
}
2015-01-30 21:56:27 +01:00
argumentList . push_back ( Variable ( nameTok , startTok , endTok , count + + , Argument , argType , functionScope , & symbolDatabase - > _settings - > library ) ) ;
2011-03-31 03:59:43 +02:00
2011-02-26 21:51:12 +01:00
if ( tok - > str ( ) = = " ) " )
break ;
}
2013-02-27 06:59:04 +01:00
2013-06-26 09:47:02 +02:00
// count default arguments
2013-02-27 06:59:04 +01:00
for ( const Token * tok = argDef - > next ( ) ; tok & & tok ! = argDef - > link ( ) ; tok = tok - > next ( ) ) {
if ( tok - > str ( ) = = " = " )
initArgCount + + ;
}
2011-02-26 21:51:12 +01:00
}
}
2012-04-17 19:50:44 +02:00
bool Function : : isImplicitlyVirtual ( bool defaultVal ) const
{
2015-01-08 05:45:31 +01:00
if ( isVirtual ( ) )
2012-04-17 19:50:44 +02:00
return true ;
else if ( access = = Private | | access = = Public | | access = = Protected ) {
bool safe = true ;
2013-03-05 15:28:40 +01:00
bool hasVirt = isImplicitlyVirtual_rec ( nestedIn - > definedType , safe ) ;
2012-04-17 19:50:44 +02:00
if ( hasVirt )
return true ;
else if ( safe )
return false ;
else
return defaultVal ;
} else
return false ;
}
2014-03-29 12:21:35 +01:00
bool Function : : isImplicitlyVirtual_rec ( const : : Type * baseType , bool & safe ) const
2012-04-17 19:50:44 +02:00
{
// check each base class
2014-01-05 19:10:16 +01:00
for ( std : : size_t i = 0 ; i < baseType - > derivedFrom . size ( ) ; + + i ) {
2014-07-23 16:51:23 +02:00
const : : Type * derivedFromType = baseType - > derivedFrom [ i ] . type ;
2012-04-17 19:50:44 +02:00
// check if base class exists in database
2014-07-23 16:51:23 +02:00
if ( derivedFromType & & derivedFromType - > classScope ) {
const Scope * parent = derivedFromType - > classScope ;
2012-04-17 19:50:44 +02:00
// check if function defined in base class
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : const_iterator it = parent - > functionMap . find ( tokenDef - > str ( ) ) ; it ! = parent - > functionMap . end ( ) & & it - > first = = tokenDef - > str ( ) ; + + it ) {
const Function * func = it - > second ;
2015-01-08 05:45:31 +01:00
if ( func - > isVirtual ( ) ) { // Base is virtual and of same name
2012-04-17 19:50:44 +02:00
const Token * temp1 = func - > tokenDef - > previous ( ) ;
const Token * temp2 = tokenDef - > previous ( ) ;
bool returnMatch = true ;
// check for matching return parameters
while ( temp1 - > str ( ) ! = " virtual " ) {
if ( temp1 - > str ( ) ! = temp2 - > str ( ) ) {
returnMatch = false ;
break ;
}
temp1 = temp1 - > previous ( ) ;
temp2 = temp2 - > previous ( ) ;
}
// check for matching function parameters
2013-03-05 16:07:59 +01:00
if ( returnMatch & & argsMatch ( baseType - > classScope , func - > argDef , argDef , " " , 0 ) ) {
2012-04-17 19:50:44 +02:00
return true ;
}
}
}
2014-07-23 16:51:23 +02:00
if ( ! derivedFromType - > derivedFrom . empty ( ) & & ! derivedFromType - > hasCircularDependencies ( ) ) {
2014-03-22 08:49:28 +01:00
// avoid endless recursion, see #5289 Crash: Stack overflow in isImplicitlyVirtual_rec when checking SVN and
// #5590 with a loop within the class hierarchie.
2014-07-23 16:51:23 +02:00
if ( isImplicitlyVirtual_rec ( derivedFromType , safe ) ) {
2014-03-29 12:21:35 +01:00
return true ;
2014-03-22 08:49:28 +01:00
}
2014-01-05 19:10:16 +01:00
}
2012-04-17 19:50:44 +02:00
} else {
// unable to find base class so assume it has no virtual function
safe = false ;
return false ;
}
}
return false ;
}
2013-12-14 08:35:46 +01:00
const Variable * Function : : getArgumentVar ( std : : size_t num ) const
2012-05-03 13:29:41 +02:00
{
for ( std : : list < Variable > : : const_iterator i = argumentList . begin ( ) ; i ! = argumentList . end ( ) ; + + i ) {
if ( i - > index ( ) = = num )
2013-08-07 16:27:37 +02:00
return ( & * i ) ;
2012-05-03 13:29:41 +02:00
else if ( i - > index ( ) > num )
return 0 ;
}
return 0 ;
}
2012-04-17 19:50:44 +02:00
2010-11-13 08:08:45 +01:00
//---------------------------------------------------------------------------
2013-03-05 18:42:42 +01:00
Scope : : Scope ( const SymbolDatabase * check_ , const Token * classDef_ , const Scope * nestedIn_ , ScopeType type_ , const Token * start_ ) :
2011-02-26 21:53:57 +01:00
check ( check_ ) ,
classDef ( classDef_ ) ,
classStart ( start_ ) ,
classEnd ( start_ - > link ( ) ) ,
nestedIn ( nestedIn_ ) ,
numConstructors ( 0 ) ,
2013-04-10 21:57:22 +02:00
numCopyOrMoveConstructors ( 0 ) ,
2012-05-14 20:46:23 +02:00
type ( type_ ) ,
2014-02-16 10:32:10 +01:00
definedType ( nullptr ) ,
functionOf ( nullptr ) ,
function ( nullptr )
2011-02-26 21:53:57 +01:00
{
}
2013-03-05 18:42:42 +01:00
Scope : : Scope ( const SymbolDatabase * check_ , const Token * classDef_ , const Scope * nestedIn_ ) :
2010-11-13 08:08:45 +01:00
check ( check_ ) ,
classDef ( classDef_ ) ,
2014-02-16 10:32:10 +01:00
classStart ( nullptr ) ,
classEnd ( nullptr ) ,
2010-11-13 08:08:45 +01:00
nestedIn ( nestedIn_ ) ,
2010-12-02 07:35:01 +01:00
numConstructors ( 0 ) ,
2013-04-10 21:57:22 +02:00
numCopyOrMoveConstructors ( 0 ) ,
2014-02-16 10:32:10 +01:00
definedType ( nullptr ) ,
functionOf ( nullptr ) ,
function ( nullptr )
2010-11-13 08:08:45 +01:00
{
2013-03-05 13:33:38 +01:00
const Token * nameTok = classDef ;
2011-10-13 20:53:06 +02:00
if ( ! classDef ) {
2011-01-17 18:29:19 +01:00
type = Scope : : eGlobal ;
2011-10-13 20:53:06 +02:00
} else if ( classDef - > str ( ) = = " class " ) {
2011-01-17 18:29:19 +01:00
type = Scope : : eClass ;
2013-03-05 13:33:38 +01:00
nameTok = nameTok - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( classDef - > str ( ) = = " struct " ) {
2011-01-17 18:29:19 +01:00
type = Scope : : eStruct ;
2013-03-05 13:33:38 +01:00
nameTok = nameTok - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( classDef - > str ( ) = = " union " ) {
2011-01-17 18:29:19 +01:00
type = Scope : : eUnion ;
2013-03-05 13:33:38 +01:00
nameTok = nameTok - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else if ( classDef - > str ( ) = = " namespace " ) {
2011-01-17 18:29:19 +01:00
type = Scope : : eNamespace ;
2013-03-05 13:33:38 +01:00
nameTok = nameTok - > next ( ) ;
2011-10-13 20:53:06 +02:00
} else {
2011-01-17 18:29:19 +01:00
type = Scope : : eFunction ;
2010-11-13 08:08:45 +01:00
}
2013-03-05 13:33:38 +01:00
// skip over qualification if present
if ( nameTok & & nameTok - > str ( ) = = " :: " )
nameTok = nameTok - > next ( ) ;
while ( nameTok & & Token : : Match ( nameTok , " %type% :: " ) )
nameTok = nameTok - > tokAt ( 2 ) ;
if ( nameTok & & nameTok - > str ( ) ! = " { " ) // anonymous and unnamed structs/unions don't have a name
className = nameTok - > str ( ) ;
2010-11-13 08:08:45 +01:00
}
2011-12-17 19:04:03 +01:00
bool Scope : : hasDefaultConstructor ( ) const
2010-12-21 08:13:40 +01:00
{
2011-10-13 20:53:06 +02:00
if ( numConstructors ) {
2011-01-17 18:29:19 +01:00
std : : list < Function > : : const_iterator func ;
2010-12-21 08:13:40 +01:00
2011-10-13 20:53:06 +02:00
for ( func = functionList . begin ( ) ; func ! = functionList . end ( ) ; + + func ) {
2011-03-31 03:59:43 +02:00
if ( func - > type = = Function : : eConstructor & & func - > argCount ( ) = = 0 )
2010-12-21 08:13:40 +01:00
return true ;
}
}
return false ;
}
2011-02-26 15:08:59 +01:00
AccessControl Scope : : defaultAccess ( ) const
{
2011-10-13 20:53:06 +02:00
switch ( type ) {
2011-02-26 15:08:59 +01:00
case eGlobal :
return Global ;
case eClass :
return Private ;
case eStruct :
return Public ;
case eUnion :
return Public ;
case eNamespace :
return Namespace ;
2011-02-26 21:53:57 +01:00
default :
2011-02-26 15:08:59 +01:00
return Local ;
}
}
2010-11-20 07:26:50 +01:00
// Get variable list..
2015-01-30 21:56:27 +01:00
void Scope : : getVariableList ( const Library * lib )
2010-11-13 08:08:45 +01:00
{
2010-11-20 07:26:50 +01:00
const Token * start ;
if ( classStart )
start = classStart - > next ( ) ;
2012-12-16 07:18:03 +01:00
// global scope
else if ( className . empty ( ) )
2010-11-20 07:26:50 +01:00
start = check - > _tokenizer - > tokens ( ) ;
2012-12-16 07:18:03 +01:00
// forward declaration
else
return ;
2014-07-10 09:23:14 +02:00
AccessControl varaccess = defaultAccess ( ) ;
2015-03-15 12:37:50 +01:00
for ( const Token * tok = start ; tok & & tok ! = classEnd ; tok = tok - > next ( ) ) {
2010-12-16 19:04:47 +01:00
// syntax error?
2015-03-15 12:37:50 +01:00
if ( tok - > next ( ) = = nullptr )
2010-12-16 19:04:47 +01:00
break ;
2010-11-20 07:26:50 +01:00
// Is it a function?
2011-10-13 20:53:06 +02:00
else if ( tok - > str ( ) = = " { " ) {
2010-11-20 07:26:50 +01:00
tok = tok - > link ( ) ;
continue ;
2010-11-13 08:08:45 +01:00
}
2010-11-20 07:26:50 +01:00
// Is it a nested class or structure?
2011-10-13 20:53:06 +02:00
else if ( Token : : Match ( tok , " class|struct|union|namespace %type% :|{ " ) ) {
2010-11-20 07:26:50 +01:00
tok = tok - > tokAt ( 2 ) ;
while ( tok & & tok - > str ( ) ! = " { " )
tok = tok - > next ( ) ;
2011-10-13 20:53:06 +02:00
if ( tok ) {
2010-11-20 07:26:50 +01:00
// skip implementation
tok = tok - > link ( ) ;
continue ;
2011-10-13 20:53:06 +02:00
} else
2010-11-20 07:26:50 +01:00
break ;
2014-06-25 20:41:22 +02:00
} else if ( Token : : Match ( tok , " struct|union { " ) ) {
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( tok - > next ( ) - > link ( ) , " } %name% ;|[ " ) ) {
2014-06-25 20:41:22 +02:00
tok = tok - > next ( ) - > link ( ) - > tokAt ( 2 ) ;
continue ;
} else if ( Token : : simpleMatch ( tok - > next ( ) - > link ( ) , " } ; " )) {
tok = tok - > next ( ) ;
continue ;
}
2011-04-01 02:54:23 +02:00
}
2010-11-13 08:08:45 +01:00
// Borland C++: Skip all variables in the __published section.
2010-12-15 18:45:53 +01:00
// These are automatically initialized.
2011-10-13 20:53:06 +02:00
else if ( tok - > str ( ) = = " __published: " ) {
for ( ; tok ; tok = tok - > next ( ) ) {
2010-11-13 08:08:45 +01:00
if ( tok - > str ( ) = = " { " )
tok = tok - > link ( ) ;
if ( Token : : Match ( tok - > next ( ) , " private:|protected:|public: " ) )
break ;
}
if ( tok )
continue ;
else
break ;
}
// "private:" "public:" "protected:" etc
2011-10-13 20:53:06 +02:00
else if ( tok - > str ( ) = = " public: " ) {
2010-11-13 08:08:45 +01:00
varaccess = Public ;
2010-11-20 07:26:50 +01:00
continue ;
2011-10-13 20:53:06 +02:00
} else if ( tok - > str ( ) = = " protected: " ) {
2010-11-13 08:08:45 +01:00
varaccess = Protected ;
2010-11-20 07:26:50 +01:00
continue ;
2011-10-13 20:53:06 +02:00
} else if ( tok - > str ( ) = = " private: " ) {
2010-11-13 08:08:45 +01:00
varaccess = Private ;
continue ;
2010-11-20 07:26:50 +01:00
}
2010-11-13 08:08:45 +01:00
// Is it a forward declaration?
2015-01-31 10:50:39 +01:00
else if ( Token : : Match ( tok , " class|struct|union %name% ; " ) ) {
2010-11-13 08:08:45 +01:00
tok = tok - > tokAt ( 2 ) ;
continue ;
}
2010-11-20 07:26:50 +01:00
// Borland C++: Ignore properties..
else if ( tok - > str ( ) = = " __property " )
2010-11-13 08:08:45 +01:00
continue ;
2015-03-15 12:37:50 +01:00
// skip return, goto and delete
else if ( Token : : Match ( tok , " return|delete|goto " ) ) {
2013-08-31 14:22:06 +02:00
while ( tok - > next ( ) & &
tok - > next ( ) - > str ( ) ! = " ; " & &
tok - > next ( ) - > str ( ) ! = " } " /* ticket #4994 */ ) {
2011-02-27 16:13:40 +01:00
tok = tok - > next ( ) ;
2013-08-31 14:22:06 +02:00
}
2011-02-27 16:13:40 +01:00
continue ;
}
2011-12-30 20:01:33 +01:00
2010-11-20 07:26:50 +01:00
// Search for start of statement..
2011-02-25 13:17:55 +01:00
else if ( tok - > previous ( ) & & ! Token : : Match ( tok - > previous ( ) , " ; | { | } | public : | protected : | private : " ))
2010-11-13 08:08:45 +01:00
continue ;
2015-03-15 12:37:50 +01:00
else if ( tok - > str ( ) = = " ; " )
2010-12-02 07:35:01 +01:00
continue ;
2010-11-13 08:08:45 +01:00
2015-01-30 21:56:27 +01:00
tok = checkVariable ( tok , varaccess , lib ) ;
2011-03-16 03:33:14 +01:00
if ( ! tok )
break ;
2011-03-06 21:21:42 +01:00
}
}
2010-11-20 07:26:50 +01:00
2015-01-30 21:56:27 +01:00
const Token * Scope : : checkVariable ( const Token * tok , AccessControl varaccess , const Library * lib )
2011-03-06 21:21:42 +01:00
{
2012-01-26 04:05:29 +01:00
// Is it a throw..?
if ( Token : : Match ( tok , " throw %any% ( " ) & &
Token : : simpleMatch ( tok - > linkAt ( 2 ) , " ) ; " ) ) {
return tok - > linkAt ( 2 ) ;
} else if ( ( Token : : Match ( tok , " throw %any% :: %any% ( " ) & &
Token : : simpleMatch ( tok - > linkAt ( 4 ) , " ) ; " ) ) ) {
return tok - > linkAt ( 4 ) ;
}
2012-08-12 12:01:24 +02:00
// friend?
2013-03-14 06:34:12 +01:00
if ( Token : : Match ( tok , " friend %type% " ) & & tok - > next ( ) - > varId ( ) = = 0 ) {
const Token * next = Token : : findmatch ( tok - > tokAt ( 2 ) , " ;|{ " ) ;
if ( next & & next - > str ( ) = = " { " )
next = next - > link ( ) ;
return next ;
}
2012-08-12 12:01:24 +02:00
2012-06-22 11:23:50 +02:00
// skip const|static|mutable|extern
while ( Token : : Match ( tok , " const|static|mutable|extern " ) ) {
2011-03-06 21:21:42 +01:00
tok = tok - > next ( ) ;
}
2010-11-13 08:08:45 +01:00
2011-04-22 15:12:08 +02:00
// the start of the type tokens does not include the above modifiers
const Token * typestart = tok ;
2015-01-19 06:38:54 +01:00
if ( Token : : Match ( tok , " class|struct|union " ) ) {
2011-03-06 21:21:42 +01:00
tok = tok - > next ( ) ;
}
2010-11-13 08:08:45 +01:00
2014-09-19 09:29:31 +02:00
// This is the start of a statement
const Token * vartok = nullptr ;
const Token * typetok = nullptr ;
2012-05-11 17:56:47 +02:00
if ( tok & & isVariableDeclaration ( tok , vartok , typetok ) ) {
// If the vartok was set in the if-blocks above, create a entry for this variable..
tok = vartok - > next ( ) ;
2014-09-14 11:35:04 +02:00
while ( Token : : Match ( tok , " [|{ " ) )
2012-05-11 17:56:47 +02:00
tok = tok - > link ( ) - > next ( ) ;
2011-01-01 01:19:32 +01:00
2013-12-31 17:51:56 +01:00
if ( vartok - > varId ( ) = = 0 ) {
if ( ! vartok - > isBoolean ( ) )
check - > debugMessage ( vartok , " Scope::checkVariable found variable \' " + vartok - > str ( ) + " \' with varid 0. " ) ;
return tok ;
}
2010-11-13 08:08:45 +01:00
2014-02-16 10:32:10 +01:00
const Type * vType = nullptr ;
2010-12-02 07:35:01 +01:00
2012-12-20 06:53:04 +01:00
if ( typetok ) {
2015-12-18 08:24:02 +01:00
vType = findVariableTypeIncludingUsedNamespaces ( check , this , typetok ) ;
2012-12-20 06:53:04 +01:00
}
2010-12-02 07:35:01 +01:00
2015-01-30 21:56:27 +01:00
addVariable ( vartok , typestart , vartok - > previous ( ) , varaccess , vType , this , lib ) ;
2010-12-02 07:35:01 +01:00
}
2011-03-06 21:21:42 +01:00
return tok ;
2010-12-02 07:35:01 +01:00
}
2011-09-03 03:07:29 +02:00
const Variable * Scope : : getVariable ( const std : : string & varname ) const
{
std : : list < Variable > : : const_iterator iter ;
2011-10-13 20:53:06 +02:00
for ( iter = varlist . begin ( ) ; iter ! = varlist . end ( ) ; + + iter ) {
2011-09-03 03:07:29 +02:00
if ( iter - > name ( ) = = varname )
return & * iter ;
}
2014-02-16 10:32:10 +01:00
return nullptr ;
2011-09-03 03:07:29 +02:00
}
2012-02-11 12:26:48 +01:00
static const Token * skipScopeIdentifiers ( const Token * tok )
2010-12-30 09:46:44 +01:00
{
2013-03-01 12:42:04 +01:00
if ( tok & & tok - > str ( ) = = " :: " ) {
2012-02-11 12:26:48 +01:00
tok = tok - > next ( ) ;
2010-12-30 09:46:44 +01:00
}
2012-02-11 12:26:48 +01:00
while ( Token : : Match ( tok , " %type% :: " ) ) {
tok = tok - > tokAt ( 2 ) ;
2010-12-30 09:46:44 +01:00
}
2010-12-30 22:57:43 +01:00
2012-02-11 12:26:48 +01:00
return tok ;
2010-12-30 22:57:43 +01:00
}
2012-02-11 12:26:48 +01:00
static const Token * skipPointers ( const Token * tok )
2010-12-30 22:57:43 +01:00
{
2014-03-28 17:43:40 +01:00
while ( Token : : Match ( tok , " *|&|&& " ) | | ( tok & & tok - > str ( ) = = " ( " & & Token : : Match ( tok - > link ( ) - > next ( ) , " (|[ " ) ) ) {
2012-02-11 12:26:48 +01:00
tok = tok - > next ( ) ;
2014-04-12 11:43:10 +02:00
if ( tok - > strAt ( - 1 ) = = " ( " & & Token : : Match ( tok , " %type% :: " ) )
tok = tok - > tokAt ( 2 ) ;
2010-12-30 09:46:44 +01:00
}
2010-12-30 22:57:43 +01:00
2012-02-11 12:26:48 +01:00
return tok ;
2010-12-30 22:57:43 +01:00
}
2016-01-30 14:04:48 +01:00
bool Scope : : isVariableDeclaration ( const Token * const tok , const Token * & vartok , const Token * & typetok ) const
2010-12-30 22:57:43 +01:00
{
2014-09-14 11:35:04 +02:00
if ( check & & check - > _tokenizer - > isCPP ( ) & & Token : : Match ( tok , " throw|new " ) )
2012-01-28 01:24:01 +01:00
return false ;
2011-01-02 00:36:22 +01:00
const Token * localTypeTok = skipScopeIdentifiers ( tok ) ;
2014-02-16 10:32:10 +01:00
const Token * localVarTok = nullptr ;
2010-12-30 22:57:43 +01:00
2011-10-13 20:53:06 +02:00
if ( Token : : Match ( localTypeTok , " %type% < " ) ) {
2014-10-12 16:06:50 +02:00
if ( Token : : Match ( tok , " const_cast|dynamic_cast|reinterpret_cast|static_cast < " ) )
return false ;
2013-04-15 22:48:28 +02:00
const Token * closeTok = localTypeTok - > next ( ) - > link ( ) ;
if ( closeTok ) {
2011-01-18 10:51:57 +01:00
localVarTok = skipPointers ( closeTok - > next ( ) ) ;
2011-01-18 10:14:12 +01:00
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( localVarTok , " :: %type% %name% [;=({] " ) ) {
2013-03-30 13:49:56 +01:00
if ( localVarTok - > strAt ( 3 ) ! = " ( " | |
2014-08-05 15:33:57 +02:00
Token : : Match ( localVarTok - > linkAt ( 3 ) , " [)}] ; " ) ) {
2013-03-30 13:49:56 +01:00
localTypeTok = localVarTok - > next ( ) ;
localVarTok = localVarTok - > tokAt ( 2 ) ;
}
2011-01-02 00:36:22 +01:00
}
}
2011-10-13 20:53:06 +02:00
} else if ( Token : : Match ( localTypeTok , " %type% " ) ) {
2012-03-24 15:10:06 +01:00
localVarTok = skipPointers ( localTypeTok - > strAt ( 1 ) = = " const " ? localTypeTok - > tokAt ( 2 ) : localTypeTok - > next ( ) ) ;
2011-01-18 10:51:57 +01:00
}
2010-12-30 22:57:43 +01:00
2015-10-26 19:20:42 +01:00
if ( ! localVarTok )
return false ;
if ( localVarTok - > str ( ) = = " const " )
2012-05-14 21:47:02 +02:00
localVarTok = localVarTok - > next ( ) ;
2016-02-03 22:49:57 +01:00
if ( Token : : Match ( localVarTok , " %name% ;|= " ) | | ( localVarTok & & localVarTok - > varId ( ) & & localVarTok - > strAt ( 1 ) = = " : " ) ) {
2011-02-27 16:21:14 +01:00
vartok = localVarTok ;
typetok = localTypeTok ;
2015-01-31 10:50:39 +01:00
} else if ( Token : : Match ( localVarTok , " %name% ) | [ " ) && localVarTok->str() != " operator " ) {
2011-01-18 10:51:57 +01:00
vartok = localVarTok ;
typetok = localTypeTok ;
2016-02-03 22:49:57 +01:00
} else if ( localVarTok & & localVarTok - > varId ( ) & & Token : : Match ( localVarTok , " %name% (| { " ) &&
2014-08-05 15:33:57 +02:00
Token : : Match ( localVarTok - > next ( ) - > link ( ) , " )|} ; " ) ) {
2011-07-23 21:12:30 +02:00
vartok = localVarTok ;
typetok = localTypeTok ;
2012-01-26 04:05:29 +01:00
} else if ( type = = eCatch & &
2015-01-31 10:50:39 +01:00
Token : : Match ( localVarTok , " %name% ) " ) ) {
2012-01-26 04:05:29 +01:00
vartok = localVarTok ;
typetok = localTypeTok ;
2011-07-23 21:12:30 +02:00
}
2012-01-26 04:05:29 +01:00
2014-02-16 10:32:10 +01:00
return nullptr ! = vartok ;
2010-12-30 09:46:44 +01:00
}
2011-01-02 00:36:22 +01:00
2015-01-02 21:38:19 +01:00
2010-12-02 07:35:01 +01:00
//---------------------------------------------------------------------------
2013-03-05 13:33:38 +01:00
const Type * SymbolDatabase : : findVariableType ( const Scope * start , const Token * typeTok ) const
2010-12-02 07:35:01 +01:00
{
2013-03-05 13:33:38 +01:00
std : : list < Type > : : const_iterator type ;
2010-12-02 07:35:01 +01:00
2013-03-05 13:33:38 +01:00
for ( type = typeList . begin ( ) ; type ! = typeList . end ( ) ; + + type ) {
2010-12-02 07:35:01 +01:00
// do the names match?
2016-01-30 14:04:48 +01:00
if ( type - > name ( ) ! = typeTok - > str ( ) )
continue ;
2010-12-02 07:35:01 +01:00
2016-01-30 14:04:48 +01:00
// check if type does not have a namespace
if ( typeTok - > strAt ( - 1 ) ! = " :: " ) {
const Scope * parent = start ;
// check if in same namespace
while ( parent ) {
// out of line class function belongs to class
if ( parent - > type = = Scope : : eFunction & & parent - > functionOf )
parent = parent - > functionOf ;
else if ( parent ! = type - > enclosingScope )
parent = parent - > nestedIn ;
else
break ;
2010-12-02 07:35:01 +01:00
}
2016-01-30 14:04:48 +01:00
if ( type - > enclosingScope = = parent )
2013-03-05 13:33:38 +01:00
return & ( * type ) ;
2016-01-30 14:04:48 +01:00
}
// type has a namespace
else {
bool match = true ;
const Scope * scope = type - > enclosingScope ;
const Token * typeTok2 = typeTok - > tokAt ( - 2 ) ;
while ( match & & scope & & Token : : Match ( typeTok2 , " %any% :: " ) ) {
// A::B..
if ( typeTok2 - > isName ( ) & & typeTok2 - > str ( ) . find ( " : " ) = = std : : string : : npos ) {
match & = bool ( scope - > className = = typeTok2 - > str ( ) ) ;
typeTok2 = typeTok2 - > tokAt ( - 2 ) ;
scope = scope - > nestedIn ;
} else {
// ::A..
match & = bool ( scope - > type = = Scope : : eGlobal ) ;
break ;
}
2010-12-02 07:35:01 +01:00
}
2016-01-30 14:04:48 +01:00
if ( match )
return & ( * type ) ;
2010-11-13 08:08:45 +01:00
}
}
2010-12-02 07:35:01 +01:00
2014-02-16 10:32:10 +01:00
return nullptr ;
2010-11-13 08:08:45 +01:00
}
//---------------------------------------------------------------------------
2011-01-28 08:33:02 +01:00
2015-01-02 21:38:19 +01:00
void Scope : : findFunctionInBase ( const std : : string & name , size_t args , std : : vector < const Function * > & matches ) const
2014-11-17 16:04:44 +01:00
{
if ( isClassOrStruct ( ) & & definedType & & ! definedType - > derivedFrom . empty ( ) ) {
for ( std : : size_t i = 0 ; i < definedType - > derivedFrom . size ( ) ; + + i ) {
const Type * base = definedType - > derivedFrom [ i ] . type ;
if ( base & & base - > classScope ) {
if ( base - > classScope = = this ) // Ticket #5120, #5125: Recursive class; tok should have been found already
continue ;
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : const_iterator it = base - > classScope - > functionMap . find ( name ) ; it ! = base - > classScope - > functionMap . end ( ) & & it - > first = = name ; + + it ) {
const Function * func = it - > second ;
if ( args = = func - > argCount ( ) | | ( args < func - > argCount ( ) & & args > = func - > minArgCount ( ) ) ) {
matches . push_back ( func ) ;
2014-11-17 16:04:44 +01:00
}
}
2015-01-02 21:38:19 +01:00
base - > classScope - > findFunctionInBase ( name , args , matches ) ;
2014-11-17 16:04:44 +01:00
}
}
}
}
//---------------------------------------------------------------------------
2014-11-20 06:18:29 +01:00
/** @todo This function does not take into account argument types when they don't match.
This can be difficult because of promotion and conversion operators and casts
and because the argument can also be a function call .
2012-10-14 17:30:37 +02:00
*/
2015-05-23 11:56:11 +02:00
const Function * Scope : : findFunction ( const Token * tok , bool requireConst ) const
2013-01-01 09:53:40 +01:00
{
2014-11-17 16:04:44 +01:00
// make sure this is a function call
const Token * end = tok - > linkAt ( 1 ) ;
if ( ! end )
return nullptr ;
std : : vector < const Token * > arguments ;
2014-10-30 13:47:20 +01:00
2014-11-17 16:04:44 +01:00
// find all the arguments for this function call
const Token * arg = tok - > tokAt ( 2 ) ;
while ( arg & & arg ! = end ) {
arguments . push_back ( arg ) ;
arg = arg - > nextArgument ( ) ;
}
std : : vector < const Function * > matches ;
// find all the possible functions that could match
const std : : size_t args = arguments . size ( ) ;
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : const_iterator it = functionMap . find ( tok - > str ( ) ) ; it ! = functionMap . end ( ) & & it - > first = = tok - > str ( ) ; + + it ) {
const Function * func = it - > second ;
if ( args = = func - > argCount ( ) | | ( args < func - > argCount ( ) & & args > = func - > minArgCount ( ) ) ) {
matches . push_back ( func ) ;
2014-11-17 16:04:44 +01:00
}
}
// check in base classes
2015-01-02 21:38:19 +01:00
findFunctionInBase ( tok - > str ( ) , args , matches ) ;
2014-11-17 16:04:44 +01:00
// check each function against the arguments in the function call for a match
2014-11-20 06:18:29 +01:00
for ( std : : size_t i = 0 ; i < matches . size ( ) ; ) {
bool erased = false ;
2014-11-17 16:04:44 +01:00
const Function * func = matches [ i ] ;
size_t same = 0 ;
for ( std : : size_t j = 0 ; j < args ; + + j ) {
const Variable * funcarg = func - > getArgumentVar ( j ) ;
// check for a match with a variable
2015-01-31 10:50:39 +01:00
if ( Token : : Match ( arguments [ j ] , " %var% ,|) " ) ) {
2014-11-17 16:04:44 +01:00
const Variable * callarg = check - > getVariableFromVarId ( arguments [ j ] - > varId ( ) ) ;
2014-11-17 23:10:00 +01:00
if ( callarg & &
callarg - > typeStartToken ( ) - > str ( ) = = funcarg - > typeStartToken ( ) - > str ( ) & &
2014-11-17 16:04:44 +01:00
callarg - > typeStartToken ( ) - > isUnsigned ( ) = = funcarg - > typeStartToken ( ) - > isUnsigned ( ) & &
callarg - > typeStartToken ( ) - > isLong ( ) = = funcarg - > typeStartToken ( ) - > isLong ( ) ) {
same + + ;
}
}
// check for a match with a numeric literal
else if ( Token : : Match ( arguments [ j ] , " %num% ,|) " )) {
if ( MathLib : : isInt ( arguments [ j ] - > str ( ) ) ) {
if ( arguments [ j ] - > str ( ) . find ( " ll " ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( " LL " ) ! = std : : string : : npos ) {
2015-08-15 20:24:26 +02:00
if ( arguments [ j ] - > str ( ) . find ( ' u ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' U ' ) ! = std : : string : : npos ) {
2014-11-17 16:04:44 +01:00
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " long " & &
funcarg - > typeStartToken ( ) - > isLong ( ) & &
funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
}
} else {
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " long " & &
funcarg - > typeStartToken ( ) - > isLong ( ) & &
! funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
}
}
2015-08-15 20:24:26 +02:00
} else if ( arguments [ j ] - > str ( ) . find ( ' l ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' L ' ) ! = std : : string : : npos ) {
if ( arguments [ j ] - > str ( ) . find ( ' u ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' U ' ) ! = std : : string : : npos ) {
2014-11-17 16:04:44 +01:00
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " long " & &
! funcarg - > typeStartToken ( ) - > isLong ( ) & &
funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
}
} else {
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " long " & &
! funcarg - > typeStartToken ( ) - > isLong ( ) & &
! funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
}
}
2015-08-15 20:24:26 +02:00
} else if ( arguments [ j ] - > str ( ) . find ( ' u ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' U ' ) ! = std : : string : : npos ) {
2014-11-17 16:04:44 +01:00
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " int " & &
funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
} else if ( Token : : Match ( funcarg - > typeStartToken ( ) , " char|short " ) ) {
same + + ;
}
} else {
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " int " & &
! funcarg - > typeStartToken ( ) - > isUnsigned ( ) ) {
same + + ;
} else if ( Token : : Match ( funcarg - > typeStartToken ( ) , " char|short|int " ) ) {
same + + ;
}
}
} else {
2015-08-15 20:24:26 +02:00
if ( arguments [ j ] - > str ( ) . find ( ' f ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' F ' ) ! = std : : string : : npos ) {
2014-11-17 16:04:44 +01:00
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " float " ) {
same + + ;
}
2015-08-15 20:24:26 +02:00
} else if ( arguments [ j ] - > str ( ) . find ( ' l ' ) ! = std : : string : : npos | |
arguments [ j ] - > str ( ) . find ( ' L ' ) ! = std : : string : : npos ) {
2014-11-17 16:04:44 +01:00
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " double " & &
funcarg - > typeStartToken ( ) - > isLong ( ) ) {
same + + ;
}
} else {
if ( funcarg - > typeStartToken ( ) - > str ( ) = = " double " & &
! funcarg - > typeStartToken ( ) - > isLong ( ) ) {
same + + ;
2014-02-19 06:35:51 +01:00
}
}
2013-01-01 09:53:40 +01:00
}
2014-11-17 16:04:44 +01:00
}
2013-01-01 09:53:40 +01:00
2014-11-17 16:04:44 +01:00
// check that function argument type is not mismatching
else if ( arguments [ j ] - > str ( ) = = " & " & & funcarg & & funcarg - > isReference ( ) ) {
// can't match so remove this function from possible matches
2014-11-20 06:18:29 +01:00
matches . erase ( matches . begin ( ) + i ) ;
erased = true ;
2014-11-17 16:04:44 +01:00
break ;
2013-01-01 09:53:40 +01:00
}
}
2014-11-17 16:04:44 +01:00
// check if all arguments matched
if ( same = = args ) {
2015-05-23 11:56:11 +02:00
if ( requireConst & & func - > isConst ( ) )
return func ;
2014-11-20 06:18:29 +01:00
// get the function this call is in
const Scope * scope = tok - > scope ( ) ;
// check if this function is a member function
if ( scope & & scope - > functionOf & & scope - > functionOf - > isClassOrStruct ( ) ) {
// check if isConst match
2015-01-08 05:45:31 +01:00
if ( scope - > function & & scope - > function - > isConst ( ) = = func - > isConst ( ) )
2014-11-20 06:18:29 +01:00
return func ;
} else
return func ;
2014-11-01 14:36:17 +01:00
}
2014-11-20 06:18:29 +01:00
if ( ! erased )
+ + i ;
2014-11-01 14:36:17 +01:00
}
2014-11-17 16:04:44 +01:00
// no exact match so just return first function found
if ( ! matches . empty ( ) ) {
return matches [ 0 ] ;
}
2014-10-30 13:47:20 +01:00
return nullptr ;
2013-01-01 09:53:40 +01:00
}
//---------------------------------------------------------------------------
2013-01-28 06:47:48 +01:00
const Function * SymbolDatabase : : findFunction ( const Token * tok ) const
2012-10-14 17:30:37 +02:00
{
2013-01-22 06:55:25 +01:00
// find the scope this function is in
2013-01-28 06:47:48 +01:00
const Scope * currScope = tok - > scope ( ) ;
2012-10-14 17:30:37 +02:00
while ( currScope & & currScope - > isExecutable ( ) ) {
if ( currScope - > functionOf )
currScope = currScope - > functionOf ;
else
currScope = currScope - > nestedIn ;
}
2013-01-22 06:55:25 +01:00
// check for a qualified name and use it when given
if ( tok - > strAt ( - 1 ) = = " :: " ) {
// find start of qualified function name
const Token * tok1 = tok ;
while ( Token : : Match ( tok1 - > tokAt ( - 2 ) , " %type% :: " ) )
tok1 = tok1 - > tokAt ( - 2 ) ;
// check for global scope
if ( tok1 - > strAt ( - 1 ) = = " :: " ) {
currScope = & scopeList . front ( ) ;
currScope = currScope - > findRecordInNestedList ( tok1 - > str ( ) ) ;
}
// find start of qualification
else {
while ( currScope ) {
if ( currScope - > className = = tok1 - > str ( ) )
break ;
else {
const Scope * scope = currScope - > findRecordInNestedList ( tok1 - > str ( ) ) ;
if ( scope ) {
currScope = scope ;
break ;
} else
currScope = currScope - > nestedIn ;
}
}
}
if ( currScope ) {
while ( currScope & & ! Token : : Match ( tok1 , " %type% :: %any% ( " ) ) {
tok1 = tok1 - > tokAt ( 2 ) ;
2015-08-15 20:24:26 +02:00
currScope = currScope - > findRecordInNestedList ( tok1 - > str ( ) ) ;
2013-01-22 06:55:25 +01:00
}
tok1 = tok1 - > tokAt ( 2 ) ;
if ( currScope & & tok1 )
2013-01-28 06:47:48 +01:00
return currScope - > findFunction ( tok1 ) ;
2013-01-22 06:55:25 +01:00
}
}
2013-01-23 16:53:55 +01:00
// check for member function
2013-07-08 06:38:33 +02:00
else if ( Token : : Match ( tok - > tokAt ( - 2 ) , " !!this . " ) ) {
2014-12-02 14:19:52 +01:00
const Token * tok1 = tok - > tokAt ( - 2 ) ;
if ( Token : : Match ( tok1 , " %var% . " ) ) {
2015-01-31 10:50:39 +01:00
const Variable * var = getVariableFromVarId ( tok1 - > varId ( ) ) ;
if ( var & & var - > typeScope ( ) )
2015-05-23 11:56:11 +02:00
return var - > typeScope ( ) - > findFunction ( tok , var - > isConst ( ) ) ;
2013-01-23 16:53:55 +01:00
}
}
2013-01-22 06:55:25 +01:00
// check in enclosing scopes
else {
while ( currScope ) {
2013-01-28 06:47:48 +01:00
const Function * func = currScope - > findFunction ( tok ) ;
2013-01-22 06:55:25 +01:00
if ( func )
return func ;
currScope = currScope - > nestedIn ;
}
2012-10-14 17:30:37 +02:00
}
return 0 ;
}
2011-01-28 08:33:02 +01:00
//---------------------------------------------------------------------------
2010-11-13 08:08:45 +01:00
2013-01-03 22:37:19 +01:00
const Scope * SymbolDatabase : : findScopeByName ( const std : : string & name ) const
2012-02-24 20:45:56 +01:00
{
for ( std : : list < Scope > : : const_iterator it = scopeList . begin ( ) ; it ! = scopeList . end ( ) ; + + it ) {
if ( it - > className = = name )
return & * it ;
}
return 0 ;
}
//---------------------------------------------------------------------------
2013-01-03 22:37:19 +01:00
Scope * Scope : : findInNestedList ( const std : : string & name )
2010-11-13 08:08:45 +01:00
{
2011-01-17 18:29:19 +01:00
std : : list < Scope * > : : iterator it ;
2010-11-13 08:08:45 +01:00
2011-10-13 20:53:06 +02:00
for ( it = nestedList . begin ( ) ; it ! = nestedList . end ( ) ; + + it ) {
2010-11-13 08:08:45 +01:00
if ( ( * it ) - > className = = name )
return ( * it ) ;
}
return 0 ;
}
//---------------------------------------------------------------------------
2013-01-03 22:37:19 +01:00
const Scope * Scope : : findRecordInNestedList ( const std : : string & name ) const
2012-11-30 06:03:58 +01:00
{
std : : list < Scope * > : : const_iterator it ;
for ( it = nestedList . begin ( ) ; it ! = nestedList . end ( ) ; + + it ) {
if ( ( * it ) - > className = = name & & ( * it ) - > type ! = eFunction )
return ( * it ) ;
}
return 0 ;
}
//---------------------------------------------------------------------------
2013-03-05 13:33:38 +01:00
const Type * Scope : : findType ( const std : : string & name ) const
{
std : : list < Type * > : : const_iterator it ;
for ( it = definedTypes . begin ( ) ; it ! = definedTypes . end ( ) ; + + it ) {
if ( ( * it ) - > name ( ) = = name )
return ( * it ) ;
}
return 0 ;
}
//---------------------------------------------------------------------------
2013-01-03 22:37:19 +01:00
Scope * Scope : : findInNestedListRecursive ( const std : : string & name )
2011-02-03 07:57:10 +01:00
{
std : : list < Scope * > : : iterator it ;
2011-10-13 20:53:06 +02:00
for ( it = nestedList . begin ( ) ; it ! = nestedList . end ( ) ; + + it ) {
2011-02-03 07:57:10 +01:00
if ( ( * it ) - > className = = name )
return ( * it ) ;
}
2011-10-13 20:53:06 +02:00
for ( it = nestedList . begin ( ) ; it ! = nestedList . end ( ) ; + + it ) {
2011-02-03 07:57:10 +01:00
Scope * child = ( * it ) - > findInNestedListRecursive ( name ) ;
2011-02-19 19:40:02 +01:00
if ( child )
return child ;
2011-02-03 07:57:10 +01:00
}
return 0 ;
}
//---------------------------------------------------------------------------
2011-01-17 18:29:19 +01:00
const Function * Scope : : getDestructor ( ) const
2010-11-13 08:08:45 +01:00
{
2011-01-17 18:29:19 +01:00
std : : list < Function > : : const_iterator it ;
2011-10-13 20:53:06 +02:00
for ( it = functionList . begin ( ) ; it ! = functionList . end ( ) ; + + it ) {
2011-01-17 18:29:19 +01:00
if ( it - > type = = Function : : eDestructor )
2011-03-11 01:43:29 +01:00
return & ( * it ) ;
2010-11-13 08:08:45 +01:00
}
return 0 ;
}
//---------------------------------------------------------------------------
2012-11-26 16:34:44 +01:00
bool SymbolDatabase : : isCPP ( ) const
{
return _tokenizer - > isCPP ( ) ;
}
2012-11-30 06:03:58 +01:00
//---------------------------------------------------------------------------
2013-01-03 22:37:19 +01:00
const Scope * SymbolDatabase : : findScope ( const Token * tok , const Scope * startScope ) const
2012-11-30 06:03:58 +01:00
{
2014-02-16 11:47:52 +01:00
const Scope * scope = nullptr ;
2012-11-30 06:03:58 +01:00
// absolute path
if ( tok - > str ( ) = = " :: " ) {
tok = tok - > next ( ) ;
2013-03-05 13:33:38 +01:00
scope = & scopeList . front ( ) ;
}
// relative path
else if ( tok - > isName ( ) ) {
scope = startScope ;
}
2012-11-30 06:03:58 +01:00
2013-03-05 13:33:38 +01:00
while ( scope & & tok & & tok - > isName ( ) ) {
if ( tok - > strAt ( 1 ) = = " :: " ) {
2012-11-30 06:03:58 +01:00
scope = scope - > findRecordInNestedList ( tok - > str ( ) ) ;
2013-03-05 13:33:38 +01:00
tok = tok - > tokAt ( 2 ) ;
} else
return scope - > findRecordInNestedList ( tok - > str ( ) ) ;
}
2012-11-30 06:03:58 +01:00
2013-03-05 13:33:38 +01:00
// not a valid path
return 0 ;
}
2012-11-30 06:03:58 +01:00
2013-08-12 06:21:03 +02:00
//---------------------------------------------------------------------------
2013-03-05 15:28:40 +01:00
const Type * SymbolDatabase : : findType ( const Token * startTok , const Scope * startScope ) const
2013-03-05 13:33:38 +01:00
{
2013-08-12 06:21:03 +02:00
// skip over struct or union
if ( Token : : Match ( startTok , " struct|union " ) )
startTok = startTok - > next ( ) ;
// type same as scope
if ( startTok - > str ( ) = = startScope - > className & & startScope - > isClassOrStruct ( ) )
return startScope - > definedType ;
2013-03-05 15:28:40 +01:00
// absolute path - directly start in global scope
if ( startTok - > str ( ) = = " :: " ) {
startTok = startTok - > next ( ) ;
startScope = & scopeList . front ( ) ;
2013-03-05 13:33:38 +01:00
}
2012-11-30 06:03:58 +01:00
2013-03-05 15:28:40 +01:00
const Token * tok = startTok ;
const Scope * scope = startScope ;
2013-03-05 13:33:38 +01:00
while ( scope & & tok & & tok - > isName ( ) ) {
if ( tok - > strAt ( 1 ) = = " :: " ) {
2012-11-30 06:03:58 +01:00
scope = scope - > findRecordInNestedList ( tok - > str ( ) ) ;
2013-03-05 15:28:40 +01:00
if ( scope ) {
tok = tok - > tokAt ( 2 ) ;
} else {
startScope = startScope - > nestedIn ;
if ( ! startScope )
break ;
scope = startScope ;
tok = startTok ;
}
2013-03-05 13:33:38 +01:00
} else
return scope - > findType ( tok - > str ( ) ) ;
2012-11-30 06:03:58 +01:00
}
// not a valid path
2013-03-05 13:33:38 +01:00
return 0 ;
2012-11-30 06:03:58 +01:00
}
2014-04-10 16:11:11 +02:00
2013-08-17 18:43:15 +02:00
//---------------------------------------------------------------------------
const Type * SymbolDatabase : : findTypeInNested ( const Token * startTok , const Scope * startScope ) const
{
// skip over struct or union
if ( Token : : Match ( startTok , " struct|union " ) )
startTok = startTok - > next ( ) ;
// type same as scope
if ( startTok - > str ( ) = = startScope - > className & & startScope - > isClassOrStruct ( ) )
return startScope - > definedType ;
bool hasPath = false ;
// absolute path - directly start in global scope
if ( startTok - > str ( ) = = " :: " ) {
hasPath = true ;
startTok = startTok - > next ( ) ;
startScope = & scopeList . front ( ) ;
}
const Token * tok = startTok ;
const Scope * scope = startScope ;
while ( scope & & tok & & tok - > isName ( ) ) {
if ( tok - > strAt ( 1 ) = = " :: " ) {
hasPath = true ;
scope = scope - > findRecordInNestedList ( tok - > str ( ) ) ;
if ( scope ) {
tok = tok - > tokAt ( 2 ) ;
} else {
startScope = startScope - > nestedIn ;
if ( ! startScope )
break ;
scope = startScope ;
tok = startTok ;
}
} else {
const Type * type = scope - > findType ( tok - > str ( ) ) ;
if ( hasPath | | type )
return type ;
else {
scope = scope - > nestedIn ;
if ( ! scope )
break ;
}
}
}
// not a valid path
return 0 ;
}
2014-04-10 16:11:11 +02:00
//---------------------------------------------------------------------------
const Scope * SymbolDatabase : : findNamespace ( const Token * tok , const Scope * scope ) const
{
const Scope * s = findScope ( tok , scope ) ;
if ( s )
return s ;
else if ( scope - > nestedIn )
return findNamespace ( tok , scope - > nestedIn ) ;
return 0 ;
}
//---------------------------------------------------------------------------
Function * SymbolDatabase : : findFunctionInScope ( const Token * func , const Scope * ns )
{
2014-04-11 05:40:37 +02:00
const Function * function = nullptr ;
2015-01-12 06:11:22 +01:00
const bool destructor = func - > strAt ( - 1 ) = = " ~ " ;
2014-04-11 05:40:37 +02:00
2015-01-02 21:38:19 +01:00
for ( std : : multimap < std : : string , const Function * > : : const_iterator it = ns - > functionMap . find ( func - > str ( ) ) ;
it ! = ns - > functionMap . end ( ) & & it - > first = = func - > str ( ) ; + + it ) {
2015-01-12 06:11:22 +01:00
if ( Function : : argsMatch ( ns , func - > tokAt ( 2 ) , it - > second - > argDef - > next ( ) , " " , 0 ) & &
it - > second - > isDestructor ( ) = = destructor ) {
2015-01-02 21:38:19 +01:00
function = it - > second ;
break ;
2014-04-11 05:40:37 +02:00
}
}
2014-04-10 16:11:11 +02:00
if ( ! function ) {
const Scope * scope = ns - > findRecordInNestedList ( func - > str ( ) ) ;
if ( scope & & func - > strAt ( 1 ) = = " :: " ) {
2014-12-09 21:36:09 +01:00
func = func - > tokAt ( 2 ) ;
if ( func - > str ( ) = = " ~ " )
func = func - > next ( ) ;
function = findFunctionInScope ( func , scope ) ;
2014-04-10 16:11:11 +02:00
}
}
return const_cast < Function * > ( function ) ;
}
2015-05-22 22:18:42 +02:00
//---------------------------------------------------------------------------
2015-06-18 19:07:51 +02:00
namespace {
2015-06-30 08:46:30 +02:00
const std : : set < std : : string > c_keywords = make_container < std : : set < std : : string > > ( ) < <
2015-05-23 20:51:15 +02:00
" auto " < < " break " < < " case " < < " char " < < " const " < < " continue " < < " default " < < " do " < <
" double " < < " else " < < " enum " < < " extern " < < " float " < < " for " < < " goto " < < " if " < < " inline " < <
" int " < < " long " < < " register " < < " restrict " < < " return " < < " short " < < " signed " < < " sizeof " < <
" static " < < " struct " < < " switch " < < " typedef " < < " union " < < " unsigned " < < " void " < < " volatile " < <
" while " ;
2015-06-30 08:46:30 +02:00
const std : : set < std : : string > cpp_keywords = make_container < std : : set < std : : string > > ( ) < <
2015-11-29 13:23:13 +01:00
c_keywords < <
2015-05-23 20:51:15 +02:00
" alignas " < < " alignof " < < " and " < < " and_eq " < < " asm " < < " auto " < < " bitand " < < " bitor " < < " bool " < <
" break " < < " case " < < " catch " < < " char " < < " char16_t " < < " char32_t " < < " class " < < " compl " < <
" concept " < < " const " < < " constexpr " < < " const_cast " < < " continue " < < " decltype " < < " default " < <
" delete " < < " do " < < " double " < < " dynamic_cast " < < " else " < < " enum " < < " explicit " < < " export " < <
" extern " < < " false " < < " float " < < " for " < < " friend " < < " goto " < < " if " < < " inline " < < " int " < < " long " < <
" mutable " < < " namespace " < < " new " < < " noexcept " < < " not " < < " not_eq " < < " nullptr " < < " operator " < <
" or " < < " or_eq " < < " private " < < " protected " < < " public " < < " register " < < " reinterpret_cast " < <
" requires " < < " return " < < " short " < < " signed " < < " sizeof " < < " static " < < " static_assert " < <
" static_cast " < < " struct " < < " switch " < < " template " < < " this " < < " thread_local " < < " throw " < <
" true " < < " try " < < " typedef " < < " typeid " < < " typename " < < " union " < < " unsigned " < < " using " < <
" virtual " < < " void " < < " volatile " < < " wchar_t " < < " while " < < " xor " < < " xor_eq " ;
2015-06-18 19:07:51 +02:00
}
2015-11-29 13:23:13 +01:00
2015-06-18 19:07:51 +02:00
bool SymbolDatabase : : isReservedName ( const std : : string & iName ) const
{
2015-11-29 13:23:13 +01:00
if ( isCPP ( ) )
return cpp_keywords . find ( iName ) ! = cpp_keywords . cend ( ) ;
else
return c_keywords . find ( iName ) ! = c_keywords . cend ( ) ;
2015-05-22 22:18:42 +02:00
}
2015-10-04 19:42:58 +02:00
2016-01-01 17:33:59 +01:00
static const Token * parsedecl ( const Token * type , ValueType * const valuetype , ValueType : : Sign defaultSignedness ) ;
2016-01-07 10:46:19 +01:00
static void setValueType ( Token * tok , const ValueType & valuetype , bool cpp , ValueType : : Sign defaultSignedness ) ;
2015-10-11 10:48:08 +02:00
2016-01-07 10:46:19 +01:00
static void setValueType ( Token * tok , const Variable & var , bool cpp , ValueType : : Sign defaultSignedness )
2015-10-11 10:48:08 +02:00
{
ValueType valuetype ;
valuetype . pointer = var . dimensions ( ) . size ( ) ;
valuetype . typeScope = var . typeScope ( ) ;
2016-01-01 17:33:59 +01:00
if ( parsedecl ( var . typeStartToken ( ) , & valuetype , defaultSignedness ) )
2016-01-07 10:46:19 +01:00
setValueType ( tok , valuetype , cpp , defaultSignedness ) ;
2015-10-11 10:48:08 +02:00
}
2016-01-07 10:46:19 +01:00
static void setValueType ( Token * tok , const ValueType & valuetype , bool cpp , ValueType : : Sign defaultSignedness )
2015-10-04 19:42:58 +02:00
{
2015-10-07 20:24:17 +02:00
tok - > setValueType ( new ValueType ( valuetype ) ) ;
2015-10-04 19:42:58 +02:00
Token * parent = const_cast < Token * > ( tok - > astParent ( ) ) ;
if ( ! parent | | parent - > valueType ( ) )
return ;
2016-02-04 20:49:13 +01:00
if ( ! parent - > astOperand1 ( ) )
2015-10-04 19:42:58 +02:00
return ;
2016-02-04 20:49:13 +01:00
const ValueType * vt1 = parent - > astOperand1 ( ) ? parent - > astOperand1 ( ) - > valueType ( ) : nullptr ;
const ValueType * vt2 = parent - > astOperand2 ( ) ? parent - > astOperand2 ( ) - > valueType ( ) : nullptr ;
if ( vt1 & & Token : : Match ( parent , " <<|>> " ) ) {
if ( ! cpp | | ( vt2 & & vt2 - > isIntegral ( ) ) )
setValueType ( parent , * vt1 , cpp , defaultSignedness ) ;
2015-12-31 20:18:35 +01:00
return ;
}
2015-10-07 20:24:17 +02:00
if ( parent - > str ( ) = = " [ " & & valuetype . pointer > 0U ) {
2015-12-30 11:36:46 +01:00
ValueType vt ( valuetype ) ;
vt . pointer - = 1U ;
vt . constness > > = 1 ;
2016-01-07 10:46:19 +01:00
setValueType ( parent , vt , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
2015-10-07 20:24:17 +02:00
if ( parent - > str ( ) = = " * " & & ! parent - > astOperand2 ( ) & & valuetype . pointer > 0U ) {
2015-12-30 11:36:46 +01:00
ValueType vt ( valuetype ) ;
vt . pointer - = 1U ;
vt . constness > > = 1 ;
2016-01-07 10:46:19 +01:00
setValueType ( parent , vt , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
2015-10-08 19:50:10 +02:00
if ( parent - > str ( ) = = " & " & & ! parent - > astOperand2 ( ) ) {
2015-12-30 11:36:46 +01:00
ValueType vt ( valuetype ) ;
vt . pointer + = 1U ;
2016-01-07 10:46:19 +01:00
setValueType ( parent , vt , cpp , defaultSignedness ) ;
2015-10-08 19:50:10 +02:00
return ;
}
2015-10-04 19:42:58 +02:00
2016-02-04 20:49:13 +01:00
if ( ( parent - > str ( ) = = " . " | | parent - > str ( ) = = " :: " ) & &
parent - > astOperand2 ( ) & & parent - > astOperand2 ( ) - > isName ( ) ) {
const Variable * var = parent - > astOperand2 ( ) - > variable ( ) ;
if ( ! var & & valuetype . typeScope & & vt1 ) {
const std : : string & name = parent - > astOperand2 ( ) - > str ( ) ;
const Scope * typeScope = vt1 - > typeScope ;
if ( ! typeScope )
2015-10-11 10:48:08 +02:00
return ;
2016-02-04 20:49:13 +01:00
for ( std : : list < Variable > : : const_iterator it = typeScope - > varlist . begin ( ) ; it ! = typeScope - > varlist . end ( ) ; + + it ) {
if ( it - > nameToken ( ) - > str ( ) = = name ) {
var = & * it ;
break ;
}
2015-10-11 10:48:08 +02:00
}
}
2016-02-04 20:49:13 +01:00
if ( var )
setValueType ( parent , * var , cpp , defaultSignedness ) ;
return ;
2015-10-11 10:48:08 +02:00
}
2016-02-04 20:49:13 +01:00
if ( ! vt1 )
2015-10-04 19:42:58 +02:00
return ;
2016-02-04 20:49:13 +01:00
if ( parent - > astOperand2 ( ) & & ! vt2 )
return ;
2015-10-04 19:42:58 +02:00
if ( parent - > isArithmeticalOp ( ) & & vt2 ) {
if ( vt1 - > pointer ! = 0U & & vt2 - > pointer = = 0U ) {
2016-01-07 10:46:19 +01:00
setValueType ( parent , * vt1 , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
if ( vt1 - > pointer = = 0U & & vt2 - > pointer ! = 0U ) {
2016-01-07 10:46:19 +01:00
setValueType ( parent , * vt2 , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
if ( vt1 - > pointer ! = 0U ) { // result is pointer diff
2016-01-07 10:46:19 +01:00
setValueType ( parent , ValueType ( ValueType : : Sign : : UNSIGNED , ValueType : : Type : : INT , 0U , 0U , " ptrdiff_t " ) , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
2015-10-08 19:50:10 +02:00
if ( vt1 - > type = = ValueType : : Type : : LONGDOUBLE | | vt2 - > type = = ValueType : : Type : : LONGDOUBLE ) {
2016-01-07 10:46:19 +01:00
setValueType ( parent , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : LONGDOUBLE , 0U ) , cpp , defaultSignedness ) ;
2015-10-08 19:50:10 +02:00
return ;
}
2015-10-04 19:42:58 +02:00
if ( vt1 - > type = = ValueType : : Type : : DOUBLE | | vt2 - > type = = ValueType : : Type : : DOUBLE ) {
2016-01-07 10:46:19 +01:00
setValueType ( parent , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : DOUBLE , 0U ) , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
if ( vt1 - > type = = ValueType : : Type : : FLOAT | | vt2 - > type = = ValueType : : Type : : FLOAT ) {
2016-01-07 10:46:19 +01:00
setValueType ( parent , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : FLOAT , 0U ) , cpp , defaultSignedness ) ;
2015-10-04 19:42:58 +02:00
return ;
}
2015-10-11 12:20:40 +02:00
}
2015-10-07 18:10:48 +02:00
2015-10-11 12:20:40 +02:00
if ( vt2 & &
vt1 - > isIntegral ( ) & & vt1 - > pointer = = 0U & &
vt2 - > isIntegral ( ) & & vt2 - > pointer = = 0U & &
2016-02-02 17:17:33 +01:00
( parent - > isArithmeticalOp ( ) | | parent - > tokType ( ) = = Token : : eBitOp | | parent - > isAssignmentOp ( ) ) ) {
2015-10-11 12:20:40 +02:00
ValueType vt ;
if ( vt1 - > type = = vt2 - > type ) {
vt . type = vt1 - > type ;
if ( vt1 - > sign = = ValueType : : Sign : : UNSIGNED | | vt2 - > sign = = ValueType : : Sign : : UNSIGNED )
vt . sign = ValueType : : Sign : : UNSIGNED ;
else if ( vt1 - > sign = = ValueType : : Sign : : UNKNOWN_SIGN | | vt2 - > sign = = ValueType : : Sign : : UNKNOWN_SIGN )
vt . sign = ValueType : : Sign : : UNKNOWN_SIGN ;
else
vt . sign = ValueType : : Sign : : SIGNED ;
vt . originalTypeName = ( vt1 - > originalTypeName . empty ( ) ? vt2 : vt1 ) - > originalTypeName ;
} else if ( vt1 - > type > vt2 - > type ) {
vt . type = vt1 - > type ;
vt . sign = vt1 - > sign ;
vt . originalTypeName = vt1 - > originalTypeName ;
} else {
vt . type = vt2 - > type ;
vt . sign = vt2 - > sign ;
vt . originalTypeName = vt2 - > originalTypeName ;
}
if ( vt . type < ValueType : : Type : : INT ) {
vt . type = ValueType : : Type : : INT ;
vt . sign = ValueType : : Sign : : SIGNED ;
vt . originalTypeName . clear ( ) ;
2015-10-04 19:42:58 +02:00
}
2015-10-11 12:20:40 +02:00
2016-01-07 10:46:19 +01:00
setValueType ( parent , vt , cpp , defaultSignedness ) ;
2015-10-11 12:20:40 +02:00
return ;
2015-10-04 19:42:58 +02:00
}
}
2016-01-01 17:33:59 +01:00
static const Token * parsedecl ( const Token * type , ValueType * const valuetype , ValueType : : Sign defaultSignedness )
2015-10-04 20:32:16 +02:00
{
2015-10-10 08:32:45 +02:00
const unsigned int pointer0 = valuetype - > pointer ;
while ( Token : : Match ( type - > previous ( ) , " %name% " ) )
type = type - > previous ( ) ;
2016-01-01 20:03:31 +01:00
valuetype - > sign = ValueType : : Sign : : UNKNOWN_SIGN ;
2015-10-04 20:32:16 +02:00
valuetype - > type = ValueType : : Type : : UNKNOWN_TYPE ;
while ( Token : : Match ( type , " %name%|*|& " ) & & ! type - > variable ( ) ) {
2015-10-04 23:27:58 +02:00
if ( type - > isSigned ( ) )
2015-10-04 20:32:16 +02:00
valuetype - > sign = ValueType : : Sign : : SIGNED ;
2015-10-04 23:27:58 +02:00
else if ( type - > isUnsigned ( ) )
2015-10-04 20:32:16 +02:00
valuetype - > sign = ValueType : : Sign : : UNSIGNED ;
2015-10-10 08:32:45 +02:00
if ( type - > str ( ) = = " const " )
valuetype - > constness | = ( 1 < < ( valuetype - > pointer - pointer0 ) ) ;
2015-12-29 19:58:51 +01:00
else if ( type - > str ( ) = = " void " )
valuetype - > type = ValueType : : Type : : VOID ;
2015-10-10 08:32:45 +02:00
else if ( type - > str ( ) = = " bool " )
2015-10-04 20:32:16 +02:00
valuetype - > type = ValueType : : Type : : BOOL ;
else if ( type - > str ( ) = = " char " )
valuetype - > type = ValueType : : Type : : CHAR ;
else if ( type - > str ( ) = = " short " )
valuetype - > type = ValueType : : Type : : SHORT ;
else if ( type - > str ( ) = = " int " )
valuetype - > type = ValueType : : Type : : INT ;
else if ( type - > str ( ) = = " long " )
2015-10-05 19:20:42 +02:00
valuetype - > type = type - > isLong ( ) ? ValueType : : Type : : LONGLONG : ValueType : : Type : : LONG ;
2015-10-04 20:32:16 +02:00
else if ( type - > str ( ) = = " float " )
valuetype - > type = ValueType : : Type : : FLOAT ;
else if ( type - > str ( ) = = " double " )
2015-10-08 19:50:10 +02:00
valuetype - > type = type - > isLong ( ) ? ValueType : : Type : : LONGDOUBLE : ValueType : : Type : : DOUBLE ;
2015-10-04 20:32:16 +02:00
else if ( type - > str ( ) = = " struct " )
valuetype - > type = ValueType : : Type : : NONSTD ;
2016-01-05 19:47:11 +01:00
else if ( type - > isName ( ) & & valuetype - > sign ! = ValueType : : Sign : : UNKNOWN_SIGN & & valuetype - > pointer = = 0U )
return nullptr ;
2015-10-04 20:32:16 +02:00
else if ( type - > str ( ) = = " * " )
valuetype - > pointer + + ;
2015-10-07 20:24:17 +02:00
if ( ! type - > originalName ( ) . empty ( ) )
valuetype - > originalTypeName = type - > originalName ( ) ;
2015-10-04 20:32:16 +02:00
type = type - > next ( ) ;
}
2016-01-01 20:03:31 +01:00
2016-01-31 14:34:26 +01:00
// Set signedness for integral types..
if ( valuetype - > isIntegral ( ) & & valuetype - > sign = = ValueType : : Sign : : UNKNOWN_SIGN ) {
if ( valuetype - > type = = ValueType : : Type : : CHAR )
valuetype - > sign = defaultSignedness ;
else if ( valuetype - > type > = ValueType : : Type : : SHORT )
valuetype - > sign = ValueType : : Sign : : SIGNED ;
}
2016-01-01 20:03:31 +01:00
2015-10-04 20:32:16 +02:00
return ( type & & valuetype - > type ! = ValueType : : Type : : UNKNOWN_TYPE ) ? type : nullptr ;
}
2016-01-07 10:46:19 +01:00
void SymbolDatabase : : setValueTypeInTokenList ( Token * tokens , bool cpp , char defaultSignedness )
2015-10-04 19:42:58 +02:00
{
2016-01-01 17:33:59 +01:00
ValueType : : Sign defsign ;
if ( defaultSignedness = = ' s ' | | defaultSignedness = = ' S ' )
defsign = ValueType : : SIGNED ;
else if ( defaultSignedness = = ' u ' | | defaultSignedness = = ' U ' )
defsign = ValueType : : UNSIGNED ;
else
defsign = ValueType : : UNKNOWN_SIGN ;
2015-10-04 19:42:58 +02:00
for ( Token * tok = tokens ; tok ; tok = tok - > next ( ) )
tok - > setValueType ( nullptr ) ;
for ( Token * tok = tokens ; tok ; tok = tok - > next ( ) ) {
if ( tok - > isNumber ( ) ) {
2015-10-07 18:10:48 +02:00
if ( MathLib : : isFloat ( tok - > str ( ) ) ) {
ValueType : : Type type = ValueType : : Type : : DOUBLE ;
2016-01-03 13:34:47 +01:00
const char suffix = tok - > str ( ) [ tok - > str ( ) . size ( ) - 1U ] ;
if ( suffix = = ' f ' | | suffix = = ' F ' )
2015-10-07 18:10:48 +02:00
type = ValueType : : Type : : FLOAT ;
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , type , 0U ) , cpp , defsign ) ;
2015-10-07 18:10:48 +02:00
} else if ( MathLib : : isInt ( tok - > str ( ) ) ) {
ValueType : : Sign sign = ValueType : : Sign : : SIGNED ;
ValueType : : Type type = ValueType : : Type : : INT ;
if ( MathLib : : isIntHex ( tok - > str ( ) ) )
sign = ValueType : : Sign : : UNSIGNED ;
2016-01-03 13:34:47 +01:00
for ( unsigned int pos = tok - > str ( ) . size ( ) - 1U ; pos > 0U & & std : : isalpha ( tok - > str ( ) [ pos ] ) ; - - pos ) {
const char suffix = tok - > str ( ) [ pos ] ;
if ( suffix = = ' u ' | | suffix = = ' U ' )
sign = ValueType : : Sign : : UNSIGNED ;
if ( suffix = = ' l ' | | suffix = = ' L ' )
type = ( type = = ValueType : : Type : : INT ) ? ValueType : : Type : : LONG : ValueType : : Type : : LONGLONG ;
2015-10-07 18:10:48 +02:00
}
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , ValueType ( sign , type , 0U ) , cpp , defsign ) ;
2015-10-07 18:10:48 +02:00
}
2015-10-04 19:42:58 +02:00
} else if ( tok - > isComparisonOp ( ) )
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : BOOL , 0U ) , cpp , defsign ) ;
2015-10-04 19:42:58 +02:00
else if ( tok - > tokType ( ) = = Token : : eChar )
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , ValueType ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : CHAR , 0U ) , cpp , defsign ) ;
2015-10-11 08:13:30 +02:00
else if ( tok - > tokType ( ) = = Token : : eString ) {
ValueType valuetype ( ValueType : : Sign : : UNKNOWN_SIGN , ValueType : : Type : : CHAR , 1U , 1U ) ;
if ( tok - > isLong ( ) ) {
valuetype . originalTypeName = " wchar_t " ;
valuetype . type = ValueType : : Type : : SHORT ;
}
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , valuetype , cpp , defsign ) ;
2015-10-11 08:13:30 +02:00
} else if ( tok - > str ( ) = = " ( " ) {
2015-10-04 19:42:58 +02:00
// cast
2015-10-04 23:27:58 +02:00
if ( ! tok - > astOperand2 ( ) & & Token : : Match ( tok , " ( %name% " ) ) {
2015-10-04 20:32:16 +02:00
ValueType valuetype ;
2016-01-01 17:33:59 +01:00
if ( Token : : simpleMatch ( parsedecl ( tok - > next ( ) , & valuetype , defsign ) , " ) " ) )
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , valuetype , cpp , defsign ) ;
2015-10-04 19:42:58 +02:00
}
2015-10-11 08:42:31 +02:00
2015-12-26 15:20:17 +01:00
// C++ cast
if ( tok - > astOperand2 ( ) & & Token : : Match ( tok - > astOperand1 ( ) , " static_cast|const_cast|dynamic_cast|reinterpret_cast < %name% " ) & & tok - > astOperand1 ( ) - > linkAt ( 1 ) ) {
ValueType valuetype ;
2016-01-01 17:33:59 +01:00
if ( Token : : simpleMatch ( parsedecl ( tok - > astOperand1 ( ) - > tokAt ( 2 ) , & valuetype , defsign ) , " > " ) )
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , valuetype , cpp , defsign ) ;
2015-12-26 15:20:17 +01:00
}
2015-10-11 08:42:31 +02:00
// function
2015-10-11 12:20:40 +02:00
else if ( tok - > previous ( ) & & tok - > previous ( ) - > function ( ) & & tok - > previous ( ) - > function ( ) - > retDef ) {
2015-10-11 08:42:31 +02:00
ValueType valuetype ;
2016-01-01 17:33:59 +01:00
if ( Token : : simpleMatch ( parsedecl ( tok - > previous ( ) - > function ( ) - > retDef , & valuetype , defsign ) , " ( " ) )
2016-01-07 10:46:19 +01:00
: : setValueType ( tok , valuetype , cpp , defsign ) ;
2015-10-11 08:42:31 +02:00
}
2015-10-11 12:20:40 +02:00
else if ( Token : : simpleMatch ( tok - > previous ( ) , " sizeof ( " ) ) {
// TODO: use specified size_t type
ValueType valuetype ( ValueType : : Sign : : UNSIGNED , ValueType : : Type : : LONG , 0U ) ;
valuetype . originalTypeName = " size_t " ;
2016-01-07 10:46:19 +01:00
setValueType ( tok , valuetype , cpp , defsign ) ;
2015-10-11 12:20:40 +02:00
}
2015-10-04 19:42:58 +02:00
} else if ( tok - > variable ( ) ) {
2016-01-07 10:46:19 +01:00
setValueType ( tok , * tok - > variable ( ) , cpp , defsign ) ;
2015-10-04 19:42:58 +02:00
}
}
}
2015-10-07 13:38:34 +02:00
std : : string ValueType : : str ( ) const
{
std : : string ret ;
2015-10-07 19:08:26 +02:00
if ( constness & 1 )
ret = " const " ;
2015-12-29 19:58:51 +01:00
if ( type = = VOID )
ret + = " void " ;
else if ( isIntegral ( ) ) {
2015-10-07 13:38:34 +02:00
if ( sign = = SIGNED )
2015-10-07 19:08:26 +02:00
ret + = " signed " ;
2015-10-07 13:38:34 +02:00
else if ( sign = = UNSIGNED )
2015-10-07 19:08:26 +02:00
ret + = " unsigned " ;
2015-10-07 13:38:34 +02:00
if ( type = = BOOL )
2015-10-07 19:08:26 +02:00
ret + = " bool " ;
2015-10-07 13:38:34 +02:00
else if ( type = = CHAR )
2015-10-07 19:08:26 +02:00
ret + = " char " ;
2015-10-07 13:38:34 +02:00
else if ( type = = SHORT )
2015-10-07 19:08:26 +02:00
ret + = " short " ;
2015-10-07 13:38:34 +02:00
else if ( type = = INT )
2015-10-07 19:08:26 +02:00
ret + = " int " ;
2015-10-07 13:38:34 +02:00
else if ( type = = LONG )
2015-10-07 19:08:26 +02:00
ret + = " long " ;
2015-10-07 13:38:34 +02:00
else if ( type = = LONGLONG )
2015-10-07 19:08:26 +02:00
ret + = " long long " ;
2015-10-07 13:38:34 +02:00
} else if ( type = = FLOAT )
2015-10-08 19:50:10 +02:00
ret + = " float " ;
2015-10-07 13:38:34 +02:00
else if ( type = = DOUBLE )
2015-10-08 19:50:10 +02:00
ret + = " double " ;
else if ( type = = LONGDOUBLE )
ret + = " long double " ;
2015-12-13 19:58:27 +01:00
for ( int p = 0 ; p < pointer ; p + + ) {
2015-10-07 19:08:26 +02:00
ret + = " * " ;
if ( constness & ( 2 < < p ) )
2015-10-07 19:50:50 +02:00
ret + = " const " ;
}
2016-01-01 15:13:50 +01:00
return ret . empty ( ) ? ret : ret . substr ( 1 ) ;
2015-10-07 13:38:34 +02:00
}