forked from qt-creator/qt-creator
		
	
		
			
				
	
	
		
			259 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			259 lines
		
	
	
		
			8.4 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /***************************************************************************
 | |
| **
 | |
| ** This file is part of Qt Creator
 | |
| **
 | |
| ** Copyright (c) 2008 Nokia Corporation and/or its subsidiary(-ies).
 | |
| **
 | |
| ** Contact:  Qt Software Information (qt-info@nokia.com)
 | |
| **
 | |
| **
 | |
| ** Non-Open Source Usage
 | |
| **
 | |
| ** Licensees may use this file in accordance with the Qt Beta Version
 | |
| ** License Agreement, Agreement version 2.2 provided with the Software or,
 | |
| ** alternatively, in accordance with the terms contained in a written
 | |
| ** agreement between you and Nokia.
 | |
| **
 | |
| ** GNU General Public License Usage
 | |
| **
 | |
| ** Alternatively, this file may be used under the terms of the GNU General
 | |
| ** Public License versions 2.0 or 3.0 as published by the Free Software
 | |
| ** Foundation and appearing in the file LICENSE.GPL included in the packaging
 | |
| ** of this file.  Please review the following information to ensure GNU
 | |
| ** General Public Licensing requirements will be met:
 | |
| **
 | |
| ** http://www.fsf.org/licensing/licenses/info/GPLv2.html and
 | |
| ** http://www.gnu.org/copyleft/gpl.html.
 | |
| **
 | |
| ** In addition, as a special exception, Nokia gives you certain additional
 | |
| ** rights. These rights are described in the Nokia Qt GPL Exception
 | |
| ** version 1.3, included in the file GPL_EXCEPTION.txt in this package.
 | |
| **
 | |
| ***************************************************************************/
 | |
| // 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 "CheckDeclarator.h"
 | |
| #include "Semantic.h"
 | |
| #include "AST.h"
 | |
| #include "Control.h"
 | |
| #include "TranslationUnit.h"
 | |
| #include "Literals.h"
 | |
| #include "CoreTypes.h"
 | |
| #include "Symbols.h"
 | |
| 
 | |
| CPLUSPLUS_BEGIN_NAMESPACE
 | |
| 
 | |
| CheckDeclarator::CheckDeclarator(Semantic *semantic)
 | |
|     : SemanticCheck(semantic),
 | |
|       _declarator(0),
 | |
|       _scope(0),
 | |
|       _name(0)
 | |
| { }
 | |
| 
 | |
| CheckDeclarator::~CheckDeclarator()
 | |
| { }
 | |
| 
 | |
| FullySpecifiedType CheckDeclarator::check(DeclaratorAST *declarator,
 | |
|                                                FullySpecifiedType type,
 | |
|                                                Scope *scope,
 | |
|                                                Name **name)
 | |
| {
 | |
|     FullySpecifiedType previousType = switchFullySpecifiedType(type);
 | |
|     Scope *previousScope = switchScope(scope);
 | |
|     DeclaratorAST *previousDeclarator = switchDeclarator(declarator);
 | |
|     Name **previousName = switchName(name);
 | |
|     accept(declarator);
 | |
|     (void) switchName(previousName);
 | |
|     (void) switchDeclarator(previousDeclarator);
 | |
|     (void) switchScope(previousScope);
 | |
|     return switchFullySpecifiedType(previousType);
 | |
| }
 | |
| 
 | |
| FullySpecifiedType CheckDeclarator::check(PtrOperatorAST *ptrOperators,
 | |
|                                                FullySpecifiedType type,
 | |
|                                                Scope *scope)
 | |
| {
 | |
|     FullySpecifiedType previousType = switchFullySpecifiedType(type);
 | |
|     Scope *previousScope = switchScope(scope);
 | |
|     accept(ptrOperators);
 | |
|     (void) switchScope(previousScope);
 | |
|     return switchFullySpecifiedType(previousType);
 | |
| }
 | |
| 
 | |
| DeclaratorAST *CheckDeclarator::switchDeclarator(DeclaratorAST *declarator)
 | |
| {
 | |
|     DeclaratorAST *previousDeclarator = _declarator;
 | |
|     _declarator = declarator;
 | |
|     return previousDeclarator;
 | |
| }
 | |
| 
 | |
| FullySpecifiedType CheckDeclarator::switchFullySpecifiedType(FullySpecifiedType type)
 | |
| {
 | |
|     FullySpecifiedType previousType = _fullySpecifiedType;
 | |
|     _fullySpecifiedType = type;
 | |
|     return previousType;
 | |
| }
 | |
| 
 | |
| Scope *CheckDeclarator::switchScope(Scope *scope)
 | |
| {
 | |
|     Scope *previousScope = _scope;
 | |
|     _scope = scope;
 | |
|     return previousScope;
 | |
| }
 | |
| 
 | |
| Name **CheckDeclarator::switchName(Name **name)
 | |
