| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | /**************************************************************************
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Commercial Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Licensees holding valid Qt Commercial licenses may use this file in | 
					
						
							|  |  |  | ** accordance with the Qt Commercial License Agreement provided with the | 
					
						
							|  |  |  | ** Software or, alternatively, in accordance with the terms contained in | 
					
						
							|  |  |  | ** a written agreement between you and Nokia. | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** GNU Lesser General Public License Usage | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** Alternatively, this file may be used under the terms of the GNU Lesser | 
					
						
							|  |  |  | ** General Public License version 2.1 as published by the Free Software | 
					
						
							|  |  |  | ** Foundation and appearing in the file LICENSE.LGPL included in the | 
					
						
							|  |  |  | ** packaging of this file.  Please review the following information to | 
					
						
							|  |  |  | ** ensure the GNU Lesser General Public License version 2.1 requirements | 
					
						
							|  |  |  | ** will be met: http://www.gnu.org/licenses/old-licenses/lgpl-2.1.html.
 | 
					
						
							|  |  |  | ** | 
					
						
							|  |  |  | ** If you are unsure which license is appropriate for your use, please | 
					
						
							| 
									
										
										
										
											2009-08-14 09:30:56 +02:00
										 |  |  | ** contact the sales department at http://qt.nokia.com/contact.
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | ** | 
					
						
							|  |  |  | **************************************************************************/ | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "CheckUndefinedSymbols.h"
 | 
					
						
							|  |  |  | #include "Overview.h"
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include <Names.h>
 | 
					
						
							|  |  |  | #include <Literals.h>
 | 
					
						
							|  |  |  | #include <Symbols.h>
 | 
					
						
							|  |  |  | #include <TranslationUnit.h>
 | 
					
						
							|  |  |  | #include <Scope.h>
 | 
					
						
							|  |  |  | #include <AST.h>
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | using namespace CPlusPlus; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CheckUndefinedSymbols::CheckUndefinedSymbols(Document::Ptr doc) | 
					
						
							| 
									
										
										
										
											2009-11-17 13:28:20 +01:00
										 |  |  |     : ASTVisitor(doc->translationUnit()), _doc(doc) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CheckUndefinedSymbols::~CheckUndefinedSymbols() | 
					
						
							|  |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::setGlobalNamespaceBinding(NamespaceBindingPtr globalNamespaceBinding) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _globalNamespaceBinding = globalNamespaceBinding; | 
					
						
							|  |  |  |     _types.clear(); | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     _protocols.clear(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (_globalNamespaceBinding) { | 
					
						
							|  |  |  |         QSet<NamespaceBinding *> processed; | 
					
						
							|  |  |  |         buildTypeMap(_globalNamespaceBinding.data(), &processed); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::operator()(AST *ast) | 
					
						
							|  |  |  | { accept(ast); } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  | QByteArray CheckUndefinedSymbols::templateParameterName(NameAST *ast) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast && ast->name) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |         if (const Identifier *id = ast->name->identifier()) | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |             return QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return QByteArray(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | QByteArray CheckUndefinedSymbols::templateParameterName(DeclarationAST *ast) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast) { | 
					
						
							|  |  |  |         if (TypenameTypeParameterAST *d = ast->asTypenameTypeParameter()) | 
					
						
							|  |  |  |             return templateParameterName(d->name); | 
					
						
							|  |  |  |         else if (TemplateTypeParameterAST *d = ast->asTemplateTypeParameter()) | 
					
						
							|  |  |  |             return templateParameterName(d->name); | 
					
						
							| 
									
										
										
										
											2009-12-08 12:57:41 +01:00
										 |  |  |         else if (ParameterDeclarationAST *d = ast->asParameterDeclaration()) { | 
					
						
							|  |  |  |             if (d->symbol) { | 
					
						
							|  |  |  |                 if (const Identifier *id = d->symbol->identifier()) | 
					
						
							|  |  |  |                     return QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     return QByteArray(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::isType(const QByteArray &name) const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  |     for (int i = _compoundStatementStack.size() - 1; i != -1; --i) { | 
					
						
							|  |  |  |         Scope *members = _compoundStatementStack.at(i)->symbol->members(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (unsigned m = 0; m < members->symbolCount(); ++m) { | 
					
						
							|  |  |  |             Symbol *member = members->symbolAt(m); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (member->isTypedef() && member->isDeclaration()) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |                 if (const Identifier *id = member->identifier()) { | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  |                     if (name == id->chars()) | 
					
						
							|  |  |  |                         return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     for (int i = _templateDeclarationStack.size() - 1; i != - 1; --i) { | 
					
						
							|  |  |  |         TemplateDeclarationAST *templateDeclaration = _templateDeclarationStack.at(i); | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         for (DeclarationListAST *it = templateDeclaration->template_parameter_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             DeclarationAST *templateParameter = it->value; | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |             if (templateParameterName(templateParameter) == name) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     return _types.contains(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  | bool CheckUndefinedSymbols::isType(const Identifier *id) const | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! id) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return isType(QByteArray::fromRawData(id->chars(), id->size())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  | void CheckUndefinedSymbols::addType(const Name *name) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (! name) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |     if (const Identifier *id = name->identifier()) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |         _types.insert(QByteArray(id->chars(), id->size())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  | void CheckUndefinedSymbols::addProtocol(const Name *name) | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (!name) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |     if (const Identifier *id = name->identifier()) | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |         _protocols.insert(QByteArray(id->chars(), id->size())); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::isProtocol(const QByteArray &name) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return _protocols.contains(name); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | void CheckUndefinedSymbols::buildTypeMap(Class *klass) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     addType(klass->name()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (unsigned i = 0; i < klass->memberCount(); ++i) { | 
					
						
							| 
									
										
										
										
											2009-10-02 10:28:05 +02:00
										 |  |  |         buildMemberTypeMap(klass->memberAt(i)); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::buildMemberTypeMap(Symbol *member) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (member == 0) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (Class *klass = member->asClass()) { | 
					
						
							|  |  |  |         buildTypeMap(klass); | 
					
						
							|  |  |  |     } else if (Enum *e = member->asEnum()) { | 
					
						
							|  |  |  |         addType(e->name()); | 
					
						
							|  |  |  |     } else if (ForwardClassDeclaration *fwd = member->asForwardClassDeclaration()) { | 
					
						
							|  |  |  |         addType(fwd->name()); | 
					
						
							|  |  |  |     } else if (Declaration *decl = member->asDeclaration()) { | 
					
						
							|  |  |  |         if (decl->isTypedef()) | 
					
						
							|  |  |  |             addType(decl->name()); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::buildTypeMap(NamespaceBinding *binding, QSet<NamespaceBinding *> *processed) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! processed->contains(binding)) { | 
					
						
							|  |  |  |         processed->insert(binding); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |         if (const Identifier *id = binding->identifier()) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |             _namespaceNames.insert(QByteArray(id->chars(), id->size())); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach (Namespace *ns, binding->symbols) { | 
					
						
							|  |  |  |             for (unsigned i = 0; i < ns->memberCount(); ++i) { | 
					
						
							|  |  |  |                 Symbol *member = ns->memberAt(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (Class *klass = member->asClass()) { | 
					
						
							|  |  |  |                     buildTypeMap(klass); | 
					
						
							|  |  |  |                 } else if (Enum *e = member->asEnum()) { | 
					
						
							|  |  |  |                     addType(e->name()); | 
					
						
							|  |  |  |                 } else if (ForwardClassDeclaration *fwd = member->asForwardClassDeclaration()) { | 
					
						
							|  |  |  |                     addType(fwd->name()); | 
					
						
							|  |  |  |                 } else if (Declaration *decl = member->asDeclaration()) { | 
					
						
							|  |  |  |                     if (decl->isTypedef()) | 
					
						
							|  |  |  |                         addType(decl->name()); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |                 } else if (ObjCForwardClassDeclaration *fKlass = member->asObjCForwardClassDeclaration()) { | 
					
						
							|  |  |  |                     addType(fKlass->name()); | 
					
						
							|  |  |  |                 } else if (ObjCClass *klass = member->asObjCClass()) { | 
					
						
							|  |  |  |                     addType(klass->name()); | 
					
						
							| 
									
										
										
										
											2009-10-02 10:28:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     for (unsigned i = 0; i < klass->memberCount(); ++i) | 
					
						
							|  |  |  |                         buildMemberTypeMap(klass->memberAt(i)); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |                 } else if (ObjCForwardProtocolDeclaration *fProto = member->asObjCForwardProtocolDeclaration()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |                     addProtocol(fProto->name()); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |                 } else if (ObjCProtocol *proto = member->asObjCProtocol()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |                     addProtocol(proto->name()); | 
					
						
							| 
									
										
										
										
											2009-10-02 10:28:05 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                     for (unsigned i = 0; i < proto->memberCount(); ++i) | 
					
						
							|  |  |  |                         buildMemberTypeMap(proto->memberAt(i)); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         foreach (NamespaceBinding *childBinding, binding->children) { | 
					
						
							|  |  |  |             buildTypeMap(childBinding, processed); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | FunctionDeclaratorAST *CheckUndefinedSymbols::currentFunctionDeclarator() const | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     if (_functionDeclaratorStack.isEmpty()) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     return _functionDeclaratorStack.last(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | CompoundStatementAST *CheckUndefinedSymbols::compoundStatement() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (_compoundStatementStack.isEmpty()) | 
					
						
							|  |  |  |         return 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return _compoundStatementStack.last(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | bool CheckUndefinedSymbols::visit(FunctionDeclaratorAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     _functionDeclaratorStack.append(ast); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::endVisit(FunctionDeclaratorAST *) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |     _functionDeclaratorStack.removeLast(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool CheckUndefinedSymbols::visit(TypeofSpecifierAST *) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(NamedTypeSpecifierAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->name) { | 
					
						
							|  |  |  |         if (! ast->name->name) { | 
					
						
							|  |  |  |             unsigned line, col; | 
					
						
							|  |  |  |             getTokenStartPosition(ast->firstToken(), &line, &col); | 
					
						
							|  |  |  |             // qWarning() << _doc->fileName() << line << col;
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |         } else if (const Identifier *id = ast->name->name->identifier()) { | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |             if (! isType(id)) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 if (FunctionDeclaratorAST *functionDeclarator = currentFunctionDeclarator()) { | 
					
						
							|  |  |  |                     if (functionDeclarator->as_cpp_initializer) | 
					
						
							|  |  |  |                         return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 Overview oo; | 
					
						
							|  |  |  |                 translationUnit()->warning(ast->firstToken(), "`%s' is not a type name", | 
					
						
							|  |  |  |                                            qPrintable(oo(ast->name->name))); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  | bool CheckUndefinedSymbols::visit(TemplateDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _templateDeclarationStack.append(ast); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::endVisit(TemplateDeclarationAST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _templateDeclarationStack.removeLast(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | bool CheckUndefinedSymbols::visit(ClassSpecifierAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool hasQ_OBJECT_CHECK = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ast->symbol) { | 
					
						
							|  |  |  |         Class *klass = ast->symbol->asClass(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         for (unsigned i = 0; i < klass->memberCount(); ++i) { | 
					
						
							|  |  |  |             Symbol *symbol = klass->memberAt(i); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (symbol->name() && symbol->name()->isNameId()) { | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |                 const NameId *nameId = symbol->name()->asNameId(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (! qstrcmp(nameId->identifier()->chars(), "qt_check_for_QOBJECT_macro")) { | 
					
						
							|  |  |  |                     hasQ_OBJECT_CHECK = true; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _qobjectStack.append(hasQ_OBJECT_CHECK); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::endVisit(ClassSpecifierAST *) | 
					
						
							|  |  |  | { _qobjectStack.removeLast(); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::qobjectCheck() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (_qobjectStack.isEmpty()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return _qobjectStack.last(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(FunctionDefinitionAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->symbol) { | 
					
						
							|  |  |  |         Function *fun = ast->symbol->asFunction(); | 
					
						
							|  |  |  |         if ((fun->isSignal() || fun->isSlot()) && ! qobjectCheck()) { | 
					
						
							|  |  |  |             translationUnit()->warning(ast->firstToken(), | 
					
						
							|  |  |  |                                        "you forgot the Q_OBJECT macro"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  | void CheckUndefinedSymbols::endVisit(FunctionDefinitionAST *) | 
					
						
							|  |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 14:22:00 +02:00
										 |  |  | bool CheckUndefinedSymbols::visit(CompoundStatementAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _compoundStatementStack.append(ast); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckUndefinedSymbols::endVisit(CompoundStatementAST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     _compoundStatementStack.removeLast(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  | bool CheckUndefinedSymbols::visit(SimpleDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const bool check = qobjectCheck(); | 
					
						
							|  |  |  |     for (List<Declaration *> *it = ast->symbols; it; it = it->next) { | 
					
						
							|  |  |  |         Declaration *decl = it->value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (Function *fun = decl->type()->asFunctionType()) { | 
					
						
							|  |  |  |             if ((fun->isSignal() || fun->isSlot()) && ! check) { | 
					
						
							|  |  |  |                 translationUnit()->warning(ast->firstToken(), | 
					
						
							|  |  |  |                                            "you forgot the Q_OBJECT macro"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(BaseSpecifierAST *base) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (NameAST *nameAST = base->name) { | 
					
						
							|  |  |  |         bool resolvedBaseClassName = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |         if (const Name *name = nameAST->name) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |             const Identifier *id = name->identifier(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |             const QByteArray spell = QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |             if (isType(spell)) | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 resolvedBaseClassName = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         if (! resolvedBaseClassName) | 
					
						
							|  |  |  |             translationUnit()->warning(nameAST->firstToken(), "expected class-name"); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(UsingDirectiveAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->symbol && ast->symbol->name() && _globalNamespaceBinding) { | 
					
						
							|  |  |  |         const Location loc = Location(ast->symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         NamespaceBinding *binding = _globalNamespaceBinding.data(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (Scope *enclosingNamespaceScope = ast->symbol->enclosingNamespaceScope()) | 
					
						
							|  |  |  |             binding = NamespaceBinding::find(enclosingNamespaceScope->owner()->asNamespace(), binding); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (! binding || ! binding->resolveNamespace(loc, ast->symbol->name())) { | 
					
						
							|  |  |  |             translationUnit()->warning(ast->name->firstToken(), | 
					
						
							|  |  |  |                                        "expected a namespace"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(QualifiedNameAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->name) { | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |         const QualifiedNameId *q = ast->name->asQualifiedNameId(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |         for (unsigned i = 0; i < q->nameCount() - 1; ++i) { | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |             const Name *name = q->nameAt(i); | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |             if (const Identifier *id = name->identifier()) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 const QByteArray spell = QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							| 
									
										
										
										
											2009-07-13 12:26:44 +02:00
										 |  |  |                 if (! (_namespaceNames.contains(spell) || isType(id))) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                     translationUnit()->warning(ast->firstToken(), | 
					
						
							|  |  |  |                                                "`%s' is not a namespace or class name", | 
					
						
							|  |  |  |                                                spell.constData()); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(CastExpressionAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->lparen_token && ast->type_id && ast->rparen_token && ast->expression) { | 
					
						
							|  |  |  |         if (TypeIdAST *cast_type_id = ast->type_id->asTypeId()) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             SpecifierListAST *type_specifier = cast_type_id->type_specifier_list; | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |             if (! cast_type_id->declarator && type_specifier && ! type_specifier->next && | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                 type_specifier->value->asNamedTypeSpecifier() && ast->expression && | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 ast->expression->asUnaryExpression()) { | 
					
						
							|  |  |  |                 // this ast node is ambigious, e.g.
 | 
					
						
							|  |  |  |                 //   (a) + b
 | 
					
						
							|  |  |  |                 // it can be parsed as
 | 
					
						
							|  |  |  |                 //   ((a) + b)
 | 
					
						
							|  |  |  |                 // or
 | 
					
						
							|  |  |  |                 //   (a) (+b)
 | 
					
						
							|  |  |  |                 accept(ast->expression); | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(SizeofExpressionAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (ast->lparen_token && ast->expression && ast->rparen_token) { | 
					
						
							|  |  |  |         if (TypeIdAST *type_id = ast->expression->asTypeId()) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             SpecifierListAST *type_specifier = type_id->type_specifier_list; | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |             if (! type_id->declarator && type_specifier && ! type_specifier->next && | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                 type_specifier->value->asNamedTypeSpecifier()) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 // this sizeof expression is ambiguos, e.g.
 | 
					
						
							|  |  |  |                 // sizeof (a)
 | 
					
						
							|  |  |  |                 //   `a' can be a typeid or a nested-expression.
 | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } else if (type_id->declarator | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                        &&   type_id->declarator->postfix_declarator_list | 
					
						
							|  |  |  |                        && ! type_id->declarator->postfix_declarator_list->next | 
					
						
							|  |  |  |                        &&   type_id->declarator->postfix_declarator_list->value->asArrayDeclarator() != 0) { | 
					
						
							| 
									
										
										
										
											2009-07-13 11:40:54 +02:00
										 |  |  |                 // this sizeof expression is ambiguos, e.g.
 | 
					
						
							|  |  |  |                 // sizeof(a[10])
 | 
					
						
							|  |  |  |                 //   `a' can be a typeid or an expression.
 | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(ObjCClassDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (NameAST *nameAST = ast->superclass) { | 
					
						
							|  |  |  |         bool resolvedSuperClassName = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |         if (const Name *name = nameAST->name) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |             const Identifier *id = name->identifier(); | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             const QByteArray spell = QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							|  |  |  |             if (isType(spell)) | 
					
						
							|  |  |  |                 resolvedSuperClassName = true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (! resolvedSuperClassName) { | 
					
						
							|  |  |  |             translationUnit()->warning(nameAST->firstToken(), | 
					
						
							|  |  |  |                                        "expected class-name after ':' token"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(ObjCProtocolRefsAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     for (NameListAST *iter = ast->identifier_list; iter; iter = iter->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         if (NameAST *nameAST = iter->value) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             bool resolvedProtocolName = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 12:46:15 +01:00
										 |  |  |             if (const Name *name = nameAST->name) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |                 const Identifier *id = name->identifier(); | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |                 const QByteArray spell = QByteArray::fromRawData(id->chars(), id->size()); | 
					
						
							|  |  |  |                 if (isProtocol(spell)) | 
					
						
							|  |  |  |                     resolvedProtocolName = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (!resolvedProtocolName) { | 
					
						
							|  |  |  |                 char after; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (iter == ast->identifier_list) | 
					
						
							|  |  |  |                     after = '<'; | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     after = ','; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 translationUnit()->warning(nameAST->firstToken(), "expected protocol name after '%c' token", after); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-11-11 09:32:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool CheckUndefinedSymbols::visit(ObjCPropertyDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     for (List<ObjCPropertyDeclaration *> *iter = ast->symbols; iter; iter = iter->next) { | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |         if (/*Name *getterName = */ iter->value->getterName()) { | 
					
						
							| 
									
										
										
										
											2009-11-12 17:35:48 +01:00
										 |  |  |             // FIXME: resolve the symbol for the name, and check its signature.
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-11 09:32:05 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |         if (/*Name *setterName = */ iter->value->setterName()) { | 
					
						
							| 
									
										
										
										
											2009-11-12 17:35:48 +01:00
										 |  |  |             // FIXME: resolve the symbol for the name, and check its signature.
 | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-11 09:32:05 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } |