| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** Copyright (c) 2009 Nokia Corporation and/or its subsidiary(-ies). | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-06-17 00:01:27 +10:00
										 |  |  | ** Contact: Nokia Corporation (qt-info@nokia.com) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** Commercial Usage | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** 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. | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** GNU Lesser General Public License Usage | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2008-12-02 14:17:16 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | ** 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.
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | **************************************************************************/ | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | // Copyright (c) 2008 Roberto Raggi <roberto.raggi@gmail.com>
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Permission is hereby granted, free of charge, to any person obtaining a copy
 | 
					
						
							|  |  |  | // of this software and associated documentation files (the "Software"), to deal
 | 
					
						
							|  |  |  | // in the Software without restriction, including without limitation the rights
 | 
					
						
							|  |  |  | // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
 | 
					
						
							|  |  |  | // copies of the Software, and to permit persons to whom the Software is
 | 
					
						
							|  |  |  | // furnished to do so, subject to the following conditions:
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // The above copyright notice and this permission notice shall be included in
 | 
					
						
							|  |  |  | // all copies or substantial portions of the Software.
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 | 
					
						
							|  |  |  | // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
 | 
					
						
							|  |  |  | // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
 | 
					
						
							|  |  |  | // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
 | 
					
						
							|  |  |  | // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
 | 
					
						
							|  |  |  | // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
 | 
					
						
							|  |  |  | // THE SOFTWARE.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #include "CheckDeclaration.h"
 | 
					
						
							|  |  |  | #include "Semantic.h"
 | 
					
						
							|  |  |  | #include "AST.h"
 | 
					
						
							|  |  |  | #include "TranslationUnit.h"
 | 
					
						
							|  |  |  | #include "Scope.h"
 | 
					
						
							|  |  |  | #include "Names.h"
 | 
					
						
							|  |  |  | #include "CoreTypes.h"
 | 
					
						
							|  |  |  | #include "Symbols.h"
 | 
					
						
							|  |  |  | #include "Control.h"
 | 
					
						
							| 
									
										
										
										
											2009-03-31 13:56:28 +02:00
										 |  |  | #include "Literals.h"
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | #include <cassert>
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 11:21:25 +02:00
										 |  |  | using namespace CPlusPlus; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | CheckDeclaration::CheckDeclaration(Semantic *semantic) | 
					
						
							|  |  |  |     : SemanticCheck(semantic), | 
					
						
							|  |  |  |       _declaration(0), | 
					
						
							|  |  |  |       _scope(0), | 
					
						
							|  |  |  |       _templateParameters(0), | 
					
						
							|  |  |  |       _checkAnonymousArguments(false) | 
					
						
							|  |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | CheckDeclaration::~CheckDeclaration() | 
					
						
							|  |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckDeclaration::check(DeclarationAST *declaration, | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  |                              Scope *scope, TemplateParameters *templateParameters) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							|  |  |  |     Scope *previousScope = switchScope(scope); | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  |     TemplateParameters *previousTemplateParameters = switchTemplateParameters(templateParameters); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     DeclarationAST *previousDeclaration = switchDeclaration(declaration); | 
					
						
							|  |  |  |     accept(declaration); | 
					
						
							|  |  |  |     (void) switchDeclaration(previousDeclaration); | 
					
						
							|  |  |  |     (void) switchTemplateParameters(previousTemplateParameters); | 
					
						
							|  |  |  |     (void) switchScope(previousScope); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | DeclarationAST *CheckDeclaration::switchDeclaration(DeclarationAST *declaration) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DeclarationAST *previousDeclaration = _declaration; | 
					
						
							|  |  |  |     _declaration = declaration; | 
					
						
							|  |  |  |     return previousDeclaration; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Scope *CheckDeclaration::switchScope(Scope *scope) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Scope *previousScope = _scope; | 
					
						
							|  |  |  |     _scope = scope; | 
					
						
							|  |  |  |     return previousScope; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  | TemplateParameters *CheckDeclaration::switchTemplateParameters(TemplateParameters *templateParameters) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  |     TemplateParameters *previousTemplateParameters = _templateParameters; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _templateParameters = templateParameters; | 
					
						
							|  |  |  |     return previousTemplateParameters; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void CheckDeclaration::checkFunctionArguments(Function *fun) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! _checkAnonymousArguments) | 
					
						
							|  |  |  |         return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (_scope->isClassScope() && fun->isPublic()) { | 
					
						
							|  |  |  |         for (unsigned argc = 0; argc < fun->argumentCount(); ++argc) { | 
					
						
							|  |  |  |             Argument *arg = fun->argumentAt(argc)->asArgument(); | 
					
						
							|  |  |  |             assert(arg != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (! arg->name()) { | 
					
						
							|  |  |  |                 translationUnit()->warning(arg->sourceLocation(), | 
					
						
							|  |  |  |                                            "anonymous argument"); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-25 11:02:02 +02:00
										 |  |  | unsigned CheckDeclaration::locationOfDeclaratorId(DeclaratorAST *declarator) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (declarator && declarator->core_declarator) { | 
					
						
							|  |  |  |         if (DeclaratorIdAST *declaratorId = declarator->core_declarator->asDeclaratorId()) | 
					
						
							|  |  |  |             return declaratorId->firstToken(); | 
					
						
							|  |  |  |         else if (NestedDeclaratorAST *nested = declarator->core_declarator->asNestedDeclarator()) | 
					
						
							|  |  |  |             return locationOfDeclaratorId(nested->declarator); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool CheckDeclaration::visit(SimpleDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     FullySpecifiedType ty = semantic()->check(ast->decl_specifier_list, _scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     FullySpecifiedType qualTy = ty.qualifiedType(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 17:44:06 +01:00
										 |  |  |     if (_templateParameters && ty) { | 
					
						
							|  |  |  |         if (Class *klass = ty->asClassType()) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             klass->setTemplateParameters(_templateParameters); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     if (! ast->declarator_list && ast->decl_specifier_list && ! ast->decl_specifier_list->next) { | 
					
						
							|  |  |  |         if (ElaboratedTypeSpecifierAST *elab_type_spec = ast->decl_specifier_list->value->asElaboratedTypeSpecifier()) { | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             unsigned sourceLocation = elab_type_spec->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (elab_type_spec->name) | 
					
						
							|  |  |  |                 sourceLocation = elab_type_spec->name->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-16 15:43:24 +01:00
										 |  |  |             Name *name = semantic()->check(elab_type_spec->name, _scope); | 
					
						
							|  |  |  |             ForwardClassDeclaration *symbol = | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |                     control()->newForwardClassDeclaration(sourceLocation, name); | 
					
						
							| 
									
										
										
										
											2009-02-16 15:43:24 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (_templateParameters) { | 
					
						
							|  |  |  |                 symbol->setTemplateParameters(_templateParameters); | 
					
						
							|  |  |  |                 _templateParameters = 0; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             _scope->enterSymbol(symbol); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |     const bool isQ_SLOT   = ast->qt_invokable_token && tokenKind(ast->qt_invokable_token) == T_Q_SLOT; | 
					
						
							|  |  |  |     const bool isQ_SIGNAL = ast->qt_invokable_token && tokenKind(ast->qt_invokable_token) == T_Q_SIGNAL; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     List<Declaration *> **decl_it = &ast->symbols; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (DeclaratorListAST *it = ast->declarator_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         Name *name = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |         FullySpecifiedType declTy = semantic()->check(it->value, qualTy, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                                                       _scope, &name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |         unsigned location = locationOfDeclaratorId(it->value); | 
					
						
							| 
									
										
										
										
											2009-06-25 11:02:02 +02:00
										 |  |  |         if (! location) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |             if (it->value) | 
					
						
							|  |  |  |                 location = it->value->firstToken(); | 
					
						
							| 
									
										
										
										
											2009-06-25 11:02:02 +02:00
										 |  |  |             else | 
					
						
							|  |  |  |                 location = ast->firstToken(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 17:44:06 +01:00
										 |  |  |         Function *fun = 0; | 
					
						
							|  |  |  |         if (declTy && 0 != (fun = declTy->asFunctionType())) { | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |             fun->setSourceLocation(location); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             fun->setScope(_scope); | 
					
						
							|  |  |  |             fun->setName(name); | 
					
						
							|  |  |  |             fun->setMethodKey(semantic()->currentMethodKey()); | 
					
						
							| 
									
										
										
										
											2009-10-27 14:59:27 +01:00
										 |  |  |             fun->setVirtual(ty.isVirtual()); | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |             if (isQ_SIGNAL) | 
					
						
							|  |  |  |                 fun->setMethodKey(Function::SignalMethod); | 
					
						
							|  |  |  |             else if (isQ_SLOT) | 
					
						
							|  |  |  |                 fun->setMethodKey(Function::SlotMethod); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             fun->setVisibility(semantic()->currentVisibility()); | 
					
						
							|  |  |  |         } else if (semantic()->currentMethodKey() != Function::NormalMethod) { | 
					
						
							|  |  |  |             translationUnit()->warning(ast->firstToken(), | 
					
						
							|  |  |  |                                        "expected a function declaration"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         Declaration *symbol = control()->newDeclaration(location, name); | 
					
						
							| 
									
										
										
										
											2009-03-31 13:56:28 +02:00
										 |  |  |         symbol->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |         symbol->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         symbol->setType(control()->integerType(IntegerType::Int)); | 
					
						
							|  |  |  |         symbol->setType(declTy); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (_templateParameters && it == ast->declarator_list && ty && ! ty->isClassType()) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             symbol->setTemplateParameters(_templateParameters); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         symbol->setVisibility(semantic()->currentVisibility()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (ty.isFriend()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Friend); | 
					
						
							|  |  |  |         else if (ty.isRegister()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Register); | 
					
						
							|  |  |  |         else if (ty.isStatic()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Static); | 
					
						
							|  |  |  |         else if (ty.isExtern()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Extern); | 
					
						
							|  |  |  |         else if (ty.isMutable()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Mutable); | 
					
						
							|  |  |  |         else if (ty.isTypedef()) | 
					
						
							|  |  |  |             symbol->setStorage(Symbol::Typedef); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |         if (it->value && it->value->initializer) { | 
					
						
							|  |  |  |             FullySpecifiedType initTy = semantic()->check(it->value->initializer, _scope); | 
					
						
							| 
									
										
										
										
											2009-06-04 13:26:11 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |         *decl_it = new (translationUnit()->memoryPool()) List<Declaration *>(); | 
					
						
							|  |  |  |         (*decl_it)->value = symbol; | 
					
						
							|  |  |  |         decl_it = &(*decl_it)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         _scope->enterSymbol(symbol); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(EmptyDeclarationAST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(AccessDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int accessSpecifier = tokenKind(ast->access_specifier_token); | 
					
						
							|  |  |  |     int visibility = semantic()->visibilityForAccessSpecifier(accessSpecifier); | 
					
						
							|  |  |  |     semantic()->switchVisibility(visibility); | 
					
						
							|  |  |  |     if (ast->slots_token) | 
					
						
							|  |  |  |         semantic()->switchMethodKey(Function::SlotMethod); | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |     else if (accessSpecifier == T_Q_SIGNALS) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         semantic()->switchMethodKey(Function::SignalMethod); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         semantic()->switchMethodKey(Function::NormalMethod); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(AsmDefinitionAST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-28 16:06:26 +01:00
										 |  |  | bool CheckDeclaration::visit(ExceptionDeclarationAST *ast) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     FullySpecifiedType ty = semantic()->check(ast->type_specifier_list, _scope); | 
					
						
							| 
									
										
										
										
											2009-10-28 16:06:26 +01:00
										 |  |  |     FullySpecifiedType qualTy = ty.qualifiedType(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Name *name = 0; | 
					
						
							|  |  |  |     FullySpecifiedType declTy = semantic()->check(ast->declarator, qualTy, | 
					
						
							|  |  |  |                                                   _scope, &name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned location = locationOfDeclaratorId(ast->declarator); | 
					
						
							|  |  |  |     if (! location) { | 
					
						
							|  |  |  |         if (ast->declarator) | 
					
						
							|  |  |  |             location = ast->declarator->firstToken(); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             location = ast->firstToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Declaration *symbol = control()->newDeclaration(location, name); | 
					
						
							|  |  |  |     symbol->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     symbol->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  |     symbol->setType(declTy); | 
					
						
							|  |  |  |     _scope->enterSymbol(symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(FunctionDefinitionAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     FullySpecifiedType ty = semantic()->check(ast->decl_specifier_list, _scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     FullySpecifiedType qualTy = ty.qualifiedType(); | 
					
						
							|  |  |  |     Name *name = 0; | 
					
						
							|  |  |  |     FullySpecifiedType funTy = semantic()->check(ast->declarator, qualTy, | 
					
						
							|  |  |  |                                                  _scope, &name); | 
					
						
							| 
									
										
										
										
											2009-02-09 17:44:06 +01:00
										 |  |  |     if (! (funTy && funTy->isFunctionType())) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         translationUnit()->error(ast->firstToken(), | 
					
						
							|  |  |  |                                  "expected a function prototype"); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 17:44:06 +01:00
										 |  |  |     Function *fun = funTy->asFunctionType(); | 
					
						
							| 
									
										
										
										
											2009-10-27 14:59:27 +01:00
										 |  |  |     fun->setVirtual(ty.isVirtual()); | 
					
						
							| 
									
										
										
										
											2009-03-31 13:56:28 +02:00
										 |  |  |     fun->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     fun->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     if (ast->declarator) | 
					
						
							|  |  |  |         fun->setSourceLocation(ast->declarator->firstToken()); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     fun->setName(name); | 
					
						
							|  |  |  |     fun->setTemplateParameters(_templateParameters); | 
					
						
							|  |  |  |     fun->setVisibility(semantic()->currentVisibility()); | 
					
						
							|  |  |  |     fun->setMethodKey(semantic()->currentMethodKey()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |     const bool isQ_SLOT   = ast->qt_invokable_token && tokenKind(ast->qt_invokable_token) == T_Q_SLOT; | 
					
						
							|  |  |  |     const bool isQ_SIGNAL = ast->qt_invokable_token && tokenKind(ast->qt_invokable_token) == T_Q_SIGNAL; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (isQ_SIGNAL) | 
					
						
							|  |  |  |         fun->setMethodKey(Function::SignalMethod); | 
					
						
							|  |  |  |     else if (isQ_SLOT) | 
					
						
							|  |  |  |         fun->setMethodKey(Function::SlotMethod); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     checkFunctionArguments(fun); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = fun; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(fun); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 09:11:14 +02:00
										 |  |  |     if (! semantic()->skipFunctionBodies()) { | 
					
						
							|  |  |  |         if (ast->ctor_initializer) { | 
					
						
							|  |  |  |             bool looksLikeCtor = false; | 
					
						
							|  |  |  |             if (ty.isValid() || ! fun->identity()) | 
					
						
							|  |  |  |                 looksLikeCtor = false; | 
					
						
							|  |  |  |             else if (fun->identity()->isNameId() || fun->identity()->isTemplateNameId()) | 
					
						
							|  |  |  |                 looksLikeCtor = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (! looksLikeCtor) { | 
					
						
							|  |  |  |                 translationUnit()->error(ast->ctor_initializer->firstToken(), | 
					
						
							|  |  |  |                                          "only constructors take base initializers"); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-09-25 13:45:13 +02:00
										 |  |  |             accept(ast->ctor_initializer); | 
					
						
							| 
									
										
										
										
											2008-12-03 11:59:11 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 09:11:14 +02:00
										 |  |  |         const int previousVisibility = semantic()->switchVisibility(Symbol::Public); | 
					
						
							|  |  |  |         const int previousMethodKey = semantic()->switchMethodKey(Function::NormalMethod); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 09:11:14 +02:00
										 |  |  |         semantic()->check(ast->function_body, fun->members()); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-26 09:11:14 +02:00
										 |  |  |         semantic()->switchMethodKey(previousMethodKey); | 
					
						
							|  |  |  |         semantic()->switchVisibility(previousVisibility); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-06 12:53:05 +02:00
										 |  |  | bool CheckDeclaration::visit(MemInitializerAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     (void) semantic()->check(ast->name, _scope); | 
					
						
							|  |  |  |     FullySpecifiedType ty = semantic()->check(ast->expression, _scope); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool CheckDeclaration::visit(LinkageBodyAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (DeclarationListAST *decl = ast->declaration_list; decl; decl = decl->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |        semantic()->check(decl->value, _scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(LinkageSpecificationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |     semantic()->check(ast->declaration, _scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(NamespaceAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Identifier *id = identifier(ast->identifier_token); | 
					
						
							|  |  |  |     Name *namespaceName = control()->nameId(id); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ast->identifier_token) | 
					
						
							|  |  |  |         sourceLocation = ast->identifier_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Namespace *ns = control()->newNamespace(sourceLocation, namespaceName); | 
					
						
							| 
									
										
										
										
											2009-03-31 13:56:28 +02:00
										 |  |  |     ns->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     ns->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = ns; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(ns); | 
					
						
							|  |  |  |     semantic()->check(ast->linkage_body, ns->members()); // ### we'll do the merge later.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(NamespaceAliasDefinitionAST *) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(ParameterDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-06-25 11:02:02 +02:00
										 |  |  |     unsigned sourceLocation = locationOfDeclaratorId(ast->declarator); | 
					
						
							|  |  |  |     if (! sourceLocation) { | 
					
						
							|  |  |  |         if (ast->declarator) | 
					
						
							|  |  |  |             sourceLocation = ast->declarator->firstToken(); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     Name *argName = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     FullySpecifiedType ty = semantic()->check(ast->type_specifier_list, _scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     FullySpecifiedType argTy = semantic()->check(ast->declarator, ty.qualifiedType(), | 
					
						
							|  |  |  |                                                  _scope, &argName); | 
					
						
							|  |  |  |     FullySpecifiedType exprTy = semantic()->check(ast->expression, _scope); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     Argument *arg = control()->newArgument(sourceLocation, argName); | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = arg; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (ast->expression) | 
					
						
							|  |  |  |         arg->setInitializer(true); | 
					
						
							|  |  |  |     arg->setType(argTy); | 
					
						
							|  |  |  |     _scope->enterSymbol(arg); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(TemplateDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  |     Scope *scope = new Scope(_scope->owner()); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (DeclarationListAST *param = ast->template_parameter_list; param; param = param->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |        semantic()->check(param->value, scope); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-26 11:41:20 +02:00
										 |  |  |     semantic()->check(ast->declaration, _scope, | 
					
						
							|  |  |  |                       new TemplateParameters(_templateParameters, scope)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(TypenameTypeParameterAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  |     if (ast->name) | 
					
						
							|  |  |  |         sourceLocation = ast->name->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     Name *name = semantic()->check(ast->name, _scope); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     Argument *arg = control()->newArgument(sourceLocation, name); // ### new template type
 | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = arg; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(arg); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(TemplateTypeParameterAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  |     if (ast->name) | 
					
						
							|  |  |  |         sourceLocation = ast->name->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     Name *name = semantic()->check(ast->name, _scope); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  |     Argument *arg = control()->newArgument(sourceLocation, name); // ### new template type
 | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = arg; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(arg); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(UsingAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Name *name = semantic()->check(ast->name, _scope); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  |     if (ast->name) | 
					
						
							|  |  |  |         sourceLocation = ast->name->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UsingDeclaration *u = control()->newUsingDeclaration(sourceLocation, name); | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = u; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(u); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(UsingDirectiveAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     Name *name = semantic()->check(ast->name, _scope); | 
					
						
							| 
									
										
										
										
											2009-03-03 13:46:37 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  |     if (ast->name) | 
					
						
							|  |  |  |         sourceLocation = ast->name->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UsingNamespaceDirective *u = control()->newUsingNamespaceDirective(sourceLocation, name); | 
					
						
							| 
									
										
										
										
											2009-02-09 12:19:17 +01:00
										 |  |  |     ast->symbol = u; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     _scope->enterSymbol(u); | 
					
						
							| 
									
										
										
										
											2009-02-19 13:52:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (! (_scope->isBlockScope() || _scope->isNamespaceScope())) | 
					
						
							|  |  |  |         translationUnit()->error(ast->firstToken(), | 
					
						
							|  |  |  |                                  "using-directive not within namespace or block scope"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  | bool CheckDeclaration::visit(ObjCProtocolForwardDeclarationAST *ast) | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     const unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     List<ObjCForwardProtocolDeclaration *> **symbolIter = &ast->symbols; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |     for (ObjCIdentifierListAST *it = ast->identifier_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         unsigned declarationLocation; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         if (it->value) | 
					
						
							|  |  |  |             declarationLocation = it->value->firstToken(); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         else | 
					
						
							|  |  |  |             declarationLocation = sourceLocation; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         Name *protocolName = semantic()->check(it->value, _scope); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ObjCForwardProtocolDeclaration *fwdProtocol = control()->newObjCForwardProtocolDeclaration(sourceLocation, protocolName); | 
					
						
							|  |  |  |         fwdProtocol->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |         fwdProtocol->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         _scope->enterSymbol(fwdProtocol); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         *symbolIter = new (translationUnit()->memoryPool()) List<ObjCForwardProtocolDeclaration *>(); | 
					
						
							|  |  |  |         (*symbolIter)->value = fwdProtocol; | 
					
						
							|  |  |  |         symbolIter = &(*symbolIter)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  | bool CheckDeclaration::visit(ObjCProtocolDeclarationAST *ast) | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned sourceLocation; | 
					
						
							|  |  |  |     if (ast->name) | 
					
						
							|  |  |  |         sourceLocation = ast->name->firstToken(); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Name *protocolName = semantic()->check(ast->name, _scope); | 
					
						
							|  |  |  |     ObjCProtocol *protocol = control()->newObjCProtocol(sourceLocation, protocolName); | 
					
						
							|  |  |  |     protocol->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     protocol->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     if (ast->protocol_refs && ast->protocol_refs->identifier_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         for (ObjCIdentifierListAST *iter = ast->protocol_refs->identifier_list; iter; iter = iter->next) { | 
					
						
							|  |  |  |             NameAST* name = iter->value; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             Name *protocolName = semantic()->check(name, _scope); | 
					
						
							|  |  |  |             ObjCBaseProtocol *baseProtocol = control()->newObjCBaseProtocol(name->firstToken(), protocolName); | 
					
						
							|  |  |  |             protocol->addProtocol(baseProtocol); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int previousObjCVisibility = semantic()->switchObjCVisibility(Function::Public); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (DeclarationListAST *it = ast->member_declaration_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |         semantic()->check(it->value, protocol->members()); | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     } | 
					
						
							|  |  |  |     (void) semantic()->switchObjCVisibility(previousObjCVisibility); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ast->symbol = protocol; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     _scope->enterSymbol(protocol); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  | bool CheckDeclaration::visit(ObjCClassForwardDeclarationAST *ast) | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     const unsigned sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     List<ObjCForwardClassDeclaration *> **symbolIter = &ast->symbols; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |     for (ObjCIdentifierListAST *it = ast->identifier_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         unsigned declarationLocation; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         if (it->value) | 
					
						
							|  |  |  |             declarationLocation = it->value->firstToken(); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         else | 
					
						
							|  |  |  |             declarationLocation = sourceLocation; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         Name *className = semantic()->check(it->value, _scope); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ObjCForwardClassDeclaration *fwdClass = control()->newObjCForwardClassDeclaration(sourceLocation, className); | 
					
						
							|  |  |  |         fwdClass->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |         fwdClass->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         _scope->enterSymbol(fwdClass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         *symbolIter = new (translationUnit()->memoryPool()) List<ObjCForwardClassDeclaration *>(); | 
					
						
							|  |  |  |         (*symbolIter)->value = fwdClass; | 
					
						
							|  |  |  |         symbolIter = &(*symbolIter)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  | bool CheckDeclaration::visit(ObjCClassDeclarationAST *ast) | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     unsigned sourceLocation; | 
					
						
							|  |  |  |     if (ast->class_name) | 
					
						
							|  |  |  |         sourceLocation = ast->class_name->firstToken(); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         sourceLocation = ast->firstToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Name *className = semantic()->check(ast->class_name, _scope); | 
					
						
							|  |  |  |     ObjCClass *klass = control()->newObjCClass(sourceLocation, className); | 
					
						
							|  |  |  |     klass->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     klass->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							|  |  |  |     ast->symbol = klass; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |     klass->setInterface(ast->interface_token != 0); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ast->category_name) { | 
					
						
							|  |  |  |         Name *categoryName = semantic()->check(ast->category_name, _scope); | 
					
						
							|  |  |  |         klass->setCategoryName(categoryName); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     if (ast->superclass) { | 
					
						
							|  |  |  |         Name *superClassName = semantic()->check(ast->superclass, _scope); | 
					
						
							|  |  |  |         ObjCBaseClass *superKlass = control()->newObjCBaseClass(ast->superclass->firstToken(), superClassName); | 
					
						
							|  |  |  |         klass->setBaseClass(superKlass); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ast->protocol_refs && ast->protocol_refs->identifier_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         for (ObjCIdentifierListAST *iter = ast->protocol_refs->identifier_list; iter; iter = iter->next) { | 
					
						
							|  |  |  |             NameAST* name = iter->value; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             Name *protocolName = semantic()->check(name, _scope); | 
					
						
							|  |  |  |             ObjCBaseProtocol *baseProtocol = control()->newObjCBaseProtocol(name->firstToken(), protocolName); | 
					
						
							|  |  |  |             klass->addProtocol(baseProtocol); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     _scope->enterSymbol(klass); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     int previousObjCVisibility = semantic()->switchObjCVisibility(Function::Protected); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (ast->inst_vars_decl) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         for (DeclarationListAST *it = ast->inst_vars_decl->instance_variable_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             semantic()->check(it->value, klass->members()); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |     (void) semantic()->switchObjCVisibility(Function::Public); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (DeclarationListAST *it = ast->member_declaration_list; it; it = it->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |         semantic()->check(it->value, klass->members()); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     (void) semantic()->switchObjCVisibility(previousObjCVisibility); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(ObjCMethodDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (!ast->method_prototype) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     FullySpecifiedType ty = semantic()->check(ast->method_prototype, _scope); | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |     ObjCMethod *methodType = ty.type()->asObjCMethodType(); | 
					
						
							|  |  |  |     if (!methodType) | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |     Symbol *symbol; | 
					
						
							| 
									
										
										
										
											2009-09-21 18:26:37 +02:00
										 |  |  |     if (ast->function_body) { | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |         if (!semantic()->skipFunctionBodies()) { | 
					
						
							|  |  |  |             semantic()->check(ast->function_body, methodType->members()); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |         symbol = methodType; | 
					
						
							| 
									
										
										
										
											2009-09-21 18:26:37 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         Declaration *decl = control()->newDeclaration(ast->firstToken(), methodType->name()); | 
					
						
							|  |  |  |         decl->setType(methodType); | 
					
						
							|  |  |  |         symbol = decl; | 
					
						
							| 
									
										
										
										
											2009-11-11 09:21:06 +01:00
										 |  |  |         symbol->setStorage(methodType->storage()); | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 18:30:18 +02:00
										 |  |  |     symbol->setStartOffset(tokenAt(ast->firstToken()).offset); | 
					
						
							|  |  |  |     symbol->setEndOffset(tokenAt(ast->lastToken()).offset); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     symbol->setVisibility(semantic()->currentVisibility()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _scope->enterSymbol(symbol); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | bool CheckDeclaration::visit(ObjCVisibilityDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int accessSpecifier = tokenKind(ast->visibility_token); | 
					
						
							|  |  |  |     int visibility = semantic()->visibilityForObjCAccessSpecifier(accessSpecifier); | 
					
						
							|  |  |  |     semantic()->switchObjCVisibility(visibility); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  | enum PropertyAttributes { | 
					
						
							|  |  |  |     None = 0, | 
					
						
							|  |  |  |     Assign = 1 << 0, | 
					
						
							|  |  |  |     Retain = 1 << 1, | 
					
						
							|  |  |  |     Copy = 1 << 2, | 
					
						
							|  |  |  |     ReadOnly = 1 << 3, | 
					
						
							|  |  |  |     ReadWrite = 1 << 4, | 
					
						
							|  |  |  |     Getter = 1 << 5, | 
					
						
							|  |  |  |     Setter = 1 << 6, | 
					
						
							|  |  |  |     NonAtomic = 1 << 7, | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     WritabilityMask = ReadOnly | ReadWrite, | 
					
						
							|  |  |  |     SetterSemanticsMask = Assign | Retain | Copy, | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::checkPropertyAttribute(ObjCPropertyAttributeAST *attrAst, | 
					
						
							|  |  |  |                                               int &flags, | 
					
						
							|  |  |  |                                               int attr) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (flags & attr) { | 
					
						
							|  |  |  |         translationUnit()->warning(attrAst->attribute_identifier_token, | 
					
						
							|  |  |  |                                    "duplicate property attribute \"%s\"", | 
					
						
							|  |  |  |                                    spell(attrAst->attribute_identifier_token)); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         flags |= attr; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool CheckDeclaration::visit(ObjCPropertyDeclarationAST *ast) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int propAttrs = None; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     for (ObjCPropertyAttributeListAST *iter= ast->property_attribute_list; iter; iter = iter->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:47:08 +01:00
										 |  |  |         ObjCPropertyAttributeAST *attrAst = iter->value; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |         if (!attrAst) | 
					
						
							|  |  |  |             continue; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         Identifier *attrId = identifier(attrAst->attribute_identifier_token); | 
					
						
							|  |  |  |         if (attrId == control()->objcGetterId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             if (checkPropertyAttribute(attrAst, propAttrs, Getter)) { | 
					
						
							|  |  |  |                 // TODO: find method declaration for getter
 | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcSetterId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             if (checkPropertyAttribute(attrAst, propAttrs, Setter)) { | 
					
						
							|  |  |  |                 // TODO: find method declaration for setter
 | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcReadwriteId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, ReadWrite); | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcReadonlyId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, ReadOnly); | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcAssignId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, Assign); | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcRetainId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, Retain); | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcCopyId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, Copy); | 
					
						
							| 
									
										
										
										
											2009-10-20 12:47:54 +02:00
										 |  |  |         } else if (attrId == control()->objcNonatomicId()) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             checkPropertyAttribute(attrAst, propAttrs, NonAtomic); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (propAttrs & ReadOnly && propAttrs & ReadWrite) | 
					
						
							|  |  |  |         // Should this be an error instead of only a warning?
 | 
					
						
							|  |  |  |         translationUnit()->warning(ast->property_token, | 
					
						
							|  |  |  |                                    "property can have at most one attribute \"readonly\" or \"readwrite\" specified"); | 
					
						
							|  |  |  |     int setterSemAttrs = propAttrs & SetterSemanticsMask; | 
					
						
							|  |  |  |     if (setterSemAttrs | 
					
						
							|  |  |  |             && setterSemAttrs != Assign | 
					
						
							|  |  |  |             && setterSemAttrs != Retain | 
					
						
							|  |  |  |             && setterSemAttrs != Copy) { | 
					
						
							|  |  |  |         // Should this be an error instead of only a warning?
 | 
					
						
							|  |  |  |         translationUnit()->warning(ast->property_token, | 
					
						
							|  |  |  |                                    "property can have at most one attribute \"assign\", \"retain\", or \"copy\" specified"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // TODO: Check if the next line is correct (EV)
 | 
					
						
							|  |  |  |     semantic()->check(ast->simple_declaration, _scope); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 11:21:25 +02:00
										 |  |  | 
 |