| {
 | |
|     Name **previousName = _name;
 | |
|     _name = name;
 | |
|     return previousName;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(DeclaratorAST *ast)
 | |
| {
 | |
|     accept(ast->ptr_operators);
 | |
|     accept(ast->postfix_declarators);
 | |
|     accept(ast->core_declarator);
 | |
| 
 | |
|     // ### check the initializer
 | |
|     // FullySpecifiedType exprTy = semantic()->check(ast->initializer, _scope);
 | |
| 
 | |
|     if (ast->initializer && _fullySpecifiedType->isFunction()) {
 | |
|         _fullySpecifiedType->asFunction()->setPureVirtual(true);
 | |
|     }
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(DeclaratorIdAST *ast)
 | |
| {
 | |
|     Name *name = semantic()->check(ast->name, _scope);
 | |
|     if (_name)
 | |
|         *_name = name;
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(NestedDeclaratorAST *ast)
 | |
| {
 | |
|     accept(ast->declarator);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(FunctionDeclaratorAST *ast)
 | |
| {
 | |
|     Function *fun = control()->newFunction(ast->firstToken());
 | |
|     fun->setReturnType(_fullySpecifiedType);
 | |
| 
 | |
|     if (ast->parameters) {
 | |
|         DeclarationAST *parameter_declarations = ast->parameters->parameter_declarations;
 | |
|         for (DeclarationAST *decl = parameter_declarations; decl; decl = decl->next) {
 | |
|             semantic()->check(decl, fun->arguments());
 | |
|         }
 | |
| 
 | |
|         if (ast->parameters->dot_dot_dot_token)
 | |
|             fun->setVariadic(true);
 | |
|     }
 | |
| 
 | |
|     // check the arguments
 | |
|     bool hasDefaultArguments = false;
 | |
|     for (unsigned i = 0; i < fun->argumentCount(); ++i) {
 | |
|         Argument *arg = fun->argumentAt(i)->asArgument();
 | |
|         if (hasDefaultArguments && ! arg->hasInitializer()) {
 | |
|             translationUnit()->error(ast->firstToken(),
 | |
|                 "default argument missing for parameter at position %d", i + 1);
 | |
|         } else if (! hasDefaultArguments) {
 | |
|             hasDefaultArguments = arg->hasInitializer();
 | |
|         }
 | |
|     }
 | |
| 
 | |
|     FullySpecifiedType funTy(fun);
 | |
|     _fullySpecifiedType = funTy;
 | |
| 
 | |
|     for (SpecifierAST *it = ast->cv_qualifier_seq; it; it = it->next) {
 | |
|         SimpleSpecifierAST *cv = static_cast<SimpleSpecifierAST *>(it);
 | |
|         int k = tokenKind(cv->specifier_token);
 | |
|         if (k == T_CONST)
 | |
|             fun->setConst(true);
 | |
|         else if (k == T_VOLATILE)
 | |
|             fun->setVolatile(true);
 | |
|     }
 | |
| 
 | |
|     accept(ast->next);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(ArrayDeclaratorAST *ast)
 | |
| {
 | |
|     ArrayType *ty = control()->arrayType(_fullySpecifiedType); // ### set the dimension
 | |
|     FullySpecifiedType exprTy = semantic()->check(ast->expression, _scope);
 | |
|     FullySpecifiedType arrTy(ty);
 | |
|     _fullySpecifiedType = ty;
 | |
|     accept(ast->next);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(PointerToMemberAST *ast)
 | |
| {
 | |
|     Name *memberName = semantic()->check(ast->nested_name_specifier, _scope);
 | |
|     PointerToMemberType *ptrTy = control()->pointerToMemberType(memberName, _fullySpecifiedType);
 | |
|     FullySpecifiedType ty(ptrTy);
 | |
|     _fullySpecifiedType = ty;
 | |
|     applyCvQualifiers(ast->cv_qualifier_seq);
 | |
|     accept(ast->next);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(PointerAST *ast)
 | |
| {
 | |
|     PointerType *ptrTy = control()->pointerType(_fullySpecifiedType);
 | |
|     FullySpecifiedType ty(ptrTy);
 | |
|     _fullySpecifiedType = ty;
 | |
|     applyCvQualifiers(ast->cv_qualifier_seq);
 | |
|     accept(ast->next);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| bool CheckDeclarator::visit(ReferenceAST *ast)
 | |
| {
 | |
|     ReferenceType *refTy = control()->referenceType(_fullySpecifiedType);
 | |
|     FullySpecifiedType ty(refTy);
 | |
|     _fullySpecifiedType = ty;
 | |
|     accept(ast->next);
 | |
|     return false;
 | |
| }
 | |
| 
 | |
| void CheckDeclarator::applyCvQualifiers(SpecifierAST *cv)
 | |
| {
 | |
|     for (; cv; cv = cv->next) {
 | |
|         SimpleSpecifierAST *spec = static_cast<SimpleSpecifierAST *>(cv);
 | |
|         switch (translationUnit()->tokenKind(spec->specifier_token)) {
 | |
|         case T_VOLATILE:
 | |
|             _fullySpecifiedType.setVolatile(true);
 | |
|             break;
 | |
|         case T_CONST:
 | |
|             _fullySpecifiedType.setConst(true);
 | |
|             break;
 | |
|         default:
 | |
|             break;
 | |
|         } // switch
 | |
|     }
 | |
| }
 | |
| 
 | |
| CPLUSPLUS_END_NAMESPACE
 |