📄 qscriptparser.cpp
字号:
/******************************************************************************** Copyright (C) 1992-2007 Trolltech ASA. All rights reserved.**** This file is part of the QtScript module of the Qt Toolkit.**** This file may be used under the terms of the GNU General Public** License version 2.0 as published by the Free Software Foundation** and appearing in the file LICENSE.GPL included in the packaging of** this file. Please review the following information to ensure GNU** General Public Licensing requirements will be met:** http://trolltech.com/products/qt/licenses/licensing/opensource/**** If you are unsure which license is appropriate for your use, please** review the following information:** http://trolltech.com/products/qt/licenses/licensing/licensingoverview** or contact the sales department at sales@trolltech.com.**** In addition, as a special exception, Trolltech gives you certain** additional rights. These rights are described in the Trolltech GPL** Exception version 1.0, which can be found at** http://www.trolltech.com/products/qt/gplexception/ and in the file** GPL_EXCEPTION.txt in this package.**** In addition, as a special exception, Trolltech, as the sole copyright** holder for Qt Designer, grants users of the Qt/Eclipse Integration** plug-in the right for the Qt/Eclipse Integration to link to** functionality provided by Qt Designer and its related libraries.**** Trolltech reserves all rights not expressly granted herein.**** This file is provided AS IS with NO WARRANTY OF ANY KIND, INCLUDING THE** WARRANTY OF DESIGN, MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE.******************************************************************************/#include <QtCore/QtDebug>#ifndef QT_NO_SCRIPT#include <string.h>#include "qscriptengine.h"#include "qscriptengine_p.h"#include "qscriptvalueimpl_p.h"#include "qscriptcontext_p.h"#include "qscriptmember_p.h"#include "qscriptobject_p.h"#include "qscriptlexer_p.h"#include "qscriptast_p.h"#include "qscriptnodepool_p.h"#define Q_SCRIPT_UPDATE_POSITION(node, startloc, endloc) do { \ node->startLine = startloc.startLine; \ node->startColumn = startloc.startColumn; \ node->endLine = endloc.endLine; \ node->endColumn = endloc.endColumn; \} while (0)#include "qscriptparser_p.h"inline static bool automatic(QScriptEnginePrivate *driver, int token){ return token == QScriptGrammar::T_RBRACE || token == 0 || driver->lexer()->prevTerminator();}QScriptParser::QScriptParser(): tos(0), stack_size(0), sym_stack(0), state_stack(0), location_stack(0){}QScriptParser::~QScriptParser(){ if (stack_size) { qFree(sym_stack); qFree(state_stack); qFree(location_stack); }}static inline QScriptParser::Location location(QScript::Lexer *lexer){ QScriptParser::Location loc; loc.startLine = lexer->startLineNo(); loc.startColumn = lexer->startColumnNo(); loc.endLine = lexer->endLineNo(); loc.endColumn = lexer->endColumnNo(); return loc;}bool QScriptParser::parse(QScriptEnginePrivate *driver){ const int INITIAL_STATE = 0; QScript::Lexer *lexer = driver->lexer(); int yytoken = -1; int saved_yytoken = -1; reallocateStack(); tos = 0; state_stack[++tos] = INITIAL_STATE; while (true) { const int state = state_stack [tos]; if (yytoken == -1 && - TERMINAL_COUNT != action_index [state]) { if (saved_yytoken == -1) { yytoken = lexer->lex(); location_stack [tos] = location(lexer); } else { yytoken = saved_yytoken; saved_yytoken = -1; } } int act = t_action (state, yytoken); if (act == ACCEPT_STATE) return true; else if (act > 0) { if (++tos == stack_size) reallocateStack(); sym_stack [tos].dval = lexer->dval (); state_stack [tos] = act; location_stack [tos] = location(lexer); yytoken = -1; } else if (act < 0) { int r = - act - 1; tos -= rhs [r]; act = state_stack [tos++]; switch (r) {case 0: { sym(1).Node = QScript::makeAstNode<QScript::AST::ThisExpression> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 1: { sym(1).Node = QScript::makeAstNode<QScript::AST::IdentifierExpression> (driver->nodePool(), sym(1).sval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 2: { sym(1).Node = QScript::makeAstNode<QScript::AST::NullExpression> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 3: { sym(1).Node = QScript::makeAstNode<QScript::AST::TrueLiteral> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 4: { sym(1).Node = QScript::makeAstNode<QScript::AST::FalseLiteral> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 5: { sym(1).Node = QScript::makeAstNode<QScript::AST::NumericLiteral> (driver->nodePool(), sym(1).dval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 6: { sym(1).Node = QScript::makeAstNode<QScript::AST::StringLiteral> (driver->nodePool(), sym(1).sval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 7: { bool rx = lexer->scanRegExp(); if (!rx) { error_message = lexer->errorMessage(); return false; } sym(1).Node = QScript::makeAstNode<QScript::AST::RegExpLiteral> (driver->nodePool(), lexer->pattern, lexer->flags); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 8: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArrayLiteral> (driver->nodePool(), sym(2).Elision); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 9: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArrayLiteral> (driver->nodePool(), sym(2).ElementList->finish ()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 10: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArrayLiteral> (driver->nodePool(), sym(2).ElementList->finish (), sym(4).Elision); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(5));} break;case 11: { if (sym(2).Node) sym(1).Node = QScript::makeAstNode<QScript::AST::ObjectLiteral> (driver->nodePool(), sym(2).PropertyNameAndValueList->finish ()); else sym(1).Node = QScript::makeAstNode<QScript::AST::ObjectLiteral> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 12: { sym(1) = sym(2); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 13: { sym(1).Node = QScript::makeAstNode<QScript::AST::ElementList> (driver->nodePool(), sym(1).Elision, sym(2).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 14: { sym(1).Node = QScript::makeAstNode<QScript::AST::ElementList> (driver->nodePool(), sym(1).ElementList, sym(3).Elision, sym(4).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(4));} break;case 15: { sym(1).Node = QScript::makeAstNode<QScript::AST::Elision> (driver->nodePool()); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 16: { sym(1).Node = QScript::makeAstNode<QScript::AST::Elision> (driver->nodePool(), sym(1).Elision); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 17: { sym(1).Node = 0;} break;case 18: { sym(1).Elision = sym(1).Elision->finish (); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 19: { sym(1).Node = QScript::makeAstNode<QScript::AST::PropertyNameAndValueList> (driver->nodePool(), sym(1).PropertyName, sym(3).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 20: { sym(1).Node = QScript::makeAstNode<QScript::AST::PropertyNameAndValueList> (driver->nodePool(), sym(1).PropertyNameAndValueList, sym(3).PropertyName, sym(5).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(5));} break;case 21: { sym(1).Node = QScript::makeAstNode<QScript::AST::IdentifierPropertyName> (driver->nodePool(), sym(1).sval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 22: { sym(1).Node = QScript::makeAstNode<QScript::AST::StringLiteralPropertyName> (driver->nodePool(), sym(1).sval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 23: { sym(1).Node = QScript::makeAstNode<QScript::AST::NumericLiteralPropertyName> (driver->nodePool(), sym(1).dval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 26: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArrayMemberExpression> (driver->nodePool(), sym(1).Expression, sym(3).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(4));} break;case 27: { sym(1).Node = QScript::makeAstNode<QScript::AST::FieldMemberExpression> (driver->nodePool(), sym(1).Expression, sym(4).sval); lexer->scanExtraIdentifiers(false); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(4));} break;case 28: { lexer->scanExtraIdentifiers(true);} break;case 29: { sym(1).Node = QScript::makeAstNode<QScript::AST::NewMemberExpression> (driver->nodePool(), sym(2).Expression, sym(3).ArgumentList); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 31: { sym(1).Node = QScript::makeAstNode<QScript::AST::NewExpression> (driver->nodePool(), sym(2).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 32: { sym(1).Node = QScript::makeAstNode<QScript::AST::CallExpression> (driver->nodePool(), sym(1).Expression, sym(2).ArgumentList); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 33: { sym(1).Node = QScript::makeAstNode<QScript::AST::CallExpression> (driver->nodePool(), sym(1).Expression, sym(2).ArgumentList); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 34: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArrayMemberExpression> (driver->nodePool(), sym(1).Expression, sym(3).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(4));} break;case 35: { sym(1).Node = QScript::makeAstNode<QScript::AST::FieldMemberExpression> (driver->nodePool(), sym(1).Expression, sym(3).sval); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 36: { sym(1).Node = 0;} break;case 37: { sym(1).Node = sym(2).ArgumentList->finish (); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 38: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArgumentList> (driver->nodePool(), sym(1).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(1));} break;case 39: { sym(1).Node = QScript::makeAstNode<QScript::AST::ArgumentList> (driver->nodePool(), sym(1).ArgumentList, sym(3).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(3));} break;case 43: { sym(1).Node = QScript::makeAstNode<QScript::AST::PostIncrementExpression> (driver->nodePool(), sym(1).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;case 44: { sym(1).Node = QScript::makeAstNode<QScript::AST::PostDecrementExpression> (driver->nodePool(), sym(1).Expression); Q_SCRIPT_UPDATE_POSITION(sym(1).Node, loc(1), loc(2));} break;
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -