/* * Copyright (C) 1999-2002 Harri Porten (porten@kde.org) * Copyright (C) 2001 Peter Kelly (pmk@post.com) * Copyright (C) 2003, 2004, 2005, 2006, 2007, 2008 Apple Inc. All rights reserved. * Copyright (C) 2007 Cameron Zwarich (cwzwarich@uwaterloo.ca) * Copyright (C) 2007 Maks Orlovich * Copyright (C) 2007 Eric Seidel * * 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. * */ #include "config.h" #include "nodes.h" #include "CodeGenerator.h" #include "ExecState.h" #include "JSGlobalObject.h" #include "JSStaticScopeObject.h" #include "LabelScope.h" #include "Parser.h" #include "PropertyNameArray.h" #include "RegExpObject.h" #include "SamplingTool.h" #include "Debugger.h" #include "lexer.h" #include "operations.h" #include #include #include #include #include #include #include using namespace WTF; namespace JSC { // ------------------------------ Node ----------------------------------------- #ifndef NDEBUG static RefCountedLeakCounter parserRefCountedCounter("JSC::Node"); #endif ParserRefCounted::ParserRefCounted(JSGlobalData* globalData) : m_globalData(globalData) { #ifndef NDEBUG parserRefCountedCounter.increment(); #endif if (!m_globalData->newParserObjects) m_globalData->newParserObjects = new HashSet; m_globalData->newParserObjects->add(this); ASSERT(m_globalData->newParserObjects->contains(this)); } ParserRefCounted::~ParserRefCounted() { #ifndef NDEBUG parserRefCountedCounter.decrement(); #endif } void ParserRefCounted::ref() { // bumping from 0 to 1 is just removing from the new nodes set if (m_globalData->newParserObjects) { HashSet::iterator it = m_globalData->newParserObjects->find(this); if (it != m_globalData->newParserObjects->end()) { m_globalData->newParserObjects->remove(it); ASSERT(!m_globalData->parserObjectExtraRefCounts || !m_globalData->parserObjectExtraRefCounts->contains(this)); return; } } ASSERT(!m_globalData->newParserObjects || !m_globalData->newParserObjects->contains(this)); if (!m_globalData->parserObjectExtraRefCounts) m_globalData->parserObjectExtraRefCounts = new HashCountedSet; m_globalData->parserObjectExtraRefCounts->add(this); } void ParserRefCounted::deref() { ASSERT(!m_globalData->newParserObjects || !m_globalData->newParserObjects->contains(this)); if (!m_globalData->parserObjectExtraRefCounts) { delete this; return; } HashCountedSet::iterator it = m_globalData->parserObjectExtraRefCounts->find(this); if (it == m_globalData->parserObjectExtraRefCounts->end()) delete this; else m_globalData->parserObjectExtraRefCounts->remove(it); } bool ParserRefCounted::hasOneRef() { if (m_globalData->newParserObjects && m_globalData->newParserObjects->contains(this)) { ASSERT(!m_globalData->parserObjectExtraRefCounts || !m_globalData->parserObjectExtraRefCounts->contains(this)); return false; } ASSERT(!m_globalData->newParserObjects || !m_globalData->newParserObjects->contains(this)); if (!m_globalData->parserObjectExtraRefCounts) return true; return !m_globalData->parserObjectExtraRefCounts->contains(this); } void ParserRefCounted::deleteNewObjects(JSGlobalData* globalData) { if (!globalData->newParserObjects) return; #ifndef NDEBUG HashSet::iterator end = globalData->newParserObjects->end(); for (HashSet::iterator it = globalData->newParserObjects->begin(); it != end; ++it) ASSERT(!globalData->parserObjectExtraRefCounts || !globalData->parserObjectExtraRefCounts->contains(*it)); #endif deleteAllValues(*globalData->newParserObjects); delete globalData->newParserObjects; globalData->newParserObjects = 0; } Node::Node(JSGlobalData* globalData) : ParserRefCounted(globalData) { m_line = globalData->lexer->lineNo(); } static void substitute(UString& string, const UString& substring) JSC_FAST_CALL; static void substitute(UString& string, const UString& substring) { int position = string.find("%s"); ASSERT(position != -1); UString newString = string.substr(0, position); newString.append(substring); newString.append(string.substr(position + 2)); string = newString; } RegisterID* ThrowableExpressionData::emitThrowError(CodeGenerator& generator, ErrorType e, const char* msg) { generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RegisterID* exception = generator.emitNewError(generator.newTemporary(), e, jsString(generator.globalData(), msg)); generator.emitThrow(exception); return exception; } RegisterID* ThrowableExpressionData::emitThrowError(CodeGenerator& generator, ErrorType e, const char* msg, const Identifier& label) { UString message = msg; substitute(message, label.ustring()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RegisterID* exception = generator.emitNewError(generator.newTemporary(), e, jsString(generator.globalData(), message)); generator.emitThrow(exception); return exception; } // ------------------------------ StatementNode -------------------------------- StatementNode::StatementNode(JSGlobalData* globalData) : Node(globalData) , m_lastLine(-1) { } void StatementNode::setLoc(int firstLine, int lastLine) { m_line = firstLine; m_lastLine = lastLine; } // ------------------------------ SourceElements -------------------------------- void SourceElements::append(PassRefPtr statement) { if (statement->isEmptyStatement()) return; m_statements.append(statement); } // ------------------------------ NullNode ------------------------------------- RegisterID* NullNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) return 0; return generator.emitLoad(dst, jsNull()); } // ------------------------------ BooleanNode ---------------------------------- RegisterID* BooleanNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) return 0; return generator.emitLoad(dst, m_value); } // ------------------------------ NumberNode ----------------------------------- RegisterID* NumberNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) return 0; return generator.emitLoad(dst, m_double); } // ------------------------------ StringNode ----------------------------------- RegisterID* StringNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) return 0; return generator.emitLoad(dst, m_value); } // ------------------------------ RegExpNode ----------------------------------- RegisterID* RegExpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr regExp = RegExp::create(generator.globalData(), m_pattern, m_flags); if (!regExp->isValid()) return emitThrowError(generator, SyntaxError, ("Invalid regular expression: " + UString(regExp->errorMessage())).UTF8String().c_str()); if (dst == ignoredResult()) return 0; return generator.emitNewRegExp(generator.finalDestination(dst), regExp.get()); } // ------------------------------ ThisNode ------------------------------------- RegisterID* ThisNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) return 0; return generator.moveToDestinationIfNeeded(dst, generator.thisRegister()); } // ------------------------------ ResolveNode ---------------------------------- bool ResolveNode::isPure(CodeGenerator& generator) const { return generator.isLocal(m_ident); } RegisterID* ResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (dst == ignoredResult()) return 0; return generator.moveToDestinationIfNeeded(dst, local); } generator.emitExpressionInfo(m_startOffset + m_ident.size(), m_ident.size(), 0); return generator.emitResolve(generator.finalDestination(dst), m_ident); } // ------------------------------ ArrayNode ------------------------------------ RegisterID* ArrayNode::emitCode(CodeGenerator& generator, RegisterID* dst) { // FIXME: Should we put all of this code into emitNewArray? unsigned length = 0; ElementNode* firstPutElement; for (firstPutElement = m_element.get(); firstPutElement; firstPutElement = firstPutElement->next()) { if (firstPutElement->elision()) break; ++length; } if (!firstPutElement && !m_elision) return generator.emitNewArray(generator.finalDestination(dst), m_element.get()); RefPtr array = generator.emitNewArray(generator.tempDestination(dst), m_element.get()); for (ElementNode* n = firstPutElement; n; n = n->next()) { RegisterID* value = generator.emitNode(n->value()); length += n->elision(); generator.emitPutByIndex(array.get(), length++, value); } if (m_elision) { RegisterID* value = generator.emitLoad(0, jsNumber(generator.globalData(), m_elision + length)); generator.emitPutById(array.get(), generator.propertyNames().length, value); } return generator.moveToDestinationIfNeeded(dst, array.get()); } // ------------------------------ ObjectLiteralNode ---------------------------- RegisterID* ObjectLiteralNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (!m_list) { if (dst == ignoredResult()) return 0; return generator.emitNewObject(generator.finalDestination(dst)); } return generator.emitNode(dst, m_list.get()); } // ------------------------------ PropertyListNode ----------------------------- RegisterID* PropertyListNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr newObj = generator.tempDestination(dst); generator.emitNewObject(newObj.get()); for (PropertyListNode* p = this; p; p = p->m_next.get()) { RegisterID* value = generator.emitNode(p->m_node->m_assign.get()); switch (p->m_node->m_type) { case PropertyNode::Constant: { generator.emitPutById(newObj.get(), p->m_node->name(), value); break; } case PropertyNode::Getter: { generator.emitPutGetter(newObj.get(), p->m_node->name(), value); break; } case PropertyNode::Setter: { generator.emitPutSetter(newObj.get(), p->m_node->name(), value); break; } default: ASSERT_NOT_REACHED(); } } return generator.moveToDestinationIfNeeded(dst, newObj.get()); } // ------------------------------ BracketAccessorNode -------------------------------- RegisterID* BracketAccessorNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNodeForLeftHandSide(m_base.get(), m_subscriptHasAssignments, m_subscript->isPure(generator)); RegisterID* property = generator.emitNode(m_subscript.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitGetByVal(generator.finalDestination(dst), base.get(), property); } // ------------------------------ DotAccessorNode -------------------------------- RegisterID* DotAccessorNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RegisterID* base = generator.emitNode(m_base.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitGetById(generator.finalDestination(dst), base, m_ident); } // ------------------------------ ArgumentListNode ----------------------------- RegisterID* ArgumentListNode::emitCode(CodeGenerator& generator, RegisterID* dst) { ASSERT(m_expr); return generator.emitNode(dst, m_expr.get()); } // ------------------------------ NewExprNode ---------------------------------- RegisterID* NewExprNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr func = generator.emitNode(m_expr.get()); return generator.emitConstruct(generator.finalDestination(dst), func.get(), m_args.get(), m_divot, m_startOffset, m_endOffset); } RegisterID* EvalFunctionCallNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.tempDestination(dst); RefPtr func = generator.newTemporary(); generator.emitResolveWithBase(base.get(), func.get(), generator.propertyNames().eval); return generator.emitCallEval(generator.finalDestination(dst, base.get()), func.get(), base.get(), m_args.get(), m_divot, m_startOffset, m_endOffset); } RegisterID* FunctionCallValueNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr func = generator.emitNode(m_expr.get()); return generator.emitCall(generator.finalDestination(dst), func.get(), 0, m_args.get(), m_divot, m_startOffset, m_endOffset); } RegisterID* FunctionCallResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RefPtr local = generator.registerFor(m_ident)) return generator.emitCall(generator.finalDestination(dst), local.get(), 0, m_args.get(), m_divot, m_startOffset, m_endOffset); int index = 0; size_t depth = 0; JSObject* globalObject = 0; if (generator.findScopedProperty(m_ident, index, depth, false, globalObject) && index != missingSymbolMarker()) { RefPtr func = generator.emitGetScopedVar(generator.newTemporary(), depth, index, globalObject); return generator.emitCall(generator.finalDestination(dst), func.get(), 0, m_args.get(), m_divot, m_startOffset, m_endOffset); } RefPtr base = generator.tempDestination(dst); RefPtr func = generator.newTemporary(); int identifierStart = m_divot - m_startOffset; generator.emitExpressionInfo(identifierStart + m_ident.size(), m_ident.size(), 0); generator.emitResolveFunction(base.get(), func.get(), m_ident); return generator.emitCall(generator.finalDestination(dst, base.get()), func.get(), base.get(), m_args.get(), m_divot, m_startOffset, m_endOffset); } RegisterID* FunctionCallBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); RegisterID* property = generator.emitNode(m_subscript.get()); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr function = generator.emitGetByVal(generator.newTemporary(), base.get(), property); return generator.emitCall(generator.finalDestination(dst, base.get()), function.get(), base.get(), m_args.get(), m_divot, m_startOffset, m_endOffset); } RegisterID* FunctionCallDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr function = generator.emitGetById(generator.newTemporary(), base.get(), m_ident); return generator.emitCall(generator.finalDestination(dst, base.get()), function.get(), base.get(), m_args.get(), m_divot, m_startOffset, m_endOffset); } // ------------------------------ PostfixResolveNode ---------------------------------- static RegisterID* emitPreIncOrDec(CodeGenerator& generator, RegisterID* srcDst, Operator oper) { return (oper == OpPlusPlus) ? generator.emitPreInc(srcDst) : generator.emitPreDec(srcDst); } static RegisterID* emitPostIncOrDec(CodeGenerator& generator, RegisterID* dst, RegisterID* srcDst, Operator oper) { return (oper == OpPlusPlus) ? generator.emitPostInc(dst, srcDst) : generator.emitPostDec(dst, srcDst); } RegisterID* PostfixResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (generator.isLocalConstant(m_ident)) { if (dst == ignoredResult()) return 0; return generator.emitToJSNumber(generator.finalDestination(dst), local); } if (dst == ignoredResult()) return emitPreIncOrDec(generator, local, m_operator); return emitPostIncOrDec(generator, generator.finalDestination(dst), local, m_operator); } int index = 0; size_t depth = 0; JSObject* globalObject = 0; if (generator.findScopedProperty(m_ident, index, depth, true, globalObject) && index != missingSymbolMarker()) { RefPtr value = generator.emitGetScopedVar(generator.newTemporary(), depth, index, globalObject); RegisterID* oldValue; if (dst == ignoredResult()) { oldValue = 0; emitPreIncOrDec(generator, value.get(), m_operator); } else { oldValue = emitPostIncOrDec(generator, generator.finalDestination(dst), value.get(), m_operator); } generator.emitPutScopedVar(depth, index, value.get(), globalObject); return oldValue; } generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RefPtr value = generator.newTemporary(); RefPtr base = generator.emitResolveWithBase(generator.newTemporary(), value.get(), m_ident); RegisterID* oldValue; if (dst == ignoredResult()) { oldValue = 0; emitPreIncOrDec(generator, value.get(), m_operator); } else { oldValue = emitPostIncOrDec(generator, generator.finalDestination(dst), value.get(), m_operator); } generator.emitPutById(base.get(), m_ident, value.get()); return oldValue; } // ------------------------------ PostfixBracketNode ---------------------------------- RegisterID* PostfixBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); RefPtr property = generator.emitNode(m_subscript.get()); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr value = generator.emitGetByVal(generator.newTemporary(), base.get(), property.get()); RegisterID* oldValue; if (dst == ignoredResult()) { oldValue = 0; if (m_operator == OpPlusPlus) generator.emitPreInc(value.get()); else generator.emitPreDec(value.get()); } else { oldValue = (m_operator == OpPlusPlus) ? generator.emitPostInc(generator.finalDestination(dst), value.get()) : generator.emitPostDec(generator.finalDestination(dst), value.get()); } generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutByVal(base.get(), property.get(), value.get()); return oldValue; } // ------------------------------ PostfixDotNode ---------------------------------- RegisterID* PostfixDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr value = generator.emitGetById(generator.newTemporary(), base.get(), m_ident); RegisterID* oldValue; if (dst == ignoredResult()) { oldValue = 0; if (m_operator == OpPlusPlus) generator.emitPreInc(value.get()); else generator.emitPreDec(value.get()); } else { oldValue = (m_operator == OpPlusPlus) ? generator.emitPostInc(generator.finalDestination(dst), value.get()) : generator.emitPostDec(generator.finalDestination(dst), value.get()); } generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutById(base.get(), m_ident, value.get()); return oldValue; } // ------------------------------ PostfixErrorNode ----------------------------------- RegisterID* PostfixErrorNode::emitCode(CodeGenerator& generator, RegisterID*) { return emitThrowError(generator, ReferenceError, m_operator == OpPlusPlus ? "Postfix ++ operator applied to value that is not a reference." : "Postfix -- operator applied to value that is not a reference."); } // ------------------------------ DeleteResolveNode ----------------------------------- RegisterID* DeleteResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (generator.registerFor(m_ident)) return generator.emitUnexpectedLoad(generator.finalDestination(dst), false); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RegisterID* base = generator.emitResolveBase(generator.tempDestination(dst), m_ident); return generator.emitDeleteById(generator.finalDestination(dst, base), base, m_ident); } // ------------------------------ DeleteBracketNode ----------------------------------- RegisterID* DeleteBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr r0 = generator.emitNode(m_base.get()); RegisterID* r1 = generator.emitNode(m_subscript.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitDeleteByVal(generator.finalDestination(dst), r0.get(), r1); } // ------------------------------ DeleteDotNode ----------------------------------- RegisterID* DeleteDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RegisterID* r0 = generator.emitNode(m_base.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitDeleteById(generator.finalDestination(dst), r0, m_ident); } // ------------------------------ DeleteValueNode ----------------------------------- RegisterID* DeleteValueNode::emitCode(CodeGenerator& generator, RegisterID* dst) { generator.emitNode(ignoredResult(), m_expr.get()); // delete on a non-location expression ignores the value and returns true return generator.emitUnexpectedLoad(generator.finalDestination(dst), true); } // ------------------------------ VoidNode ------------------------------------- RegisterID* VoidNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) { generator.emitNode(ignoredResult(), m_expr.get()); return 0; } RefPtr r0 = generator.emitNode(m_expr.get()); return generator.emitLoad(dst, jsUndefined()); } // ------------------------------ TypeOfValueNode ----------------------------------- RegisterID* TypeOfResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (dst == ignoredResult()) return 0; return generator.emitTypeOf(generator.finalDestination(dst), local); } RefPtr scratch = generator.emitResolveBase(generator.tempDestination(dst), m_ident); generator.emitGetById(scratch.get(), scratch.get(), m_ident); if (dst == ignoredResult()) return 0; return generator.emitTypeOf(generator.finalDestination(dst, scratch.get()), scratch.get()); } // ------------------------------ TypeOfValueNode ----------------------------------- RegisterID* TypeOfValueNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) { generator.emitNode(ignoredResult(), m_expr.get()); return 0; } RefPtr src = generator.emitNode(m_expr.get()); return generator.emitTypeOf(generator.finalDestination(dst), src.get()); } // ------------------------------ PrefixResolveNode ---------------------------------- RegisterID* PrefixResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (generator.isLocalConstant(m_ident)) { if (dst == ignoredResult()) return 0; RefPtr r0 = generator.emitUnexpectedLoad(generator.finalDestination(dst), (m_operator == OpPlusPlus) ? 1.0 : -1.0); return generator.emitBinaryOp(op_add, r0.get(), local, r0.get(), OperandTypes()); } emitPreIncOrDec(generator, local, m_operator); return generator.moveToDestinationIfNeeded(dst, local); } int index = 0; size_t depth = 0; JSObject* globalObject = 0; if (generator.findScopedProperty(m_ident, index, depth, false, globalObject) && index != missingSymbolMarker()) { RefPtr propDst = generator.emitGetScopedVar(generator.tempDestination(dst), depth, index, globalObject); emitPreIncOrDec(generator, propDst.get(), m_operator); generator.emitPutScopedVar(depth, index, propDst.get(), globalObject); return generator.moveToDestinationIfNeeded(dst, propDst.get()); } generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RefPtr propDst = generator.tempDestination(dst); RefPtr base = generator.emitResolveWithBase(generator.newTemporary(), propDst.get(), m_ident); emitPreIncOrDec(generator, propDst.get(), m_operator); generator.emitPutById(base.get(), m_ident, propDst.get()); return generator.moveToDestinationIfNeeded(dst, propDst.get()); } // ------------------------------ PrefixBracketNode ---------------------------------- RegisterID* PrefixBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); RefPtr property = generator.emitNode(m_subscript.get()); RefPtr propDst = generator.tempDestination(dst); generator.emitExpressionInfo(m_divot + m_subexpressionDivotOffset, m_subexpressionStartOffset, m_endOffset - m_subexpressionDivotOffset); RegisterID* value = generator.emitGetByVal(propDst.get(), base.get(), property.get()); if (m_operator == OpPlusPlus) generator.emitPreInc(value); else generator.emitPreDec(value); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutByVal(base.get(), property.get(), value); return generator.moveToDestinationIfNeeded(dst, propDst.get()); } // ------------------------------ PrefixDotNode ---------------------------------- RegisterID* PrefixDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNode(m_base.get()); RefPtr propDst = generator.tempDestination(dst); generator.emitExpressionInfo(m_divot + m_subexpressionDivotOffset, m_subexpressionStartOffset, m_endOffset - m_subexpressionDivotOffset); RegisterID* value = generator.emitGetById(propDst.get(), base.get(), m_ident); if (m_operator == OpPlusPlus) generator.emitPreInc(value); else generator.emitPreDec(value); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutById(base.get(), m_ident, value); return generator.moveToDestinationIfNeeded(dst, propDst.get()); } // ------------------------------ PrefixErrorNode ----------------------------------- RegisterID* PrefixErrorNode::emitCode(CodeGenerator& generator, RegisterID*) { return emitThrowError(generator, ReferenceError, m_operator == OpPlusPlus ? "Prefix ++ operator applied to value that is not a reference." : "Prefix -- operator applied to value that is not a reference."); } // ------------------------------ Unary Operation Nodes ----------------------------------- RegisterID* UnaryOpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RegisterID* src = generator.emitNode(m_expr.get()); return generator.emitUnaryOp(opcode(), generator.finalDestination(dst), src, m_expr->resultDescriptor()); } // ------------------------------ Binary Operation Nodes ----------------------------------- RegisterID* BinaryOpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { OpcodeID opcode = this->opcode(); if (opcode == op_neq) { if (m_expr1->isNull() || m_expr2->isNull()) { RefPtr src = generator.emitNode(dst, m_expr1->isNull() ? m_expr2.get() : m_expr1.get()); return generator.emitUnaryOp(op_neq_null, generator.finalDestination(dst, src.get()), src.get(), ResultType::unknown()); } } RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RegisterID* src2 = generator.emitNode(m_expr2.get()); return generator.emitBinaryOp(opcode, generator.finalDestination(dst, src1.get()), src1.get(), src2, OperandTypes(m_expr1->resultDescriptor(), m_expr2->resultDescriptor())); } RegisterID* EqualNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (m_expr1->isNull() || m_expr2->isNull()) { RefPtr src = generator.emitNode(dst, m_expr1->isNull() ? m_expr2.get() : m_expr1.get()); return generator.emitUnaryOp(op_eq_null, generator.finalDestination(dst, src.get()), src.get(), ResultType::unknown()); } RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RegisterID* src2 = generator.emitNode(m_expr2.get()); return generator.emitEqualityOp(op_eq, generator.finalDestination(dst, src1.get()), src1.get(), src2); } RegisterID* StrictEqualNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RegisterID* src2 = generator.emitNode(m_expr2.get()); return generator.emitEqualityOp(op_stricteq, generator.finalDestination(dst, src1.get()), src1.get(), src2); } RegisterID* ReverseBinaryOpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RegisterID* src2 = generator.emitNode(m_expr2.get()); return generator.emitBinaryOp(opcode(), generator.finalDestination(dst, src1.get()), src2, src1.get(), OperandTypes(m_expr2->resultDescriptor(), m_expr1->resultDescriptor())); } RegisterID* ThrowableBinaryOpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RegisterID* src2 = generator.emitNode(m_expr2.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitBinaryOp(opcode(), generator.finalDestination(dst, src1.get()), src1.get(), src2, OperandTypes(m_expr1->resultDescriptor(), m_expr2->resultDescriptor())); } RegisterID* InstanceOfNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr src1 = generator.emitNodeForLeftHandSide(m_expr1.get(), m_rightHasAssignments, m_expr2->isPure(generator)); RefPtr src2 = generator.emitNode(m_expr2.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RegisterID* src2Prototype = generator.emitGetById(generator.newTemporary(), src2.get(), generator.globalData()->propertyNames->prototype); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitInstanceOf(generator.finalDestination(dst, src1.get()), src1.get(), src2.get(), src2Prototype); } // ------------------------------ Binary Logical Nodes ---------------------------- RegisterID* LogicalOpNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr temp = generator.tempDestination(dst); RefPtr target = generator.newLabel(); generator.emitNode(temp.get(), m_expr1.get()); if (m_operator == OpLogicalAnd) generator.emitJumpIfFalse(temp.get(), target.get()); else generator.emitJumpIfTrue(temp.get(), target.get()); generator.emitNode(temp.get(), m_expr2.get()); generator.emitLabel(target.get()); return generator.moveToDestinationIfNeeded(dst, temp.get()); } // ------------------------------ ConditionalNode ------------------------------ RegisterID* ConditionalNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr newDst = generator.finalDestination(dst); RefPtr beforeElse = generator.newLabel(); RefPtr afterElse = generator.newLabel(); RegisterID* cond = generator.emitNode(m_logical.get()); generator.emitJumpIfFalse(cond, beforeElse.get()); generator.emitNode(newDst.get(), m_expr1.get()); generator.emitJump(afterElse.get()); generator.emitLabel(beforeElse.get()); generator.emitNode(newDst.get(), m_expr2.get()); generator.emitLabel(afterElse.get()); return newDst.get(); } // ------------------------------ ReadModifyResolveNode ----------------------------------- // FIXME: should this be moved to be a method on CodeGenerator? static ALWAYS_INLINE RegisterID* emitReadModifyAssignment(CodeGenerator& generator, RegisterID* dst, RegisterID* src1, RegisterID* src2, Operator oper, OperandTypes types) { OpcodeID opcode; switch (oper) { case OpMultEq: opcode = op_mul; break; case OpDivEq: opcode = op_div; break; case OpPlusEq: opcode = op_add; break; case OpMinusEq: opcode = op_sub; break; case OpLShift: opcode = op_lshift; break; case OpRShift: opcode = op_rshift; break; case OpURShift: opcode = op_urshift; break; case OpAndEq: opcode = op_bitand; break; case OpXOrEq: opcode = op_bitxor; break; case OpOrEq: opcode = op_bitor; break; case OpModEq: opcode = op_mod; break; default: ASSERT_NOT_REACHED(); return dst; } return generator.emitBinaryOp(opcode, dst, src1, src2, types); } RegisterID* ReadModifyResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (generator.isLocalConstant(m_ident)) { RegisterID* src2 = generator.emitNode(m_right.get()); return emitReadModifyAssignment(generator, generator.finalDestination(dst), local, src2, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); } if (generator.leftHandSideNeedsCopy(m_rightHasAssignments, m_right->isPure(generator))) { RefPtr result = generator.newTemporary(); generator.emitMove(result.get(), local); RegisterID* src2 = generator.emitNode(m_right.get()); emitReadModifyAssignment(generator, result.get(), result.get(), src2, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); generator.emitMove(local, result.get()); return generator.moveToDestinationIfNeeded(dst, result.get()); } RegisterID* src2 = generator.emitNode(m_right.get()); RegisterID* result = emitReadModifyAssignment(generator, local, local, src2, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); return generator.moveToDestinationIfNeeded(dst, result); } int index = 0; size_t depth = 0; JSObject* globalObject = 0; if (generator.findScopedProperty(m_ident, index, depth, true, globalObject) && index != missingSymbolMarker()) { RefPtr src1 = generator.emitGetScopedVar(generator.tempDestination(dst), depth, index, globalObject); RegisterID* src2 = generator.emitNode(m_right.get()); RegisterID* result = emitReadModifyAssignment(generator, generator.finalDestination(dst, src1.get()), src1.get(), src2, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); generator.emitPutScopedVar(depth, index, result, globalObject); return result; } RefPtr src1 = generator.tempDestination(dst); generator.emitExpressionInfo(m_divot - m_startOffset + m_ident.size(), m_ident.size(), 0); RefPtr base = generator.emitResolveWithBase(generator.newTemporary(), src1.get(), m_ident); RegisterID* src2 = generator.emitNode(m_right.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); RegisterID* result = emitReadModifyAssignment(generator, generator.finalDestination(dst, src1.get()), src1.get(), src2, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); return generator.emitPutById(base.get(), m_ident, result); } // ------------------------------ AssignResolveNode ----------------------------------- RegisterID* AssignResolveNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (RegisterID* local = generator.registerFor(m_ident)) { if (generator.isLocalConstant(m_ident)) return generator.emitNode(dst, m_right.get()); RegisterID* result = generator.emitNode(local, m_right.get()); return generator.moveToDestinationIfNeeded(dst, result); } int index = 0; size_t depth = 0; JSObject* globalObject = 0; if (generator.findScopedProperty(m_ident, index, depth, true, globalObject) && index != missingSymbolMarker()) { if (dst == ignoredResult()) dst = 0; RegisterID* value = generator.emitNode(dst, m_right.get()); generator.emitPutScopedVar(depth, index, value, globalObject); return value; } RefPtr base = generator.emitResolveBase(generator.newTemporary(), m_ident); if (dst == ignoredResult()) dst = 0; RegisterID* value = generator.emitNode(dst, m_right.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitPutById(base.get(), m_ident, value); } // ------------------------------ AssignDotNode ----------------------------------- RegisterID* AssignDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNodeForLeftHandSide(m_base.get(), m_rightHasAssignments, m_right->isPure(generator)); RefPtr value = generator.destinationForAssignResult(dst); RegisterID* result = generator.emitNode(value.get(), m_right.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutById(base.get(), m_ident, result); return generator.moveToDestinationIfNeeded(dst, result); } // ------------------------------ ReadModifyDotNode ----------------------------------- RegisterID* ReadModifyDotNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNodeForLeftHandSide(m_base.get(), m_rightHasAssignments, m_right->isPure(generator)); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr value = generator.emitGetById(generator.tempDestination(dst), base.get(), m_ident); RegisterID* change = generator.emitNode(m_right.get()); RegisterID* updatedValue = emitReadModifyAssignment(generator, generator.finalDestination(dst, value.get()), value.get(), change, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); return generator.emitPutById(base.get(), m_ident, updatedValue); } // ------------------------------ AssignErrorNode ----------------------------------- RegisterID* AssignErrorNode::emitCode(CodeGenerator& generator, RegisterID*) { return emitThrowError(generator, ReferenceError, "Left side of assignment is not a reference."); } // ------------------------------ AssignBracketNode ----------------------------------- RegisterID* AssignBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNodeForLeftHandSide(m_base.get(), m_subscriptHasAssignments || m_rightHasAssignments, m_subscript->isPure(generator) && m_right->isPure(generator)); RefPtr property = generator.emitNodeForLeftHandSide(m_subscript.get(), m_rightHasAssignments, m_right->isPure(generator)); RefPtr value = generator.destinationForAssignResult(dst); RegisterID* result = generator.emitNode(value.get(), m_right.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutByVal(base.get(), property.get(), result); return generator.moveToDestinationIfNeeded(dst, result); } RegisterID* ReadModifyBracketNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr base = generator.emitNodeForLeftHandSide(m_base.get(), m_subscriptHasAssignments || m_rightHasAssignments, m_subscript->isPure(generator) && m_right->isPure(generator)); RefPtr property = generator.emitNodeForLeftHandSide(m_subscript.get(), m_rightHasAssignments, m_right->isPure(generator)); generator.emitExpressionInfo(m_divot - m_subexpressionDivotOffset, m_startOffset - m_subexpressionDivotOffset, m_subexpressionEndOffset); RefPtr value = generator.emitGetByVal(generator.tempDestination(dst), base.get(), property.get()); RegisterID* change = generator.emitNode(m_right.get()); RegisterID* updatedValue = emitReadModifyAssignment(generator, generator.finalDestination(dst, value.get()), value.get(), change, m_operator, OperandTypes(ResultType::unknown(), m_right->resultDescriptor())); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutByVal(base.get(), property.get(), updatedValue); return updatedValue; } // ------------------------------ CommaNode ------------------------------------ RegisterID* CommaNode::emitCode(CodeGenerator& generator, RegisterID* dst) { generator.emitNode(ignoredResult(), m_expr1.get()); return generator.emitNode(dst, m_expr2.get()); } // ------------------------------ ConstDeclNode ---------------------------------- ConstDeclNode::ConstDeclNode(JSGlobalData* globalData, const Identifier& ident, ExpressionNode* init) : ExpressionNode(globalData) , m_ident(ident) , m_init(init) { } RegisterID* ConstDeclNode::emitCodeSingle(CodeGenerator& generator) { if (RegisterID* local = generator.constRegisterFor(m_ident)) { if (!m_init) return local; return generator.emitNode(local, m_init.get()); } // FIXME: While this code should only be hit in eval code, it will potentially // assign to the wrong base if m_ident exists in an intervening dynamic scope. RefPtr base = generator.emitResolveBase(generator.newTemporary(), m_ident); RegisterID* value = m_init ? generator.emitNode(m_init.get()) : generator.emitLoad(0, jsUndefined()); return generator.emitPutById(base.get(), m_ident, value); } RegisterID* ConstDeclNode::emitCode(CodeGenerator& generator, RegisterID*) { RegisterID* result = 0; for (ConstDeclNode* n = this; n; n = n->m_next.get()) result = n->emitCodeSingle(generator); return result; } // ------------------------------ ConstStatementNode ----------------------------- RegisterID* ConstStatementNode::emitCode(CodeGenerator& generator, RegisterID*) { return generator.emitNode(m_next.get()); } // ------------------------------ Helper functions for handling Vectors of StatementNode ------------------------------- static inline RegisterID* statementListEmitCode(StatementVector& statements, CodeGenerator& generator, RegisterID* dst) { StatementVector::iterator end = statements.end(); for (StatementVector::iterator it = statements.begin(); it != end; ++it) { StatementNode* n = it->get(); if (!n->isLoop()) generator.emitDebugHook(WillExecuteStatement, n->firstLine(), n->lastLine()); generator.emitNode(dst, n); } return 0; } static inline void statementListPushFIFO(StatementVector& statements, DeclarationStacks::NodeStack& stack) { StatementVector::iterator it = statements.end(); StatementVector::iterator begin = statements.begin(); while (it != begin) { --it; stack.append((*it).get()); } } // ------------------------------ BlockNode ------------------------------------ BlockNode::BlockNode(JSGlobalData* globalData, SourceElements* children) : StatementNode(globalData) { if (children) children->releaseContentsIntoVector(m_children); } RegisterID* BlockNode::emitCode(CodeGenerator& generator, RegisterID* dst) { return statementListEmitCode(m_children, generator, dst); } // ------------------------------ EmptyStatementNode --------------------------- RegisterID* EmptyStatementNode::emitCode(CodeGenerator&, RegisterID* dst) { return dst; } // ------------------------------ DebuggerStatementNode --------------------------- RegisterID* DebuggerStatementNode::emitCode(CodeGenerator& generator, RegisterID* dst) { generator.emitDebugHook(DidReachBreakpoint, firstLine(), lastLine()); return dst; } // ------------------------------ ExprStatementNode ---------------------------- RegisterID* ExprStatementNode::emitCode(CodeGenerator& generator, RegisterID* dst) { ASSERT(m_expr); return generator.emitNode(dst, m_expr.get()); } // ------------------------------ VarStatementNode ---------------------------- RegisterID* VarStatementNode::emitCode(CodeGenerator& generator, RegisterID*) { ASSERT(m_expr); return generator.emitNode(m_expr.get()); } // ------------------------------ IfNode --------------------------------------- RegisterID* IfNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr afterThen = generator.newLabel(); RegisterID* cond = generator.emitNode(m_condition.get()); generator.emitJumpIfFalse(cond, afterThen.get()); if (!m_ifBlock->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_ifBlock->firstLine(), m_ifBlock->lastLine()); generator.emitNode(dst, m_ifBlock.get()); generator.emitLabel(afterThen.get()); // FIXME: This should return the last statement exectuted so that it can be returned as a Completion return 0; } RegisterID* IfElseNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr beforeElse = generator.newLabel(); RefPtr afterElse = generator.newLabel(); RegisterID* cond = generator.emitNode(m_condition.get()); generator.emitJumpIfFalse(cond, beforeElse.get()); if (!m_ifBlock->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_ifBlock->firstLine(), m_ifBlock->lastLine()); generator.emitNode(dst, m_ifBlock.get()); generator.emitJump(afterElse.get()); generator.emitLabel(beforeElse.get()); if (!m_elseBlock->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_elseBlock->firstLine(), m_elseBlock->lastLine()); generator.emitNode(dst, m_elseBlock.get()); generator.emitLabel(afterElse.get()); // FIXME: This should return the last statement exectuted so that it can be returned as a Completion return 0; } // ------------------------------ DoWhileNode ---------------------------------- RegisterID* DoWhileNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr scope = generator.newLabelScope(LabelScope::Loop); RefPtr topOfLoop = generator.newLabel(); generator.emitLabel(topOfLoop.get()); generator.emitDebugHook(WillExecuteStatement, firstLine(), lastLine()); if (!m_statement->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_statement->firstLine(), m_statement->lastLine()); RefPtr result = generator.emitNode(dst, m_statement.get()); generator.emitLabel(scope->continueTarget()); generator.emitDebugHook(WillExecuteStatement, m_expr->lineNo(), m_expr->lineNo()); RegisterID* cond = generator.emitNode(m_expr.get()); generator.emitJumpIfTrue(cond, topOfLoop.get()); generator.emitLabel(scope->breakTarget()); return result.get(); } // ------------------------------ WhileNode ------------------------------------ RegisterID* WhileNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr scope = generator.newLabelScope(LabelScope::Loop); generator.emitJump(scope->continueTarget()); RefPtr topOfLoop = generator.newLabel(); generator.emitLabel(topOfLoop.get()); if (!m_statement->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_statement->firstLine(), m_statement->lastLine()); generator.emitNode(dst, m_statement.get()); generator.emitLabel(scope->continueTarget()); generator.emitDebugHook(WillExecuteStatement, m_expr->lineNo(), m_expr->lineNo()); RegisterID* cond = generator.emitNode(m_expr.get()); generator.emitJumpIfTrue(cond, topOfLoop.get()); generator.emitLabel(scope->breakTarget()); // FIXME: This should return the last statement executed so that it can be returned as a Completion return 0; } // ------------------------------ ForNode -------------------------------------- RegisterID* ForNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) dst = 0; RefPtr scope = generator.newLabelScope(LabelScope::Loop); generator.emitDebugHook(WillExecuteStatement, firstLine(), lastLine()); if (m_expr1) generator.emitNode(ignoredResult(), m_expr1.get()); RefPtr condition = generator.newLabel(); generator.emitJump(condition.get()); RefPtr topOfLoop = generator.newLabel(); generator.emitLabel(topOfLoop.get()); if (!m_statement->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_statement->firstLine(), m_statement->lastLine()); RefPtr result = generator.emitNode(dst, m_statement.get()); generator.emitLabel(scope->continueTarget()); if (m_expr3) generator.emitNode(ignoredResult(), m_expr3.get()); generator.emitLabel(condition.get()); if (m_expr2) { RegisterID* cond = generator.emitNode(m_expr2.get()); generator.emitJumpIfTrue(cond, topOfLoop.get()); } else generator.emitJump(topOfLoop.get()); generator.emitLabel(scope->breakTarget()); return result.get(); } // ------------------------------ ForInNode ------------------------------------ ForInNode::ForInNode(JSGlobalData* globalData, ExpressionNode* l, ExpressionNode* expr, StatementNode* statement) : StatementNode(globalData) , m_init(0L) , m_lexpr(l) , m_expr(expr) , m_statement(statement) , m_identIsVarDecl(false) { } ForInNode::ForInNode(JSGlobalData* globalData, const Identifier& ident, ExpressionNode* in, ExpressionNode* expr, StatementNode* statement, int divot, int startOffset, int endOffset) : StatementNode(globalData) , m_ident(ident) , m_lexpr(new ResolveNode(globalData, ident, divot - startOffset)) , m_expr(expr) , m_statement(statement) , m_identIsVarDecl(true) { if (in) { AssignResolveNode* node = new AssignResolveNode(globalData, ident, in, true); node->setExceptionSourceCode(divot, divot - startOffset, endOffset - divot); m_init = node; } // for( var foo = bar in baz ) } RegisterID* ForInNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr scope = generator.newLabelScope(LabelScope::Loop); if (!m_lexpr->isLocation()) return emitThrowError(generator, ReferenceError, "Left side of for-in statement is not a reference."); RefPtr continueTarget = generator.newLabel(); generator.emitDebugHook(WillExecuteStatement, firstLine(), lastLine()); if (m_init) generator.emitNode(ignoredResult(), m_init.get()); RegisterID* forInBase = generator.emitNode(m_expr.get()); RefPtr iter = generator.emitGetPropertyNames(generator.newTemporary(), forInBase); generator.emitJump(scope->continueTarget()); RefPtr loopStart = generator.newLabel(); generator.emitLabel(loopStart.get()); RegisterID* propertyName; if (m_lexpr->isResolveNode()) { const Identifier& ident = static_cast(m_lexpr.get())->identifier(); propertyName = generator.registerFor(ident); if (!propertyName) { propertyName = generator.newTemporary(); RefPtr protect = propertyName; RegisterID* base = generator.emitResolveBase(generator.newTemporary(), ident); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitPutById(base, ident, propertyName); } } else if (m_lexpr->isDotAccessorNode()) { DotAccessorNode* assignNode = static_cast(m_lexpr.get()); const Identifier& ident = assignNode->identifier(); propertyName = generator.newTemporary(); RefPtr protect = propertyName; RegisterID* base = generator.emitNode(assignNode->base()); generator.emitExpressionInfo(assignNode->divot(), assignNode->startOffset(), assignNode->endOffset()); generator.emitPutById(base, ident, propertyName); } else { ASSERT(m_lexpr->isBracketAccessorNode()); BracketAccessorNode* assignNode = static_cast(m_lexpr.get()); propertyName = generator.newTemporary(); RefPtr protect = propertyName; RefPtr base = generator.emitNode(assignNode->base()); RegisterID* subscript = generator.emitNode(assignNode->subscript()); generator.emitExpressionInfo(assignNode->divot(), assignNode->startOffset(), assignNode->endOffset()); generator.emitPutByVal(base.get(), subscript, propertyName); } if (!m_statement->isBlock()) generator.emitDebugHook(WillExecuteStatement, m_statement->firstLine(), m_statement->lastLine()); generator.emitNode(dst, m_statement.get()); generator.emitLabel(scope->continueTarget()); generator.emitNextPropertyName(propertyName, iter.get(), loopStart.get()); generator.emitLabel(scope->breakTarget()); return dst; } // ------------------------------ ContinueNode --------------------------------- // ECMA 12.7 RegisterID* ContinueNode::emitCode(CodeGenerator& generator, RegisterID* dst) { LabelScope* scope = generator.continueTarget(m_ident); if (!scope) return m_ident.isEmpty() ? emitThrowError(generator, SyntaxError, "Invalid continue statement.") : emitThrowError(generator, SyntaxError, "Undefined label: '%s'.", m_ident); generator.emitJumpScopes(scope->continueTarget(), scope->scopeDepth()); return dst; } // ------------------------------ BreakNode ------------------------------------ // ECMA 12.8 RegisterID* BreakNode::emitCode(CodeGenerator& generator, RegisterID* dst) { LabelScope* scope = generator.breakTarget(m_ident); if (!scope) return m_ident.isEmpty() ? emitThrowError(generator, SyntaxError, "Invalid break statement.") : emitThrowError(generator, SyntaxError, "Undefined label: '%s'.", m_ident); generator.emitJumpScopes(scope->breakTarget(), scope->scopeDepth()); return dst; } // ------------------------------ ReturnNode ----------------------------------- RegisterID* ReturnNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (generator.codeType() != FunctionCode) return emitThrowError(generator, SyntaxError, "Invalid return statement."); if (dst == ignoredResult()) dst = 0; RegisterID* r0 = m_value ? generator.emitNode(dst, m_value.get()) : generator.emitLoad(dst, jsUndefined()); if (generator.scopeDepth()) { RefPtr l0 = generator.newLabel(); generator.emitJumpScopes(l0.get(), 0); generator.emitLabel(l0.get()); } generator.emitDebugHook(WillLeaveCallFrame, firstLine(), lastLine()); return generator.emitReturn(r0); } // ------------------------------ WithNode ------------------------------------- RegisterID* WithNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr scope = generator.emitNode(m_expr.get()); // scope must be protected until popped generator.emitExpressionInfo(m_divot, m_expressionLength, 0); generator.emitPushScope(scope.get()); RegisterID* result = generator.emitNode(dst, m_statement.get()); generator.emitPopScope(); return result; } // ------------------------------ CaseBlockNode -------------------------------- enum SwitchKind { SwitchUnset = 0, SwitchNumber = 1, SwitchString = 2, SwitchNeither = 3 }; static void processClauseList(ClauseListNode* list, Vector& literalVector, SwitchKind& typeForTable, bool& singleCharacterSwitch, int32_t& min_num, int32_t& max_num) { for (; list; list = list->getNext()) { ExpressionNode* clauseExpression = list->getClause()->expr(); literalVector.append(clauseExpression); if (clauseExpression->isNumber()) { double value = static_cast(clauseExpression)->value(); if ((typeForTable & ~SwitchNumber) || !JSImmediate::from(value)) { typeForTable = SwitchNeither; break; } int32_t intVal = static_cast(value); ASSERT(intVal == value); if (intVal < min_num) min_num = intVal; if (intVal > max_num) max_num = intVal; typeForTable = SwitchNumber; continue; } if (clauseExpression->isString()) { if (typeForTable & ~SwitchString) { typeForTable = SwitchNeither; break; } const UString& value = static_cast(clauseExpression)->value().ustring(); if (singleCharacterSwitch &= value.size() == 1) { int32_t intVal = value.rep()->data()[0]; if (intVal < min_num) min_num = intVal; if (intVal > max_num) max_num = intVal; } typeForTable = SwitchString; continue; } typeForTable = SwitchNeither; break; } } SwitchInfo::SwitchType CaseBlockNode::tryOptimizedSwitch(Vector& literalVector, int32_t& min_num, int32_t& max_num) { SwitchKind typeForTable = SwitchUnset; bool singleCharacterSwitch = true; processClauseList(m_list1.get(), literalVector, typeForTable, singleCharacterSwitch, min_num, max_num); processClauseList(m_list2.get(), literalVector, typeForTable, singleCharacterSwitch, min_num, max_num); if (typeForTable == SwitchUnset || typeForTable == SwitchNeither) return SwitchInfo::SwitchNone; if (typeForTable == SwitchNumber) { int32_t range = max_num - min_num; if (min_num <= max_num && range <= 1000 && (range / literalVector.size()) < 10) return SwitchInfo::SwitchImmediate; return SwitchInfo::SwitchNone; } ASSERT(typeForTable == SwitchString); if (singleCharacterSwitch) { int32_t range = max_num - min_num; if (min_num <= max_num && range <= 1000 && (range / literalVector.size()) < 10) return SwitchInfo::SwitchCharacter; } return SwitchInfo::SwitchString; } RegisterID* CaseBlockNode::emitCodeForBlock(CodeGenerator& generator, RegisterID* switchExpression, RegisterID* dst) { RefPtr defaultLabel; Vector, 8> labelVector; Vector literalVector; int32_t min_num = std::numeric_limits::max(); int32_t max_num = std::numeric_limits::min(); SwitchInfo::SwitchType switchType = tryOptimizedSwitch(literalVector, min_num, max_num); if (switchType != SwitchInfo::SwitchNone) { // Prepare the various labels for (uint32_t i = 0; i < literalVector.size(); i++) labelVector.append(generator.newLabel()); defaultLabel = generator.newLabel(); generator.beginSwitch(switchExpression, switchType); } else { // Setup jumps for (ClauseListNode* list = m_list1.get(); list; list = list->getNext()) { RefPtr clauseVal = generator.newTemporary(); generator.emitNode(clauseVal.get(), list->getClause()->expr()); generator.emitBinaryOp(op_stricteq, clauseVal.get(), clauseVal.get(), switchExpression, OperandTypes()); labelVector.append(generator.newLabel()); generator.emitJumpIfTrue(clauseVal.get(), labelVector[labelVector.size() - 1].get()); } for (ClauseListNode* list = m_list2.get(); list; list = list->getNext()) { RefPtr clauseVal = generator.newTemporary(); generator.emitNode(clauseVal.get(), list->getClause()->expr()); generator.emitBinaryOp(op_stricteq, clauseVal.get(), clauseVal.get(), switchExpression, OperandTypes()); labelVector.append(generator.newLabel()); generator.emitJumpIfTrue(clauseVal.get(), labelVector[labelVector.size() - 1].get()); } defaultLabel = generator.newLabel(); generator.emitJump(defaultLabel.get()); } RegisterID* result = 0; size_t i = 0; for (ClauseListNode* list = m_list1.get(); list; list = list->getNext()) { generator.emitLabel(labelVector[i++].get()); result = statementListEmitCode(list->getClause()->children(), generator, dst); } if (m_defaultClause) { generator.emitLabel(defaultLabel.get()); result = statementListEmitCode(m_defaultClause->children(), generator, dst); } for (ClauseListNode* list = m_list2.get(); list; list = list->getNext()) { generator.emitLabel(labelVector[i++].get()); result = statementListEmitCode(list->getClause()->children(), generator, dst); } if (!m_defaultClause) generator.emitLabel(defaultLabel.get()); ASSERT(i == labelVector.size()); if (switchType != SwitchInfo::SwitchNone) { ASSERT(labelVector.size() == literalVector.size()); generator.endSwitch(labelVector.size(), labelVector.data(), literalVector.data(), defaultLabel.get(), min_num, max_num); } return result; } // ------------------------------ SwitchNode ----------------------------------- RegisterID* SwitchNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr scope = generator.newLabelScope(LabelScope::Switch); RefPtr r0 = generator.emitNode(m_expr.get()); RegisterID* r1 = m_block->emitCodeForBlock(generator, r0.get(), dst); generator.emitLabel(scope->breakTarget()); return r1; } // ------------------------------ LabelNode ------------------------------------ RegisterID* LabelNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (generator.breakTarget(m_name)) return emitThrowError(generator, SyntaxError, "Duplicate label: %s.", m_name); RefPtr scope = generator.newLabelScope(LabelScope::NamedLabel, &m_name); RegisterID* r0 = generator.emitNode(dst, m_statement.get()); generator.emitLabel(scope->breakTarget()); return r0; } // ------------------------------ ThrowNode ------------------------------------ RegisterID* ThrowNode::emitCode(CodeGenerator& generator, RegisterID* dst) { if (dst == ignoredResult()) dst = 0; RefPtr expr = generator.emitNode(dst, m_expr.get()); generator.emitExpressionInfo(m_divot, m_startOffset, m_endOffset); generator.emitThrow(expr.get()); return dst; } // ------------------------------ TryNode -------------------------------------- RegisterID* TryNode::emitCode(CodeGenerator& generator, RegisterID* dst) { RefPtr tryStartLabel = generator.newLabel(); RefPtr tryEndLabel = generator.newLabel(); RefPtr finallyStart; RefPtr finallyReturnAddr; if (m_finallyBlock) { finallyStart = generator.newLabel(); finallyReturnAddr = generator.newTemporary(); generator.pushFinallyContext(finallyStart.get(), finallyReturnAddr.get()); } generator.emitLabel(tryStartLabel.get()); generator.emitNode(dst, m_tryBlock.get()); generator.emitLabel(tryEndLabel.get()); if (m_catchBlock) { RefPtr handlerEndLabel = generator.newLabel(); generator.emitJump(handlerEndLabel.get()); RefPtr exceptionRegister = generator.emitCatch(generator.newTemporary(), tryStartLabel.get(), tryEndLabel.get()); generator.emitPushNewScope(exceptionRegister.get(), m_exceptionIdent, exceptionRegister.get()); generator.emitNode(dst, m_catchBlock.get()); generator.emitPopScope(); generator.emitLabel(handlerEndLabel.get()); } if (m_finallyBlock) { generator.popFinallyContext(); // there may be important registers live at the time we jump // to a finally block (such as for a return or throw) so we // ref the highest register ever used as a conservative // approach to not clobbering anything important RefPtr highestUsedRegister = generator.highestUsedRegister(); RefPtr finallyEndLabel = generator.newLabel(); generator.emitJumpSubroutine(finallyReturnAddr.get(), finallyStart.get()); // Use a label to record the subtle fact that sret will return to the // next instruction. sret is the only way to jump without an explicit label. generator.emitLabel(generator.newLabel().get()); generator.emitJump(finallyEndLabel.get()); // Finally block for exception path RefPtr tempExceptionRegister = generator.emitCatch(generator.newTemporary(), tryStartLabel.get(), generator.emitLabel(generator.newLabel().get()).get()); generator.emitJumpSubroutine(finallyReturnAddr.get(), finallyStart.get()); // Use a label to record the subtle fact that sret will return to the // next instruction. sret is the only way to jump without an explicit label. generator.emitLabel(generator.newLabel().get()); generator.emitThrow(tempExceptionRegister.get()); // emit the finally block itself generator.emitLabel(finallyStart.get()); generator.emitNode(dst, m_finallyBlock.get()); generator.emitSubroutineReturn(finallyReturnAddr.get()); generator.emitLabel(finallyEndLabel.get()); } return dst; } // ------------------------------ ScopeNode ----------------------------- ScopeNode::ScopeNode(JSGlobalData* globalData, const SourceCode& source, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, CodeFeatures features, int numConstants) : BlockNode(globalData, children) , m_source(source) , m_features(features) , m_numConstants(numConstants) { if (varStack) m_varStack = *varStack; if (funcStack) m_functionStack = *funcStack; #if ENABLE(OPCODE_SAMPLING) globalData->machine->sampler()->notifyOfScope(this); #endif } // ------------------------------ ProgramNode ----------------------------- ProgramNode::ProgramNode(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& source, CodeFeatures features, int numConstants) : ScopeNode(globalData, source, children, varStack, funcStack, features, numConstants) { } ProgramNode* ProgramNode::create(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& source, CodeFeatures features, int numConstants) { return new ProgramNode(globalData, children, varStack, funcStack, source, features, numConstants); } // ------------------------------ EvalNode ----------------------------- EvalNode::EvalNode(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& source, CodeFeatures features, int numConstants) : ScopeNode(globalData, source, children, varStack, funcStack, features, numConstants) { } RegisterID* EvalNode::emitCode(CodeGenerator& generator, RegisterID*) { generator.emitDebugHook(WillExecuteProgram, firstLine(), lastLine()); RefPtr dstRegister = generator.newTemporary(); generator.emitLoad(dstRegister.get(), jsUndefined()); statementListEmitCode(m_children, generator, dstRegister.get()); generator.emitDebugHook(DidExecuteProgram, firstLine(), lastLine()); generator.emitEnd(dstRegister.get()); return 0; } void EvalNode::generateCode(ScopeChainNode* scopeChainNode) { ScopeChain scopeChain(scopeChainNode); JSGlobalObject* globalObject = scopeChain.globalObject(); SymbolTable symbolTable; m_code.set(new EvalCodeBlock(this, globalObject, source().provider())); CodeGenerator generator(this, globalObject->debugger(), scopeChain, &symbolTable, m_code.get()); generator.generate(); } EvalNode* EvalNode::create(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& source, CodeFeatures features, int numConstants) { return new EvalNode(globalData, children, varStack, funcStack, source, features, numConstants); } // ------------------------------ FunctionBodyNode ----------------------------- FunctionBodyNode::FunctionBodyNode(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& sourceCode, CodeFeatures features, int numConstants) : ScopeNode(globalData, sourceCode, children, varStack, funcStack, features, numConstants) , m_parameters(0) , m_parameterCount(0) , m_refCount(0) { } FunctionBodyNode::~FunctionBodyNode() { if (m_parameters) fastFree(m_parameters); } void FunctionBodyNode::finishParsing(const SourceCode& source, ParameterNode* firstParameter) { Vector parameters; for (ParameterNode* parameter = firstParameter; parameter; parameter = parameter->nextParam()) parameters.append(parameter->ident()); size_t count = parameters.size(); setSource(source); finishParsing(parameters.releaseBuffer(), count); } void FunctionBodyNode::finishParsing(Identifier* parameters, size_t parameterCount) { ASSERT(!source().isNull()); m_parameters = parameters; m_parameterCount = parameterCount; } void FunctionBodyNode::mark() { if (m_code) m_code->mark(); } FunctionBodyNode* FunctionBodyNode::create(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, CodeFeatures features, int numConstants) { return new FunctionBodyNode(globalData, children, varStack, funcStack, SourceCode(), features, numConstants); } FunctionBodyNode* FunctionBodyNode::create(JSGlobalData* globalData, SourceElements* children, VarStack* varStack, FunctionStack* funcStack, const SourceCode& sourceCode, CodeFeatures features, int numConstants) { return new FunctionBodyNode(globalData, children, varStack, funcStack, sourceCode, features, numConstants); } void FunctionBodyNode::generateCode(ScopeChainNode* scopeChainNode) { ScopeChain scopeChain(scopeChainNode); JSGlobalObject* globalObject = scopeChain.globalObject(); m_code.set(new CodeBlock(this, FunctionCode, source().provider(), source().startOffset())); CodeGenerator generator(this, globalObject->debugger(), scopeChain, &m_symbolTable, m_code.get()); generator.generate(); } RegisterID* FunctionBodyNode::emitCode(CodeGenerator& generator, RegisterID*) { generator.emitDebugHook(DidEnterCallFrame, firstLine(), lastLine()); statementListEmitCode(m_children, generator, ignoredResult()); if (!m_children.size() || !m_children.last()->isReturnNode()) { RegisterID* r0 = generator.emitLoad(0, jsUndefined()); generator.emitDebugHook(WillLeaveCallFrame, firstLine(), lastLine()); generator.emitReturn(r0); } return 0; } RegisterID* ProgramNode::emitCode(CodeGenerator& generator, RegisterID*) { generator.emitDebugHook(WillExecuteProgram, firstLine(), lastLine()); RefPtr dstRegister = generator.newTemporary(); generator.emitLoad(dstRegister.get(), jsUndefined()); statementListEmitCode(m_children, generator, dstRegister.get()); generator.emitDebugHook(DidExecuteProgram, firstLine(), lastLine()); generator.emitEnd(dstRegister.get()); return 0; } void ProgramNode::generateCode(ScopeChainNode* scopeChainNode) { ScopeChain scopeChain(scopeChainNode); JSGlobalObject* globalObject = scopeChain.globalObject(); m_code.set(new ProgramCodeBlock(this, GlobalCode, globalObject, source().provider())); CodeGenerator generator(this, globalObject->debugger(), scopeChain, &globalObject->symbolTable(), m_code.get(), m_varStack, m_functionStack); generator.generate(); } UString FunctionBodyNode::paramString() const { UString s(""); for (size_t pos = 0; pos < m_parameterCount; ++pos) { if (!s.isEmpty()) s += ", "; s += parameters()[pos].ustring(); } return s; } Identifier* FunctionBodyNode::copyParameters() { Identifier* parameters = static_cast(fastMalloc(m_parameterCount * sizeof(Identifier))); VectorCopier::uninitializedCopy(m_parameters, m_parameters + m_parameterCount, parameters); return parameters; } // ------------------------------ FuncDeclNode --------------------------------- JSFunction* FuncDeclNode::makeFunction(ExecState* exec, ScopeChainNode* scopeChain) { return new (exec) JSFunction(exec, m_ident, m_body.get(), scopeChain); } RegisterID* FuncDeclNode::emitCode(CodeGenerator&, RegisterID* dst) { return dst; } // ------------------------------ FuncExprNode --------------------------------- RegisterID* FuncExprNode::emitCode(CodeGenerator& generator, RegisterID* dst) { return generator.emitNewFunctionExpression(generator.finalDestination(dst), this); } JSFunction* FuncExprNode::makeFunction(ExecState* exec, ScopeChainNode* scopeChain) { JSFunction* func = new (exec) JSFunction(exec, m_ident, m_body.get(), scopeChain); /* The Identifier in a FunctionExpression can be referenced from inside the FunctionExpression's FunctionBody to allow the function to call itself recursively. However, unlike in a FunctionDeclaration, the Identifier in a FunctionExpression cannot be referenced from and does not affect the scope enclosing the FunctionExpression. */ if (!m_ident.isNull()) { JSStaticScopeObject* functionScopeObject = new (exec) JSStaticScopeObject(exec, m_ident, func, ReadOnly | DontDelete); func->scope().push(functionScopeObject); } return func; } } // namespace JSC