| 
									
										
										
										
											2009-02-25 09:15:00 +01:00
										 |  |  | /**************************************************************************
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | ** | 
					
						
							|  |  |  | ** This file is part of Qt Creator | 
					
						
							|  |  |  | ** | 
					
						
							| 
									
										
										
										
											2010-03-05 11:25:49 +01:00
										 |  |  | ** Copyright (c) 2010 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 "Parser.h"
 | 
					
						
							|  |  |  | #include "Token.h"
 | 
					
						
							|  |  |  | #include "Lexer.h"
 | 
					
						
							|  |  |  | #include "Control.h"
 | 
					
						
							|  |  |  | #include "AST.h"
 | 
					
						
							|  |  |  | #include "Literals.h"
 | 
					
						
							| 
									
										
										
										
											2009-02-05 15:06:29 +01:00
										 |  |  | #include "ObjectiveCTypeQualifiers.h"
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  | #include "QtContextKeywords.h"
 | 
					
						
							| 
									
										
										
										
											2009-11-23 11:56:44 +01:00
										 |  |  | #include <cstdio> // for putchar
 | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  | #define CPLUSPLUS_NO_DEBUG_RULE
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | #define MAX_EXPRESSION_DEPTH 100
 | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | using namespace CPlusPlus; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  | namespace { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | class DebugRule { | 
					
						
							|  |  |  |     const char *name; | 
					
						
							|  |  |  |     static int depth; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | public: | 
					
						
							|  |  |  |     DebugRule(const char *name) | 
					
						
							|  |  |  |         : name(name) | 
					
						
							| 
									
										
										
										
											2009-11-23 11:56:44 +01:00
										 |  |  |     { | 
					
						
							|  |  |  |         for (int i = 0; i < depth; ++i) | 
					
						
							|  |  |  |             putchar(' '); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ++depth; | 
					
						
							|  |  |  |         printf("%s\n", name); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ~DebugRule() | 
					
						
							|  |  |  |     { --depth; } | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | int DebugRule::depth = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  | inline bool lookAtAssignmentOperator(int tokenKind) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (tokenKind) { | 
					
						
							|  |  |  |     case T_EQUAL: | 
					
						
							|  |  |  |     case T_AMPER_EQUAL: | 
					
						
							|  |  |  |     case T_CARET_EQUAL: | 
					
						
							|  |  |  |     case T_SLASH_EQUAL: | 
					
						
							|  |  |  |     case T_GREATER_GREATER_EQUAL: | 
					
						
							|  |  |  |     case T_LESS_LESS_EQUAL: | 
					
						
							|  |  |  |     case T_MINUS_EQUAL: | 
					
						
							|  |  |  |     case T_PERCENT_EQUAL: | 
					
						
							|  |  |  |     case T_PIPE_EQUAL: | 
					
						
							|  |  |  |     case T_PLUS_EQUAL: | 
					
						
							|  |  |  |     case T_STAR_EQUAL: | 
					
						
							|  |  |  |     case T_TILDE_EQUAL: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  | namespace Prec { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  | enum { | 
					
						
							|  |  |  |     Unknown         = 0, | 
					
						
							|  |  |  |     Comma           = 1, | 
					
						
							|  |  |  |     Assignment      = 2, | 
					
						
							|  |  |  |     Conditional     = 3, | 
					
						
							|  |  |  |     LogicalOr       = 4, | 
					
						
							|  |  |  |     LogicalAnd      = 5, | 
					
						
							|  |  |  |     InclusiveOr     = 6, | 
					
						
							|  |  |  |     ExclusiveOr     = 7, | 
					
						
							|  |  |  |     And             = 8, | 
					
						
							|  |  |  |     Equality        = 9, | 
					
						
							|  |  |  |     Relational      = 10, | 
					
						
							|  |  |  |     Shift           = 11, | 
					
						
							|  |  |  |     Additive        = 12, | 
					
						
							|  |  |  |     Multiplicative  = 13, | 
					
						
							|  |  |  |     PointerToMember = 14 | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  | } // end of namespace Precedece
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | inline int precedence(int tokenKind, bool templateArguments) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     // ### this will/might need some tuning for C++0x
 | 
					
						
							|  |  |  |     // (see: [temp.names]p3)
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     if (templateArguments && tokenKind == T_GREATER) | 
					
						
							|  |  |  |         return -1; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  |     if (lookAtAssignmentOperator(tokenKind)) | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |         return Prec::Assignment; | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     switch (tokenKind) { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_COMMA:           return Prec::Comma; | 
					
						
							|  |  |  |     case T_QUESTION:        return Prec::Conditional; | 
					
						
							|  |  |  |     case T_PIPE_PIPE:       return Prec::LogicalOr; | 
					
						
							|  |  |  |     case T_AMPER_AMPER:     return Prec::LogicalAnd; | 
					
						
							|  |  |  |     case T_PIPE:            return Prec::InclusiveOr; | 
					
						
							|  |  |  |     case T_CARET:           return Prec::ExclusiveOr; | 
					
						
							|  |  |  |     case T_AMPER:           return Prec::And; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_EQUAL_EQUAL: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_EXCLAIM_EQUAL:   return Prec::Equality; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_GREATER: | 
					
						
							|  |  |  |     case T_LESS: | 
					
						
							|  |  |  |     case T_LESS_EQUAL: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_GREATER_EQUAL:   return Prec::Relational; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_LESS_LESS: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_GREATER_GREATER: return Prec::ExclusiveOr; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_PLUS: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_MINUS:           return Prec::Additive; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_STAR: | 
					
						
							|  |  |  |     case T_SLASH: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_PERCENT:         return Prec::Multiplicative; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     case T_ARROW_STAR: | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     case T_DOT_STAR:        return Prec::PointerToMember; | 
					
						
							|  |  |  |     default:                return Prec::Unknown; | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:07:00 +01:00
										 |  |  | inline bool isBinaryOperator(int tokenKind) | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  | { return precedence(tokenKind, false) != Prec::Unknown; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | inline bool isRightAssociative(int tokenKind) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     const int prec = precedence(tokenKind, false); | 
					
						
							|  |  |  |     return prec == Prec::Conditional || prec == Prec::Assignment; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  | } // end of anonymous namespace
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | #ifndef CPLUSPLUS_NO_DEBUG_RULE
 | 
					
						
							|  |  |  | #  define DEBUG_THIS_RULE() DebugRule __debug_rule__(__func__)
 | 
					
						
							|  |  |  | #else
 | 
					
						
							|  |  |  | #  define DEBUG_THIS_RULE() do {} while (0)
 | 
					
						
							|  |  |  | #endif
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | #define PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, minPrecedence) { \
 | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  |     if (LA() == T_THROW) { \ | 
					
						
							|  |  |  |         if (!parseThrowExpression(node)) \ | 
					
						
							|  |  |  |             return false; \ | 
					
						
							|  |  |  |     } else if (!parseCastExpression(node)) \ | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |         return false; \ | 
					
						
							|  |  |  |     \ | 
					
						
							|  |  |  |     parseExpressionWithOperatorPrecedence(node, minPrecedence); \ | 
					
						
							|  |  |  |     return true; \ | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | Parser::Parser(TranslationUnit *unit) | 
					
						
							|  |  |  |     : _translationUnit(unit), | 
					
						
							|  |  |  |       _control(_translationUnit->control()), | 
					
						
							|  |  |  |       _pool(_translationUnit->memoryPool()), | 
					
						
							|  |  |  |       _tokenIndex(1), | 
					
						
							|  |  |  |       _templateArguments(0), | 
					
						
							|  |  |  |       _qtMocRunEnabled(false), | 
					
						
							| 
									
										
										
										
											2010-03-24 12:54:25 +01:00
										 |  |  |       _cxx0xEnabled(false), | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |       _objCEnabled(false), | 
					
						
							|  |  |  |       _inFunctionBody(false), | 
					
						
							| 
									
										
										
										
											2010-01-24 11:09:43 +01:00
										 |  |  |       _inObjCImplementationContext(false), | 
					
						
							| 
									
										
										
										
											2010-03-23 10:17:51 +01:00
										 |  |  |       _inExpressionStatement(false), | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |       _expressionDepth(0) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | Parser::~Parser() | 
					
						
							|  |  |  | { } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::qtMocRunEnabled() const | 
					
						
							|  |  |  | { return _qtMocRunEnabled; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Parser::setQtMocRunEnabled(bool onoff) | 
					
						
							|  |  |  | { _qtMocRunEnabled = onoff; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 12:11:33 +01:00
										 |  |  | bool Parser::cxx0xEnabled() const | 
					
						
							|  |  |  | { return _cxx0xEnabled; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 14:10:39 +01:00
										 |  |  | void Parser::setCxxOxEnabled(bool onoff) | 
					
						
							| 
									
										
										
										
											2010-03-23 12:11:33 +01:00
										 |  |  | { _cxx0xEnabled = onoff; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | bool Parser::objCEnabled() const | 
					
						
							|  |  |  | { return _objCEnabled; } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Parser::setObjCEnabled(bool onoff) | 
					
						
							|  |  |  | { _objCEnabled = onoff; } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::switchTemplateArguments(bool templateArguments) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     bool previousTemplateArguments = _templateArguments; | 
					
						
							|  |  |  |     _templateArguments = templateArguments; | 
					
						
							|  |  |  |     return previousTemplateArguments; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::blockErrors(bool block) | 
					
						
							|  |  |  | { return _translationUnit->blockErrors(block); } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::skipUntil(int token) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (int tk = LA()) { | 
					
						
							|  |  |  |         if (tk == token) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-03 12:19:50 +01:00
										 |  |  | void Parser::skipUntilDeclaration() | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-12-03 12:19:50 +01:00
										 |  |  |     for (; ; consumeToken()) { | 
					
						
							|  |  |  |         switch (LA()) { | 
					
						
							|  |  |  |         case T_EOF_SYMBOL: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 15:16:32 +01:00
										 |  |  |         // end of a block
 | 
					
						
							|  |  |  |         case T_RBRACE: | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-03 12:19:50 +01:00
										 |  |  |         // names
 | 
					
						
							|  |  |  |         case T_IDENTIFIER: | 
					
						
							|  |  |  |         case T_COLON_COLON: | 
					
						
							|  |  |  |         case T_TILDE: | 
					
						
							|  |  |  |         case T_OPERATOR: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // empty declaration
 | 
					
						
							|  |  |  |         case T_SEMICOLON: | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // member specification
 | 
					
						
							|  |  |  |         case T_USING: | 
					
						
							|  |  |  |         case T_TEMPLATE: | 
					
						
							|  |  |  |         case T_PUBLIC: | 
					
						
							|  |  |  |         case T_PROTECTED: | 
					
						
							|  |  |  |         case T_PRIVATE: | 
					
						
							|  |  |  |         case T_Q_SIGNALS: | 
					
						
							|  |  |  |         case T_Q_SLOTS: | 
					
						
							| 
									
										
										
										
											2010-03-11 18:09:36 +01:00
										 |  |  |         case T_Q_PROPERTY: | 
					
						
							| 
									
										
										
										
											2010-03-16 15:16:32 +01:00
										 |  |  |         case T_Q_ENUMS: | 
					
						
							|  |  |  |         case T_Q_FLAGS: | 
					
						
							| 
									
										
										
										
											2010-03-16 17:29:40 +01:00
										 |  |  |         case T_Q_INTERFACES: | 
					
						
							| 
									
										
										
										
											2010-03-16 15:16:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // Qt function specifiers
 | 
					
						
							|  |  |  |         case T_Q_SIGNAL: | 
					
						
							|  |  |  |         case T_Q_SLOT: | 
					
						
							|  |  |  |         case T_Q_INVOKABLE: | 
					
						
							| 
									
										
										
										
											2009-12-03 12:19:50 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // declarations
 | 
					
						
							|  |  |  |         case T_ENUM: | 
					
						
							|  |  |  |         case T_NAMESPACE: | 
					
						
							|  |  |  |         case T_ASM: | 
					
						
							|  |  |  |         case T_EXPORT: | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         case T_AT_CLASS: | 
					
						
							|  |  |  |         case T_AT_INTERFACE: | 
					
						
							|  |  |  |         case T_AT_PROTOCOL: | 
					
						
							|  |  |  |         case T_AT_IMPLEMENTATION: | 
					
						
							|  |  |  |         case T_AT_END: | 
					
						
							| 
									
										
										
										
											2009-12-03 12:19:50 +01:00
										 |  |  |             return; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         default: | 
					
						
							|  |  |  |             if (lookAtBuiltinTypeSpecifier() || lookAtClassKey() || | 
					
						
							|  |  |  |                 lookAtFunctionSpecifier() || lookAtStorageClassSpecifier()) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |         } // switch
 | 
					
						
							| 
									
										
										
										
											2010-01-29 22:49:55 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::skipUntilStatement() | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     while (int tk = LA()) { | 
					
						
							|  |  |  |         switch (tk) { | 
					
						
							|  |  |  |             case T_SEMICOLON: | 
					
						
							|  |  |  |             case T_LBRACE: | 
					
						
							|  |  |  |             case T_RBRACE: | 
					
						
							|  |  |  |             case T_CONST: | 
					
						
							|  |  |  |             case T_VOLATILE: | 
					
						
							|  |  |  |             case T_IDENTIFIER: | 
					
						
							|  |  |  |             case T_CASE: | 
					
						
							|  |  |  |             case T_DEFAULT: | 
					
						
							|  |  |  |             case T_IF: | 
					
						
							|  |  |  |             case T_SWITCH: | 
					
						
							|  |  |  |             case T_WHILE: | 
					
						
							|  |  |  |             case T_DO: | 
					
						
							|  |  |  |             case T_FOR: | 
					
						
							|  |  |  |             case T_BREAK: | 
					
						
							|  |  |  |             case T_CONTINUE: | 
					
						
							|  |  |  |             case T_RETURN: | 
					
						
							|  |  |  |             case T_GOTO: | 
					
						
							|  |  |  |             case T_TRY: | 
					
						
							|  |  |  |             case T_CATCH: | 
					
						
							|  |  |  |             case T_THROW: | 
					
						
							|  |  |  |             case T_CHAR: | 
					
						
							|  |  |  |             case T_WCHAR_T: | 
					
						
							|  |  |  |             case T_BOOL: | 
					
						
							|  |  |  |             case T_SHORT: | 
					
						
							|  |  |  |             case T_INT: | 
					
						
							|  |  |  |             case T_LONG: | 
					
						
							|  |  |  |             case T_SIGNED: | 
					
						
							|  |  |  |             case T_UNSIGNED: | 
					
						
							|  |  |  |             case T_FLOAT: | 
					
						
							|  |  |  |             case T_DOUBLE: | 
					
						
							|  |  |  |             case T_VOID: | 
					
						
							|  |  |  |             case T_CLASS: | 
					
						
							|  |  |  |             case T_STRUCT: | 
					
						
							|  |  |  |             case T_UNION: | 
					
						
							|  |  |  |             case T_ENUM: | 
					
						
							|  |  |  |             case T_COLON_COLON: | 
					
						
							|  |  |  |             case T_TEMPLATE: | 
					
						
							|  |  |  |             case T_USING: | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             case T_AT_SYNCHRONIZED: | 
					
						
							|  |  |  |                 if (objCEnabled()) | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:28 +01:00
										 |  |  |                     return true; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:53:05 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             default: | 
					
						
							|  |  |  |                 consumeToken(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::skip(int l, int r) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     int count = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (int tk = LA()) { | 
					
						
							|  |  |  |         if (tk == l) | 
					
						
							|  |  |  |             ++count; | 
					
						
							|  |  |  |         else if (tk == r) | 
					
						
							|  |  |  |             --count; | 
					
						
							|  |  |  |         else if (l != T_LBRACE && (tk == T_LBRACE || | 
					
						
							|  |  |  |                                    tk == T_RBRACE || | 
					
						
							|  |  |  |                                    tk == T_SEMICOLON)) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (count == 0) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Parser::match(int kind, unsigned *token) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (LA() == kind) | 
					
						
							|  |  |  |         *token = consumeToken(); | 
					
						
							|  |  |  |     else { | 
					
						
							|  |  |  |         *token = 0; | 
					
						
							|  |  |  |         _translationUnit->error(_tokenIndex, "expected token `%s' got `%s'", | 
					
						
							|  |  |  |                                 Token::name(kind), tok().spell()); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseClassOrNamespaceName(NameAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |         unsigned identifier_token = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA(2) == T_LESS && parseTemplateId(node) && LA() == T_COLON_COLON) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         rewind(identifier_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA(2) == T_COLON_COLON) { | 
					
						
							|  |  |  |             SimpleNameAST *ast = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |             ast->identifier_token = consumeToken(); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else if (LA() == T_TEMPLATE) { | 
					
						
							|  |  |  |         unsigned template_token = consumeToken(); | 
					
						
							|  |  |  |         if (parseTemplateId(node)) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         rewind(template_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTemplateId(NameAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-10-22 16:46:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     const unsigned start = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_IDENTIFIER && LA(2) == T_LESS) { | 
					
						
							|  |  |  |         TemplateIdAST *ast = new (_pool) TemplateIdAST; | 
					
						
							|  |  |  |         ast->identifier_token = consumeToken(); | 
					
						
							|  |  |  |         ast->less_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_GREATER || parseTemplateArgumentList( | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 ast->template_argument_list)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             if (LA() == T_GREATER) { | 
					
						
							|  |  |  |                 ast->greater_token = consumeToken(); | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-22 16:46:26 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  | bool Parser::parseNestedNameSpecifier(NestedNameSpecifierListAST *&node, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                                       bool /*acceptTemplateId*/) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |     NestedNameSpecifierListAST **nested_name_specifier = &node; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     NameAST *class_or_namespace_name = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |     if (parseClassOrNamespaceName(class_or_namespace_name) && LA() == T_COLON_COLON) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         unsigned scope_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |         NestedNameSpecifierAST *name = new (_pool) NestedNameSpecifierAST; | 
					
						
							|  |  |  |         name->class_or_namespace_name = class_or_namespace_name; | 
					
						
							|  |  |  |         name->scope_token = scope_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         *nested_name_specifier = new (_pool) NestedNameSpecifierListAST(name); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         nested_name_specifier = &(*nested_name_specifier)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |         while (parseClassOrNamespaceName(class_or_namespace_name) && LA() == T_COLON_COLON) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             scope_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             name = new (_pool) NestedNameSpecifierAST; | 
					
						
							|  |  |  |             name->class_or_namespace_name = class_or_namespace_name; | 
					
						
							|  |  |  |             name->scope_token = scope_token; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             *nested_name_specifier = new (_pool) NestedNameSpecifierListAST(name); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             nested_name_specifier = &(*nested_name_specifier)->next; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // ### ugly hack
 | 
					
						
							|  |  |  |         rewind(scope_token); | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  | bool Parser::parseNestedNameSpecifierOpt(NestedNameSpecifierListAST *&name, bool acceptTemplateId) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  |     if (! parseNestedNameSpecifier(name, acceptTemplateId)) | 
					
						
							|  |  |  |         rewind(start); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseName(NameAST *&node, bool acceptTemplateId) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned global_scope_token = 0; | 
					
						
							|  |  |  |     if (LA() == T_COLON_COLON) | 
					
						
							|  |  |  |         global_scope_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |     NestedNameSpecifierListAST *nested_name_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     parseNestedNameSpecifierOpt(nested_name_specifier, | 
					
						
							|  |  |  |                                 /*acceptTemplateId=*/ true); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NameAST *unqualified_name = 0; | 
					
						
							|  |  |  |     if (parseUnqualifiedName(unqualified_name, | 
					
						
							|  |  |  |                              /*acceptTemplateId=*/ acceptTemplateId || nested_name_specifier != 0)) { | 
					
						
							|  |  |  |         if (! global_scope_token && ! nested_name_specifier) { | 
					
						
							|  |  |  |             node = unqualified_name; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         QualifiedNameAST *ast = new (_pool) QualifiedNameAST; | 
					
						
							|  |  |  |         ast->global_scope_token = global_scope_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->nested_name_specifier_list = nested_name_specifier; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ast->unqualified_name = unqualified_name; | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTranslationUnit(TranslationUnitAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     TranslationUnitAST *ast = new (_pool) TranslationUnitAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     DeclarationListAST **decl = &ast->declaration_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     while (LA()) { | 
					
						
							|  |  |  |         unsigned start_declaration = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |         DeclarationAST *declaration = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (parseDeclaration(declaration)) { | 
					
						
							|  |  |  |             *decl = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             (*decl)->value = declaration; | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             decl = &(*decl)->next; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } else { | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  |             _translationUnit->error(start_declaration, "expected a declaration"); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             rewind(start_declaration + 1); | 
					
						
							|  |  |  |             skipUntilDeclaration(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-16 18:00:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         _templateArgumentList.clear(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseEmptyDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_SEMICOLON) { | 
					
						
							|  |  |  |         EmptyDeclarationAST *ast = new (_pool) EmptyDeclarationAST; | 
					
						
							|  |  |  |         ast->semicolon_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_SEMICOLON: | 
					
						
							|  |  |  |         return parseEmptyDeclaration(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_NAMESPACE: | 
					
						
							|  |  |  |         return parseNamespace(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_USING: | 
					
						
							|  |  |  |         return parseUsing(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_ASM: | 
					
						
							|  |  |  |         return parseAsmDefinition(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TEMPLATE: | 
					
						
							|  |  |  |     case T_EXPORT: | 
					
						
							|  |  |  |         return parseTemplateDeclaration(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     // ObjcC++
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_CLASS: | 
					
						
							|  |  |  |         return parseObjCClassForwardDeclaration(node); | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_INTERFACE: | 
					
						
							|  |  |  |         return parseObjCInterface(node); | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_PROTOCOL: | 
					
						
							|  |  |  |         return parseObjCProtocol(node); | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_IMPLEMENTATION: | 
					
						
							|  |  |  |         return parseObjCImplementation(node); | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-17 14:35:48 +01:00
										 |  |  |     case T_Q_DECLARE_INTERFACE: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         unsigned lparen_token = 0; | 
					
						
							|  |  |  |         match(T_LPAREN, &lparen_token); | 
					
						
							|  |  |  |         NameAST *name = 0; | 
					
						
							|  |  |  |         parseName(name); | 
					
						
							|  |  |  |         unsigned comma_token = 0; | 
					
						
							|  |  |  |         match(T_COMMA, &comma_token); | 
					
						
							|  |  |  |         unsigned string_literal = 0; | 
					
						
							|  |  |  |         match(T_STRING_LITERAL, &string_literal); | 
					
						
							|  |  |  |         unsigned rparen_token = 0; | 
					
						
							|  |  |  |         match(T_RPAREN, &rparen_token); | 
					
						
							|  |  |  |     }   return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_END: | 
					
						
							|  |  |  |         // TODO: should this be done here, or higher-up?
 | 
					
						
							|  |  |  |         _translationUnit->error(cursor(), "skip stray token `%s'", tok().spell()); | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         break; | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     default: { | 
					
						
							|  |  |  |         if (_objCEnabled && LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |             const unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             SpecifierListAST *attributes = 0, **attr = &attributes; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |             while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |                 attr = &(*attr)->next; | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             if (LA() == T_AT_INTERFACE) | 
					
						
							|  |  |  |                 return parseObjCInterface(node, attributes); | 
					
						
							|  |  |  |             else if (LA() == T_AT_PROTOCOL) | 
					
						
							|  |  |  |                 return parseObjCProtocol(node, attributes); | 
					
						
							|  |  |  |             else if (LA() == T_AT_PROPERTY) | 
					
						
							|  |  |  |                 return parseObjCPropertyDeclaration(node, attributes); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |             rewind(start); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (LA() == T_EXTERN && LA(2) == T_TEMPLATE) | 
					
						
							|  |  |  |             return parseTemplateDeclaration(node); | 
					
						
							|  |  |  |         else if (LA() == T_EXTERN && LA(2) == T_STRING_LITERAL) | 
					
						
							|  |  |  |             return parseLinkageSpecification(node); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             return parseSimpleDeclaration(node); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     }   break; // default
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } // end switch
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLinkageSpecification(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_EXTERN && LA(2) == T_STRING_LITERAL) { | 
					
						
							|  |  |  |         LinkageSpecificationAST *ast = new (_pool) LinkageSpecificationAST; | 
					
						
							|  |  |  |         ast->extern_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->extern_type_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LBRACE) | 
					
						
							|  |  |  |             parseLinkageBody(ast->declaration); | 
					
						
							|  |  |  |         else | 
					
						
							|  |  |  |             parseDeclaration(ast->declaration); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLinkageBody(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LBRACE) { | 
					
						
							|  |  |  |         LinkageBodyAST *ast = new (_pool) LinkageBodyAST; | 
					
						
							|  |  |  |         ast->lbrace_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         DeclarationListAST **declaration_ptr = &ast->declaration_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while (int tk = LA()) { | 
					
						
							|  |  |  |             if (tk == T_RBRACE) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             unsigned start_declaration = cursor(); | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |             if (parseDeclaration(declaration)) { | 
					
						
							|  |  |  |                 *declaration_ptr = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |                 (*declaration_ptr)->value = declaration; | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |                 declaration_ptr = &(*declaration_ptr)->next; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  |                 _translationUnit->error(start_declaration, "expected a declaration"); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 rewind(start_declaration + 1); | 
					
						
							|  |  |  |                 skipUntilDeclaration(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-11-16 18:00:20 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             _templateArgumentList.clear(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         match(T_RBRACE, &ast->rbrace_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // ### rename parseNamespaceAliarOrDeclaration?
 | 
					
						
							|  |  |  | bool Parser::parseNamespace(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() != T_NAMESPACE) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned namespace_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_IDENTIFIER && LA(2) == T_EQUAL) { | 
					
						
							|  |  |  |         NamespaceAliasDefinitionAST *ast = | 
					
						
							|  |  |  |                 new (_pool) NamespaceAliasDefinitionAST; | 
					
						
							|  |  |  |         ast->namespace_token = namespace_token; | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->namespace_name_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |         parseName(ast->name); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NamespaceAST *ast = new (_pool) NamespaceAST; | 
					
						
							|  |  |  |     ast->namespace_token = namespace_token; | 
					
						
							|  |  |  |     if (LA() == T_IDENTIFIER) | 
					
						
							|  |  |  |         ast->identifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     SpecifierListAST **attr_ptr = &ast->attribute_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |         parseAttributeSpecifier(*attr_ptr); | 
					
						
							|  |  |  |         attr_ptr = &(*attr_ptr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (LA() == T_LBRACE) | 
					
						
							|  |  |  |         parseLinkageBody(ast->linkage_body); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseUsing(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() != T_USING) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA(2) == T_NAMESPACE) | 
					
						
							|  |  |  |         return parseUsingDirective(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     UsingAST *ast = new (_pool) UsingAST; | 
					
						
							|  |  |  |     ast->using_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_TYPENAME) | 
					
						
							|  |  |  |         ast->typename_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parseName(ast->name); | 
					
						
							|  |  |  |     match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseUsingDirective(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_USING && LA(2) == T_NAMESPACE) { | 
					
						
							|  |  |  |         UsingDirectiveAST *ast = new (_pool) UsingDirectiveAST; | 
					
						
							|  |  |  |         ast->using_token = consumeToken(); | 
					
						
							|  |  |  |         ast->namespace_token = consumeToken(); | 
					
						
							|  |  |  |         if (! parseName(ast->name)) | 
					
						
							|  |  |  |             _translationUnit->warning(cursor(), "expected `namespace name' before `%s'", | 
					
						
							|  |  |  |                                       tok().spell()); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseConversionFunctionId(NameAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() != T_OPERATOR) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     unsigned operator_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! parseTypeSpecifier(type_specifier)) { | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  |     PtrOperatorListAST *ptr_operators = 0, **ptr_operators_tail = &ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (parsePtrOperator(*ptr_operators_tail)) | 
					
						
							|  |  |  |         ptr_operators_tail = &(*ptr_operators_tail)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ConversionFunctionIdAST *ast = new (_pool) ConversionFunctionIdAST; | 
					
						
							|  |  |  |     ast->operator_token = operator_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     ast->type_specifier_list = type_specifier; | 
					
						
							|  |  |  |     ast->ptr_operator_list = ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseOperatorFunctionId(NameAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() != T_OPERATOR) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     unsigned operator_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     OperatorAST *op = 0; | 
					
						
							|  |  |  |     if (! parseOperator(op)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     OperatorFunctionIdAST *ast = new (_pool) OperatorFunctionIdAST; | 
					
						
							|  |  |  |     ast->operator_token = operator_token; | 
					
						
							|  |  |  |     ast->op = op; | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  | Parser::TemplateArgumentListEntry *Parser::templateArgumentListEntry(unsigned tokenIndex) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-11-16 18:00:20 +01:00
										 |  |  |     std::map<unsigned, TemplateArgumentListEntry>::iterator it =_templateArgumentList.find(tokenIndex); | 
					
						
							|  |  |  |     if (it != _templateArgumentList.end()) | 
					
						
							|  |  |  |         return &it->second; | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseTemplateArgumentList(TemplateArgumentListAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  |     if (TemplateArgumentListEntry *entry = templateArgumentListEntry(cursor())) { | 
					
						
							|  |  |  |         rewind(entry->cursor); | 
					
						
							| 
									
										
										
										
											2009-10-23 14:31:55 +02:00
										 |  |  |         node = entry->ast; | 
					
						
							|  |  |  |         return entry->ast != 0; | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     TemplateArgumentListAST **template_argument_ptr = &node; | 
					
						
							|  |  |  |     ExpressionAST *template_argument = 0; | 
					
						
							|  |  |  |     if (parseTemplateArgument(template_argument)) { | 
					
						
							|  |  |  |         *template_argument_ptr = new (_pool) TemplateArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:37:46 +01:00
										 |  |  |         (*template_argument_ptr)->value = template_argument; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         template_argument_ptr = &(*template_argument_ptr)->next; | 
					
						
							| 
									
										
										
										
											2010-03-23 14:10:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             consumeToken(); // ### store this token in the AST
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (parseTemplateArgument(template_argument)) { | 
					
						
							|  |  |  |                 *template_argument_ptr = new (_pool) TemplateArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:37:46 +01:00
										 |  |  |                 (*template_argument_ptr)->value = template_argument; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 template_argument_ptr = &(*template_argument_ptr)->next; | 
					
						
							| 
									
										
										
										
											2010-03-23 14:10:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |                     consumeToken(); // ### store this token in the AST
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-16 18:00:20 +01:00
										 |  |  |         _templateArgumentList.insert(std::make_pair(cursor(), TemplateArgumentListEntry(start, cursor(), node))); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-16 18:00:20 +01:00
										 |  |  |     _templateArgumentList.insert(std::make_pair(cursor(), TemplateArgumentListEntry(start, cursor(), 0))); | 
					
						
							| 
									
										
										
										
											2009-10-23 11:21:36 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAsmDefinition(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-05 11:37:51 +01:00
										 |  |  |     if (LA() != T_ASM) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AsmDefinitionAST *ast = new (_pool) AsmDefinitionAST; | 
					
						
							|  |  |  |     ast->asm_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_VOLATILE) | 
					
						
							|  |  |  |         ast->volatile_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |     unsigned string_literal_token = 0; | 
					
						
							|  |  |  |     match(T_STRING_LITERAL, &string_literal_token); | 
					
						
							|  |  |  |     while (LA() == T_STRING_LITERAL) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (LA() == T_COLON) { | 
					
						
							|  |  |  |         consumeToken(); // skip T_COLON
 | 
					
						
							|  |  |  |         parseAsmOperandList(); | 
					
						
							|  |  |  |         if (LA() == T_COLON) { | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  |             parseAsmOperandList(); | 
					
						
							|  |  |  |             if (LA() == T_COLON) { | 
					
						
							|  |  |  |                 consumeToken(); | 
					
						
							|  |  |  |                 parseAsmClobberList(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if (LA() == T_COLON_COLON) { | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  |             parseAsmClobberList(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else if (LA() == T_COLON_COLON) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         parseAsmClobberList(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAsmOperandList() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  |     if (LA() != T_STRING_LITERAL) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 11:37:51 +01:00
										 |  |  |     if (parseAsmOperand()) { | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  |             parseAsmOperand(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 11:37:51 +01:00
										 |  |  | bool Parser::parseAsmOperand() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-05 11:37:51 +01:00
										 |  |  |     unsigned string_literal_token = 0; | 
					
						
							|  |  |  |     match(T_STRING_LITERAL, &string_literal_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |         /*unsigned lbracket_token = */ consumeToken(); | 
					
						
							|  |  |  |         match(T_STRING_LITERAL, &string_literal_token); | 
					
						
							|  |  |  |         unsigned rbracket_token = 0; | 
					
						
							|  |  |  |         match(T_RBRACKET, &rbracket_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned lparen_token = 0, rparen_token = 0; | 
					
						
							|  |  |  |     match(T_LPAREN, &lparen_token); | 
					
						
							|  |  |  |     ExpressionAST *expression = 0; | 
					
						
							|  |  |  |     parseExpression(expression); | 
					
						
							|  |  |  |     match(T_RPAREN, &rparen_token); | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAsmClobberList() | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-05 11:37:51 +01:00
										 |  |  |     if (LA() != T_STRING_LITERAL) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned string_literal_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (LA() == T_COMMA) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         match(T_STRING_LITERAL, &string_literal_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseTemplateDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! (LA(1) == T_TEMPLATE || ((LA(1) == T_EXPORT || LA(1) == T_EXTERN) | 
					
						
							|  |  |  |             && LA(2) == T_TEMPLATE))) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TemplateDeclarationAST *ast = new (_pool) TemplateDeclarationAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_EXPORT || LA() == T_EXPORT) | 
					
						
							|  |  |  |         ast->export_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ast->template_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_LESS) { | 
					
						
							|  |  |  |         ast->less_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (LA() == T_GREATER || parseTemplateParameterList(ast->template_parameter_list)) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             match(T_GREATER, &ast->greater_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parseDeclaration(ast->declaration); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseOperator(OperatorAST *&node) // ### FIXME
 | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     OperatorAST *ast = new (_pool) OperatorAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_NEW: | 
					
						
							|  |  |  |     case T_DELETE: { | 
					
						
							|  |  |  |         ast->op_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |             ast->open_token = consumeToken(); | 
					
						
							|  |  |  |             match(T_RBRACKET, &ast->close_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_PLUS: | 
					
						
							|  |  |  |     case T_MINUS: | 
					
						
							|  |  |  |     case T_STAR: | 
					
						
							|  |  |  |     case T_SLASH: | 
					
						
							|  |  |  |     case T_PERCENT: | 
					
						
							|  |  |  |     case T_CARET: | 
					
						
							|  |  |  |     case T_AMPER: | 
					
						
							|  |  |  |     case T_PIPE: | 
					
						
							|  |  |  |     case T_TILDE: | 
					
						
							|  |  |  |     case T_EXCLAIM: | 
					
						
							|  |  |  |     case T_LESS: | 
					
						
							|  |  |  |     case T_GREATER: | 
					
						
							|  |  |  |     case T_COMMA: | 
					
						
							|  |  |  |     case T_AMPER_EQUAL: | 
					
						
							|  |  |  |     case T_CARET_EQUAL: | 
					
						
							|  |  |  |     case T_SLASH_EQUAL: | 
					
						
							|  |  |  |     case T_EQUAL: | 
					
						
							|  |  |  |     case T_EQUAL_EQUAL: | 
					
						
							|  |  |  |     case T_EXCLAIM_EQUAL: | 
					
						
							|  |  |  |     case T_GREATER_EQUAL: | 
					
						
							|  |  |  |     case T_GREATER_GREATER_EQUAL: | 
					
						
							|  |  |  |     case T_LESS_EQUAL: | 
					
						
							|  |  |  |     case T_LESS_LESS_EQUAL: | 
					
						
							|  |  |  |     case T_MINUS_EQUAL: | 
					
						
							|  |  |  |     case T_PERCENT_EQUAL: | 
					
						
							|  |  |  |     case T_PIPE_EQUAL: | 
					
						
							|  |  |  |     case T_PLUS_EQUAL: | 
					
						
							|  |  |  |     case T_STAR_EQUAL: | 
					
						
							|  |  |  |     case T_TILDE_EQUAL: | 
					
						
							|  |  |  |     case T_LESS_LESS: | 
					
						
							|  |  |  |     case T_GREATER_GREATER: | 
					
						
							|  |  |  |     case T_AMPER_AMPER: | 
					
						
							|  |  |  |     case T_PIPE_PIPE: | 
					
						
							|  |  |  |     case T_PLUS_PLUS: | 
					
						
							|  |  |  |     case T_MINUS_MINUS: | 
					
						
							|  |  |  |     case T_ARROW_STAR: | 
					
						
							|  |  |  |     case T_DOT_STAR: | 
					
						
							|  |  |  |     case T_ARROW: | 
					
						
							|  |  |  |         ast->op_token = consumeToken(); | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         if (LA() == T_LPAREN && LA(2) == T_RPAREN) { | 
					
						
							|  |  |  |             ast->op_token = ast->open_token = consumeToken(); | 
					
						
							|  |  |  |             ast->close_token = consumeToken(); | 
					
						
							|  |  |  |         } else if (LA() == T_LBRACKET && LA(2) == T_RBRACKET) { | 
					
						
							|  |  |  |             ast->op_token = ast->open_token = consumeToken(); | 
					
						
							|  |  |  |             ast->close_token = consumeToken(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseCvQualifiers(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     SpecifierListAST **ast = &node; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (*ast) | 
					
						
							|  |  |  |         ast = &(*ast)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (int tk = LA()) { | 
					
						
							|  |  |  |         if (tk == T_CONST || tk == T_VOLATILE) { | 
					
						
							|  |  |  |             SimpleSpecifierAST *spec = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |             spec->specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             *ast = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast = &(*ast)->next; | 
					
						
							| 
									
										
										
										
											2009-01-14 15:16:02 +01:00
										 |  |  |         } else if(LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |             parseAttributeSpecifier(*ast); | 
					
						
							|  |  |  |             ast = &(*ast)->next; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } else { | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return start != cursor(); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  | bool Parser::parsePtrOperator(PtrOperatorListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-03-23 12:11:33 +01:00
										 |  |  |     if (LA() == T_AMPER || (_cxx0xEnabled && LA() == T_AMPER_AMPER)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ReferenceAST *ast = new (_pool) ReferenceAST; | 
					
						
							| 
									
										
										
										
											2010-03-23 12:11:33 +01:00
										 |  |  |         ast->reference_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  |         node = new (_pool) PtrOperatorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } else if (LA() == T_STAR) { | 
					
						
							|  |  |  |         PointerAST *ast = new (_pool) PointerAST; | 
					
						
							|  |  |  |         ast->star_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         parseCvQualifiers(ast->cv_qualifier_list); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  |         node = new (_pool) PtrOperatorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } else if (LA() == T_COLON_COLON || LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |         unsigned scope_or_identifier_token = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         unsigned global_scope_token = 0; | 
					
						
							|  |  |  |         if (LA() == T_COLON_COLON) | 
					
						
							|  |  |  |             global_scope_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:12:04 +01:00
										 |  |  |         NestedNameSpecifierListAST *nested_name_specifiers = 0; | 
					
						
							|  |  |  |         bool has_nested_name_specifier = parseNestedNameSpecifier(nested_name_specifiers, true); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (has_nested_name_specifier && LA() == T_STAR) { | 
					
						
							|  |  |  |             PointerToMemberAST *ast = new (_pool) PointerToMemberAST; | 
					
						
							|  |  |  |             ast->global_scope_token = global_scope_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->nested_name_specifier_list = nested_name_specifiers; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->star_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             parseCvQualifiers(ast->cv_qualifier_list); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  |             node = new (_pool) PtrOperatorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         rewind(scope_or_identifier_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTemplateArgument(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2010-03-23 14:10:05 +01:00
										 |  |  |     if (parseTypeId(node)) { | 
					
						
							|  |  |  |         int index = 1; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             index = 2; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA(index) == T_COMMA || LA(index) == T_GREATER) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  |     bool previousTemplateArguments = switchTemplateArguments(true); | 
					
						
							|  |  |  |     bool parsed = parseLogicalOrExpression(node); | 
					
						
							|  |  |  |     (void) switchTemplateArguments(previousTemplateArguments); | 
					
						
							|  |  |  |     return parsed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseDeclSpecifierSeq(SpecifierListAST *&decl_specifier_seq, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                                    bool onlyTypeSpecifiers, | 
					
						
							|  |  |  |                                    bool simplified) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     bool has_type_specifier = false; | 
					
						
							|  |  |  |     NameAST *named_type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST **decl_specifier_seq_ptr = &decl_specifier_seq; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     for (;;) { | 
					
						
							|  |  |  |         if (lookAtCVQualifier()) { | 
					
						
							|  |  |  |             SimpleSpecifierAST *spec = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |             spec->specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             *decl_specifier_seq_ptr = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |         } else if (! onlyTypeSpecifiers && lookAtStorageClassSpecifier()) { | 
					
						
							|  |  |  |             SimpleSpecifierAST *spec = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |             spec->specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             *decl_specifier_seq_ptr = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |         } else if (! named_type_specifier && lookAtBuiltinTypeSpecifier()) { | 
					
						
							|  |  |  |             parseBuiltinTypeSpecifier(*decl_specifier_seq_ptr); | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else if (! has_type_specifier && (LA() == T_COLON_COLON || | 
					
						
							|  |  |  |                                             LA() == T_IDENTIFIER)) { | 
					
						
							|  |  |  |             if (! parseName(named_type_specifier)) | 
					
						
							|  |  |  |                 return false; | 
					
						
							|  |  |  |             NamedTypeSpecifierAST *spec = new (_pool) NamedTypeSpecifierAST; | 
					
						
							|  |  |  |             spec->name = named_type_specifier; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             *decl_specifier_seq_ptr = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else if (! simplified && ! has_type_specifier && (LA() == T_TYPENAME || | 
					
						
							|  |  |  |                                                             LA() == T_ENUM     || | 
					
						
							|  |  |  |                                                             lookAtClassKey())) { | 
					
						
							|  |  |  |             unsigned startOfElaboratedTypeSpecifier = cursor(); | 
					
						
							|  |  |  |             if (! parseElaboratedTypeSpecifier(*decl_specifier_seq_ptr)) { | 
					
						
							|  |  |  |                 _translationUnit->error(startOfElaboratedTypeSpecifier, | 
					
						
							|  |  |  |                                         "expected an elaborated type specifier"); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return decl_specifier_seq != 0; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseDeclaratorOrAbstractDeclarator(DeclaratorAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  |     bool blocked = blockErrors(true); | 
					
						
							|  |  |  |     if (parseDeclarator(node)) { | 
					
						
							|  |  |  |         blockErrors(blocked); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     blockErrors(blocked); | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  |     return parseAbstractDeclarator(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCoreDeclarator(DeclaratorAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *attributes = 0; | 
					
						
							|  |  |  |     SpecifierListAST **attribute_ptr = &attributes; | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  |     while (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |         parseAttributeSpecifier(*attribute_ptr); | 
					
						
							|  |  |  |         attribute_ptr = &(*attribute_ptr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  |     PtrOperatorListAST *ptr_operators = 0, **ptr_operators_tail = &ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (parsePtrOperator(*ptr_operators_tail)) | 
					
						
							|  |  |  |         ptr_operators_tail = &(*ptr_operators_tail)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_COLON_COLON || LA() == T_IDENTIFIER || LA() == T_TILDE | 
					
						
							|  |  |  |             || LA() == T_OPERATOR) { | 
					
						
							|  |  |  |         NameAST *name = 0; | 
					
						
							|  |  |  |         if (parseName(name)) { | 
					
						
							|  |  |  |             DeclaratorIdAST *declarator_id = new (_pool) DeclaratorIdAST; | 
					
						
							|  |  |  |             declarator_id->name = name; | 
					
						
							|  |  |  |             DeclaratorAST *ast = new (_pool) DeclaratorAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->attribute_list = attributes; | 
					
						
							|  |  |  |             ast->ptr_operator_list = ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->core_declarator = declarator_id; | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else if (LA() == T_LPAREN) { | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  |         if (attributes) | 
					
						
							|  |  |  |             _translationUnit->warning(attributes->firstToken(), "unexpected attribtues"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         DeclaratorAST *declarator = 0; | 
					
						
							|  |  |  |         if (parseDeclarator(declarator) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             NestedDeclaratorAST *nested_declarator = new (_pool) NestedDeclaratorAST; | 
					
						
							|  |  |  |             nested_declarator->lparen_token = lparen_token; | 
					
						
							|  |  |  |             nested_declarator->declarator = declarator; | 
					
						
							|  |  |  |             nested_declarator->rparen_token = consumeToken(); | 
					
						
							|  |  |  |             DeclaratorAST *ast = new (_pool) DeclaratorAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->ptr_operator_list = ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->core_declarator = nested_declarator; | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-06-18 17:48:55 +02:00
										 |  |  |     rewind(start); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  | bool Parser::parseDeclarator(DeclaratorAST *&node, bool stopAtCppInitializer) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! parseCoreDeclarator(node)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     PostfixDeclaratorListAST **postfix_ptr = &node->postfix_declarator_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							|  |  |  |         unsigned startOfPostDeclarator = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  |             if (stopAtCppInitializer) { | 
					
						
							|  |  |  |                 unsigned lparen_token = cursor(); | 
					
						
							|  |  |  |                 ExpressionAST *initializer = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 bool blocked = blockErrors(true); | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |                 if (parseInitializer(initializer, &node->equals_token)) { | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  |                     if (NestedExpressionAST *expr = initializer->asNestedExpression()) { | 
					
						
							|  |  |  |                         if (expr->expression && expr->rparen_token && (LA() == T_COMMA || LA() == T_SEMICOLON)) { | 
					
						
							|  |  |  |                             rewind(lparen_token); | 
					
						
							| 
									
										
										
										
											2009-03-17 14:27:28 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                             // check for ambiguous declarators.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             consumeToken(); | 
					
						
							|  |  |  |                             ParameterDeclarationClauseAST *parameter_declaration_clause = 0; | 
					
						
							|  |  |  |                             if (parseParameterDeclarationClause(parameter_declaration_clause) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |                                 unsigned rparen_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                                 FunctionDeclaratorAST *ast = new (_pool) FunctionDeclaratorAST; | 
					
						
							|  |  |  |                                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                                 ast->parameters = parameter_declaration_clause; | 
					
						
							|  |  |  |                                 ast->as_cpp_initializer = initializer; | 
					
						
							|  |  |  |                                 ast->rparen_token = rparen_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |                                 *postfix_ptr = new (_pool) PostfixDeclaratorListAST(ast); | 
					
						
							| 
									
										
										
										
											2009-03-17 14:27:28 +01:00
										 |  |  |                                 postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                                 blockErrors(blocked); | 
					
						
							|  |  |  |                                 return true; | 
					
						
							|  |  |  |                             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  |                             blockErrors(blocked); | 
					
						
							| 
									
										
										
										
											2009-03-17 14:27:28 +01:00
										 |  |  |                             rewind(lparen_token); | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  |                             return true; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 blockErrors(blocked); | 
					
						
							|  |  |  |                 rewind(lparen_token); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             FunctionDeclaratorAST *ast = new (_pool) FunctionDeclaratorAST; | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |             parseParameterDeclarationClause(ast->parameters); | 
					
						
							|  |  |  |             if (LA() != T_RPAREN) { | 
					
						
							|  |  |  |                 rewind(startOfPostDeclarator); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             ast->rparen_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             parseCvQualifiers(ast->cv_qualifier_list); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             parseExceptionSpecification(ast->exception_specification); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |             *postfix_ptr = new (_pool) PostfixDeclaratorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |         } else if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |             ArrayDeclaratorAST *ast = new (_pool) ArrayDeclaratorAST; | 
					
						
							|  |  |  |             ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  |             if (LA() == T_RBRACKET || parseConstantExpression(ast->expression)) { | 
					
						
							|  |  |  |                 match(T_RBRACKET, &ast->rbracket_token); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |             *postfix_ptr = new (_pool) PostfixDeclaratorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-04 13:47:06 +01:00
										 |  |  |     if (LA() == T___ASM__ && LA(2) == T_LPAREN) { // ### store the asm specifier in the AST
 | 
					
						
							|  |  |  |         consumeToken(); // skip __asm__
 | 
					
						
							|  |  |  |         consumeToken(); // skip T_LPAREN
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (skipUntil(T_RPAREN)) | 
					
						
							|  |  |  |             consumeToken(); // skip T_RPAREN
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     SpecifierListAST **spec_ptr = &node->post_attribute_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |         parseAttributeSpecifier(*spec_ptr); | 
					
						
							|  |  |  |         spec_ptr = &(*spec_ptr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAbstractCoreDeclarator(DeclaratorAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     PtrOperatorListAST *ptr_operators = 0, **ptr_operators_tail = &ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (parsePtrOperator(*ptr_operators_tail)) | 
					
						
							|  |  |  |         ptr_operators_tail = &(*ptr_operators_tail)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned after_ptr_operators = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         DeclaratorAST *declarator = 0; | 
					
						
							|  |  |  |         if (parseAbstractDeclarator(declarator) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             NestedDeclaratorAST *nested_declarator = new (_pool) NestedDeclaratorAST; | 
					
						
							|  |  |  |             nested_declarator->lparen_token = lparen_token; | 
					
						
							|  |  |  |             nested_declarator->declarator = declarator; | 
					
						
							|  |  |  |             nested_declarator->rparen_token = consumeToken(); | 
					
						
							|  |  |  |             DeclaratorAST *ast = new (_pool) DeclaratorAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->ptr_operator_list = ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->core_declarator = nested_declarator; | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rewind(after_ptr_operators); | 
					
						
							|  |  |  |     if (ptr_operators) { | 
					
						
							|  |  |  |         DeclaratorAST *ast = new (_pool) DeclaratorAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->ptr_operator_list = ptr_operators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAbstractDeclarator(DeclaratorAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! parseAbstractCoreDeclarator(node)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |     PostfixDeclaratorListAST *postfix_declarators = 0, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         **postfix_ptr = &postfix_declarators; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |             FunctionDeclaratorAST *ast = new (_pool) FunctionDeclaratorAST; | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |             if (LA() == T_RPAREN || parseParameterDeclarationClause(ast->parameters)) { | 
					
						
							|  |  |  |                 if (LA() == T_RPAREN) | 
					
						
							|  |  |  |                     ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             parseCvQualifiers(ast->cv_qualifier_list); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             parseExceptionSpecification(ast->exception_specification); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |             *postfix_ptr = new (_pool) PostfixDeclaratorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |         } else if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |             ArrayDeclaratorAST *ast = new (_pool) ArrayDeclaratorAST; | 
					
						
							|  |  |  |             ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  |             if (LA() == T_RBRACKET || parseConstantExpression(ast->expression)) { | 
					
						
							|  |  |  |                 if (LA() == T_RBRACKET) | 
					
						
							|  |  |  |                     ast->rbracket_token = consumeToken(); | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-11-10 14:33:51 +01:00
										 |  |  |             *postfix_ptr = new (_pool) PostfixDeclaratorListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (postfix_declarators) { | 
					
						
							|  |  |  |         if (! node) | 
					
						
							|  |  |  |             node = new (_pool) DeclaratorAST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         node->postfix_declarator_list = postfix_declarators; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseEnumSpecifier(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_ENUM) { | 
					
						
							|  |  |  |         unsigned enum_token = consumeToken(); | 
					
						
							|  |  |  |         NameAST *name = 0; | 
					
						
							|  |  |  |         parseName(name); | 
					
						
							|  |  |  |         if (LA() == T_LBRACE) { | 
					
						
							|  |  |  |             EnumSpecifierAST *ast = new (_pool) EnumSpecifierAST; | 
					
						
							|  |  |  |             ast->enum_token = enum_token; | 
					
						
							|  |  |  |             ast->name = name; | 
					
						
							|  |  |  |             ast->lbrace_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |             unsigned comma_token = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             EnumeratorListAST **enumerator_ptr = &ast->enumerator_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             while (int tk = LA()) { | 
					
						
							|  |  |  |                 if (tk == T_RBRACE) | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (LA() != T_IDENTIFIER) { | 
					
						
							|  |  |  |                     _translationUnit->error(cursor(), "expected identifier before '%s'", tok().spell()); | 
					
						
							|  |  |  |                     skipUntil(T_IDENTIFIER); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |                 if (parseEnumerator(*enumerator_ptr)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                     enumerator_ptr = &(*enumerator_ptr)->next; | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |                 if (LA() != T_RBRACE) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                     match(T_COMMA, &comma_token); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             match(T_RBRACE, &ast->rbrace_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  | bool Parser::parseTemplateParameterList(DeclarationListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |     DeclarationListAST **template_parameter_ptr = &node; | 
					
						
							|  |  |  |     DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |     if (parseTemplateParameter(declaration)) { | 
					
						
							|  |  |  |         *template_parameter_ptr = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |         (*template_parameter_ptr)->value = declaration; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         template_parameter_ptr = &(*template_parameter_ptr)->next; | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |             consumeToken(); // XXX Store this token somewhere
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             declaration = 0; | 
					
						
							|  |  |  |             if (parseTemplateParameter(declaration)) { | 
					
						
							|  |  |  |                 *template_parameter_ptr = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |                 (*template_parameter_ptr)->value = declaration; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 template_parameter_ptr = &(*template_parameter_ptr)->next; | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTemplateParameter(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseTypeParameter(node)) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     bool previousTemplateArguments = switchTemplateArguments(true); | 
					
						
							|  |  |  |     bool parsed = parseParameterDeclaration(node); | 
					
						
							|  |  |  |     (void) switchTemplateArguments(previousTemplateArguments); | 
					
						
							|  |  |  |     return parsed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTypenameTypeParameter(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_CLASS || LA() == T_TYPENAME) { | 
					
						
							|  |  |  |         TypenameTypeParameterAST *ast = new (_pool) TypenameTypeParameterAST; | 
					
						
							|  |  |  |         ast->classkey_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-03-23 14:03:05 +01:00
										 |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             ast->dot_dot_dot_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         parseName(ast->name); | 
					
						
							|  |  |  |         if (LA() == T_EQUAL) { | 
					
						
							|  |  |  |             ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |             parseTypeId(ast->type_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTemplateTypeParameter(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TEMPLATE) { | 
					
						
							|  |  |  |         TemplateTypeParameterAST *ast = new (_pool) TemplateTypeParameterAST; | 
					
						
							|  |  |  |         ast->template_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_LESS) | 
					
						
							|  |  |  |             ast->less_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         parseTemplateParameterList(ast->template_parameter_list); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (LA() == T_GREATER) | 
					
						
							|  |  |  |             ast->greater_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_CLASS) | 
					
						
							|  |  |  |             ast->class_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-03-23 14:03:05 +01:00
										 |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             ast->dot_dot_dot_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // parse optional name
 | 
					
						
							|  |  |  |         parseName(ast->name); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_EQUAL) { | 
					
						
							|  |  |  |             ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |             parseTypeId(ast->type_id); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTypeParameter(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_CLASS || LA() == T_TYPENAME) | 
					
						
							|  |  |  |         return parseTypenameTypeParameter(node); | 
					
						
							|  |  |  |     else if (LA() == T_TEMPLATE) | 
					
						
							|  |  |  |         return parseTemplateTypeParameter(node); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTypeId(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseTypeSpecifier(type_specifier)) { | 
					
						
							|  |  |  |         TypeIdAST *ast = new (_pool) TypeIdAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->type_specifier_list = type_specifier; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         parseAbstractDeclarator(ast->declarator); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseParameterDeclarationClause(ParameterDeclarationClauseAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  |     if (LA() == T_RPAREN) | 
					
						
							|  |  |  |         return true; // nothing to do
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |     DeclarationListAST *parameter_declarations = 0; | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned dot_dot_dot_token = 0; | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  |     if (LA() == T_DOT_DOT_DOT) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         dot_dot_dot_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  |     else { | 
					
						
							|  |  |  |         parseParameterDeclarationList(parameter_declarations); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_DOT_DOT_DOT) { | 
					
						
							|  |  |  |             dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  |         } else if (LA() == T_COMMA && LA(2) == T_DOT_DOT_DOT) { | 
					
						
							|  |  |  |             consumeToken(); // skip comma
 | 
					
						
							|  |  |  |             dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parameter_declarations || dot_dot_dot_token) { | 
					
						
							|  |  |  |         ParameterDeclarationClauseAST *ast = new (_pool) ParameterDeclarationClauseAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->parameter_declaration_list = parameter_declarations; | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  |         ast->dot_dot_dot_token = dot_dot_dot_token; | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  | bool Parser::parseParameterDeclarationList(DeclarationListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-13 14:28:29 +02:00
										 |  |  |     if (LA() == T_DOT_DOT_DOT || (LA() == T_COMMA && LA(2) == T_DOT_DOT_DOT)) | 
					
						
							|  |  |  |         return false; // nothing to do.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |     DeclarationListAST **parameter_declaration_ptr = &node; | 
					
						
							|  |  |  |     DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |     if (parseParameterDeclaration(declaration)) { | 
					
						
							|  |  |  |         *parameter_declaration_ptr = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |         (*parameter_declaration_ptr)->value = declaration; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         parameter_declaration_ptr = &(*parameter_declaration_ptr)->next; | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             declaration = 0; | 
					
						
							|  |  |  |             if (parseParameterDeclaration(declaration)) { | 
					
						
							|  |  |  |                 *parameter_declaration_ptr = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |                 (*parameter_declaration_ptr)->value = declaration; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 parameter_declaration_ptr = &(*parameter_declaration_ptr)->next; | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseParameterDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *decl_specifier_seq = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseDeclSpecifierSeq(decl_specifier_seq)) { | 
					
						
							|  |  |  |         ParameterDeclarationAST *ast = new (_pool) ParameterDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->type_specifier_list = decl_specifier_seq; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         parseDeclaratorOrAbstractDeclarator(ast->declarator); | 
					
						
							|  |  |  |         if (LA() == T_EQUAL) { | 
					
						
							|  |  |  |             ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |             parseLogicalOrExpression(ast->expression); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseClassSpecifier(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! lookAtClassKey()) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned classkey_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *attributes = 0, **attr_ptr = &attributes; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     while (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |         parseAttributeSpecifier(*attr_ptr); | 
					
						
							|  |  |  |         attr_ptr = &(*attr_ptr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA(1) == T_IDENTIFIER && LA(2) == T_IDENTIFIER) { | 
					
						
							|  |  |  |         _translationUnit->warning(cursor(), "skip identifier `%s'", | 
					
						
							|  |  |  |                                   tok().spell()); | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NameAST *name = 0; | 
					
						
							|  |  |  |     parseName(name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-03 14:47:02 +01:00
										 |  |  |     bool parsed = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const bool previousInFunctionBody = _inFunctionBody; | 
					
						
							|  |  |  |     _inFunctionBody = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-02 16:10:00 +01:00
										 |  |  |     unsigned colon_token = 0; | 
					
						
							| 
									
										
										
										
											2010-03-23 13:52:24 +01:00
										 |  |  |     unsigned dot_dot_dot_token = 0; | 
					
						
							| 
									
										
										
										
											2009-01-02 16:10:00 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_COLON || LA() == T_LBRACE) { | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  |         BaseSpecifierListAST *base_clause_list = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (LA() == T_COLON) { | 
					
						
							| 
									
										
										
										
											2009-01-02 16:10:00 +01:00
										 |  |  |             colon_token = cursor(); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             parseBaseClause(base_clause_list); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 13:52:24 +01:00
										 |  |  |             if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |                 dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             if (LA() != T_LBRACE) { | 
					
						
							|  |  |  |                 _translationUnit->error(cursor(), "expected `{' before `%s'", tok().spell()); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 const unsigned saved = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 for (int n = 0; n < 3 && LA() != T_EOF_SYMBOL; ++n, consumeToken()) { | 
					
						
							|  |  |  |                     if (LA() == T_LBRACE) | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 if (LA() != T_LBRACE) | 
					
						
							|  |  |  |                     rewind(saved); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ClassSpecifierAST *ast = new (_pool) ClassSpecifierAST; | 
					
						
							|  |  |  |         ast->classkey_token = classkey_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ast->name = name; | 
					
						
							| 
									
										
										
										
											2009-01-02 16:10:00 +01:00
										 |  |  |         ast->colon_token = colon_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  |         ast->base_clause_list = base_clause_list; | 
					
						
							| 
									
										
										
										
											2010-03-23 13:52:24 +01:00
										 |  |  |         ast->dot_dot_dot_token = dot_dot_dot_token; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LBRACE) | 
					
						
							|  |  |  |             ast->lbrace_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         DeclarationListAST **declaration_ptr = &ast->member_specifier_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (int tk = LA()) { | 
					
						
							|  |  |  |             if (tk == T_RBRACE) { | 
					
						
							|  |  |  |                 ast->rbrace_token = consumeToken(); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             unsigned start_declaration = cursor(); | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |             if (parseMemberSpecification(declaration)) { | 
					
						
							| 
									
										
										
										
											2010-01-26 12:45:58 +01:00
										 |  |  |                 if (declaration) {  // paranoia check
 | 
					
						
							|  |  |  |                     *declaration_ptr = new (_pool) DeclarationListAST; | 
					
						
							|  |  |  |                     (*declaration_ptr)->value = declaration; | 
					
						
							|  |  |  |                     declaration_ptr = &(*declaration_ptr)->next; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (cursor() == start_declaration) { // more paranoia
 | 
					
						
							|  |  |  |                     rewind(start_declaration + 1); | 
					
						
							|  |  |  |                     skipUntilDeclaration(); | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  |                 _translationUnit->error(start_declaration, "expected a declaration"); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 rewind(start_declaration + 1); | 
					
						
							|  |  |  |                 skipUntilDeclaration(); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-03 14:47:02 +01:00
										 |  |  |         parsed = true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-03 14:47:02 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     _inFunctionBody = previousInFunctionBody; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return parsed; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAccessSpecifier(SpecifierAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_PUBLIC: | 
					
						
							|  |  |  |     case T_PROTECTED: | 
					
						
							|  |  |  |     case T_PRIVATE: { | 
					
						
							|  |  |  |         SimpleSpecifierAST *ast = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |         ast->specifier_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAccessDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-03 13:21:08 +10:00
										 |  |  |     if (LA() == T_PUBLIC || LA() == T_PROTECTED || LA() == T_PRIVATE || LA() == T_Q_SIGNALS || LA() == T_Q_SLOTS) { | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |         bool isSignals = LA() == T_Q_SIGNALS; | 
					
						
							| 
									
										
										
										
											2010-02-03 13:21:08 +10:00
										 |  |  |         bool isSlots = LA() == T_Q_SLOTS; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         AccessDeclarationAST *ast = new (_pool) AccessDeclarationAST; | 
					
						
							|  |  |  |         ast->access_specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-03 13:21:08 +10:00
										 |  |  |         if (! isSignals && (LA() == T_Q_SLOTS || isSlots)) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->slots_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_COLON, &ast->colon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  | /*
 | 
					
						
							|  |  |  |  Q_PROPERTY(type name | 
					
						
							|  |  |  |         READ getFunction | 
					
						
							|  |  |  |         [WRITE setFunction] | 
					
						
							|  |  |  |         [RESET resetFunction] | 
					
						
							|  |  |  |         [NOTIFY notifySignal] | 
					
						
							|  |  |  |         [DESIGNABLE bool] | 
					
						
							|  |  |  |         [SCRIPTABLE bool] | 
					
						
							|  |  |  |         [STORED bool] | 
					
						
							|  |  |  |         [USER bool] | 
					
						
							|  |  |  |         [CONSTANT] | 
					
						
							|  |  |  |         [FINAL]) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Note that "type" appears to be any valid type. So these are valid: | 
					
						
							|  |  |  |       Q_PROPERTY(const char *zoo READ zoo) | 
					
						
							|  |  |  |       Q_PROPERTY(const class Blah *blah READ blah) | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     Furthermore, the only restriction on the order of the items in between the | 
					
						
							|  |  |  |     parenthesis is that the type is the first parameter and the name comes after | 
					
						
							|  |  |  |     the type. | 
					
						
							|  |  |  | */ | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  | bool Parser::parseQtPropertyDeclaration(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_Q_PROPERTY) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QtPropertyDeclarationAST *ast = new (_pool)QtPropertyDeclarationAST; | 
					
						
							|  |  |  |     ast->property_specifier_token = consumeToken(); | 
					
						
							|  |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |         parseTypeId(ast->type_id); | 
					
						
							| 
									
										
										
										
											2010-03-03 16:45:18 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 16:25:12 +01:00
										 |  |  |         SimpleNameAST *property_name = new (_pool) SimpleNameAST; | 
					
						
							| 
									
										
										
										
											2010-03-03 16:45:18 +01:00
										 |  |  |         // special case: keywords are allowed for property names!
 | 
					
						
							|  |  |  |         if (tok().isKeyword()) { | 
					
						
							|  |  |  |             property_name->identifier_token = consumeToken(); | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             match(T_IDENTIFIER, &property_name->identifier_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 16:25:12 +01:00
										 |  |  |         ast->property_name = property_name; | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |         QtPropertyDeclarationItemListAST **iter = &ast->property_declaration_items; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         while (true) { | 
					
						
							|  |  |  |             if (LA() == T_RPAREN) { | 
					
						
							|  |  |  |                 ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } else if (LA() == T_IDENTIFIER) { | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 QtPropertyDeclarationItemAST *item = 0; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |                 switch (peekAtQtContextKeyword()) { | 
					
						
							|  |  |  |                 case Token_READ: | 
					
						
							|  |  |  |                 case Token_WRITE: | 
					
						
							|  |  |  |                 case Token_RESET: | 
					
						
							| 
									
										
										
										
											2010-02-16 16:54:39 +01:00
										 |  |  |                 case Token_NOTIFY: | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |                 case Token_DESIGNABLE: | 
					
						
							|  |  |  |                 case Token_SCRIPTABLE: | 
					
						
							|  |  |  |                 case Token_STORED: | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 case Token_USER: { | 
					
						
							| 
									
										
										
										
											2010-02-16 16:54:39 +01:00
										 |  |  |                     unsigned item_name_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                     ExpressionAST *expr = 0; | 
					
						
							| 
									
										
										
										
											2010-02-16 16:54:39 +01:00
										 |  |  |                     if (parsePostfixExpression(expr)) { | 
					
						
							|  |  |  |                         QtPropertyDeclarationItemAST *bItem = | 
					
						
							|  |  |  |                                 new (_pool) QtPropertyDeclarationItemAST; | 
					
						
							|  |  |  |                         bItem->item_name_token = item_name_token; | 
					
						
							|  |  |  |                         bItem->expression = expr; | 
					
						
							| 
									
										
										
										
											2010-02-09 17:01:05 +01:00
										 |  |  |                         item = bItem; | 
					
						
							|  |  |  |                     } else { | 
					
						
							| 
									
										
										
										
											2010-02-16 16:54:39 +01:00
										 |  |  |                         _translationUnit->error(cursor(), | 
					
						
							|  |  |  |                                                 "expected expression before `%s'", | 
					
						
							|  |  |  |                                                 tok().spell()); | 
					
						
							| 
									
										
										
										
											2010-02-09 17:01:05 +01:00
										 |  |  |                     } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |                     break; | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 case Token_CONSTANT: | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 case Token_FINAL: { | 
					
						
							| 
									
										
										
										
											2010-02-16 16:54:39 +01:00
										 |  |  |                     QtPropertyDeclarationItemAST *fItem = new (_pool) QtPropertyDeclarationItemAST; | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                     fItem->item_name_token = consumeToken(); | 
					
						
							|  |  |  |                     item = fItem; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |                     break; | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 default: | 
					
						
							|  |  |  |                     _translationUnit->error(cursor(), "expected `)' before `%s'", tok().spell()); | 
					
						
							| 
									
										
										
										
											2010-02-09 17:01:05 +01:00
										 |  |  |                     // skip the token
 | 
					
						
							|  |  |  |                     consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-02-06 15:47:46 +01:00
										 |  |  |                 if (item) { | 
					
						
							|  |  |  |                     *iter = new (_pool) QtPropertyDeclarationItemListAST; | 
					
						
							|  |  |  |                     (*iter)->value = item; | 
					
						
							|  |  |  |                     iter = &(*iter)->next; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-02-09 17:01:05 +01:00
										 |  |  |             } else if (!LA()) { | 
					
						
							|  |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |             } else { | 
					
						
							|  |  |  |                 _translationUnit->error(cursor(), "expected `)' before `%s'", tok().spell()); | 
					
						
							| 
									
										
										
										
											2010-02-09 17:01:05 +01:00
										 |  |  |                 // skip the token
 | 
					
						
							|  |  |  |                 consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // q-enums-decl ::= 'Q_ENUMS' '(' q-enums-list? ')'
 | 
					
						
							|  |  |  | // q-enums-list ::= identifier
 | 
					
						
							|  |  |  | // q-enums-list ::= q-enums-list identifier
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Note: Q_ENUMS is a CPP macro with exactly 1 parameter.
 | 
					
						
							|  |  |  | // Examples of valid uses:
 | 
					
						
							|  |  |  | //   Q_ENUMS()
 | 
					
						
							|  |  |  | //   Q_ENUMS(Priority)
 | 
					
						
							|  |  |  | //   Q_ENUMS(Priority Severity)
 | 
					
						
							|  |  |  | // so, these are not allowed:
 | 
					
						
							|  |  |  | //   Q_ENUMS
 | 
					
						
							|  |  |  | //   Q_ENUMS(Priority, Severity)
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  | bool Parser::parseQtEnumDeclaration(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     if (LA() != T_Q_ENUMS) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     QtEnumDeclarationAST *ast = new (_pool) QtEnumDeclarationAST; | 
					
						
							|  |  |  |     ast->enum_specifier_token = consumeToken(); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |     for (NameListAST **iter = &ast->enumerator_list; LA() && LA() != T_RPAREN; iter = &(*iter)->next) { | 
					
						
							|  |  |  |         NameAST *name_ast = 0; | 
					
						
							| 
									
										
										
										
											2010-03-16 16:38:02 +01:00
										 |  |  |         if (!parseName(name_ast)) | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         *iter = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |         (*iter)->value = name_ast; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  | // q-flags-decl ::= 'Q_FLAGS' '(' q-flags-list? ')'
 | 
					
						
							|  |  |  | // q-flags-list ::= identifier
 | 
					
						
							|  |  |  | // q-flags-list ::= q-flags-list identifier
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // Note: Q_FLAGS is a CPP macro with exactly 1 parameter.
 | 
					
						
							|  |  |  | // Examples of valid uses:
 | 
					
						
							|  |  |  | //   Q_FLAGS()
 | 
					
						
							|  |  |  | //   Q_FLAGS(Orientation)
 | 
					
						
							|  |  |  | //   Q_FLAGS(Orientation DropActions)
 | 
					
						
							|  |  |  | // so, these are not allowed:
 | 
					
						
							|  |  |  | //   Q_FLAGS
 | 
					
						
							|  |  |  | //   Q_FLAGS(Orientation, DropActions)
 | 
					
						
							|  |  |  | bool Parser::parseQtFlags(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  |     if (LA() != T_Q_FLAGS) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QtFlagsDeclarationAST *ast = new (_pool) QtFlagsDeclarationAST; | 
					
						
							|  |  |  |     ast->flags_specifier_token = consumeToken(); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |     for (NameListAST **iter = &ast->flag_enums_list; LA() && LA() != T_RPAREN; iter = &(*iter)->next) { | 
					
						
							|  |  |  |         NameAST *name_ast = 0; | 
					
						
							| 
									
										
										
										
											2010-03-16 16:38:02 +01:00
										 |  |  |         if (!parseName(name_ast)) | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |             break; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  |         *iter = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2010-02-15 16:24:10 +01:00
										 |  |  |         (*iter)->value = name_ast; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 17:29:40 +01:00
										 |  |  | // class-specifier ::=
 | 
					
						
							|  |  |  | //   c++-class-specifier
 | 
					
						
							|  |  |  | //   q-tag
 | 
					
						
							|  |  |  | //   q-enums-of-flags
 | 
					
						
							|  |  |  | //   q-class-info
 | 
					
						
							|  |  |  | //   q-interfaces
 | 
					
						
							|  |  |  | //   q-private-slot
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // declaration ::=
 | 
					
						
							|  |  |  | //   c++-declaration
 | 
					
						
							|  |  |  | //   q-declare-interface
 | 
					
						
							|  |  |  | //   q-declare-metatype
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-tag ::=
 | 
					
						
							|  |  |  | //   Q_OBJECT
 | 
					
						
							|  |  |  | //   Q_GADGET
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-enums-or-flags ::=
 | 
					
						
							|  |  |  | //   (Q_ENUMS | Q_FLAGS) LPAREN name+ RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-class-info ::=
 | 
					
						
							|  |  |  | //   Q_CLASS_INFO LPAREN string-literal COMMA STRING_LITERAL RPAREN
 | 
					
						
							|  |  |  | //   Q_CLASS_INFO LPAREN string-literal COMMA IDENTIFIER LPAREN STRING_LITERAL RPAREN RPAREN
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // q-interfaces ::=
 | 
					
						
							|  |  |  | //   Q_INTERFACES LPAREN (name q-constraints)* RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-constraints ::=
 | 
					
						
							|  |  |  | //   (COLON name)*
 | 
					
						
							|  |  |  | bool Parser::parseQtInterfaces(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_Q_INTERFACES) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     QtInterfacesDeclarationAST *ast = new (_pool) QtInterfacesDeclarationAST; | 
					
						
							|  |  |  |     ast->interfaces_token = consumeToken(); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |     for (QtInterfaceNameListAST **iter = &ast->interface_name_list; LA() && LA() != T_RPAREN; iter = &(*iter)->next) { | 
					
						
							|  |  |  |         NameAST *name_ast = 0; | 
					
						
							|  |  |  |         if (!parseName(name_ast)) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         *iter = new (_pool) QtInterfaceNameListAST; | 
					
						
							|  |  |  |         (*iter)->value = new (_pool) QtInterfaceNameAST; | 
					
						
							|  |  |  |         (*iter)->value->interface_name = name_ast; | 
					
						
							|  |  |  |         for (NameListAST **iter2 = &(*iter)->value->constraint_list; LA() && LA() == T_COLON; iter2 = &(*iter2)->next) { | 
					
						
							|  |  |  |             /*unsigned colon_token =*/ consumeToken(); | 
					
						
							|  |  |  |             NameAST *name_ast2 = 0; | 
					
						
							|  |  |  |             if (!parseName(name_ast2)) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             *iter2 = new (_pool) NameListAST; | 
					
						
							|  |  |  |             (*iter2)->value = name_ast2; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // q-private-slot ::=
 | 
					
						
							|  |  |  | //   Q_PRIVATE_SLOT LPAREN IDENTIFIER (LPAREN RPAREN)? COMMA q-function-declaration RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-function-declaration ::=
 | 
					
						
							|  |  |  | //   decl-specifier-list declarator   [+ check for the function-declarator]
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-declare-interface ::=
 | 
					
						
							|  |  |  | //   Q_DECLARE_INTERFACE LPAREN name COMMA (STRING_LITERAL | IDENTIFIER) RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // q-declare-metatype ::=
 | 
					
						
							|  |  |  | //   Q_DECLARE_METATYPE LPAREN name RPAREN SEMICOLON? [warning]
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseMemberSpecification(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							| 
									
										
										
										
											2010-03-17 14:35:48 +01:00
										 |  |  |     case T_Q_OBJECT: | 
					
						
							|  |  |  |     case T_Q_GADGET: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QtObjectTagAST *ast = new (_pool) QtObjectTagAST; | 
					
						
							|  |  |  |         ast->q_object_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_Q_PRIVATE_SLOT: | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |         QtPrivateSlotAST *ast = new (_pool) QtPrivateSlotAST; | 
					
						
							|  |  |  |         ast->q_private_slot_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         match(T_IDENTIFIER, &ast->dptr_token); | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |             ast->dptr_lparen_token = consumeToken(); | 
					
						
							|  |  |  |             match(T_RPAREN, &ast->dptr_rparen_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         match(T_COMMA, &ast->comma_token); | 
					
						
							|  |  |  |         parseTypeSpecifier(ast->type_specifiers); | 
					
						
							|  |  |  |         parseDeclarator(ast->declarator); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |     }   return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     case T_SEMICOLON: | 
					
						
							|  |  |  |         return parseEmptyDeclaration(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_USING: | 
					
						
							|  |  |  |         return parseUsing(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TEMPLATE: | 
					
						
							|  |  |  |         return parseTemplateDeclaration(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |     case T_Q_SIGNALS: | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     case T_PUBLIC: | 
					
						
							|  |  |  |     case T_PROTECTED: | 
					
						
							|  |  |  |     case T_PRIVATE: | 
					
						
							| 
									
										
										
										
											2010-02-03 13:21:08 +10:00
										 |  |  |     case T_Q_SLOTS: | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return parseAccessDeclaration(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  |     case T_Q_PROPERTY: | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  |         return parseQtPropertyDeclaration(node); | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case T_Q_ENUMS: | 
					
						
							| 
									
										
										
										
											2010-02-06 11:38:54 +01:00
										 |  |  |         return parseQtEnumDeclaration(node); | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case T_Q_FLAGS: | 
					
						
							| 
									
										
										
										
											2010-02-06 14:32:25 +01:00
										 |  |  |         return parseQtFlags(node); | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-16 17:29:40 +01:00
										 |  |  |     case T_Q_INTERFACES: | 
					
						
							|  |  |  |         return parseQtInterfaces(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         return parseSimpleDeclaration(node, /*acceptStructDeclarator=*/true); | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCtorInitializer(CtorInitializerAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_COLON) { | 
					
						
							|  |  |  |         unsigned colon_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         CtorInitializerAST *ast = new (_pool) CtorInitializerAST; | 
					
						
							|  |  |  |         ast->colon_token = colon_token; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         parseMemInitializerList(ast->member_initializer_list); | 
					
						
							| 
									
										
										
										
											2010-03-23 13:43:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             ast->dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseElaboratedTypeSpecifier(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (lookAtClassKey() || LA() == T_ENUM || LA() == T_TYPENAME) { | 
					
						
							|  |  |  |         unsigned classkey_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-03-22 18:28:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         SpecifierListAST *attributes = 0, **attr_ptr = &attributes; | 
					
						
							|  |  |  |         while (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |             parseAttributeSpecifier(*attr_ptr); | 
					
						
							|  |  |  |             attr_ptr = &(*attr_ptr)->next; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         NameAST *name = 0; | 
					
						
							|  |  |  |         if (parseName(name)) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             ElaboratedTypeSpecifierAST *ast = new (_pool) ElaboratedTypeSpecifierAST; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->classkey_token = classkey_token; | 
					
						
							| 
									
										
										
										
											2010-03-22 18:28:04 +01:00
										 |  |  |             ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->name = name; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExceptionSpecification(ExceptionSpecificationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_THROW) { | 
					
						
							|  |  |  |         ExceptionSpecificationAST *ast = new (_pool) ExceptionSpecificationAST; | 
					
						
							|  |  |  |         ast->throw_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             ast->dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  |         else | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             parseTypeIdList(ast->type_id_list); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (LA() == T_RPAREN) | 
					
						
							|  |  |  |             ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  | bool Parser::parseEnumerator(EnumeratorListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |         EnumeratorAST *ast = new (_pool) EnumeratorAST; | 
					
						
							|  |  |  |         ast->identifier_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_EQUAL) { | 
					
						
							|  |  |  |             ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |             parseConstantExpression(ast->expression); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         node = new (_pool) EnumeratorListAST; | 
					
						
							|  |  |  |         node->value = ast; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseInitDeclarator(DeclaratorAST *&node, | 
					
						
							|  |  |  |         bool acceptStructDeclarator) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (acceptStructDeclarator && LA() == T_COLON) { | 
					
						
							|  |  |  |         // anonymous bit-field declaration.
 | 
					
						
							|  |  |  |         // ### TODO create the AST
 | 
					
						
							| 
									
										
										
										
											2009-03-17 14:12:18 +01:00
										 |  |  |     } else if (! parseDeclarator(node, /*stopAtCppInitializer = */ ! acceptStructDeclarator)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_ASM && LA(2) == T_LPAREN) { // ### FIXME
 | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (skip(T_LPAREN, T_RPAREN)) | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (acceptStructDeclarator && node && | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ! node->postfix_declarator_list && | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             node->core_declarator && | 
					
						
							|  |  |  |             node->core_declarator->asNestedDeclarator()) { | 
					
						
							|  |  |  |         rewind(start); | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (acceptStructDeclarator && LA() == T_COLON | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             && (! node || ! node->postfix_declarator_list)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         unsigned colon_token = consumeToken(); | 
					
						
							|  |  |  |         ExpressionAST *expression = 0; | 
					
						
							|  |  |  |         if (parseConstantExpression(expression) && (LA() == T_COMMA || | 
					
						
							|  |  |  |                                                     LA() == T_SEMICOLON)) { | 
					
						
							|  |  |  |             // recognized a bitfielddeclarator.
 | 
					
						
							|  |  |  |             // ### TODO create the AST
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         rewind(colon_token); | 
					
						
							|  |  |  |     } else if (LA() == T_EQUAL || (! acceptStructDeclarator && LA() == T_LPAREN)) { | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |         parseInitializer(node->initializer, &node->equals_token); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | bool Parser::parseBaseClause(BaseSpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_COLON) { | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  |         consumeToken(); // ### remove me
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  |         BaseSpecifierListAST **ast = &node; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (parseBaseSpecifier(*ast)) { | 
					
						
							|  |  |  |             ast = &(*ast)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 if (parseBaseSpecifier(*ast)) | 
					
						
							| 
									
										
										
										
											2009-06-29 13:46:04 +02:00
										 |  |  |                     ast = &(*ast)->next; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  | bool Parser::parseInitializer(ExpressionAST *&node, unsigned *equals_token) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         return parsePrimaryExpression(node); | 
					
						
							|  |  |  |     } else if (LA() == T_EQUAL) { | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |         (*equals_token) = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return parseInitializerClause(node); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  | bool Parser::parseMemInitializerList(MemInitializerListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  |     MemInitializerListAST **initializer = &node; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (parseMemInitializer(*initializer)) { | 
					
						
							|  |  |  |         initializer = &(*initializer)->next; | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  |         while (true) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (LA() == T_LBRACE) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 13:43:18 +01:00
										 |  |  |             else if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT && LA(2) == T_LBRACE) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  |             else if (LA() == T_COMMA || (LA() == T_IDENTIFIER && (LA(2) == T_LPAREN || LA(2) == T_COLON_COLON))) { | 
					
						
							|  |  |  |                 if (LA() != T_COMMA) | 
					
						
							|  |  |  |                     _translationUnit->error(cursor(), "expected `,'"); | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (parseMemInitializer(*initializer)) | 
					
						
							|  |  |  |                     initializer = &(*initializer)->next; | 
					
						
							|  |  |  |                 else | 
					
						
							|  |  |  |                     _translationUnit->error(cursor(), "expected a member initializer"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             } else break; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 13:43:18 +01:00
										 |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) { | 
					
						
							|  |  |  |             if (LA(2) != T_LBRACE) | 
					
						
							|  |  |  |                 _translationUnit->error(cursor(), "expected `{'"); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         } else if (LA() != T_LBRACE) { | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  |             _translationUnit->error(cursor(), "expected `{'"); | 
					
						
							| 
									
										
										
										
											2010-03-23 13:43:18 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  | bool Parser::parseMemInitializer(MemInitializerListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     NameAST *name = 0; | 
					
						
							| 
									
										
										
										
											2009-12-07 14:33:13 +01:00
										 |  |  |     if (! parseName(name)) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-11-10 14:11:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 14:33:13 +01:00
										 |  |  |     MemInitializerAST *ast = new (_pool) MemInitializerAST; | 
					
						
							|  |  |  |     ast->name = name; | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |     parseExpressionList(ast->expression_list); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     node = new (_pool) MemInitializerListAST; | 
					
						
							|  |  |  |     node->value = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTypeIdList(ExpressionListAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     ExpressionListAST **expression_list_ptr = &node; | 
					
						
							|  |  |  |     ExpressionAST *typeId = 0; | 
					
						
							|  |  |  |     if (parseTypeId(typeId)) { | 
					
						
							|  |  |  |         *expression_list_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         (*expression_list_ptr)->value = typeId; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         expression_list_ptr = &(*expression_list_ptr)->next; | 
					
						
							| 
									
										
										
										
											2010-03-23 14:13:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |             consumeToken(); // ### store this token
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (LA() == T_COMMA) { | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (parseTypeId(typeId)) { | 
					
						
							|  |  |  |                 *expression_list_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 (*expression_list_ptr)->value = typeId; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 expression_list_ptr = &(*expression_list_ptr)->next; | 
					
						
							| 
									
										
										
										
											2010-03-23 14:13:04 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |                     consumeToken(); // ### store this token
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExpressionList(ExpressionListAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     ExpressionListAST **expression_list_ptr = &node; | 
					
						
							|  |  |  |     ExpressionAST *expression = 0; | 
					
						
							|  |  |  |     if (parseAssignmentExpression(expression)) { | 
					
						
							|  |  |  |         *expression_list_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         (*expression_list_ptr)->value = expression; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         expression_list_ptr = &(*expression_list_ptr)->next; | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-13 12:36:51 +01:00
										 |  |  |             if (parseAssignmentExpression(expression)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 *expression_list_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 (*expression_list_ptr)->value = expression; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 expression_list_ptr = &(*expression_list_ptr)->next; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | bool Parser::parseBaseSpecifier(BaseSpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     BaseSpecifierAST *ast = new (_pool) BaseSpecifierAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_VIRTUAL) { | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->virtual_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         int tk = LA(); | 
					
						
							|  |  |  |         if (tk == T_PUBLIC || tk == T_PROTECTED || tk == T_PRIVATE) | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |             ast->access_specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         int tk = LA(); | 
					
						
							|  |  |  |         if (tk == T_PUBLIC || tk == T_PROTECTED || tk == T_PRIVATE) | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |             ast->access_specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_VIRTUAL) | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |             ast->virtual_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parseName(ast->name); | 
					
						
							|  |  |  |     if (! ast->name) | 
					
						
							|  |  |  |         _translationUnit->error(cursor(), "expected class-name"); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:03:40 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     node = new (_pool) BaseSpecifierListAST; | 
					
						
							|  |  |  |     node->value = ast; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseInitializerList(ExpressionListAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     ExpressionListAST **initializer_ptr = &node; | 
					
						
							|  |  |  |     ExpressionAST *initializer = 0; | 
					
						
							|  |  |  |     if (parseInitializerClause(initializer)) { | 
					
						
							|  |  |  |         *initializer_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         (*initializer_ptr)->value = initializer; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         initializer_ptr = &(*initializer_ptr)->next; | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             initializer = 0; | 
					
						
							|  |  |  |             parseInitializerClause(initializer); | 
					
						
							|  |  |  |             *initializer_ptr = new (_pool) ExpressionListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             (*initializer_ptr)->value = initializer; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             initializer_ptr = &(*initializer_ptr)->next; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-03-23 14:16:58 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (_cxx0xEnabled && LA() == T_DOT_DOT_DOT) | 
					
						
							|  |  |  |         consumeToken(); // ### store this token
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseInitializerClause(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LBRACE) { | 
					
						
							|  |  |  |         ArrayInitializerAST *ast = new (_pool) ArrayInitializerAST; | 
					
						
							|  |  |  |         ast->lbrace_token = consumeToken(); | 
					
						
							|  |  |  |         parseInitializerList(ast->expression_list); | 
					
						
							| 
									
										
										
										
											2009-01-13 14:58:29 +01:00
										 |  |  |         match(T_RBRACE, &ast->rbrace_token); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return parseAssignmentExpression(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseUnqualifiedName(NameAST *&node, bool acceptTemplateId) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TILDE && LA(2) == T_IDENTIFIER) { | 
					
						
							|  |  |  |         DestructorNameAST *ast = new (_pool) DestructorNameAST; | 
					
						
							|  |  |  |         ast->tilde_token = consumeToken(); | 
					
						
							|  |  |  |         ast->identifier_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (LA() == T_OPERATOR) { | 
					
						
							|  |  |  |         unsigned operator_token = cursor(); | 
					
						
							|  |  |  |         if (parseOperatorFunctionId(node)) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         rewind(operator_token); | 
					
						
							|  |  |  |         return parseConversionFunctionId(node); | 
					
						
							|  |  |  |      } else if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |          unsigned identifier_token = cursor(); | 
					
						
							|  |  |  |          if (acceptTemplateId && LA(2) == T_LESS && parseTemplateId(node)) { | 
					
						
							|  |  |  |              if (! _templateArguments || (LA() == T_COMMA  || LA() == T_GREATER || | 
					
						
							|  |  |  |                                           LA() == T_LPAREN || LA() == T_RPAREN  || | 
					
						
							|  |  |  |                                           LA() == T_COLON_COLON)) | 
					
						
							|  |  |  |                  return true; | 
					
						
							|  |  |  |          } | 
					
						
							|  |  |  |          rewind(identifier_token); | 
					
						
							|  |  |  |          SimpleNameAST *ast = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |          ast->identifier_token = consumeToken(); | 
					
						
							|  |  |  |          node = ast; | 
					
						
							|  |  |  |          return true; | 
					
						
							|  |  |  |     } else if (LA() == T_TEMPLATE) { | 
					
						
							|  |  |  |         unsigned template_token = consumeToken(); | 
					
						
							|  |  |  |         if (parseTemplateId(node)) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         rewind(template_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseStringLiteral(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! (LA() == T_STRING_LITERAL || LA() == T_WIDE_STRING_LITERAL)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     StringLiteralAST **ast = reinterpret_cast<StringLiteralAST **> (&node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (LA() == T_STRING_LITERAL || LA() == T_WIDE_STRING_LITERAL) { | 
					
						
							|  |  |  |         *ast = new (_pool) StringLiteralAST; | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         (*ast)->literal_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ast = &(*ast)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExpressionStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  |     if (LA() == T_SEMICOLON) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ExpressionStatementAST *ast = new (_pool) ExpressionStatementAST; | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 10:17:51 +01:00
										 |  |  |     const bool wasInExpressionStatement = _inExpressionStatement; | 
					
						
							|  |  |  |     _inExpressionStatement = true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // switch to the temp pool
 | 
					
						
							|  |  |  |     MemoryPool *previousPool = _pool; | 
					
						
							|  |  |  |     _pool = &_expressionStatementTempPool; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool parsed = false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  |     ExpressionAST *expression = 0; | 
					
						
							|  |  |  |     if (parseExpression(expression)) { | 
					
						
							| 
									
										
										
										
											2010-03-23 10:17:51 +01:00
										 |  |  |         ExpressionStatementAST *ast = new (previousPool) ExpressionStatementAST; | 
					
						
							| 
									
										
										
										
											2010-03-24 12:54:25 +01:00
										 |  |  |         if (expression) | 
					
						
							|  |  |  |             ast->expression = expression->clone(previousPool); | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							| 
									
										
										
										
											2010-03-23 10:17:51 +01:00
										 |  |  |         parsed = true; | 
					
						
							| 
									
										
										
										
											2010-02-02 15:38:21 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-23 10:17:51 +01:00
										 |  |  |     _inExpressionStatement = wasInExpressionStatement; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (! _inExpressionStatement) { | 
					
						
							|  |  |  |         // rewind the memory pool after parsing a toplevel expression statement.
 | 
					
						
							|  |  |  |         _expressionStatementTempPool.reset(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // restore the pool
 | 
					
						
							|  |  |  |     _pool = previousPool; | 
					
						
							|  |  |  |     return parsed; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_WHILE: | 
					
						
							|  |  |  |         return parseWhileStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_DO: | 
					
						
							|  |  |  |         return parseDoStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |     case T_Q_FOREACH: | 
					
						
							|  |  |  |         return parseForeachStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     case T_FOR: | 
					
						
							|  |  |  |         return parseForStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_IF: | 
					
						
							|  |  |  |         return parseIfStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_SWITCH: | 
					
						
							|  |  |  |         return parseSwitchStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TRY: | 
					
						
							|  |  |  |         return parseTryBlockStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_CASE: | 
					
						
							|  |  |  |     case T_DEFAULT: | 
					
						
							|  |  |  |         return parseLabeledStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_BREAK: | 
					
						
							|  |  |  |         return parseBreakStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_CONTINUE: | 
					
						
							|  |  |  |         return parseContinueStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_GOTO: | 
					
						
							|  |  |  |         return parseGotoStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_RETURN: | 
					
						
							|  |  |  |         return parseReturnStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_LBRACE: | 
					
						
							|  |  |  |         return parseCompoundStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_ASM: | 
					
						
							|  |  |  |     case T_NAMESPACE: | 
					
						
							|  |  |  |     case T_USING: | 
					
						
							|  |  |  |     case T_TEMPLATE: | 
					
						
							|  |  |  |     case T_CLASS: case T_STRUCT: case T_UNION: | 
					
						
							|  |  |  |         return parseDeclarationStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_SEMICOLON: { | 
					
						
							|  |  |  |         ExpressionStatementAST *ast = new (_pool) ExpressionStatementAST; | 
					
						
							|  |  |  |         ast->semicolon_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_SYNCHRONIZED: | 
					
						
							|  |  |  |         return objCEnabled() && parseObjCSynchronizedStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-30 16:30:21 +01:00
										 |  |  |     case T_Q_D: | 
					
						
							|  |  |  |     case T_Q_Q: { | 
					
						
							|  |  |  |         QtMemberDeclarationAST *ast = new (_pool) QtMemberDeclarationAST; | 
					
						
							|  |  |  |         ast->q_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseTypeId(ast->type_id); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |     } return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         if (LA() == T_IDENTIFIER && LA(2) == T_COLON) | 
					
						
							|  |  |  |             return parseLabeledStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return parseExpressionOrDeclarationStatement(node); | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							| 
									
										
										
										
											2010-02-03 13:21:08 +10:00
										 |  |  |     return false; //Avoid compiler warning
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseBreakStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_BREAK) { | 
					
						
							|  |  |  |         BreakStatementAST *ast = new (_pool) BreakStatementAST; | 
					
						
							|  |  |  |         ast->break_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseContinueStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_CONTINUE) { | 
					
						
							|  |  |  |         ContinueStatementAST *ast = new (_pool) ContinueStatementAST; | 
					
						
							|  |  |  |         ast->continue_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseGotoStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_GOTO) { | 
					
						
							|  |  |  |         GotoStatementAST *ast = new (_pool) GotoStatementAST; | 
					
						
							|  |  |  |         ast->goto_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_IDENTIFIER, &ast->identifier_token); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseReturnStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_RETURN) { | 
					
						
							|  |  |  |         ReturnStatementAST *ast = new (_pool) ReturnStatementAST; | 
					
						
							|  |  |  |         ast->return_token = consumeToken(); | 
					
						
							|  |  |  |         parseExpression(ast->expression); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | bool Parser::isPointerDeclaration(DeclarationStatementAST *ast) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! ast) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (SimpleDeclarationAST *declaration = ast->declaration->asSimpleDeclaration()) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (SpecifierListAST *spec = declaration->decl_specifier_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             if (spec->value->asNamedTypeSpecifier() && ! spec->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 if (DeclaratorListAST *declarators = declaration->declarator_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |                     if (DeclaratorAST *declarator = declarators->value) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                         if (declarator->ptr_operator_list && declarator->equals_token && declarator->initializer) { | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |                             return true; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-15 12:24:41 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::maybeAmbiguousStatement(DeclarationStatementAST *ast) const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     if (! ast) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (SimpleDeclarationAST *declaration = ast->declaration->asSimpleDeclaration()) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (SpecifierListAST *spec = declaration->decl_specifier_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             if (spec->value->asNamedTypeSpecifier() && ! spec->next) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 if (DeclaratorListAST *declarators = declaration->declarator_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |                     if (DeclaratorAST *declarator = declarators->value) { | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |                         if (declarator->core_declarator && | 
					
						
							|  |  |  |                             declarator->core_declarator->asNestedDeclarator()) { | 
					
						
							|  |  |  |                             // recognized name(id-expression)
 | 
					
						
							|  |  |  |                             return true; | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         } else if (DeclaratorListAST *declarators = declaration->declarator_list) { | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |             // no decl_specifiers...
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |             if (DeclaratorAST *declarator = declarators->value) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 if (declarator->postfix_declarator_list && declarator->postfix_declarator_list->value->asFunctionDeclarator() | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |                                                      && ! declarator->initializer) { | 
					
						
							|  |  |  |                     return false; | 
					
						
							| 
									
										
										
										
											2010-01-29 21:33:57 +01:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-15 12:24:41 +01:00
										 |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseExpressionOrDeclarationStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_SEMICOLON) | 
					
						
							|  |  |  |         return parseExpressionStatement(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  |     bool blocked = blockErrors(true); | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseDeclarationStatement(node)) { | 
					
						
							|  |  |  |         DeclarationStatementAST *stmt = static_cast<DeclarationStatementAST *>(node); | 
					
						
							| 
									
										
										
										
											2008-12-15 12:24:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |         if (isPointerDeclaration(stmt)) { | 
					
						
							|  |  |  |             blockErrors(blocked); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (! maybeAmbiguousStatement(stmt)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             unsigned end_of_declaration_statement = cursor(); | 
					
						
							|  |  |  |             rewind(start); | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             StatementAST *expression = 0; | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |             if (parseExpressionStatement(expression) && cursor() == end_of_declaration_statement) { | 
					
						
							|  |  |  |                 // it's an ambiguous expression-or-declaration statement.
 | 
					
						
							|  |  |  |                 ExpressionOrDeclarationStatementAST *ast = new (_pool) ExpressionOrDeclarationStatementAST; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 ast->declaration = node; | 
					
						
							|  |  |  |                 ast->expression = expression; | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             rewind(end_of_declaration_statement); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             blockErrors(blocked); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-15 12:24:41 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |     // it's not a declaration statement.
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     blockErrors(blocked); | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  |     return parseExpressionStatement(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCondition(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     bool blocked = blockErrors(true); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseTypeSpecifier(type_specifier)) { | 
					
						
							|  |  |  |         DeclaratorAST *declarator = 0; | 
					
						
							|  |  |  |         if (parseInitDeclarator(declarator, /*acceptStructDeclarator=*/false)) { | 
					
						
							| 
									
										
										
										
											2010-03-24 13:56:34 +01:00
										 |  |  |             if (declarator->initializer && declarator->equals_token) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 ConditionAST *ast = new (_pool) ConditionAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 ast->type_specifier_list = type_specifier; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 ast->declarator = declarator; | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 blockErrors(blocked); | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     blockErrors(blocked); | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  |     return parseExpression(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseWhileStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_WHILE) { | 
					
						
							|  |  |  |         WhileStatementAST *ast = new (_pool) WhileStatementAST; | 
					
						
							|  |  |  |         ast->while_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseCondition(ast->condition); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseDoStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_DO) { | 
					
						
							|  |  |  |         DoStatementAST *ast = new (_pool) DoStatementAST; | 
					
						
							|  |  |  |         ast->do_token = consumeToken(); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  |         match(T_WHILE, &ast->while_token); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseExpression(ast->expression); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  | bool Parser::parseForeachStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |     if (LA() == T_Q_FOREACH) { | 
					
						
							|  |  |  |         ForeachStatementAST *ast = new (_pool) ForeachStatementAST; | 
					
						
							|  |  |  |         ast->foreach_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         unsigned startOfTypeSpecifier = cursor(); | 
					
						
							|  |  |  |         bool blocked = blockErrors(true); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (parseTypeSpecifier(ast->type_specifier_list)) | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |             parseDeclarator(ast->declarator); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (! ast->type_specifier_list || ! ast->declarator) { | 
					
						
							|  |  |  |             ast->type_specifier_list = 0; | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |             ast->declarator = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             blockErrors(blocked); | 
					
						
							|  |  |  |             rewind(startOfTypeSpecifier); | 
					
						
							| 
									
										
										
										
											2009-07-29 09:46:38 +02:00
										 |  |  |             parseAssignmentExpression(ast->initializer); | 
					
						
							| 
									
										
										
										
											2009-07-03 09:11:52 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         blockErrors(blocked); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         match(T_COMMA, &ast->comma_token); | 
					
						
							|  |  |  |         parseExpression(ast->expression); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseForStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |     if (LA() != T_FOR) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned for_token = consumeToken(); | 
					
						
							|  |  |  |     unsigned lparen_token = 0; | 
					
						
							|  |  |  |     match(T_LPAREN, &lparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |     unsigned startOfTypeSpecifier = cursor(); | 
					
						
							|  |  |  |     bool blocked = blockErrors(true); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |     if (objCEnabled()) { | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         ObjCFastEnumerationAST *ast = new (_pool) ObjCFastEnumerationAST; | 
					
						
							|  |  |  |         ast->for_token = for_token; | 
					
						
							|  |  |  |         ast->lparen_token = lparen_token; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (parseTypeSpecifier(ast->type_specifier_list)) | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |             parseDeclarator(ast->declarator); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if ((ast->type_specifier_list || ast->declarator) && !peekAtObjCContextKeyword(Token_in)) { | 
					
						
							| 
									
										
										
										
											2009-07-29 10:39:27 +02:00
										 |  |  |             // woops, probably parsed too much: "in" got parsed as a declarator. Let's redo it:
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->type_specifier_list = 0; | 
					
						
							| 
									
										
										
										
											2009-07-29 10:39:27 +02:00
										 |  |  |             ast->declarator = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             rewind(startOfTypeSpecifier); | 
					
						
							|  |  |  |             parseDeclarator(ast->declarator); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (! ast->type_specifier_list || ! ast->declarator) { | 
					
						
							|  |  |  |             ast->type_specifier_list = 0; | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |             ast->declarator = 0; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |             rewind(startOfTypeSpecifier); | 
					
						
							|  |  |  |             parseAssignmentExpression(ast->initializer); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (parseObjCContextKeyword(Token_in, ast->in_token)) { | 
					
						
							|  |  |  |             blockErrors(blocked); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             parseExpression(ast->fast_enumeratable_expression); | 
					
						
							|  |  |  |             match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2010-01-09 18:00:25 +01:00
										 |  |  |             parseStatement(ast->statement); | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         // there was no "in" token, so we continue with a normal for-statement
 | 
					
						
							|  |  |  |         rewind(startOfTypeSpecifier); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |     blockErrors(blocked); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // Normal C/C++ for-statement parsing
 | 
					
						
							|  |  |  |     ForStatementAST *ast = new (_pool) ForStatementAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ast->for_token = for_token; | 
					
						
							|  |  |  |     ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |     parseForInitStatement(ast->initializer); | 
					
						
							| 
									
										
										
										
											2009-11-12 17:35:48 +01:00
										 |  |  |     parseCondition(ast->condition); | 
					
						
							| 
									
										
										
										
											2009-07-29 10:06:14 +02:00
										 |  |  |     match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |     parseExpression(ast->expression); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     parseStatement(ast->statement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseForInitStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return parseExpressionOrDeclarationStatement(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCompoundStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LBRACE) { | 
					
						
							|  |  |  |         CompoundStatementAST *ast = new (_pool) CompoundStatementAST; | 
					
						
							|  |  |  |         ast->lbrace_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 12:09:32 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // ### TODO: the GNU "local label" extension: "__label__ X, Y, Z;"
 | 
					
						
							|  |  |  |         // These are only allowed at the start of a compound stmt regardless of the language.
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         StatementListAST **statement_ptr = &ast->statement_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (int tk = LA()) { | 
					
						
							|  |  |  |             if (tk == T_RBRACE) | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             unsigned start_statement = cursor(); | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |             StatementAST *statement = 0; | 
					
						
							|  |  |  |             if (! parseStatement(statement)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 rewind(start_statement + 1); | 
					
						
							|  |  |  |                 skipUntilStatement(); | 
					
						
							|  |  |  |             } else { | 
					
						
							| 
									
										
										
										
											2009-06-17 16:08:01 +02:00
										 |  |  |                 *statement_ptr = new (_pool) StatementListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:27:56 +01:00
										 |  |  |                 (*statement_ptr)->value = statement; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 statement_ptr = &(*statement_ptr)->next; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         match(T_RBRACE, &ast->rbrace_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseIfStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_IF) { | 
					
						
							|  |  |  |         IfStatementAST *ast = new (_pool) IfStatementAST; | 
					
						
							|  |  |  |         ast->if_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseCondition(ast->condition); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         if (! parseStatement(ast->statement)) | 
					
						
							|  |  |  |             _translationUnit->error(cursor(), "expected statement"); | 
					
						
							|  |  |  |         if (LA() == T_ELSE) { | 
					
						
							|  |  |  |             ast->else_token = consumeToken(); | 
					
						
							|  |  |  |             if (! parseStatement(ast->else_statement)) | 
					
						
							|  |  |  |                 _translationUnit->error(cursor(), "expected statement"); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseSwitchStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_SWITCH) { | 
					
						
							|  |  |  |         SwitchStatementAST *ast = new (_pool) SwitchStatementAST; | 
					
						
							|  |  |  |         ast->switch_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseCondition(ast->condition); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLabeledStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_IDENTIFIER: | 
					
						
							|  |  |  |         if (LA(2) == T_COLON) { | 
					
						
							|  |  |  |             LabeledStatementAST *ast = new (_pool) LabeledStatementAST; | 
					
						
							|  |  |  |             ast->label_token = consumeToken(); | 
					
						
							|  |  |  |             ast->colon_token = consumeToken(); | 
					
						
							|  |  |  |             parseStatement(ast->statement); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_DEFAULT: { | 
					
						
							|  |  |  |         LabeledStatementAST *ast = new (_pool) LabeledStatementAST; | 
					
						
							|  |  |  |         ast->label_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_COLON, &ast->colon_token); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_CASE: { | 
					
						
							|  |  |  |         CaseStatementAST *ast = new (_pool) CaseStatementAST; | 
					
						
							|  |  |  |         ast->case_token = consumeToken(); | 
					
						
							|  |  |  |         parseConstantExpression(ast->expression); | 
					
						
							|  |  |  |         match(T_COLON, &ast->colon_token); | 
					
						
							|  |  |  |         parseStatement(ast->statement); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseBlockDeclaration(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_USING: | 
					
						
							|  |  |  |         return parseUsing(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_ASM: | 
					
						
							|  |  |  |         return parseAsmDefinition(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_NAMESPACE: | 
					
						
							|  |  |  |         return parseNamespaceAliasDefinition(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return parseSimpleDeclaration(node); | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseNamespaceAliasDefinition(DeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_NAMESPACE && LA(2) == T_IDENTIFIER && LA(3) == T_EQUAL) { | 
					
						
							|  |  |  |         NamespaceAliasDefinitionAST *ast = new (_pool) NamespaceAliasDefinitionAST; | 
					
						
							|  |  |  |         ast->namespace_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->namespace_name_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         ast->equal_token = consumeToken(); | 
					
						
							|  |  |  |         parseName(ast->name); | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseDeclarationStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-10-20 11:41:19 +02:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |     if (! parseBlockDeclaration(declaration)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 11:41:19 +02:00
										 |  |  |     if (SimpleDeclarationAST *simpleDeclaration = declaration->asSimpleDeclaration()) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (! simpleDeclaration->decl_specifier_list) { | 
					
						
							| 
									
										
										
										
											2009-10-20 11:41:19 +02:00
										 |  |  |             rewind(start); | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     DeclarationStatementAST *ast = new (_pool) DeclarationStatementAST; | 
					
						
							|  |  |  |     ast->declaration = declaration; | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::lookAtCVQualifier() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_CONST: | 
					
						
							|  |  |  |     case T_VOLATILE: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::lookAtFunctionSpecifier() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_INLINE: | 
					
						
							|  |  |  |     case T_VIRTUAL: | 
					
						
							|  |  |  |     case T_EXPLICIT: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::lookAtStorageClassSpecifier() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_FRIEND: | 
					
						
							|  |  |  |     case T_AUTO: | 
					
						
							|  |  |  |     case T_REGISTER: | 
					
						
							|  |  |  |     case T_STATIC: | 
					
						
							|  |  |  |     case T_EXTERN: | 
					
						
							|  |  |  |     case T_MUTABLE: | 
					
						
							|  |  |  |     case T_TYPEDEF: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::lookAtBuiltinTypeSpecifier() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_CHAR: | 
					
						
							|  |  |  |     case T_WCHAR_T: | 
					
						
							|  |  |  |     case T_BOOL: | 
					
						
							|  |  |  |     case T_SHORT: | 
					
						
							|  |  |  |     case T_INT: | 
					
						
							|  |  |  |     case T_LONG: | 
					
						
							|  |  |  |     case T_SIGNED: | 
					
						
							|  |  |  |     case T_UNSIGNED: | 
					
						
							|  |  |  |     case T_FLOAT: | 
					
						
							|  |  |  |     case T_DOUBLE: | 
					
						
							|  |  |  |     case T_VOID: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     // [gcc] extensions
 | 
					
						
							|  |  |  |     case T___TYPEOF__: | 
					
						
							|  |  |  |     case T___ATTRIBUTE__: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::lookAtClassKey() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_CLASS: | 
					
						
							|  |  |  |     case T_STRUCT: | 
					
						
							|  |  |  |     case T_UNION: | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseAttributeSpecifier(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() != T___ATTRIBUTE__) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     AttributeSpecifierAST *ast = new (_pool) AttributeSpecifierAST; | 
					
						
							|  |  |  |     ast->attribute_token = consumeToken(); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->first_lparen_token); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->second_lparen_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     parseAttributeList(ast->attribute_list); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     match(T_RPAREN, &ast->first_rparen_token); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->second_rparen_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:03:37 +01:00
										 |  |  | bool Parser::parseAttributeList(AttributeListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:56:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:03:37 +01:00
										 |  |  |     AttributeListAST **iter = &node; | 
					
						
							| 
									
										
										
										
											2009-11-10 14:56:37 +01:00
										 |  |  |     while (LA() == T_CONST || LA() == T_IDENTIFIER) { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:03:37 +01:00
										 |  |  |         *iter = new (_pool) AttributeListAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_CONST) { | 
					
						
							|  |  |  |             AttributeAST *attr = new (_pool) AttributeAST; | 
					
						
							|  |  |  |             attr->identifier_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             (*iter)->value = attr; | 
					
						
							|  |  |  |             iter = &(*iter)->next; | 
					
						
							|  |  |  |         } else if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |             AttributeAST *attr = new (_pool) AttributeAST; | 
					
						
							|  |  |  |             attr->identifier_token = consumeToken(); | 
					
						
							|  |  |  |             if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |                 attr->lparen_token = consumeToken(); | 
					
						
							|  |  |  |                 parseExpressionList(attr->expression_list); | 
					
						
							|  |  |  |                 match(T_RPAREN, &attr->rparen_token); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             (*iter)->value = attr; | 
					
						
							|  |  |  |             iter = &(*iter)->next; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() != T_COMMA) | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:56:37 +01:00
										 |  |  |         consumeToken(); // skip T_COMMA
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-11-10 14:56:37 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseBuiltinTypeSpecifier(SpecifierListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |         return parseAttributeSpecifier(node); | 
					
						
							|  |  |  |     } else if (LA() == T___TYPEOF__) { | 
					
						
							|  |  |  |         TypeofSpecifierAST *ast = new (_pool) TypeofSpecifierAST; | 
					
						
							|  |  |  |         ast->typeof_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |             unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |             if (parseTypeId(ast->expression) && LA() == T_RPAREN) { | 
					
						
							| 
									
										
										
										
											2009-05-11 16:34:35 +02:00
										 |  |  |                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                 ast->rparen_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                 node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             rewind(lparen_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         parseUnaryExpression(ast->expression); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } else if (lookAtBuiltinTypeSpecifier()) { | 
					
						
							|  |  |  |         SimpleSpecifierAST *ast = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |         ast->specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         node = new (_pool) SpecifierListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseSimpleDeclaration(DeclarationAST *&node, | 
					
						
							|  |  |  |                                     bool acceptStructDeclarator) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |     unsigned qt_invokable_token = 0; | 
					
						
							| 
									
										
										
										
											2010-02-04 12:03:34 +01:00
										 |  |  |     if (acceptStructDeclarator | 
					
						
							|  |  |  |             && (LA() == T_Q_SIGNAL || LA() == T_Q_SLOT || LA() == T_Q_INVOKABLE)) | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |         qt_invokable_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     // parse a simple declaration, a function definition,
 | 
					
						
							|  |  |  |     // or a contructor declaration.
 | 
					
						
							|  |  |  |     bool has_type_specifier = false; | 
					
						
							|  |  |  |     bool has_complex_type_specifier = false; | 
					
						
							|  |  |  |     unsigned startOfNamedTypeSpecifier = 0; | 
					
						
							|  |  |  |     NameAST *named_type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *decl_specifier_seq = 0, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |          **decl_specifier_seq_ptr = &decl_specifier_seq; | 
					
						
							|  |  |  |     for (;;) { | 
					
						
							|  |  |  |         if (lookAtCVQualifier() || lookAtFunctionSpecifier() | 
					
						
							|  |  |  |                 || lookAtStorageClassSpecifier()) { | 
					
						
							|  |  |  |             SimpleSpecifierAST *spec = new (_pool) SimpleSpecifierAST; | 
					
						
							|  |  |  |             spec->specifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |             *decl_specifier_seq_ptr = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |         } else if (LA() == T___ATTRIBUTE__) { | 
					
						
							|  |  |  |             parseAttributeSpecifier(*decl_specifier_seq_ptr); | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |         } else if (! named_type_specifier && ! has_complex_type_specifier && lookAtBuiltinTypeSpecifier()) { | 
					
						
							|  |  |  |             parseBuiltinTypeSpecifier(*decl_specifier_seq_ptr); | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else if (! has_type_specifier && (LA() == T_COLON_COLON || | 
					
						
							|  |  |  |                                             LA() == T_IDENTIFIER)) { | 
					
						
							|  |  |  |             startOfNamedTypeSpecifier = cursor(); | 
					
						
							|  |  |  |             if (parseName(named_type_specifier)) { | 
					
						
							|  |  |  |                 NamedTypeSpecifierAST *spec = new (_pool) NamedTypeSpecifierAST; | 
					
						
							|  |  |  |                 spec->name = named_type_specifier; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                 *decl_specifier_seq_ptr = new (_pool) SpecifierListAST(spec); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |                 has_type_specifier = true; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 rewind(startOfNamedTypeSpecifier); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } else if (! has_type_specifier && LA() == T_ENUM) { | 
					
						
							|  |  |  |             unsigned startOfTypeSpecifier = cursor(); | 
					
						
							|  |  |  |             if (! parseElaboratedTypeSpecifier(*decl_specifier_seq_ptr) || LA() == T_LBRACE) { | 
					
						
							|  |  |  |                 rewind(startOfTypeSpecifier); | 
					
						
							|  |  |  |                 if (! parseEnumSpecifier(*decl_specifier_seq_ptr)) { | 
					
						
							|  |  |  |                     _translationUnit->error(startOfTypeSpecifier, | 
					
						
							|  |  |  |                                             "expected an enum specifier"); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 has_complex_type_specifier = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else if (! has_type_specifier && LA() == T_TYPENAME) { | 
					
						
							|  |  |  |             unsigned startOfElaboratedTypeSpecifier = cursor(); | 
					
						
							|  |  |  |             if (! parseElaboratedTypeSpecifier(*decl_specifier_seq_ptr)) { | 
					
						
							|  |  |  |                 _translationUnit->error(startOfElaboratedTypeSpecifier, | 
					
						
							|  |  |  |                                         "expected an elaborated type specifier"); | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else if (! has_type_specifier && lookAtClassKey()) { | 
					
						
							|  |  |  |             unsigned startOfTypeSpecifier = cursor(); | 
					
						
							|  |  |  |             if (! parseElaboratedTypeSpecifier(*decl_specifier_seq_ptr) || | 
					
						
							|  |  |  |                 (LA() == T_COLON || LA() == T_LBRACE || (LA(0) == T_IDENTIFIER && LA(1) == T_IDENTIFIER && | 
					
						
							|  |  |  |                                                          (LA(2) == T_COLON || LA(2) == T_LBRACE)))) { | 
					
						
							|  |  |  |                 rewind(startOfTypeSpecifier); | 
					
						
							|  |  |  |                 if (! parseClassSpecifier(*decl_specifier_seq_ptr)) { | 
					
						
							|  |  |  |                     _translationUnit->error(startOfTypeSpecifier, | 
					
						
							|  |  |  |                                             "wrong type specifier"); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 has_complex_type_specifier = true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             decl_specifier_seq_ptr = &(*decl_specifier_seq_ptr)->next; | 
					
						
							|  |  |  |             has_type_specifier = true; | 
					
						
							|  |  |  |         } else | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DeclaratorListAST *declarator_list = 0, | 
					
						
							|  |  |  |         **declarator_ptr = &declarator_list; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     DeclaratorAST *declarator = 0; | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (LA() != T_SEMICOLON) { | 
					
						
							|  |  |  |         const bool maybeCtor = (LA() == T_LPAREN && named_type_specifier); | 
					
						
							|  |  |  |         if (! parseInitDeclarator(declarator, acceptStructDeclarator) && maybeCtor) { | 
					
						
							|  |  |  |             rewind(startOfNamedTypeSpecifier); | 
					
						
							|  |  |  |             named_type_specifier = 0; | 
					
						
							|  |  |  |             // pop the named type specifier from the decl-specifier-seq
 | 
					
						
							|  |  |  |             SpecifierListAST **spec_ptr = &decl_specifier_seq; | 
					
						
							|  |  |  |             for (; *spec_ptr; spec_ptr = &(*spec_ptr)->next) { | 
					
						
							|  |  |  |                 if (! (*spec_ptr)->next) { | 
					
						
							|  |  |  |                     *spec_ptr = 0; | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  |             if (! parseInitDeclarator(declarator, acceptStructDeclarator)) | 
					
						
							|  |  |  |                 return false; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |     // if there is no valid declarator
 | 
					
						
							|  |  |  |     // and it doesn't look like a fwd or a class declaration
 | 
					
						
							|  |  |  |     // then it's not a declarations
 | 
					
						
							|  |  |  |     if (! declarator && ! maybeForwardOrClassDeclaration(decl_specifier_seq)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     DeclaratorAST *firstDeclarator = declarator; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (declarator) { | 
					
						
							|  |  |  |         *declarator_ptr = new (_pool) DeclaratorListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |         (*declarator_ptr)->value = declarator; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         declarator_ptr = &(*declarator_ptr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_COMMA || LA() == T_SEMICOLON || has_complex_type_specifier) { | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             consumeToken(); // consume T_COMMA
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             declarator = 0; | 
					
						
							|  |  |  |             if (parseInitDeclarator(declarator, acceptStructDeclarator)) { | 
					
						
							|  |  |  |                 *declarator_ptr = new (_pool) DeclaratorListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:34:29 +01:00
										 |  |  |                 (*declarator_ptr)->value = declarator; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 declarator_ptr = &(*declarator_ptr)->next; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         SimpleDeclarationAST *ast = new (_pool) SimpleDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |         ast->qt_invokable_token = qt_invokable_token; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | #ifdef ICHECK_BUILD
 | 
					
						
							|  |  |  |         ast->invoke_token = invoke_token; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->decl_specifier_list = decl_specifier_seq; | 
					
						
							|  |  |  |         ast->declarator_list = declarator_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } else if (! _inFunctionBody && declarator && (LA() == T_COLON || LA() == T_LBRACE || LA() == T_TRY)) { | 
					
						
							|  |  |  |         CtorInitializerAST *ctor_initializer = 0; | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  |         bool hasCtorInitializer = false; | 
					
						
							|  |  |  |         if (LA() == T_COLON) { | 
					
						
							|  |  |  |             hasCtorInitializer = true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             parseCtorInitializer(ctor_initializer); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-07 14:50:54 +01:00
										 |  |  |             if (LA() != T_LBRACE) { | 
					
						
							|  |  |  |                 const unsigned pos = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 for (int n = 0; n < 3 && LA(); consumeToken(), ++n) | 
					
						
							|  |  |  |                     if (LA() == T_LBRACE) | 
					
						
							|  |  |  |                         break; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 if (LA() != T_LBRACE) { | 
					
						
							|  |  |  |                     _translationUnit->error(pos, "unexpected token `%s'", _translationUnit->spell(pos)); | 
					
						
							|  |  |  |                     rewind(pos); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LBRACE || hasCtorInitializer) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             FunctionDefinitionAST *ast = new (_pool) FunctionDefinitionAST; | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |             ast->qt_invokable_token = qt_invokable_token; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | #ifdef ICHECK_BUILD
 | 
					
						
							|  |  |  |             ast->invoke_token = invoke_token; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->decl_specifier_list = decl_specifier_seq; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->declarator = firstDeclarator; | 
					
						
							|  |  |  |             ast->ctor_initializer = ctor_initializer; | 
					
						
							|  |  |  |             parseFunctionBody(ast->function_body); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; // recognized a function definition.
 | 
					
						
							|  |  |  |         } else if (LA() == T_TRY) { | 
					
						
							|  |  |  |             FunctionDefinitionAST *ast = new (_pool) FunctionDefinitionAST; | 
					
						
							| 
									
										
										
										
											2009-03-25 11:50:17 +01:00
										 |  |  |             ast->qt_invokable_token = qt_invokable_token; | 
					
						
							| 
									
										
										
										
											2010-01-19 15:26:08 +10:00
										 |  |  | #ifdef ICHECK_BUILD
 | 
					
						
							|  |  |  |             ast->invoke_token = invoke_token; | 
					
						
							|  |  |  | #endif
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->decl_specifier_list = decl_specifier_seq; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             ast->declarator = firstDeclarator; | 
					
						
							|  |  |  |             ast->ctor_initializer = ctor_initializer; | 
					
						
							|  |  |  |             parseTryBlockStatement(ast->function_body); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; // recognized a function definition.
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _translationUnit->error(cursor(), "unexpected token `%s'", tok().spell()); | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::maybeForwardOrClassDeclaration(SpecifierListAST *decl_specifier_seq) const | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     // look at the decl_specifier for possible fwd or class declarations.
 | 
					
						
							| 
									
										
										
										
											2010-01-06 16:15:05 +01:00
										 |  |  |     if (SpecifierListAST *it = decl_specifier_seq) { | 
					
						
							|  |  |  |         while (it) { | 
					
						
							|  |  |  |             SimpleSpecifierAST *spec = it->value->asSimpleSpecifier(); | 
					
						
							|  |  |  |             if (spec && _translationUnit->tokenKind(spec->specifier_token) == T_FRIEND) | 
					
						
							|  |  |  |                 it = it->next; | 
					
						
							|  |  |  |             else | 
					
						
							|  |  |  |                 break; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (it) { | 
					
						
							|  |  |  |             SpecifierAST *spec = it->value; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (! it->next && (spec->asElaboratedTypeSpecifier() || | 
					
						
							|  |  |  |                                spec->asEnumSpecifier() || | 
					
						
							|  |  |  |                                spec->asClassSpecifier())) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-10-19 17:47:49 +02:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseFunctionBody(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (_translationUnit->skipFunctionBody()) { | 
					
						
							|  |  |  |         unsigned token_lbrace = 0; | 
					
						
							|  |  |  |         match(T_LBRACE, &token_lbrace); | 
					
						
							|  |  |  |         if (! token_lbrace) | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         const Token &tk = _translationUnit->tokenAt(token_lbrace); | 
					
						
							|  |  |  |         if (tk.close_brace) | 
					
						
							|  |  |  |             rewind(tk.close_brace); | 
					
						
							|  |  |  |         unsigned token_rbrace = 0; | 
					
						
							|  |  |  |         match(T_RBRACE, &token_rbrace); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     _inFunctionBody = true; | 
					
						
							|  |  |  |     const bool parsed = parseCompoundStatement(node); | 
					
						
							|  |  |  |     _inFunctionBody = false; | 
					
						
							|  |  |  |     return parsed; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTryBlockStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TRY) { | 
					
						
							|  |  |  |         TryBlockStatementAST *ast = new (_pool) TryBlockStatementAST; | 
					
						
							|  |  |  |         ast->try_token = consumeToken(); | 
					
						
							|  |  |  |         parseCompoundStatement(ast->statement); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:15:51 +01:00
										 |  |  |         CatchClauseListAST **catch_clause_ptr = &ast->catch_clause_list; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         while (parseCatchClause(*catch_clause_ptr)) | 
					
						
							|  |  |  |             catch_clause_ptr = &(*catch_clause_ptr)->next; | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 15:15:51 +01:00
										 |  |  | bool Parser::parseCatchClause(CatchClauseListAST *&node) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_CATCH) { | 
					
						
							|  |  |  |         CatchClauseAST *ast = new (_pool) CatchClauseAST; | 
					
						
							|  |  |  |         ast->catch_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseExceptionDeclaration(ast->exception_declaration); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         parseCompoundStatement(ast->statement); | 
					
						
							| 
									
										
										
										
											2009-11-10 15:15:51 +01:00
										 |  |  |         node = new (_pool) CatchClauseListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExceptionDeclaration(ExceptionDeclarationAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_DOT_DOT_DOT) { | 
					
						
							|  |  |  |         ExceptionDeclarationAST *ast = new (_pool) ExceptionDeclarationAST; | 
					
						
							|  |  |  |         ast->dot_dot_dot_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseTypeSpecifier(type_specifier)) { | 
					
						
							|  |  |  |         ExceptionDeclarationAST *ast = new (_pool) ExceptionDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->type_specifier_list = type_specifier; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         parseDeclaratorOrAbstractDeclarator(ast->declarator); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseBoolLiteral(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TRUE || LA() == T_FALSE) { | 
					
						
							|  |  |  |         BoolLiteralAST *ast = new (_pool) BoolLiteralAST; | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->literal_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseNumericLiteral(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-10 15:44:47 +02:00
										 |  |  |     if (LA() == T_NUMERIC_LITERAL  || | 
					
						
							|  |  |  |         LA() == T_CHAR_LITERAL     || | 
					
						
							|  |  |  |         LA() == T_WIDE_CHAR_LITERAL) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         NumericLiteralAST *ast = new (_pool) NumericLiteralAST; | 
					
						
							| 
									
										
										
										
											2009-06-05 09:44:25 +02:00
										 |  |  |         ast->literal_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseThisExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_THIS) { | 
					
						
							|  |  |  |         ThisExpressionAST *ast = new (_pool) ThisExpressionAST; | 
					
						
							|  |  |  |         ast->this_token = consumeToken(); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parsePrimaryExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_STRING_LITERAL: | 
					
						
							|  |  |  |     case T_WIDE_STRING_LITERAL: | 
					
						
							|  |  |  |         return parseStringLiteral(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-10 15:44:47 +02:00
										 |  |  |     case T_CHAR_LITERAL: // ### FIXME don't use NumericLiteral for chars
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     case T_WIDE_CHAR_LITERAL: | 
					
						
							| 
									
										
										
										
											2009-07-10 15:44:47 +02:00
										 |  |  |     case T_NUMERIC_LITERAL: | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return parseNumericLiteral(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TRUE: | 
					
						
							|  |  |  |     case T_FALSE: | 
					
						
							|  |  |  |         return parseBoolLiteral(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_THIS: | 
					
						
							|  |  |  |         return parseThisExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_LPAREN: | 
					
						
							| 
									
										
										
										
											2010-02-04 14:55:18 +01:00
										 |  |  |         if (LA(2) == T_LBRACE) { | 
					
						
							|  |  |  |             // GNU extension: '(' '{' statement-list '}' ')'
 | 
					
						
							|  |  |  |             CompoundExpressionAST *ast = new (_pool) CompoundExpressionAST; | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |             StatementAST *statement = 0; | 
					
						
							|  |  |  |             parseCompoundStatement(statement); | 
					
						
							| 
									
										
										
										
											2010-03-23 10:22:41 +01:00
										 |  |  |             ast->statement = statement->asCompoundStatement(); | 
					
						
							| 
									
										
										
										
											2010-02-04 14:55:18 +01:00
										 |  |  |             match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return parseNestedExpression(node); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case T_SIGNAL: | 
					
						
							|  |  |  |     case T_SLOT: | 
					
						
							|  |  |  |         return parseQtMethod(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-03-24 12:54:25 +01:00
										 |  |  |     case T_LBRACKET: { | 
					
						
							|  |  |  |         const unsigned lbracket_token = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (_cxx0xEnabled) { | 
					
						
							|  |  |  |             if (parseLambdaExpression(node)) | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (_objCEnabled) { | 
					
						
							|  |  |  |             rewind(lbracket_token); | 
					
						
							|  |  |  |             return parseObjCExpression(node); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_STRING_LITERAL: | 
					
						
							|  |  |  |     case T_AT_ENCODE: | 
					
						
							|  |  |  |     case T_AT_PROTOCOL: | 
					
						
							|  |  |  |     case T_AT_SELECTOR: | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  |         return parseObjCExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     default: { | 
					
						
							|  |  |  |         NameAST *name = 0; | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  |         if (parseNameId(name)) { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             node = name; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  |         break; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } // default
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  | bool Parser::parseObjCExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_AT_ENCODE: | 
					
						
							| 
									
										
										
										
											2009-02-03 14:29:30 +01:00
										 |  |  |         return parseObjCEncodeExpression(node); | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_PROTOCOL: | 
					
						
							| 
									
										
										
										
											2009-02-03 14:33:25 +01:00
										 |  |  |         return parseObjCProtocolExpression(node); | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_SELECTOR: | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |         return parseObjCSelectorExpression(node); | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_LBRACKET: | 
					
						
							|  |  |  |         return parseObjCMessageExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_AT_STRING_LITERAL: | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |         return parseObjCStringLiteral(node); | 
					
						
							| 
									
										
										
										
											2009-02-03 14:26:35 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  | bool Parser::parseObjCStringLiteral(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_STRING_LITERAL) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     StringLiteralAST **ast = reinterpret_cast<StringLiteralAST **> (&node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     while (LA() == T_AT_STRING_LITERAL) { | 
					
						
							|  |  |  |         *ast = new (_pool) StringLiteralAST; | 
					
						
							|  |  |  |         (*ast)->literal_token = consumeToken(); | 
					
						
							|  |  |  |         ast = &(*ast)->next; | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |     } | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:53:05 +02:00
										 |  |  | bool Parser::parseObjCSynchronizedStatement(StatementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_SYNCHRONIZED) | 
					
						
							| 
									
										
										
										
											2009-07-31 16:53:05 +02:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ObjCSynchronizedStatementAST *ast = new (_pool) ObjCSynchronizedStatementAST; | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:53:05 +02:00
										 |  |  |     ast->synchronized_token = consumeToken(); | 
					
						
							|  |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |     parseExpression(ast->synchronized_object); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |     parseStatement(ast->statement); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  | bool Parser::parseObjCEncodeExpression(ExpressionAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:29:30 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_ENCODE) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:29:30 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  |     ObjCEncodeExpressionAST *ast = new (_pool) ObjCEncodeExpressionAST; | 
					
						
							|  |  |  |     ast->encode_token = consumeToken(); | 
					
						
							|  |  |  |     parseObjCTypeName(ast->type_name); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-02-03 14:29:30 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:18:05 +02:00
										 |  |  | bool Parser::parseObjCProtocolExpression(ExpressionAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:33:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_PROTOCOL) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:33:25 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:18:05 +02:00
										 |  |  |     ObjCProtocolExpressionAST *ast = new (_pool) ObjCProtocolExpressionAST; | 
					
						
							|  |  |  |     ast->protocol_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &ast->identifier_token); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 12:18:05 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-02-03 14:33:25 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | bool Parser::parseObjCSelectorExpression(ExpressionAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_SELECTOR) | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     ObjCSelectorExpressionAST *ast = new (_pool) ObjCSelectorExpressionAST; | 
					
						
							|  |  |  |     ast->selector_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2009-02-06 15:51:17 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							|  |  |  |     if (LA() == T_COLON) { | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *args = new (_pool) ObjCSelectorAST; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ast->selector = args; | 
					
						
							|  |  |  |         ObjCSelectorArgumentListAST *last = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         args->selector_argument_list = last; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:45:20 +01:00
										 |  |  |         last->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         last->value->name_token = identifier_token; | 
					
						
							|  |  |  |         last->value->colon_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while (LA() != T_RPAREN) { | 
					
						
							|  |  |  |             last->next = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |             last = last->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:45:20 +01:00
										 |  |  |             last->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             match(T_IDENTIFIER, &last->value->name_token); | 
					
						
							|  |  |  |             match(T_COLON, &last->value->colon_token); | 
					
						
							| 
									
										
										
										
											2009-02-06 15:51:17 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     } else { | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *args = new (_pool) ObjCSelectorAST; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ast->selector = args; | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         args->selector_argument_list = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |         args->selector_argument_list->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         args->selector_argument_list->value->name_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-02-06 15:51:17 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-02-03 14:42:45 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | bool Parser::parseObjCMessageExpression(ExpressionAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |     if (LA() != T_LBRACKET) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |     unsigned lbracket_token = consumeToken(); | 
					
						
							|  |  |  |     ExpressionAST *receiver_expression = 0; | 
					
						
							|  |  |  |     ObjCSelectorAST *selector = 0; | 
					
						
							|  |  |  |     ObjCMessageArgumentListAST *argument_list = 0; | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |     if (parseObjCMessageReceiver(receiver_expression) && | 
					
						
							|  |  |  |         parseObjCMessageArguments(selector, argument_list)) { | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ObjCMessageExpressionAST *ast = new (_pool) ObjCMessageExpressionAST; | 
					
						
							|  |  |  |         ast->lbracket_token = lbracket_token; | 
					
						
							|  |  |  |         ast->receiver_expression = receiver_expression; | 
					
						
							|  |  |  |         ast->selector = selector; | 
					
						
							|  |  |  |         ast->argument_list = argument_list; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_RBRACKET, &ast->rbracket_token); | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     rewind(start); | 
					
						
							|  |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-15 12:11:07 +02:00
										 |  |  | bool Parser::parseObjCMessageReceiver(ExpressionAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-15 12:11:07 +02:00
										 |  |  |     return parseExpression(node); | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  | bool Parser::parseObjCMessageArguments(ObjCSelectorAST *&selNode, ObjCMessageArgumentListAST *& argNode) | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |     if (LA() == T_RBRACKET) | 
					
						
							|  |  |  |         return false; // nothing to do.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |     ObjCSelectorArgumentAST *selectorArgument = 0; | 
					
						
							|  |  |  |     ObjCMessageArgumentAST *messageArgument = 0; | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |     if (parseObjCSelectorArg(selectorArgument, messageArgument)) { | 
					
						
							|  |  |  |         ObjCSelectorArgumentListAST *selAst = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:45:20 +01:00
										 |  |  |         selAst->value = selectorArgument; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |         ObjCSelectorArgumentListAST *lastSelector = selAst; | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |         ObjCMessageArgumentListAST *argAst = new (_pool) ObjCMessageArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:43:15 +01:00
										 |  |  |         argAst->value = messageArgument; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |         ObjCMessageArgumentListAST *lastArgument = argAst; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         while (parseObjCSelectorArg(selectorArgument, messageArgument)) { | 
					
						
							| 
									
										
										
										
											2009-02-04 11:45:17 +01:00
										 |  |  |             // accept the selector args.
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |             lastSelector->next = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |             lastSelector = lastSelector->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:45:20 +01:00
										 |  |  |             lastSelector->value = selectorArgument; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  |             lastArgument->next = new (_pool) ObjCMessageArgumentListAST; | 
					
						
							|  |  |  |             lastArgument = lastArgument->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:43:15 +01:00
										 |  |  |             lastArgument->value = messageArgument; | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             ExpressionAST **lastExpression = &lastArgument->value->parameter_value_expression; | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |             while (LA() == T_COMMA) { | 
					
						
							|  |  |  |                 BinaryExpressionAST *binaryExpression = new (_pool) BinaryExpressionAST; | 
					
						
							|  |  |  |                 binaryExpression->left_expression = *lastExpression; | 
					
						
							|  |  |  |                 binaryExpression->binary_op_token = consumeToken(); // T_COMMA
 | 
					
						
							|  |  |  |                 parseAssignmentExpression(binaryExpression->right_expression); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |                 lastExpression = &binaryExpression->right_expression; | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *selWithArgs = new (_pool) ObjCSelectorAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         selWithArgs->selector_argument_list = selAst; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         selNode = selWithArgs; | 
					
						
							|  |  |  |         argNode = argAst; | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-02-04 11:45:17 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         rewind(start); | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |         unsigned name_token = 0; | 
					
						
							|  |  |  |         if (!parseObjCSelector(name_token)) | 
					
						
							|  |  |  |             return false; | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *sel = new (_pool) ObjCSelectorAST; | 
					
						
							|  |  |  |         sel->selector_argument_list = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |         sel->selector_argument_list->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         sel->selector_argument_list->value->name_token = name_token; | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |         selNode = sel; | 
					
						
							|  |  |  |         argNode = 0; | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-19 14:49:51 +02:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  | bool Parser::parseObjCSelectorArg(ObjCSelectorArgumentAST *&selNode, ObjCMessageArgumentAST *&argNode) | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  |     unsigned selector_token = 0; | 
					
						
							|  |  |  |     if (!parseObjCSelector(selector_token)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |     if (LA() != T_COLON) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |     selNode = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |     selNode->name_token = selector_token; | 
					
						
							|  |  |  |     selNode->colon_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-31 16:03:48 +02:00
										 |  |  |     argNode = new (_pool) ObjCMessageArgumentAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     ExpressionAST **expr = &argNode->parameter_value_expression; | 
					
						
							| 
									
										
										
										
											2009-08-10 12:44:52 +02:00
										 |  |  |     unsigned expressionStart = cursor(); | 
					
						
							| 
									
										
										
										
											2009-12-18 16:57:15 +01:00
										 |  |  |     if (parseAssignmentExpression(*expr) && LA() == T_COLON && (*expr)->asCastExpression()) { | 
					
						
							| 
									
										
										
										
											2009-08-10 12:44:52 +02:00
										 |  |  |         rewind(expressionStart); | 
					
						
							| 
									
										
										
										
											2009-12-18 16:57:15 +01:00
										 |  |  |         parseUnaryExpression(*expr); | 
					
						
							|  |  |  |         //
 | 
					
						
							| 
									
										
										
										
											2009-08-10 12:44:52 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-03 18:47:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  | bool Parser::parseNameId(NameAST *&name) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  |     unsigned start = cursor(); | 
					
						
							|  |  |  |     if (! parseName(name)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-09 12:32:04 +01:00
										 |  |  |     if (LA() == T_RPAREN || LA() == T_COMMA) | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  |     QualifiedNameAST *qualified_name_id = name->asQualifiedName(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TemplateIdAST *template_id = 0; | 
					
						
							|  |  |  |     if (qualified_name_id) { | 
					
						
							|  |  |  |         if (NameAST *unqualified_name = qualified_name_id->unqualified_name) | 
					
						
							|  |  |  |             template_id = unqualified_name->asTemplateId(); | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         template_id = name->asTemplateId(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (! template_id) | 
					
						
							|  |  |  |         return true; // it's not a template-id, there's nothing to rewind.
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     else if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         // a template-id followed by a T_LPAREN
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         if (TemplateArgumentListAST *template_arguments = template_id->template_argument_list) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:37:46 +01:00
										 |  |  |             if (! template_arguments->next && template_arguments->value && | 
					
						
							|  |  |  |                     template_arguments->value->asBinaryExpression()) { | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-19 11:59:26 +02:00
										 |  |  |                 unsigned saved = cursor(); | 
					
						
							|  |  |  |                 ExpressionAST *expr = 0; | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-19 11:59:26 +02:00
										 |  |  |                 bool blocked = blockErrors(true); | 
					
						
							| 
									
										
										
										
											2009-06-19 14:48:07 +02:00
										 |  |  |                 bool lookAtCastExpression = parseCastExpression(expr); | 
					
						
							|  |  |  |                 (void) blockErrors(blocked); | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-06-19 14:48:07 +02:00
										 |  |  |                 if (lookAtCastExpression) { | 
					
						
							| 
									
										
										
										
											2009-06-19 11:59:26 +02:00
										 |  |  |                     if (CastExpressionAST *cast_expression = expr->asCastExpression()) { | 
					
						
							|  |  |  |                         if (cast_expression->lparen_token && cast_expression->rparen_token | 
					
						
							|  |  |  |                                 && cast_expression->type_id && cast_expression->expression) { | 
					
						
							|  |  |  |                             rewind(start); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                             name = 0; | 
					
						
							|  |  |  |                             return parseName(name, false); | 
					
						
							|  |  |  |                         } | 
					
						
							|  |  |  |                     } | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |                 rewind(saved); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_COMMA: | 
					
						
							|  |  |  |     case T_SEMICOLON: | 
					
						
							|  |  |  |     case T_LBRACKET: | 
					
						
							|  |  |  |     case T_LPAREN: | 
					
						
							| 
									
										
										
										
											2009-06-19 12:39:11 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     case T_IDENTIFIER: | 
					
						
							|  |  |  |     case T_STATIC_CAST: | 
					
						
							|  |  |  |     case T_DYNAMIC_CAST: | 
					
						
							|  |  |  |     case T_REINTERPRET_CAST: | 
					
						
							|  |  |  |     case T_CONST_CAST: | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  |         rewind(start); | 
					
						
							|  |  |  |         return parseName(name, false); | 
					
						
							| 
									
										
										
										
											2009-10-20 15:29:00 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         if (tok().isLiteral() || tok().isOperator()) { | 
					
						
							|  |  |  |             rewind(start); | 
					
						
							|  |  |  |             return parseName(name, false); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-26 14:28:44 +01:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseNestedExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         bool previousTemplateArguments = switchTemplateArguments(false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ExpressionAST *expression = 0; | 
					
						
							|  |  |  |         if (parseExpression(expression) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             NestedExpressionAST *ast = new (_pool) NestedExpressionAST; | 
					
						
							|  |  |  |             ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |             ast->expression = expression; | 
					
						
							|  |  |  |             ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             (void) switchTemplateArguments(previousTemplateArguments); | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         (void) switchTemplateArguments(previousTemplateArguments); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCppCastExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_DYNAMIC_CAST     || LA() == T_STATIC_CAST || | 
					
						
							|  |  |  |         LA() == T_REINTERPRET_CAST || LA() == T_CONST_CAST) { | 
					
						
							|  |  |  |         CppCastExpressionAST *ast = new (_pool) CppCastExpressionAST; | 
					
						
							|  |  |  |         ast->cast_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LESS, &ast->less_token); | 
					
						
							|  |  |  |         parseTypeId(ast->type_id); | 
					
						
							|  |  |  |         match(T_GREATER, &ast->greater_token); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         parseExpression(ast->expression); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // typename ::opt  nested-name-specifier identifier ( expression-listopt )
 | 
					
						
							|  |  |  | // typename ::opt  nested-name-specifier templateopt  template-id ( expression-listopt )
 | 
					
						
							|  |  |  | bool Parser::parseTypenameCallExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TYPENAME) { | 
					
						
							|  |  |  |         unsigned typename_token = consumeToken(); | 
					
						
							|  |  |  |         NameAST *name = 0; | 
					
						
							|  |  |  |         if (parseName(name) && LA() == T_LPAREN) { | 
					
						
							|  |  |  |             TypenameCallExpressionAST *ast = new (_pool) TypenameCallExpressionAST; | 
					
						
							|  |  |  |             ast->typename_token = typename_token; | 
					
						
							|  |  |  |             ast->name = name; | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |             parseExpressionList(ast->expression_list); | 
					
						
							|  |  |  |             match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // typeid ( expression )
 | 
					
						
							|  |  |  | // typeid ( type-id )
 | 
					
						
							|  |  |  | bool Parser::parseTypeidExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_TYPEID) { | 
					
						
							|  |  |  |         TypeidExpressionAST *ast = new (_pool) TypeidExpressionAST; | 
					
						
							|  |  |  |         ast->typeid_token = consumeToken(); | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) | 
					
						
							|  |  |  |             ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |         unsigned saved = cursor(); | 
					
						
							|  |  |  |         if (! (parseTypeId(ast->expression) && LA() == T_RPAREN)) { | 
					
						
							|  |  |  |             rewind(saved); | 
					
						
							|  |  |  |             parseExpression(ast->expression); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCorePostfixExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-10-20 15:01:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_DYNAMIC_CAST: | 
					
						
							|  |  |  |     case T_STATIC_CAST: | 
					
						
							|  |  |  |     case T_REINTERPRET_CAST: | 
					
						
							|  |  |  |     case T_CONST_CAST: | 
					
						
							|  |  |  |         return parseCppCastExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TYPENAME: | 
					
						
							|  |  |  |         return parseTypenameCallExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TYPEID: | 
					
						
							|  |  |  |         return parseTypeidExpression(node); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: { | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         SpecifierListAST *type_specifier = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         bool blocked = blockErrors(true); | 
					
						
							|  |  |  |         if (lookAtBuiltinTypeSpecifier() && | 
					
						
							|  |  |  |                 parseSimpleTypeSpecifier(type_specifier) && | 
					
						
							|  |  |  |                 LA() == T_LPAREN) { | 
					
						
							|  |  |  |             unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |             ExpressionListAST *expression_list = 0; | 
					
						
							|  |  |  |             parseExpressionList(expression_list); | 
					
						
							|  |  |  |             if (LA() == T_RPAREN) { | 
					
						
							|  |  |  |                 unsigned rparen_token = consumeToken(); | 
					
						
							|  |  |  |                 TypeConstructorCallAST *ast = new (_pool) TypeConstructorCallAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |                 ast->type_specifier_list = type_specifier; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                 ast->expression_list = expression_list; | 
					
						
							|  |  |  |                 ast->rparen_token = rparen_token; | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 blockErrors(blocked); | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         rewind(start); | 
					
						
							| 
									
										
										
										
											2009-01-13 14:58:29 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         // look for compound literals
 | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |             unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |             ExpressionAST *type_id = 0; | 
					
						
							|  |  |  |             if (parseTypeId(type_id) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |                 unsigned rparen_token = consumeToken(); | 
					
						
							|  |  |  |                 if (LA() == T_LBRACE) { | 
					
						
							|  |  |  |                     blockErrors(blocked); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     CompoundLiteralAST *ast = new (_pool) CompoundLiteralAST; | 
					
						
							|  |  |  |                     ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                     ast->type_id = type_id; | 
					
						
							|  |  |  |                     ast->rparen_token = rparen_token; | 
					
						
							|  |  |  |                     parseInitializerClause(ast->initializer); | 
					
						
							|  |  |  |                     node = ast; | 
					
						
							|  |  |  |                     return true; | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             rewind(start); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         blockErrors(blocked); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         return parsePrimaryExpression(node); | 
					
						
							| 
									
										
										
										
											2009-10-20 15:01:45 +02:00
										 |  |  |     } // default
 | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parsePostfixExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (parseCorePostfixExpression(node)) { | 
					
						
							| 
									
										
										
										
											2009-11-10 14:24:32 +01:00
										 |  |  |         PostfixListAST *postfix_expressions = 0, | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             **postfix_ptr = &postfix_expressions; | 
					
						
							|  |  |  |         while (LA()) { | 
					
						
							|  |  |  |             if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |                 CallAST *ast = new (_pool) CallAST; | 
					
						
							|  |  |  |                 ast->lparen_token = consumeToken(); | 
					
						
							|  |  |  |                 parseExpressionList(ast->expression_list); | 
					
						
							|  |  |  |                 match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:24:32 +01:00
										 |  |  |                 *postfix_ptr = new (_pool) PostfixListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |             } else if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |                 ArrayAccessAST *ast = new (_pool) ArrayAccessAST; | 
					
						
							|  |  |  |                 ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  |                 parseExpression(ast->expression); | 
					
						
							|  |  |  |                 match(T_RBRACKET, &ast->rbracket_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:24:32 +01:00
										 |  |  |                 *postfix_ptr = new (_pool) PostfixListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |             } else if (LA() == T_PLUS_PLUS || LA() == T_MINUS_MINUS) { | 
					
						
							|  |  |  |                 PostIncrDecrAST *ast = new (_pool) PostIncrDecrAST; | 
					
						
							|  |  |  |                 ast->incr_decr_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:24:32 +01:00
										 |  |  |                 *postfix_ptr = new (_pool) PostfixListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |             } else if (LA() == T_DOT || LA() == T_ARROW) { | 
					
						
							|  |  |  |                 MemberAccessAST *ast = new (_pool) MemberAccessAST; | 
					
						
							|  |  |  |                 ast->access_token = consumeToken(); | 
					
						
							|  |  |  |                 if (LA() == T_TEMPLATE) | 
					
						
							|  |  |  |                     ast->template_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-26 12:58:52 +01:00
										 |  |  |                 if (! parseNameId(ast->member_name)) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                     _translationUnit->error(cursor(), "expected unqualified-id before token `%s'", | 
					
						
							|  |  |  |                                             tok().spell()); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:24:32 +01:00
										 |  |  |                 *postfix_ptr = new (_pool) PostfixListAST(ast); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 postfix_ptr = &(*postfix_ptr)->next; | 
					
						
							|  |  |  |             } else break; | 
					
						
							|  |  |  |         } // while
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (postfix_expressions) { | 
					
						
							|  |  |  |             PostfixExpressionAST *ast = new (_pool) PostfixExpressionAST; | 
					
						
							|  |  |  |             ast->base_expression = node; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->postfix_expression_list = postfix_expressions; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |             node = ast; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseUnaryExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_PLUS_PLUS: | 
					
						
							|  |  |  |     case T_MINUS_MINUS: | 
					
						
							|  |  |  |     case T_STAR: | 
					
						
							|  |  |  |     case T_AMPER: | 
					
						
							|  |  |  |     case T_PLUS: | 
					
						
							|  |  |  |     case T_MINUS: | 
					
						
							|  |  |  |     case T_EXCLAIM: { | 
					
						
							| 
									
										
										
										
											2009-12-03 18:08:14 +01:00
										 |  |  |         unsigned op = cursor(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         UnaryExpressionAST *ast = new (_pool) UnaryExpressionAST; | 
					
						
							|  |  |  |         ast->unary_op_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-12-03 18:08:14 +01:00
										 |  |  |         if (! parseCastExpression(ast->expression)) { | 
					
						
							|  |  |  |             _translationUnit->error(op, "expected expression after token `%s'", | 
					
						
							|  |  |  |                                     _translationUnit->spell(op)); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_TILDE: { | 
					
						
							|  |  |  |         if (LA(2) == T_IDENTIFIER && LA(3) == T_LPAREN) | 
					
						
							|  |  |  |             break; // prefer destructor names
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         UnaryExpressionAST *ast = new (_pool) UnaryExpressionAST; | 
					
						
							|  |  |  |         ast->unary_op_token = consumeToken(); | 
					
						
							|  |  |  |         parseCastExpression(ast->expression); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_SIZEOF: { | 
					
						
							|  |  |  |         SizeofExpressionAST *ast = new (_pool) SizeofExpressionAST; | 
					
						
							|  |  |  |         ast->sizeof_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |             unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |             if (parseTypeId(ast->expression) && LA() == T_RPAREN) { | 
					
						
							| 
									
										
										
										
											2009-06-18 14:21:28 +02:00
										 |  |  |                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                 ast->rparen_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |                 node = ast; | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 rewind(lparen_token); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parseUnaryExpression(ast->expression); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         break; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_NEW || (LA(1) == T_COLON_COLON && | 
					
						
							|  |  |  |                           LA(2) == T_NEW)) | 
					
						
							|  |  |  |         return parseNewExpression(node); | 
					
						
							|  |  |  |     else if (LA() == T_DELETE || (LA(1) == T_COLON_COLON && | 
					
						
							|  |  |  |                                   LA(2) == T_DELETE)) | 
					
						
							|  |  |  |         return parseDeleteExpression(node); | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         return parsePostfixExpression(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  | // new-placement ::= T_LPAREN expression-list T_RPAREN
 | 
					
						
							|  |  |  | bool Parser::parseNewPlacement(NewPlacementAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         ExpressionListAST *expression_list = 0; | 
					
						
							|  |  |  |         if (parseExpressionList(expression_list) && expression_list && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             unsigned rparen_token = consumeToken(); | 
					
						
							|  |  |  |             NewPlacementAST *ast = new (_pool) NewPlacementAST; | 
					
						
							|  |  |  |             ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |             ast->expression_list = expression_list; | 
					
						
							|  |  |  |             ast->rparen_token = rparen_token; | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | // new-expression ::= T_COLON_COLON? T_NEW new-placement.opt
 | 
					
						
							|  |  |  | //                    new-type-id new-initializer.opt
 | 
					
						
							|  |  |  | // new-expression ::= T_COLON_COLON? T_NEW new-placement.opt
 | 
					
						
							|  |  |  | //                    T_LPAREN type-id T_RPAREN new-initializer.opt
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | bool Parser::parseNewExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     if (! (LA() == T_NEW || (LA() == T_COLON_COLON && LA(2) == T_NEW))) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     NewExpressionAST *ast = new (_pool) NewExpressionAST; | 
					
						
							|  |  |  |     if (LA() == T_COLON_COLON) | 
					
						
							|  |  |  |         ast->scope_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     ast->new_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     NewPlacementAST *new_placement = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parseNewPlacement(new_placement)) { | 
					
						
							|  |  |  |         unsigned after_new_placement = cursor(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         NewTypeIdAST *new_type_id = 0; | 
					
						
							|  |  |  |         if (parseNewTypeId(new_type_id)) { | 
					
						
							|  |  |  |             ast->new_placement = new_placement; | 
					
						
							|  |  |  |             ast->new_type_id = new_type_id; | 
					
						
							|  |  |  |             parseNewInitializer(ast->new_initializer); | 
					
						
							|  |  |  |             // recognized new-placement.opt new-type-id new-initializer.opt
 | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |         rewind(after_new_placement); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         if (LA() == T_LPAREN) { | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |             unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |             ExpressionAST *type_id = 0; | 
					
						
							|  |  |  |             if (parseTypeId(type_id) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |                 ast->new_placement = new_placement; | 
					
						
							|  |  |  |                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                 ast->type_id = type_id; | 
					
						
							|  |  |  |                 ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |                 parseNewInitializer(ast->new_initializer); | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     rewind(ast->new_token + 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         ExpressionAST *type_id = 0; | 
					
						
							|  |  |  |         if (parseTypeId(type_id) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |             ast->type_id = type_id; | 
					
						
							|  |  |  |             ast->rparen_token = consumeToken(); | 
					
						
							|  |  |  |             parseNewInitializer(ast->new_initializer); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     parseNewTypeId(ast->new_type_id); | 
					
						
							|  |  |  |     parseNewInitializer(ast->new_initializer); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseNewTypeId(NewTypeIdAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |     SpecifierListAST *typeSpec = 0; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (! parseTypeSpecifier(typeSpec)) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     NewTypeIdAST *ast = new (_pool) NewTypeIdAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     ast->type_specifier_list = typeSpec; | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     PtrOperatorListAST **ptrop_it = &ast->ptr_operator_list; | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     while (parsePtrOperator(*ptrop_it)) | 
					
						
							|  |  |  |         ptrop_it = &(*ptrop_it)->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     NewArrayDeclaratorListAST **it = &ast->new_array_declarator_list; | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     while (parseNewArrayDeclarator(*it)) | 
					
						
							|  |  |  |         it = &(*it)->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 15:30:16 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     node = ast; | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 14:16:39 +01:00
										 |  |  | bool Parser::parseNewArrayDeclarator(NewArrayDeclaratorListAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     if (LA() != T_LBRACKET) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-10 14:43:19 +01:00
										 |  |  |     NewArrayDeclaratorAST *ast = new (_pool) NewArrayDeclaratorAST; | 
					
						
							|  |  |  |     ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  |     parseExpression(ast->expression); | 
					
						
							|  |  |  |     match(T_RBRACKET, &ast->rbracket_token); | 
					
						
							| 
									
										
										
										
											2009-11-10 14:16:39 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     node = new (_pool) NewArrayDeclaratorListAST; | 
					
						
							|  |  |  |     node->value = ast; | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseNewInitializer(NewInitializerAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         ExpressionAST *expression = 0; | 
					
						
							|  |  |  |         if (LA() == T_RPAREN || parseExpression(expression)) { | 
					
						
							|  |  |  |             NewInitializerAST *ast = new (_pool) NewInitializerAST; | 
					
						
							|  |  |  |             ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |             ast->expression = expression; | 
					
						
							|  |  |  |             match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseDeleteExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_DELETE || (LA() == T_COLON_COLON && LA(2) == T_DELETE)) { | 
					
						
							|  |  |  |         DeleteExpressionAST *ast = new (_pool) DeleteExpressionAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_COLON_COLON) | 
					
						
							|  |  |  |             ast->scope_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         ast->delete_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LBRACKET) { | 
					
						
							|  |  |  |             ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  |             match(T_RBRACKET, &ast->rbracket_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parseCastExpression(ast->expression); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCastExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         unsigned lparen_token = consumeToken(); | 
					
						
							|  |  |  |         ExpressionAST *type_id = 0; | 
					
						
							|  |  |  |         if (parseTypeId(type_id) && LA() == T_RPAREN) { | 
					
						
							|  |  |  |             unsigned rparen_token = consumeToken(); | 
					
						
							|  |  |  |             ExpressionAST *expression = 0; | 
					
						
							|  |  |  |             if (parseCastExpression(expression)) { | 
					
						
							|  |  |  |                 CastExpressionAST *ast = new (_pool) CastExpressionAST; | 
					
						
							|  |  |  |                 ast->lparen_token = lparen_token; | 
					
						
							|  |  |  |                 ast->type_id = type_id; | 
					
						
							|  |  |  |                 ast->rparen_token = rparen_token; | 
					
						
							|  |  |  |                 ast->expression = expression; | 
					
						
							|  |  |  |                 node = ast; | 
					
						
							|  |  |  |                 return true; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |         rewind(lparen_token); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return parseUnaryExpression(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parsePmExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::PointerToMember) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseMultiplicativeExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Multiplicative) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAdditiveExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Additive) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseShiftExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Shift) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseRelationalExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Relational) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseEqualityExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Equality) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAndExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::And) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExclusiveOrExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::ExclusiveOr) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseInclusiveOrExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::InclusiveOr) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLogicalAndExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::LogicalAnd) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLogicalOrExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::LogicalOr) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseConditionalExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Conditional) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseAssignmentExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_THROW) | 
					
						
							|  |  |  |         return parseThrowExpression(node); | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     else | 
					
						
							|  |  |  |         PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Assignment) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseQtMethod(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_SIGNAL || LA() == T_SLOT) { | 
					
						
							|  |  |  |         QtMethodAST *ast = new (_pool) QtMethodAST; | 
					
						
							|  |  |  |         ast->method_token = consumeToken(); | 
					
						
							|  |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							|  |  |  |         if (! parseDeclarator(ast->declarator)) | 
					
						
							|  |  |  |             _translationUnit->error(cursor(), "expected a function declarator before token `%s'", | 
					
						
							|  |  |  |                                     tok().spell()); | 
					
						
							|  |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseConstantExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     return parseConditionalExpression(node); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (_expressionDepth > MAX_EXPRESSION_DEPTH) | 
					
						
							| 
									
										
										
										
											2010-01-22 16:06:41 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     ++_expressionDepth; | 
					
						
							|  |  |  |     bool success = parseCommaExpression(node); | 
					
						
							|  |  |  |     --_expressionDepth; | 
					
						
							|  |  |  |     return success; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | void Parser::parseExpressionWithOperatorPrecedence(ExpressionAST *&lhs, int minPrecedence) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     while (precedence(tok().kind(), _templateArguments) >= minPrecedence) { | 
					
						
							|  |  |  |         const int operPrecedence = precedence(tok().kind(), _templateArguments); | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |         const int oper = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ConditionalExpressionAST *condExpr = 0; | 
					
						
							|  |  |  |         if (operPrecedence == Prec::Conditional) { | 
					
						
							|  |  |  |             condExpr = new (_pool) ConditionalExpressionAST; | 
					
						
							|  |  |  |             condExpr->question_token = oper; | 
					
						
							|  |  |  |             if (oper == T_COLON) { | 
					
						
							|  |  |  |                 // GNU extension:
 | 
					
						
							|  |  |  |                 //   logical-or-expression '?' ':' conditional-expression
 | 
					
						
							|  |  |  |                 condExpr->left_expression = 0; | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 parseExpression(condExpr->left_expression); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             match(T_COLON, &condExpr->colon_token); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |         ExpressionAST *rhs = 0; | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |         const bool isCPlusPlus = true; | 
					
						
							|  |  |  |         if (operPrecedence <= Prec::Conditional && isCPlusPlus) { | 
					
						
							|  |  |  |             // in C++ you can put a throw in the right-most expression of a conditional expression,
 | 
					
						
							|  |  |  |             // or an assignment, so some special handling:
 | 
					
						
							|  |  |  |             if (!parseAssignmentExpression(rhs)) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             // for C & all other expressions:
 | 
					
						
							|  |  |  |             if (!parseCastExpression(rhs)) | 
					
						
							|  |  |  |                 return; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |         for (int tokenKindAhead = tok().kind(), precedenceAhead = precedence(tokenKindAhead, _templateArguments); | 
					
						
							| 
									
										
										
										
											2010-02-05 13:28:42 +01:00
										 |  |  |                 (precedenceAhead > operPrecedence && isBinaryOperator(tokenKindAhead)) | 
					
						
							|  |  |  |                         || (precedenceAhead == operPrecedence && isRightAssociative(tokenKindAhead)); | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |                 tokenKindAhead = tok().kind(), precedenceAhead = precedence(tokenKindAhead, _templateArguments)) { | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |             parseExpressionWithOperatorPrecedence(rhs, precedenceAhead); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |         if (condExpr) { // we were parsing a ternairy conditional expression
 | 
					
						
							|  |  |  |             condExpr->condition = lhs; | 
					
						
							|  |  |  |             condExpr->right_expression = rhs; | 
					
						
							|  |  |  |             lhs = condExpr; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             BinaryExpressionAST *expr = new (_pool) BinaryExpressionAST; | 
					
						
							|  |  |  |             expr->left_expression = lhs; | 
					
						
							|  |  |  |             expr->binary_op_token = oper; | 
					
						
							|  |  |  |             expr->right_expression = rhs; | 
					
						
							|  |  |  |             lhs = expr; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-01-28 15:21:57 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCommaExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2010-02-02 12:08:01 +01:00
										 |  |  |     PARSE_EXPRESSION_WITH_OPERATOR_PRECEDENCE(node, Prec::Comma) | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseThrowExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2008-12-02 12:01:29 +01:00
										 |  |  |     if (LA() == T_THROW) { | 
					
						
							|  |  |  |         ThrowExpressionAST *ast = new (_pool) ThrowExpressionAST; | 
					
						
							|  |  |  |         ast->throw_token = consumeToken(); | 
					
						
							|  |  |  |         parseAssignmentExpression(ast->expression); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | bool Parser::lookAtObjCSelector() const | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |     case T_IDENTIFIER: | 
					
						
							|  |  |  |     case T_OR: | 
					
						
							|  |  |  |     case T_AND: | 
					
						
							|  |  |  |     case T_NOT: | 
					
						
							|  |  |  |     case T_XOR: | 
					
						
							|  |  |  |     case T_BITOR: | 
					
						
							|  |  |  |     case T_COMPL: | 
					
						
							|  |  |  |     case T_OR_EQ: | 
					
						
							|  |  |  |     case T_AND_EQ: | 
					
						
							|  |  |  |     case T_BITAND: | 
					
						
							|  |  |  |     case T_NOT_EQ: | 
					
						
							|  |  |  |     case T_XOR_EQ: | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     default: | 
					
						
							|  |  |  |         if (tok().isKeyword()) | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     return false; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-class-declaraton ::= T_AT_CLASS (T_IDENTIFIER @ T_COMMA) T_SEMICOLON
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  | bool Parser::parseObjCClassForwardDeclaration(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_CLASS) | 
					
						
							| 
									
										
										
										
											2009-01-08 12:02:07 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |     ObjCClassForwardDeclarationAST *ast = new (_pool) ObjCClassForwardDeclarationAST; | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     ast->class_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     ast->identifier_list = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |     SimpleNameAST *name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |     name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |     ast->identifier_list->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     NameListAST **nextId = &ast->identifier_list->next; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         match(T_IDENTIFIER, &identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         *nextId = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         (*nextId)->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         nextId = &(*nextId)->next; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:02:07 +01:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-interface ::= attribute-specifier-list-opt objc-class-interface
 | 
					
						
							|  |  |  | // objc-interface ::= objc-category-interface
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // objc-class-interface ::= T_AT_INTERFACE T_IDENTIFIER (T_COLON T_IDENTIFIER)?
 | 
					
						
							|  |  |  | //                          objc-protocol-refs-opt
 | 
					
						
							|  |  |  | //                          objc-class-instance-variables-opt
 | 
					
						
							|  |  |  | //                          objc-interface-declaration-list
 | 
					
						
							|  |  |  | //                          T_AT_END
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // objc-category-interface ::= T_AT_INTERFACE T_IDENTIFIER
 | 
					
						
							|  |  |  | //                             T_LPAREN T_IDENTIFIER? T_RPAREN
 | 
					
						
							|  |  |  | //                             objc-protocol-refs-opt
 | 
					
						
							|  |  |  | //                             objc-interface-declaration-list
 | 
					
						
							|  |  |  | //                             T_AT_END
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | bool Parser::parseObjCInterface(DeclarationAST *&node, | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                                 SpecifierListAST *attributes) | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (! attributes && LA() == T___ATTRIBUTE__) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         SpecifierListAST **attr = &attributes; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |             attr = &(*attr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_INTERFACE) | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     unsigned objc_interface_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         // a category interface
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (attributes) | 
					
						
							|  |  |  |             _translationUnit->error(attributes->firstToken(), | 
					
						
							|  |  |  |                                     "invalid attributes for category interface declaration"); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCClassDeclarationAST *ast = new (_pool) ObjCClassDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         ast->interface_token = objc_interface_token; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *class_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         class_name->identifier_token= identifier_token; | 
					
						
							|  |  |  |         ast->class_name = class_name; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |             SimpleNameAST *category_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |             category_name->identifier_token = consumeToken(); | 
					
						
							|  |  |  |             ast->category_name = category_name; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         parseObjCProtocolRefs(ast->protocol_refs); | 
					
						
							| 
									
										
										
										
											2009-08-05 09:45:12 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         DeclarationListAST **nextMembers = &ast->member_declaration_list; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |         while (parseObjCInterfaceMemberDeclaration(declaration)) { | 
					
						
							|  |  |  |             *nextMembers = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             (*nextMembers)->value = declaration; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             nextMembers = &(*nextMembers)->next; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         match(T_AT_END, &ast->end_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // a class interface declaration
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCClassDeclarationAST *ast = new (_pool) ObjCClassDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         ast->interface_token = objc_interface_token; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST* class_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         class_name->identifier_token = identifier_token; | 
					
						
							|  |  |  |         ast->class_name = class_name; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         if (LA() == T_COLON) { | 
					
						
							|  |  |  |             ast->colon_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |             SimpleNameAST *superclass = new (_pool) SimpleNameAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             match(T_IDENTIFIER, &superclass->identifier_token); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |             ast->superclass = superclass; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         parseObjCProtocolRefs(ast->protocol_refs); | 
					
						
							| 
									
										
										
										
											2009-07-16 14:31:13 +02:00
										 |  |  |         parseObjClassInstanceVariables(ast->inst_vars_decl); | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         DeclarationListAST **nextMembers = &ast->member_declaration_list; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |         while (parseObjCInterfaceMemberDeclaration(declaration)) { | 
					
						
							|  |  |  |             *nextMembers = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             (*nextMembers)->value = declaration; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             nextMembers = &(*nextMembers)->next; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         match(T_AT_END, &ast->end_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-protocol ::= T_AT_PROTOCOL (T_IDENTIFIER @ T_COMMA) T_SEMICOLON
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | bool Parser::parseObjCProtocol(DeclarationAST *&node, | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |                                SpecifierListAST *attributes) | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (! attributes && LA() == T___ATTRIBUTE__) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  |         SpecifierListAST **attr = &attributes; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |             attr = &(*attr)->next; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_PROTOCOL) | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     unsigned protocol_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_COMMA || LA() == T_SEMICOLON) { | 
					
						
							|  |  |  |         // a protocol forward declaration
 | 
					
						
							| 
									
										
										
										
											2009-01-08 11:47:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCProtocolForwardDeclarationAST *ast = new (_pool) ObjCProtocolForwardDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         ast->protocol_token = protocol_token; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         ast->identifier_list = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         ast->identifier_list->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         NameListAST **nextId = &ast->identifier_list->next; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-08 12:08:06 +01:00
										 |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |             consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |             match(T_IDENTIFIER, &identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |             *nextId = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |             name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |             name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |             (*nextId)->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             nextId = &(*nextId)->next; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:08:06 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-08 13:06:09 +01:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         // a protocol definition
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCProtocolDeclarationAST *ast = new (_pool) ObjCProtocolDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         ast->protocol_token = protocol_token; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         name->identifier_token = identifier_token; | 
					
						
							|  |  |  |         ast->name = name; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:02:07 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         parseObjCProtocolRefs(ast->protocol_refs); | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         DeclarationListAST **nextMembers = &ast->member_declaration_list; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         DeclarationAST *declaration = 0; | 
					
						
							|  |  |  |         while (parseObjCInterfaceMemberDeclaration(declaration)) { | 
					
						
							|  |  |  |             *nextMembers = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             (*nextMembers)->value = declaration; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             nextMembers = &(*nextMembers)->next; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         match(T_AT_END, &ast->end_token); | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  | // objc-implementation ::= T_AT_IMPLEMENTAION T_IDENTIFIER (T_COLON T_IDENTIFIER)?
 | 
					
						
							|  |  |  | //                         objc-class-instance-variables-opt
 | 
					
						
							|  |  |  | // objc-implementation ::= T_AT_IMPLEMENTAION T_IDENTIFIER T_LPAREN T_IDENTIFIER T_RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | bool Parser::parseObjCImplementation(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_IMPLEMENTATION) | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |     unsigned implementation_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_LPAREN) { | 
					
						
							|  |  |  |         // a category implementation
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCClassDeclarationAST *ast = new (_pool) ObjCClassDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         ast->implementation_token = implementation_token; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *class_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         class_name->identifier_token = identifier_token; | 
					
						
							|  |  |  |         ast->class_name = class_name; | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *category_name = new (_pool) SimpleNameAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_IDENTIFIER, &category_name->identifier_token); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ast->category_name = category_name; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         parseObjCMethodDefinitionList(ast->member_declaration_list); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         match(T_AT_END, &ast->end_token); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |         // a class implementation
 | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         ObjCClassDeclarationAST *ast = new (_pool) ObjCClassDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         ast->implementation_token = implementation_token; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         SimpleNameAST *class_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         class_name->identifier_token = identifier_token; | 
					
						
							|  |  |  |         ast->class_name = class_name; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_COLON) { | 
					
						
							|  |  |  |             ast->colon_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |             SimpleNameAST *superclass = new (_pool) SimpleNameAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             match(T_IDENTIFIER, &superclass->identifier_token); | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |             ast->superclass = superclass; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         parseObjClassInstanceVariables(ast->inst_vars_decl); | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         parseObjCMethodDefinitionList(ast->member_declaration_list); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         match(T_AT_END, &ast->end_token); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | bool Parser::parseObjCMethodDefinitionList(DeclarationListAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |     DeclarationListAST **next = &node; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     while (LA() && LA() != T_AT_END) { | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |         unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         DeclarationAST *declaration = 0; | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |         switch (LA()) { | 
					
						
							|  |  |  |         case T_PLUS: | 
					
						
							|  |  |  |         case T_MINUS: | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |             parseObjCMethodDefinition(declaration); | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |             if (start == cursor()) | 
					
						
							|  |  |  |                 consumeToken(); | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-06 11:36:39 +01:00
										 |  |  |         case T_SEMICOLON: | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |             consumeToken(); | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2009-02-06 11:36:39 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         case T_AT_SYNTHESIZE: { | 
					
						
							|  |  |  |             ObjCSynthesizedPropertiesDeclarationAST *ast = new (_pool) ObjCSynthesizedPropertiesDeclarationAST; | 
					
						
							|  |  |  |             ast->synthesized_token = consumeToken(); | 
					
						
							|  |  |  |             ObjCSynthesizedPropertyListAST *last = new (_pool) ObjCSynthesizedPropertyListAST; | 
					
						
							|  |  |  |             ast->property_identifier_list = last; | 
					
						
							|  |  |  |             last->value = new (_pool) ObjCSynthesizedPropertyAST; | 
					
						
							|  |  |  |             match(T_IDENTIFIER, &last->value->property_identifier_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (LA() == T_EQUAL) { | 
					
						
							|  |  |  |                 last->value->equals_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 match(T_IDENTIFIER, &last->value->alias_identifier_token); | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             while (LA() == T_COMMA) { | 
					
						
							|  |  |  |                 consumeToken(); // consume T_COMMA
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                 last->next = new (_pool) ObjCSynthesizedPropertyListAST; | 
					
						
							|  |  |  |                 last = last->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 12:49:49 +01:00
										 |  |  |                 last->value = new (_pool) ObjCSynthesizedPropertyAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |                 match(T_IDENTIFIER, &last->value->property_identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |                 if (LA() == T_EQUAL) { | 
					
						
							| 
									
										
										
										
											2009-11-10 12:49:49 +01:00
										 |  |  |                     last->value->equals_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |                     match(T_IDENTIFIER, &last->value->alias_identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  |                 } | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 11:21:06 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             declaration = ast; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-14 14:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         case T_AT_DYNAMIC: { | 
					
						
							|  |  |  |             ObjCDynamicPropertiesDeclarationAST *ast = new (_pool) ObjCDynamicPropertiesDeclarationAST; | 
					
						
							|  |  |  |             ast->dynamic_token = consumeToken(); | 
					
						
							|  |  |  |             ast->property_identifier_list = new (_pool) NameListAST; | 
					
						
							|  |  |  |             SimpleNameAST *name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |             match(T_IDENTIFIER, &name->identifier_token); | 
					
						
							|  |  |  |             ast->property_identifier_list->value = name; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             NameListAST *last = ast->property_identifier_list; | 
					
						
							|  |  |  |             while (LA() == T_COMMA) { | 
					
						
							|  |  |  |                 consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2010-02-14 14:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |                 last->next = new (_pool) NameListAST; | 
					
						
							|  |  |  |                 last = last->next; | 
					
						
							|  |  |  |                 name = new (_pool) SimpleNameAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 match(T_IDENTIFIER, &name->identifier_token); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |                 last->value = name; | 
					
						
							|  |  |  |             } | 
					
						
							| 
									
										
										
										
											2010-02-14 14:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							| 
									
										
										
										
											2010-02-14 14:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |             declaration = ast; | 
					
						
							|  |  |  |             break; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-02-06 11:36:39 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             if (LA() == T_EXTERN && LA(2) == T_STRING_LITERAL) { | 
					
						
							|  |  |  |                 parseDeclaration(declaration); | 
					
						
							|  |  |  |             } else { | 
					
						
							|  |  |  |                 if (! parseBlockDeclaration(declaration)) { | 
					
						
							|  |  |  |                     rewind(start); | 
					
						
							|  |  |  |                     _translationUnit->error(cursor(), | 
					
						
							|  |  |  |                                             "skip token `%s'", tok().spell()); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |                     consumeToken(); | 
					
						
							|  |  |  |                 } | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |         } // switch
 | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         if (declaration) { | 
					
						
							|  |  |  |             *next = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |             (*next)->value = declaration; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             next = &(*next)->next; | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-20 10:04:44 +02:00
										 |  |  | bool Parser::parseObjCMethodDefinition(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-08-05 09:45:12 +02:00
										 |  |  |     ObjCMethodPrototypeAST *method_prototype = 0; | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  |     if (! parseObjCMethodPrototype(method_prototype)) | 
					
						
							| 
									
										
										
										
											2009-02-05 12:41:57 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-04 16:22:24 +02:00
										 |  |  |     ObjCMethodDeclarationAST *ast = new (_pool) ObjCMethodDeclarationAST; | 
					
						
							|  |  |  |     ast->method_prototype = method_prototype; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 13:44:33 +02:00
										 |  |  |     // Objective-C allows you to write:
 | 
					
						
							|  |  |  |     // - (void) foo; { body; }
 | 
					
						
							|  |  |  |     // so a method is a forward declaration when it doesn't have a _body_.
 | 
					
						
							|  |  |  |     // However, we still need to read the semicolon.
 | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  |     if (LA() == T_SEMICOLON) { | 
					
						
							|  |  |  |         ast->semicolon_token = consumeToken(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-02-06 15:13:12 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-06 13:44:33 +02:00
										 |  |  |     parseFunctionBody(ast->function_body); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-04 16:22:24 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-12 15:20:24 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-protocol-refs ::= T_LESS (T_IDENTIFIER @ T_COMMA) T_GREATER
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | bool Parser::parseObjCProtocolRefs(ObjCProtocolRefsAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (LA() != T_LESS) | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     ObjCProtocolRefsAST *ast = new (_pool) ObjCProtocolRefsAST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_LESS, &ast->less_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     unsigned identifier_token = 0; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, &identifier_token); | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     ast->identifier_list = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |     SimpleNameAST *name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |     name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |     ast->identifier_list->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     NameListAST **nextId = &ast->identifier_list->next; | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |         consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         match(T_IDENTIFIER, &identifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |         *nextId = new (_pool) NameListAST; | 
					
						
							| 
									
										
										
										
											2009-08-05 17:14:08 +02:00
										 |  |  |         name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |         name->identifier_token = identifier_token; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:39:29 +01:00
										 |  |  |         (*nextId)->value = name; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         nextId = &(*nextId)->next; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_GREATER, &ast->greater_token); | 
					
						
							| 
									
										
										
										
											2009-07-13 09:45:28 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-class-instance-variables ::= T_LBRACE
 | 
					
						
							|  |  |  | //                                   objc-instance-variable-decl-list-opt
 | 
					
						
							|  |  |  | //                                   T_RBRACE
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-16 14:31:13 +02:00
										 |  |  | bool Parser::parseObjClassInstanceVariables(ObjCInstanceVariablesDeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (LA() != T_LBRACE) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 14:31:13 +02:00
										 |  |  |     ObjCInstanceVariablesDeclarationAST *ast = new (_pool) ObjCInstanceVariablesDeclarationAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_LBRACE, &ast->lbrace_token); | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     for (DeclarationListAST **next = &ast->instance_variable_list; LA(); next = &(*next)->next) { | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         if (LA() == T_RBRACE) | 
					
						
							|  |  |  |             break; | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         const unsigned start = cursor(); | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         *next = new (_pool) DeclarationListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 11:23:35 +01:00
										 |  |  |         parseObjCInstanceVariableDeclaration((*next)->value); | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         if (start == cursor()) { | 
					
						
							|  |  |  |             // skip stray token.
 | 
					
						
							|  |  |  |             _translationUnit->error(cursor(), "skip stray token `%s'", tok().spell()); | 
					
						
							|  |  |  |             consumeToken(); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_RBRACE, &ast->rbrace_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 14:31:13 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-interface-declaration ::= T_AT_REQUIRED
 | 
					
						
							|  |  |  | // objc-interface-declaration ::= T_AT_OPTIONAL
 | 
					
						
							|  |  |  | // objc-interface-declaration ::= T_SEMICOLON
 | 
					
						
							|  |  |  | // objc-interface-declaration ::= objc-property-declaration
 | 
					
						
							|  |  |  | // objc-interface-declaration ::= objc-method-prototype
 | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | bool Parser::parseObjCInterfaceMemberDeclaration(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     case T_AT_END: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_AT_REQUIRED: | 
					
						
							|  |  |  |     case T_AT_OPTIONAL: | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_SEMICOLON: | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     case T_AT_PROPERTY: { | 
					
						
							|  |  |  |         return parseObjCPropertyDeclaration(node); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     case T_PLUS: | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  |     case T_MINUS: { | 
					
						
							|  |  |  |         ObjCMethodDeclarationAST *ast = new (_pool) ObjCMethodDeclarationAST; | 
					
						
							|  |  |  |         if (parseObjCMethodPrototype(ast->method_prototype)) { | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |             match(T_SEMICOLON, &ast->semicolon_token); | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } else { | 
					
						
							|  |  |  |             return false; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-13 15:05:48 +01:00
										 |  |  |     case T_ENUM: | 
					
						
							|  |  |  |     case T_CLASS: | 
					
						
							|  |  |  |     case T_STRUCT: | 
					
						
							|  |  |  |     case T_UNION: { | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         return parseSimpleDeclaration(node, /*accept struct declarators */ true); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-13 15:05:48 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     default: { | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         return parseSimpleDeclaration(node, /*accept struct declarators */ true); | 
					
						
							| 
									
										
										
										
											2009-01-13 15:05:48 +01:00
										 |  |  |     } // default
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } // switch
 | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-instance-variable-declaration ::= objc-visibility-specifier
 | 
					
						
							|  |  |  | // objc-instance-variable-declaration ::= block-declaration
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | bool Parser::parseObjCInstanceVariableDeclaration(DeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     switch (LA()) { | 
					
						
							|  |  |  |         case T_AT_PRIVATE: | 
					
						
							|  |  |  |         case T_AT_PROTECTED: | 
					
						
							|  |  |  |         case T_AT_PUBLIC: | 
					
						
							|  |  |  |         case T_AT_PACKAGE: { | 
					
						
							|  |  |  |             ObjCVisibilityDeclarationAST *ast = new (_pool) ObjCVisibilityDeclarationAST; | 
					
						
							|  |  |  |             ast->visibility_token = consumeToken(); | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							| 
									
										
										
										
											2009-07-16 14:31:13 +02:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2010-02-14 14:41:51 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |         default: | 
					
						
							|  |  |  |             return parseSimpleDeclaration(node, true); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-property-declaration ::=
 | 
					
						
							|  |  |  | //    T_AT_PROPERTY T_LPAREN (property-attribute @ T_COMMA) T_RPAREN simple-declaration
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:00:22 +01:00
										 |  |  | bool Parser::parseObjCPropertyDeclaration(DeclarationAST *&node, SpecifierListAST *attributes) | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2010-02-15 12:23:48 +01:00
										 |  |  |     if (LA() != T_AT_PROPERTY) | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |     ObjCPropertyDeclarationAST *ast = new (_pool) ObjCPropertyDeclarationAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     ast->attribute_list = attributes; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |     ast->property_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (LA() == T_LPAREN) { | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ObjCPropertyAttributeAST *property_attribute = 0; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |         if (parseObjCPropertyAttribute(property_attribute)) { | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |             ast->property_attribute_list = new (_pool) ObjCPropertyAttributeListAST; | 
					
						
							|  |  |  |             ast->property_attribute_list->value = property_attribute; | 
					
						
							|  |  |  |             ObjCPropertyAttributeListAST *last = ast->property_attribute_list; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  |             while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-11-10 11:13:57 +01:00
										 |  |  |                 consumeToken(); // consume T_COMMA
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |                 last->next = new (_pool) ObjCPropertyAttributeListAST; | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  |                 last = last->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:47:08 +01:00
										 |  |  |                 if (!parseObjCPropertyAttribute(last->value)) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |                     _translationUnit->error(_tokenIndex, "expected token `%s' got `%s'", | 
					
						
							|  |  |  |                                             Token::name(T_IDENTIFIER), tok().spell()); | 
					
						
							|  |  |  |                     break; | 
					
						
							|  |  |  |                 } | 
					
						
							| 
									
										
										
										
											2009-07-13 11:15:45 +02:00
										 |  |  |             } | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-01-22 09:44:40 +01:00
										 |  |  |     if (parseSimpleDeclaration(ast->simple_declaration, /*accept-struct-declarators = */ true)) | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |     else | 
					
						
							|  |  |  |         _translationUnit->error(_tokenIndex, "expected a simple declaration"); | 
					
						
							| 
									
										
										
										
											2009-07-16 15:50:42 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-08 18:05:33 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-01-08 12:54:56 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-method-prototype ::= (T_PLUS | T_MINUS) objc-method-decl objc-method-attrs-opt
 | 
					
						
							|  |  |  | //
 | 
					
						
							|  |  |  | // objc-method-decl ::= objc-type-name? objc-selector
 | 
					
						
							|  |  |  | // objc-method-decl ::= objc-type-name? objc-keyword-decl-list objc-parmlist-opt
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  | bool Parser::parseObjCMethodPrototype(ObjCMethodPrototypeAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (LA() != T_PLUS && LA() != T_MINUS) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 17:38:24 +02:00
										 |  |  |     ObjCMethodPrototypeAST *ast = new (_pool) ObjCMethodPrototypeAST; | 
					
						
							|  |  |  |     ast->method_type_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 17:38:24 +02:00
										 |  |  |     parseObjCTypeName(ast->type_name); | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if ((lookAtObjCSelector() && LA(2) == T_COLON) || LA() == T_COLON) { | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ObjCSelectorArgumentAST *argument = 0; | 
					
						
							| 
									
										
										
										
											2009-07-16 17:38:24 +02:00
										 |  |  |         ObjCMessageArgumentDeclarationAST *declaration = 0; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         parseObjCKeywordDeclaration(argument, declaration); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *sel = new (_pool) ObjCSelectorAST; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ast->selector = sel; | 
					
						
							|  |  |  |         ObjCSelectorArgumentListAST *lastSel = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         sel->selector_argument_list = lastSel; | 
					
						
							|  |  |  |         sel->selector_argument_list->value = argument; | 
					
						
							| 
									
										
										
										
											2009-07-16 17:38:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |         ast->argument_list = new (_pool) ObjCMessageArgumentDeclarationListAST; | 
					
						
							|  |  |  |         ast->argument_list->value = declaration; | 
					
						
							|  |  |  |         ObjCMessageArgumentDeclarationListAST *lastArg = ast->argument_list; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | 
 | 
					
						
							|  |  |  |         while (parseObjCKeywordDeclaration(argument, declaration)) { | 
					
						
							|  |  |  |             lastSel->next = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |             lastSel = lastSel->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:45:20 +01:00
										 |  |  |             lastSel->value = argument; | 
					
						
							| 
									
										
										
										
											2009-07-16 17:38:24 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |             lastArg->next = new (_pool) ObjCMessageArgumentDeclarationListAST; | 
					
						
							|  |  |  |             lastArg = lastArg->next; | 
					
						
							| 
									
										
										
										
											2009-11-10 12:48:33 +01:00
										 |  |  |             lastArg->value = declaration; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |         while (LA() == T_COMMA) { | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |             consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |             if (LA() == T_DOT_DOT_DOT) { | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |                 ast->dot_dot_dot_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |                 break; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |             // TODO: Is this still valid, and if so, should it be stored in the AST? (EV)
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |             DeclarationAST *parameter_declaration = 0; | 
					
						
							|  |  |  |             parseParameterDeclaration(parameter_declaration); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } else if (lookAtObjCSelector()) { | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *sel = new (_pool) ObjCSelectorAST; | 
					
						
							|  |  |  |         sel->selector_argument_list = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |         sel->selector_argument_list->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         parseObjCSelector(sel->selector_argument_list->value->name_token); | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |         ast->selector = sel; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     } else { | 
					
						
							|  |  |  |         _translationUnit->error(cursor(), "expected a selector"); | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     SpecifierListAST **attr = &ast->attribute_list; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |         attr = &(*attr)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-08-05 09:45:12 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-property-attribute ::= getter '=' identifier
 | 
					
						
							|  |  |  | // objc-property-attribute ::= setter '=' identifier ':'
 | 
					
						
							|  |  |  | // objc-property-attribute ::= readonly
 | 
					
						
							|  |  |  | // objc-property-attribute ::= readwrite
 | 
					
						
							|  |  |  | // objc-property-attribute ::= assign
 | 
					
						
							|  |  |  | // objc-property-attribute ::= retain
 | 
					
						
							|  |  |  | // objc-property-attribute ::= copy
 | 
					
						
							|  |  |  | // objc-property-attribute ::= nonatomic
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | bool Parser::parseObjCPropertyAttribute(ObjCPropertyAttributeAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (LA() != T_IDENTIFIER) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     node = new (_pool) ObjCPropertyAttributeAST; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |     const Identifier *id = tok().identifier; | 
					
						
							| 
									
										
										
										
											2010-02-04 10:58:20 +01:00
										 |  |  |     const int k = classifyObjectiveCContextKeyword(id->chars(), id->size()); | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     switch (k) { | 
					
						
							|  |  |  |     case Token_copy: | 
					
						
							|  |  |  |     case Token_assign: | 
					
						
							|  |  |  |     case Token_retain: | 
					
						
							|  |  |  |     case Token_readonly: | 
					
						
							|  |  |  |     case Token_readwrite: | 
					
						
							|  |  |  |     case Token_nonatomic: | 
					
						
							|  |  |  |         node->attribute_identifier_token = consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     case Token_getter: { | 
					
						
							|  |  |  |         node->attribute_identifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_EQUAL, &node->equals_token); | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *sel = new (_pool) ObjCSelectorAST; | 
					
						
							|  |  |  |         sel->selector_argument_list = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |         sel->selector_argument_list->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         match(T_IDENTIFIER, &sel->selector_argument_list->value->name_token); | 
					
						
							|  |  |  |         node->method_selector = sel; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |         return true; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |     case Token_setter: { | 
					
						
							|  |  |  |         node->attribute_identifier_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |         match(T_EQUAL, &node->equals_token); | 
					
						
							| 
									
										
										
										
											2010-02-23 17:43:40 +01:00
										 |  |  |         ObjCSelectorAST *sel = new (_pool) ObjCSelectorAST; | 
					
						
							|  |  |  |         sel->selector_argument_list = new (_pool) ObjCSelectorArgumentListAST; | 
					
						
							|  |  |  |         sel->selector_argument_list->value = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							|  |  |  |         match(T_IDENTIFIER, &sel->selector_argument_list->value->name_token); | 
					
						
							|  |  |  |         match(T_COLON, &sel->selector_argument_list->value->colon_token); | 
					
						
							|  |  |  |         node->method_selector = sel; | 
					
						
							| 
									
										
										
										
											2009-10-05 18:02:01 +02:00
										 |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     default: | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-type-name ::= T_LPAREN objc-type-qualifiers-opt type-id T_RPAREN
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  | bool Parser::parseObjCTypeName(ObjCTypeNameAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     if (LA() != T_LPAREN) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  |     ObjCTypeNameAST *ast = new (_pool) ObjCTypeNameAST; | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_LPAREN, &ast->lparen_token); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:40:50 +01:00
										 |  |  |     parseObjCTypeQualifiers(ast->type_qualifier_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  |     parseTypeId(ast->type_id); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-selector ::= T_IDENTIFIER | keyword
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  | bool Parser::parseObjCSelector(unsigned &selector_token) | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (! lookAtObjCSelector()) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 11:27:45 +02:00
										 |  |  |     selector_token = consumeToken(); | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | // objc-keyword-decl ::= objc-selector? T_COLON objc-type-name? objc-keyword-attributes-opt T_IDENTIFIER
 | 
					
						
							|  |  |  | //
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  | bool Parser::parseObjCKeywordDeclaration(ObjCSelectorArgumentAST *&argument, ObjCMessageArgumentDeclarationAST *&node) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     if (! (LA() == T_COLON || (lookAtObjCSelector() && LA(2) == T_COLON))) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     node = new (_pool) ObjCMessageArgumentDeclarationAST; | 
					
						
							|  |  |  |     argument = new (_pool) ObjCSelectorArgumentAST; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     parseObjCSelector(argument->name_token); | 
					
						
							| 
									
										
										
										
											2010-02-02 11:54:03 +01:00
										 |  |  |     match(T_COLON, &argument->colon_token); | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-28 16:34:15 +02:00
										 |  |  |     parseObjCTypeName(node->type_name); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-11-10 16:47:16 +01:00
										 |  |  |     SpecifierListAST **attr = &node->attribute_list; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  |     while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |         attr = &(*attr)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-17 16:25:12 +01:00
										 |  |  |     SimpleNameAST *param_name = new (_pool) SimpleNameAST; | 
					
						
							|  |  |  |     match(T_IDENTIFIER, ¶m_name->identifier_token); | 
					
						
							|  |  |  |     node->param_name = param_name; | 
					
						
							| 
									
										
										
										
											2009-01-12 14:55:33 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-16 12:44:47 +02:00
										 |  |  | bool Parser::parseObjCTypeQualifiers(unsigned &type_qualifier) | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							| 
									
										
										
										
											2009-01-13 15:05:48 +01:00
										 |  |  |     if (LA() != T_IDENTIFIER) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |     const Identifier *id = tok().identifier; | 
					
						
							| 
									
										
										
										
											2010-02-04 11:03:14 +01:00
										 |  |  |     switch (classifyObjectiveCContextKeyword(id->chars(), id->size())) { | 
					
						
							|  |  |  |     case Token_bycopy: | 
					
						
							|  |  |  |     case Token_byref: | 
					
						
							|  |  |  |     case Token_in: | 
					
						
							|  |  |  |     case Token_inout: | 
					
						
							|  |  |  |     case Token_oneway: | 
					
						
							|  |  |  |     case Token_out: | 
					
						
							|  |  |  |         type_qualifier = consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     default: | 
					
						
							| 
									
										
										
										
											2009-02-05 15:06:29 +01:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-02-04 11:03:14 +01:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2009-01-09 16:55:25 +01:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-07-29 10:39:27 +02:00
										 |  |  | bool Parser::peekAtObjCContextKeyword(int kind) | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  | { | 
					
						
							|  |  |  |     if (LA() != T_IDENTIFIER) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2009-12-01 11:33:13 +01:00
										 |  |  |     const Identifier *id = tok().identifier; | 
					
						
							| 
									
										
										
										
											2010-02-04 10:58:20 +01:00
										 |  |  |     const int k = classifyObjectiveCContextKeyword(id->chars(), id->size()); | 
					
						
							| 
									
										
										
										
											2009-07-29 10:39:27 +02:00
										 |  |  |     return k == kind; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseObjCContextKeyword(int kind, unsigned &in_token) | 
					
						
							|  |  |  | { | 
					
						
							| 
									
										
										
										
											2009-10-20 14:56:36 +02:00
										 |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-04 14:55:18 +01:00
										 |  |  |     if (!peekAtObjCContextKeyword(kind)) | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2010-02-04 14:55:18 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  |     in_token = consumeToken(); | 
					
						
							|  |  |  |     return true; | 
					
						
							| 
									
										
										
										
											2009-07-20 11:46:59 +02:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  | int Parser::peekAtQtContextKeyword() const | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_IDENTIFIER) | 
					
						
							|  |  |  |         return false; | 
					
						
							| 
									
										
										
										
											2009-10-20 11:21:25 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2010-02-06 11:05:43 +01:00
										 |  |  |     const Identifier *id = tok().identifier; | 
					
						
							|  |  |  |     return classifyQtContextKeyword(id->chars(), id->size()); | 
					
						
							|  |  |  | } | 
					
						
							| 
									
										
										
										
											2010-03-24 12:54:25 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLambdaExpression(ExpressionAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LambdaIntroducerAST *lambda_introducer = 0; | 
					
						
							|  |  |  |     if (parseLambdaIntroducer(lambda_introducer)) { | 
					
						
							|  |  |  |         LambdaExpressionAST *ast = new (_pool) LambdaExpressionAST; | 
					
						
							|  |  |  |         ast->lambda_introducer = lambda_introducer; | 
					
						
							|  |  |  |         parseLambdaDeclarator(ast->lambda_declarator); | 
					
						
							|  |  |  |         parseCompoundStatement(ast->statement); | 
					
						
							|  |  |  |         node = ast; | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLambdaIntroducer(LambdaIntroducerAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_LBRACKET) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LambdaIntroducerAST *ast = new (_pool) LambdaIntroducerAST; | 
					
						
							|  |  |  |     ast->lbracket_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() != T_RBRACKET) | 
					
						
							|  |  |  |         parseLambdaCapture(ast->lambda_capture); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_RBRACKET) { | 
					
						
							|  |  |  |         ast->rbracket_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         if (LA() == T_LPAREN || LA() == T_LBRACE) { | 
					
						
							|  |  |  |             node = ast; | 
					
						
							|  |  |  |             return true; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLambdaCapture(LambdaCaptureAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     bool startsWithDefaultCapture = false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     unsigned default_capture = 0; | 
					
						
							|  |  |  |     CaptureListAST *capture_list = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_AMPER || LA() == T_EQUAL) { | 
					
						
							|  |  |  |         if (LA(2) == T_COMMA || LA(2) == T_RBRACKET) { | 
					
						
							|  |  |  |             startsWithDefaultCapture = true; | 
					
						
							|  |  |  |             default_capture = consumeToken(); // consume capture-default
 | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (startsWithDefaultCapture && LA() == T_COMMA) { | 
					
						
							|  |  |  |         consumeToken(); // consume ','
 | 
					
						
							|  |  |  |         parseCaptureList(capture_list); // required
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else if (LA() != T_RBRACKET) { | 
					
						
							|  |  |  |         parseCaptureList(capture_list); // optional
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LambdaCaptureAST *ast = new (_pool) LambdaCaptureAST; | 
					
						
							|  |  |  |     ast->default_capture = default_capture; | 
					
						
							|  |  |  |     ast->capture_list = capture_list; | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCapture(CaptureAST *&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() == T_IDENTIFIER) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else if (LA() == T_AMPER && LA(2) == T_IDENTIFIER) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     } else if (LA() == T_THIS) { | 
					
						
							|  |  |  |         consumeToken(); | 
					
						
							|  |  |  |         return true; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return false; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseCaptureList(CaptureListAST *&) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     CaptureAST *capture = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (parseCapture(capture)) { | 
					
						
							|  |  |  |         while (LA() == T_COMMA) { | 
					
						
							|  |  |  |             consumeToken(); // consume `,'
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             parseCapture(capture); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseLambdaDeclarator(LambdaDeclaratorAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_LPAREN) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     LambdaDeclaratorAST *ast = new (_pool) LambdaDeclaratorAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ast->lparen_token = consumeToken(); // consume `('
 | 
					
						
							|  |  |  |     parseParameterDeclarationClause(ast->parameter_declaration_clause); | 
					
						
							|  |  |  |     match(T_RPAREN, &ast->rparen_token); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SpecifierListAST **attr = &ast->attributes; | 
					
						
							|  |  |  |     while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |         attr = &(*attr)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (LA() == T_MUTABLE) | 
					
						
							|  |  |  |         ast->mutable_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parseExceptionSpecification(ast->exception_specification); | 
					
						
							|  |  |  |     parseTrailingReturnType(ast->trailing_return_type); | 
					
						
							|  |  |  |     node = ast; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTrailingReturnType(TrailingReturnTypeAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     if (LA() != T_ARROW) | 
					
						
							|  |  |  |         return false; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     TrailingReturnTypeAST *ast = new (_pool) TrailingReturnTypeAST; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     ast->arrow_token = consumeToken(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     SpecifierListAST **attr = &ast->attributes; | 
					
						
							|  |  |  |     while (parseAttributeSpecifier(*attr)) | 
					
						
							|  |  |  |         attr = &(*attr)->next; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     parseTrailingTypeSpecifierSeq(ast->type_specifiers); | 
					
						
							|  |  |  |     parseAbstractDeclarator(ast->declarator); | 
					
						
							| 
									
										
										
										
											2010-03-24 13:56:34 +01:00
										 |  |  |     node = ast; | 
					
						
							| 
									
										
										
										
											2010-03-24 12:54:25 +01:00
										 |  |  |     return true; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | bool Parser::parseTrailingTypeSpecifierSeq(SpecifierListAST *&node) | 
					
						
							|  |  |  | { | 
					
						
							|  |  |  |     DEBUG_THIS_RULE(); | 
					
						
							|  |  |  |     return parseSimpleTypeSpecifier(node); | 
					
						
							|  |  |  | } |