/* * Copyright (C) 1999-2001 Harri Porten (porten@kde.org) * Copyright (C) 2001 Peter Kelly (pmk@post.com) * Copyright (C) 2003, 2006, 2007, 2008, 2009 Apple Inc. All rights reserved. * * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Library General Public * License as published by the Free Software Foundation; either * version 2 of the License, or (at your option) any later version. * * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Library General Public License for more details. * * You should have received a copy of the GNU Library General Public License * along with this library; see the file COPYING.LIB. If not, write to * the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, * Boston, MA 02110-1301, USA. * */ #ifndef Parser_h #define Parser_h #include "Debugger.h" #include "Executable.h" #include "JSGlobalObject.h" #include "Nodes.h" #include "SourceProvider.h" #include #include #include #include namespace JSC { class FunctionBodyNode; class ProgramNode; class UString; template struct ParserArenaData : ParserArenaDeletable { T data; }; class Parser : public Noncopyable { public: template PassRefPtr parse(ExecState*, Debugger*, const SourceCode&, int* errLine = 0, UString* errMsg = 0); template PassRefPtr reparse(JSGlobalData*, ParsedNode*); void reparseInPlace(JSGlobalData*, FunctionBodyNode*); PassRefPtr parseFunctionFromGlobalCode(ExecState*, Debugger*, const SourceCode&, int* errLine = 0, UString* errMsg = 0); void didFinishParsing(SourceElements*, ParserArenaData*, ParserArenaData*, CodeFeatures features, int lastLine, int numConstants); ParserArena& arena() { return m_arena; } private: void parse(JSGlobalData*, int* errLine, UString* errMsg); ParserArena m_arena; const SourceCode* m_source; SourceElements* m_sourceElements; ParserArenaData* m_varDeclarations; ParserArenaData* m_funcDeclarations; CodeFeatures m_features; int m_lastLine; int m_numConstants; }; template PassRefPtr Parser::parse(ExecState* exec, Debugger* debugger, const SourceCode& source, int* errLine, UString* errMsg) { m_source = &source; parse(&exec->globalData(), errLine, errMsg); RefPtr result; if (m_sourceElements) { result = ParsedNode::create(&exec->globalData(), m_sourceElements, m_varDeclarations ? &m_varDeclarations->data : 0, m_funcDeclarations ? &m_funcDeclarations->data : 0, *m_source, m_features, m_numConstants); result->setLoc(m_source->firstLine(), m_lastLine); } m_arena.reset(); m_source = 0; m_varDeclarations = 0; m_funcDeclarations = 0; if (debugger) debugger->sourceParsed(exec, source, *errLine, *errMsg); return result.release(); } template PassRefPtr Parser::reparse(JSGlobalData* globalData, ParsedNode* oldParsedNode) { m_source = &oldParsedNode->source(); parse(globalData, 0, 0); RefPtr result; if (m_sourceElements) { result = ParsedNode::create(globalData, m_sourceElements, m_varDeclarations ? &m_varDeclarations->data : 0, m_funcDeclarations ? &m_funcDeclarations->data : 0, *m_source, oldParsedNode->features(), m_numConstants); result->setLoc(m_source->firstLine(), m_lastLine); } m_arena.reset(); m_source = 0; m_varDeclarations = 0; m_funcDeclarations = 0; return result.release(); } inline PassRefPtr Parser::parseFunctionFromGlobalCode(ExecState* exec, Debugger* debugger, const SourceCode& source, int* errLine, UString* errMsg) { RefPtr program = parse(exec, debugger, source, errLine, errMsg); if (!program) return 0; StatementVector& children = program->children(); if (children.size() != 1) return 0; StatementNode* exprStatement = children[0]; ASSERT(exprStatement); ASSERT(exprStatement->isExprStatement()); if (!exprStatement || !exprStatement->isExprStatement()) return 0; ExpressionNode* funcExpr = static_cast(exprStatement)->expr(); ASSERT(funcExpr); ASSERT(funcExpr->isFuncExprNode()); if (!funcExpr || !funcExpr->isFuncExprNode()) return 0; RefPtr body = static_cast(funcExpr)->body(); ASSERT(body); return body.release(); } inline JSObject* EvalExecutable::parse(ExecState* exec, bool allowDebug) { int errLine; UString errMsg; m_node = exec->globalData().parser->parse(exec, allowDebug ? exec->dynamicGlobalObject()->debugger() : 0, m_source, &errLine, &errMsg); if (!m_node) return Error::create(exec, SyntaxError, errMsg, errLine, m_source.provider()->asID(), m_source.provider()->url()); return 0; } inline JSObject* ProgramExecutable::parse(ExecState* exec, bool allowDebug) { int errLine; UString errMsg; m_node = exec->globalData().parser->parse(exec, allowDebug ? exec->dynamicGlobalObject()->debugger() : 0, m_source, &errLine, &errMsg); if (!m_node) return Error::create(exec, SyntaxError, errMsg, errLine, m_source.provider()->asID(), m_source.provider()->url()); return 0; } } // namespace JSC #endif // Parser_h