| /* |
| * Copyright (C) 1999-2002 Harri Porten (porten@kde.org) |
| * Copyright (C) 2001 Peter Kelly (pmk@post.com) |
| * Copyright (C) 2003-2022 Apple Inc. All rights reserved. |
| * Copyright (C) 2007 Cameron Zwarich (cwzwarich@uwaterloo.ca) |
| * Copyright (C) 2007 Maks Orlovich |
| * Copyright (C) 2007 Eric Seidel <eric@webkit.org> |
| * Copyright (C) 2012 Igalia, S.L. |
| * |
| * 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 "NodeConstructors.h" |
| |
| #include "AbstractModuleRecord.h" |
| #include "BuiltinNames.h" |
| #include "BytecodeGenerator.h" |
| #include "BytecodeGeneratorBaseInlines.h" |
| #include "JSArrayIterator.h" |
| #include "JSAsyncGenerator.h" |
| #include "JSCInlines.h" |
| #include "JSGenerator.h" |
| #include "JSImmutableButterfly.h" |
| #include "JSMapIterator.h" |
| #include "JSSetIterator.h" |
| #include "JSStringIterator.h" |
| #include "LabelScope.h" |
| #include "LinkTimeConstant.h" |
| #include "ModuleScopeData.h" |
| #include "StackAlignment.h" |
| #include "UnlinkedMetadataTableInlines.h" |
| #include "YarrFlags.h" |
| #include <wtf/Assertions.h> |
| #include <wtf/text/StringBuilder.h> |
| |
| namespace JSC { |
| |
| /* |
| Details of the emitBytecode function. |
| |
| Return value: The register holding the production's value. |
| dst: An optional parameter specifying the most efficient destination at |
| which to store the production's value. |
| If dst is null, you may return whatever VirtualRegister you want. Otherwise you have to return dst. |
| |
| The dst argument provides for a crude form of copy propagation. For example, |
| |
| x = 1 |
| |
| becomes |
| |
| load r[x], 1 |
| |
| instead of |
| |
| load r0, 1 |
| mov r[x], r0 |
| |
| because the assignment node, "x =", passes r[x] as dst to the number node, "1". |
| */ |
| |
| void ExpressionNode::emitBytecodeInConditionContext(BytecodeGenerator& generator, Label& trueTarget, Label& falseTarget, FallThroughMode fallThroughMode) |
| { |
| RegisterID* result = generator.emitNode(this); |
| if (fallThroughMode == FallThroughMeansTrue) |
| generator.emitJumpIfFalse(result, falseTarget); |
| else |
| generator.emitJumpIfTrue(result, trueTarget); |
| } |
| |
| // ------------------------------ ThrowableExpressionData -------------------------------- |
| |
| RegisterID* ThrowableExpressionData::emitThrowReferenceError(BytecodeGenerator& generator, ASCIILiteral message, RegisterID* dst) |
| { |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitThrowReferenceError(message); |
| if (dst) |
| return dst; |
| return generator.newTemporary(); |
| } |
| |
| // ------------------------------ ConstantNode ---------------------------------- |
| |
| void ConstantNode::emitBytecodeInConditionContext(BytecodeGenerator& generator, Label& trueTarget, Label& falseTarget, FallThroughMode fallThroughMode) |
| { |
| TriState value = TriState::Indeterminate; |
| JSValue constant = jsValue(generator); |
| if (LIKELY(constant)) |
| value = constant.pureToBoolean(); |
| |
| if (UNLIKELY(needsDebugHook())) { |
| if (value != TriState::Indeterminate) |
| generator.emitDebugHook(this); |
| } |
| |
| if (value == TriState::Indeterminate) |
| ExpressionNode::emitBytecodeInConditionContext(generator, trueTarget, falseTarget, fallThroughMode); |
| else if (value == TriState::True && fallThroughMode == FallThroughMeansFalse) |
| generator.emitJump(trueTarget); |
| else if (value == TriState::False && fallThroughMode == FallThroughMeansTrue) |
| generator.emitJump(falseTarget); |
| |
| // All other cases are unconditional fall-throughs, like "if (true)". |
| } |
| |
| RegisterID* ConstantNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| JSValue constant = jsValue(generator); |
| if (UNLIKELY(!constant)) { |
| // This can happen if we try to parse a string or BigInt so enormous that we OOM. |
| return generator.emitThrowExpressionTooDeepException(); |
| } |
| return generator.emitLoad(dst, constant); |
| } |
| |
| JSValue StringNode::jsValue(BytecodeGenerator& generator) const |
| { |
| return generator.addStringConstant(m_value); |
| } |
| |
| JSValue BigIntNode::jsValue(BytecodeGenerator& generator) const |
| { |
| return generator.addBigIntConstant(m_value, m_radix, m_sign); |
| } |
| |
| // ------------------------------ NumberNode ---------------------------------- |
| |
| RegisterID* NumberNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.emitLoad(dst, jsValue(generator), isIntegerNode() ? SourceCodeRepresentation::Integer : SourceCodeRepresentation::Double); |
| } |
| |
| // ------------------------------ RegExpNode ----------------------------------- |
| |
| RegisterID* RegExpNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| |
| auto flags = Yarr::parseFlags(m_flags.string()); |
| ASSERT(flags); |
| RegExp* regExp = RegExp::create(generator.vm(), m_pattern.string(), flags.value()); |
| if (regExp->isValid()) |
| return generator.emitNewRegExp(generator.finalDestination(dst), regExp); |
| |
| const char* messageCharacters = regExp->errorMessage(); |
| const Identifier& message = generator.parserArena().identifierArena().makeIdentifier(generator.vm(), bitwise_cast<const LChar*>(messageCharacters), strlen(messageCharacters)); |
| generator.emitThrowStaticError(ErrorTypeWithExtension::SyntaxError, message); |
| return generator.emitLoad(generator.finalDestination(dst), jsUndefined()); |
| } |
| |
| // ------------------------------ ThisNode ------------------------------------- |
| |
| RegisterID* ThisNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| generator.ensureThis(); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| |
| RegisterID* result = generator.move(dst, generator.thisRegister()); |
| static const unsigned thisLength = strlen("this"); |
| generator.emitProfileType(generator.thisRegister(), position(), position() + thisLength); |
| return result; |
| } |
| |
| // ------------------------------ SuperNode ------------------------------------- |
| |
| static RegisterID* emitHomeObjectForCallee(BytecodeGenerator& generator) |
| { |
| if ((generator.isDerivedClassContext() || generator.isDerivedConstructorContext()) && generator.parseMode() != SourceParseMode::ClassFieldInitializerMode) { |
| RegisterID* derivedConstructor = generator.emitLoadDerivedConstructorFromArrowFunctionLexicalEnvironment(); |
| return generator.emitGetById(generator.newTemporary(), derivedConstructor, generator.propertyNames().builtinNames().homeObjectPrivateName()); |
| } |
| |
| RegisterID callee; |
| callee.setIndex(CallFrameSlot::callee); |
| return generator.emitGetById(generator.newTemporary(), &callee, generator.propertyNames().builtinNames().homeObjectPrivateName()); |
| } |
| |
| static RegisterID* emitSuperBaseForCallee(BytecodeGenerator& generator) |
| { |
| RefPtr<RegisterID> homeObject = emitHomeObjectForCallee(generator); |
| return generator.emitGetPrototypeOf(generator.newTemporary(), homeObject.get()); |
| } |
| |
| static RegisterID* emitGetSuperFunctionForConstruct(BytecodeGenerator& generator) |
| { |
| if (generator.isDerivedConstructorContext()) |
| return generator.emitGetPrototypeOf(generator.newTemporary(), generator.emitLoadDerivedConstructorFromArrowFunctionLexicalEnvironment()); |
| |
| RegisterID callee; |
| callee.setIndex(CallFrameSlot::callee); |
| return generator.emitGetPrototypeOf(generator.newTemporary(), &callee); |
| } |
| |
| RegisterID* SuperNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RegisterID* result = emitSuperBaseForCallee(generator); |
| return generator.move(generator.finalDestination(dst), result); |
| } |
| |
| // ------------------------------ ImportNode ------------------------------------- |
| |
| RegisterID* ImportNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> importModule = generator.moveLinkTimeConstant(nullptr, LinkTimeConstant::importModule); |
| CallArguments arguments(generator, nullptr, m_option ? 2 : 1); |
| generator.emitLoad(arguments.thisRegister(), jsUndefined()); |
| generator.emitNode(arguments.argumentRegister(0), m_expr); |
| if (m_option) |
| generator.emitNode(arguments.argumentRegister(1), m_option); |
| return generator.emitCall(generator.finalDestination(dst, importModule.get()), importModule.get(), NoExpectedFunction, arguments, divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| } |
| |
| // ------------------------------ NewTargetNode ---------------------------------- |
| |
| RegisterID* NewTargetNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| |
| return generator.move(dst, generator.newTarget()); |
| } |
| |
| // ------------------------------ ImportMetaNode --------------------------------- |
| |
| RegisterID* ImportMetaNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| return generator.emitNode(dst, m_expr); |
| } |
| |
| // ------------------------------ ResolveNode ---------------------------------- |
| |
| bool ResolveNode::isPure(BytecodeGenerator& generator) const |
| { |
| return generator.variable(m_ident).offset().isStack(); |
| } |
| |
| RegisterID* ResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| Variable var = generator.variable(m_ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| |
| generator.emitProfileType(local, var, m_position, m_position + m_ident.length()); |
| return generator.move(dst, local); |
| } |
| |
| JSTextPosition divot = m_start + m_ident.length(); |
| generator.emitExpressionInfo(divot, m_start, divot); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(dst, var); |
| RegisterID* finalDest = generator.finalDestination(dst); |
| RefPtr<RegisterID> uncheckedResult = generator.newTemporary(); |
| generator.emitGetFromScope(uncheckedResult.get(), scope.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, uncheckedResult.get(), nullptr); |
| generator.move(finalDest, uncheckedResult.get()); |
| generator.emitProfileType(finalDest, var, m_position, m_position + m_ident.length()); |
| return finalDest; |
| } |
| |
| // ------------------------------ TemplateStringNode ----------------------------------- |
| |
| RegisterID* TemplateStringNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| ASSERT(cooked()); |
| return generator.emitLoad(dst, JSValue(generator.addStringConstant(*cooked()))); |
| } |
| |
| // ------------------------------ TemplateLiteralNode ----------------------------------- |
| |
| RegisterID* TemplateLiteralNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_templateExpressions) { |
| TemplateStringNode* templateString = m_templateStrings->value(); |
| ASSERT_WITH_MESSAGE(!m_templateStrings->next(), "Only one template element exists because there's no expression in a given template literal."); |
| return generator.emitNode(dst, templateString); |
| } |
| |
| Vector<RefPtr<RegisterID>, 16> temporaryRegisters; |
| |
| TemplateStringListNode* templateString = m_templateStrings; |
| TemplateExpressionListNode* templateExpression = m_templateExpressions; |
| for (; templateExpression; templateExpression = templateExpression->next(), templateString = templateString->next()) { |
| // Evaluate TemplateString. |
| ASSERT(templateString->value()->cooked()); |
| if (!templateString->value()->cooked()->isEmpty()) { |
| temporaryRegisters.append(generator.newTemporary()); |
| generator.emitNode(temporaryRegisters.last().get(), templateString->value()); |
| } |
| |
| // Evaluate Expression. |
| temporaryRegisters.append(generator.newTemporary()); |
| generator.emitNode(temporaryRegisters.last().get(), templateExpression->value()); |
| generator.emitToString(temporaryRegisters.last().get(), temporaryRegisters.last().get()); |
| } |
| |
| // Evaluate tail TemplateString. |
| ASSERT(templateString->value()->cooked()); |
| if (!templateString->value()->cooked()->isEmpty()) { |
| temporaryRegisters.append(generator.newTemporary()); |
| generator.emitNode(temporaryRegisters.last().get(), templateString->value()); |
| } |
| |
| if (temporaryRegisters.size() == 1) |
| return generator.emitToString(generator.finalDestination(dst, temporaryRegisters[0].get()), temporaryRegisters[0].get()); |
| |
| return generator.emitStrcat(generator.finalDestination(dst, temporaryRegisters[0].get()), temporaryRegisters[0].get(), temporaryRegisters.size()); |
| } |
| |
| // ------------------------------ TaggedTemplateNode ----------------------------------- |
| |
| RegisterID* TaggedTemplateNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ExpectedFunction expectedFunction = NoExpectedFunction; |
| RefPtr<RegisterID> tag = nullptr; |
| RefPtr<RegisterID> base = nullptr; |
| if (!m_tag->isLocation()) { |
| tag = generator.newTemporary(); |
| tag = generator.emitNode(tag.get(), m_tag); |
| } else if (m_tag->isResolveNode()) { |
| ResolveNode* resolve = static_cast<ResolveNode*>(m_tag); |
| const Identifier& identifier = resolve->identifier(); |
| expectedFunction = generator.expectedFunctionForIdentifier(identifier); |
| |
| Variable var = generator.variable(identifier); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| tag = generator.move(generator.newTemporary(), local); |
| } else { |
| tag = generator.newTemporary(); |
| base = generator.newTemporary(); |
| |
| JSTextPosition newDivot = divotStart() + identifier.length(); |
| generator.emitExpressionInfo(newDivot, divotStart(), newDivot); |
| generator.move(base.get(), generator.emitResolveScope(base.get(), var)); |
| generator.emitGetFromScope(tag.get(), base.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, tag.get(), nullptr); |
| } |
| } else if (m_tag->isBracketAccessorNode()) { |
| BracketAccessorNode* bracket = static_cast<BracketAccessorNode*>(m_tag); |
| base = generator.newTemporary(); |
| base = generator.emitNode(base.get(), bracket->base()); |
| RefPtr<RegisterID> property = generator.emitNodeForProperty(bracket->subscript()); |
| if (bracket->base()->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| tag = generator.emitGetByVal(generator.newTemporary(), base.get(), thisValue.get(), property.get()); |
| } else |
| tag = generator.emitGetByVal(generator.newTemporary(), base.get(), property.get()); |
| } else { |
| ASSERT(m_tag->isDotAccessorNode()); |
| DotAccessorNode* dot = static_cast<DotAccessorNode*>(m_tag); |
| tag = generator.newTemporary(); |
| base = generator.newTemporary(); |
| base = generator.emitNode(base.get(), dot->base()); |
| tag = dot->emitGetPropertyValue(generator, tag.get(), base.get()); |
| } |
| |
| RefPtr<RegisterID> templateObject = generator.emitGetTemplateObject(nullptr, this); |
| |
| unsigned expressionsCount = 0; |
| for (TemplateExpressionListNode* templateExpression = m_templateLiteral->templateExpressions(); templateExpression; templateExpression = templateExpression->next()) |
| ++expressionsCount; |
| |
| CallArguments callArguments(generator, nullptr, 1 + expressionsCount); |
| if (base) |
| generator.move(callArguments.thisRegister(), base.get()); |
| else |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| |
| unsigned argumentIndex = 0; |
| generator.move(callArguments.argumentRegister(argumentIndex++), templateObject.get()); |
| for (TemplateExpressionListNode* templateExpression = m_templateLiteral->templateExpressions(); templateExpression; templateExpression = templateExpression->next()) |
| generator.emitNode(callArguments.argumentRegister(argumentIndex++), templateExpression->value()); |
| |
| return generator.emitCallInTailPosition(generator.finalDestination(dst, tag.get()), tag.get(), expectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| |
| // ------------------------------ ArrayNode ------------------------------------ |
| |
| RegisterID* ArrayNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| bool hadVariableExpression = false; |
| unsigned length = 0; |
| |
| IndexingType recommendedIndexingType = ArrayWithUndecided; |
| ElementNode* firstPutElement; |
| for (firstPutElement = m_element; firstPutElement; firstPutElement = firstPutElement->next()) { |
| if (firstPutElement->elision() || firstPutElement->value()->isSpreadExpression()) |
| break; |
| if (!firstPutElement->value()->isConstant()) |
| hadVariableExpression = true; |
| else { |
| JSValue constant = static_cast<ConstantNode*>(firstPutElement->value())->jsValue(generator); |
| if (UNLIKELY(!constant)) |
| hadVariableExpression = true; |
| else |
| recommendedIndexingType = leastUpperBoundOfIndexingTypeAndValue(recommendedIndexingType, constant); |
| } |
| |
| ++length; |
| } |
| |
| auto newArray = [&] (RegisterID* dst, ElementNode* elements, unsigned length, bool hadVariableExpression) { |
| if (length && !hadVariableExpression) { |
| recommendedIndexingType |= CopyOnWrite; |
| ASSERT(generator.vm().heap.isDeferred()); // We run bytecode generator under a DeferGC. If we stopped doing that, we'd need to put a DeferGC here as we filled in these slots. |
| auto* array = JSImmutableButterfly::create(generator.vm(), recommendedIndexingType, length); |
| unsigned index = 0; |
| for (ElementNode* element = elements; index < length; element = element->next()) { |
| ASSERT(element->value()->isConstant()); |
| JSValue constant = static_cast<ConstantNode*>(element->value())->jsValue(generator); |
| ASSERT(constant); |
| array->setIndex(generator.vm(), index++, constant); |
| } |
| return generator.emitNewArrayBuffer(dst, array, recommendedIndexingType); |
| } |
| return generator.emitNewArray(dst, elements, length, recommendedIndexingType); |
| }; |
| |
| if (!firstPutElement && !m_elision) |
| return newArray(generator.finalDestination(dst), m_element, length, hadVariableExpression); |
| |
| if (firstPutElement && firstPutElement->value()->isSpreadExpression()) { |
| bool hasElision = m_elision; |
| if (!hasElision) { |
| for (ElementNode* node = firstPutElement; node; node = node->next()) { |
| if (node->elision()) { |
| hasElision = true; |
| break; |
| } |
| } |
| } |
| |
| if (!hasElision) |
| return generator.emitNewArrayWithSpread(generator.finalDestination(dst), m_element); |
| } |
| |
| RefPtr<RegisterID> array = newArray(generator.tempDestination(dst), m_element, length, hadVariableExpression); |
| ElementNode* n = firstPutElement; |
| for (; n; n = n->next()) { |
| if (n->value()->isSpreadExpression()) |
| goto handleSpread; |
| RefPtr<RegisterID> value = generator.emitNode(n->value()); |
| length += n->elision(); |
| |
| RefPtr<RegisterID> index = generator.emitLoad(nullptr, jsNumber(length++)); |
| generator.emitDirectPutByVal(array.get(), index.get(), value.get()); |
| } |
| |
| if (m_elision) { |
| RegisterID* value = generator.emitLoad(nullptr, jsNumber(m_elision + length)); |
| generator.emitPutById(array.get(), generator.propertyNames().length, value); |
| } |
| |
| return generator.move(dst, array.get()); |
| |
| handleSpread: |
| RefPtr<RegisterID> index = generator.emitLoad(generator.newTemporary(), jsNumber(length)); |
| auto spreader = scopedLambda<void(BytecodeGenerator&, RegisterID*)>([array, index](BytecodeGenerator& generator, RegisterID* value) |
| { |
| generator.emitDirectPutByVal(array.get(), index.get(), value); |
| generator.emitInc(index.get()); |
| }); |
| for (; n; n = n->next()) { |
| if (n->elision()) |
| generator.emitBinaryOp<OpAdd>(index.get(), index.get(), generator.emitLoad(nullptr, jsNumber(n->elision())), OperandTypes(ResultType::numberTypeIsInt32(), ResultType::numberTypeIsInt32())); |
| if (n->value()->isSpreadExpression()) { |
| SpreadExpressionNode* spread = static_cast<SpreadExpressionNode*>(n->value()); |
| generator.emitEnumeration(spread, spread->expression(), spreader); |
| } else { |
| generator.emitDirectPutByVal(array.get(), index.get(), generator.emitNode(n->value())); |
| generator.emitInc(index.get()); |
| } |
| } |
| |
| if (m_elision) { |
| generator.emitBinaryOp<OpAdd>(index.get(), index.get(), generator.emitLoad(nullptr, jsNumber(m_elision)), OperandTypes(ResultType::numberTypeIsInt32(), ResultType::numberTypeIsInt32())); |
| generator.emitPutById(array.get(), generator.propertyNames().length, index.get()); |
| } |
| return generator.move(dst, array.get()); |
| } |
| |
| bool ArrayNode::isSimpleArray() const |
| { |
| if (m_elision) |
| return false; |
| for (ElementNode* ptr = m_element; ptr; ptr = ptr->next()) { |
| if (ptr->elision()) |
| return false; |
| if (ptr->value()->isSpreadExpression()) |
| return false; |
| } |
| return true; |
| } |
| |
| ArgumentListNode* ArrayNode::toArgumentList(ParserArena& parserArena, int lineNumber, int startPosition) const |
| { |
| ASSERT(!m_elision); |
| ElementNode* ptr = m_element; |
| if (!ptr) |
| return nullptr; |
| JSTokenLocation location; |
| location.line = lineNumber; |
| location.startOffset = startPosition; |
| ArgumentListNode* head = new (parserArena) ArgumentListNode(location, ptr->value()); |
| ArgumentListNode* tail = head; |
| ptr = ptr->next(); |
| for (; ptr; ptr = ptr->next()) { |
| ASSERT(!ptr->elision()); |
| tail = new (parserArena) ArgumentListNode(location, tail, ptr->value()); |
| } |
| return head; |
| } |
| |
| // ------------------------------ ObjectLiteralNode ---------------------------- |
| |
| RegisterID* ObjectLiteralNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_list) { |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.emitNewObject(generator.finalDestination(dst)); |
| } |
| RefPtr<RegisterID> newObj = generator.emitNewObject(generator.tempDestination(dst)); |
| generator.emitNode(newObj.get(), m_list); |
| return generator.move(dst, newObj.get()); |
| } |
| |
| // ------------------------------ PropertyListNode ----------------------------- |
| |
| static inline void emitPutHomeObject(BytecodeGenerator& generator, RegisterID* function, RegisterID* homeObject) |
| { |
| generator.emitPutById(function, generator.propertyNames().builtinNames().homeObjectPrivateName(), homeObject); |
| } |
| |
| void PropertyListNode::emitDeclarePrivateFieldNames(BytecodeGenerator& generator, RegisterID* scope) |
| { |
| // Walk the list and declare any Private property names (e.g. `#foo`) in the provided scope. |
| RefPtr<RegisterID> createPrivateSymbol; |
| for (PropertyListNode* p = this; p; p = p->m_next) { |
| const PropertyNode& node = *p->m_node; |
| if (node.type() & PropertyNode::PrivateField) { |
| if (!createPrivateSymbol) |
| createPrivateSymbol = generator.moveLinkTimeConstant(nullptr, LinkTimeConstant::createPrivateSymbol); |
| |
| CallArguments arguments(generator, nullptr, 0); |
| generator.emitLoad(arguments.thisRegister(), jsUndefined()); |
| RefPtr<RegisterID> symbol = generator.emitCall(generator.finalDestination(nullptr, createPrivateSymbol.get()), createPrivateSymbol.get(), NoExpectedFunction, arguments, position(), position(), position(), DebuggableCall::No); |
| |
| Variable var = generator.variable(*node.name()); |
| generator.emitPutToScope(scope, var, symbol.get(), DoNotThrowIfNotFound, InitializationMode::ConstInitialization); |
| } |
| } |
| } |
| |
| RegisterID* PropertyListNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dstOrConstructor, RegisterID* prototype, Vector<JSTextPosition>* instanceFieldLocations, Vector<JSTextPosition>* staticFieldLocations) |
| { |
| using GetterSetterPair = std::pair<PropertyNode*, PropertyNode*>; |
| using GetterSetterMap = HashMap<UniquedStringImpl*, GetterSetterPair, IdentifierRepHash>; |
| |
| if (hasPrivateAccessors()) { |
| GetterSetterMap privateAccessorMap; |
| |
| for (PropertyListNode* propertyList = this; propertyList; propertyList = propertyList->m_next) { |
| if (!(propertyList->m_node->type() & (PropertyNode::PrivateGetter | PropertyNode::PrivateSetter))) |
| continue; |
| |
| // We group private getters and setters to store them in a object |
| GetterSetterPair pair(propertyList->m_node, static_cast<PropertyNode*>(nullptr)); |
| GetterSetterMap::AddResult result = privateAccessorMap.add(propertyList->m_node->name()->impl(), pair); |
| auto& resultPair = result.iterator->value; |
| // If the map already contains an element with node->name(), |
| // we need to store this node in the second part. |
| if (!result.isNewEntry) |
| resultPair.second = propertyList->m_node; |
| continue; |
| } |
| |
| // Then we declare private accessors |
| for (auto& it : privateAccessorMap) { |
| // FIXME: Use GetterSetter to store private accessors |
| // https://bugs.webkit.org/show_bug.cgi?id=221915 |
| RefPtr<RegisterID> getterSetterObj = generator.emitNewObject(generator.newTemporary()); |
| GetterSetterPair pair = it.value; |
| |
| auto emitPutAccessor = [&] (PropertyNode* propertyNode) { |
| RegisterID* base = propertyNode->isInstanceClassProperty() ? prototype : dstOrConstructor; |
| |
| RefPtr<RegisterID> value = generator.emitNode(propertyNode->m_assign); |
| if (propertyNode->needsSuperBinding()) |
| emitPutHomeObject(generator, value.get(), base); |
| auto setterOrGetterIdent = propertyNode->m_type & PropertyNode::PrivateGetter |
| ? generator.propertyNames().builtinNames().getPrivateName() |
| : generator.propertyNames().builtinNames().setPrivateName(); |
| generator.emitDirectPutById(getterSetterObj.get(), setterOrGetterIdent, value.get()); |
| }; |
| |
| if (pair.first) |
| emitPutAccessor(pair.first); |
| |
| if (pair.second) |
| emitPutAccessor(pair.second); |
| |
| Variable var = generator.variable(*pair.first->name()); |
| generator.emitPutToScope(generator.scopeRegister(), var, getterSetterObj.get(), DoNotThrowIfNotFound, InitializationMode::ConstInitialization); |
| } |
| } |
| |
| PropertyListNode* p = this; |
| RegisterID* dst = nullptr; |
| |
| // Fast case: this loop just handles regular value properties. |
| for (; p && (p->m_node->m_type & PropertyNode::Constant); p = p->m_next) { |
| dst = p->m_node->isInstanceClassProperty() ? prototype : dstOrConstructor; |
| |
| if (p->m_node->type() & (PropertyNode::PrivateGetter | PropertyNode::PrivateSetter)) |
| continue; |
| |
| if (p->isComputedClassField()) |
| emitSaveComputedFieldName(generator, *p->m_node); |
| |
| if (p->isInstanceClassField() && !(p->m_node->type() & PropertyNode::PrivateMethod)) { |
| ASSERT(instanceFieldLocations); |
| instanceFieldLocations->append(p->position()); |
| continue; |
| } |
| |
| if (p->isStaticClassField()) { |
| ASSERT(staticFieldLocations); |
| staticFieldLocations->append(p->position()); |
| continue; |
| } |
| |
| emitPutConstantProperty(generator, dst, *p->m_node); |
| } |
| |
| // Were there any get/set properties? |
| if (p) { |
| // Build a list of getter/setter pairs to try to put them at the same time. If we encounter |
| // a constant property by the same name as accessor or a computed property or a spread, |
| // just emit everything as that may override previous values. |
| bool canOverrideProperties = false; |
| |
| GetterSetterMap instanceMap; |
| GetterSetterMap staticMap; |
| |
| // Build a map, pairing get/set values together. |
| for (PropertyListNode* q = p; q; q = q->m_next) { |
| PropertyNode* node = q->m_node; |
| if (node->m_type & PropertyNode::Computed || node->m_type & PropertyNode::Spread) { |
| canOverrideProperties = true; |
| break; |
| } |
| |
| GetterSetterMap& map = node->isStaticClassProperty() ? staticMap : instanceMap; |
| if (node->m_type & PropertyNode::Constant) { |
| if (map.contains(node->name()->impl())) { |
| canOverrideProperties = true; |
| break; |
| } |
| continue; |
| } |
| |
| // Duplicates are possible. |
| GetterSetterPair pair(node, static_cast<PropertyNode*>(nullptr)); |
| GetterSetterMap::AddResult result = map.add(node->name()->impl(), pair); |
| auto& resultPair = result.iterator->value; |
| if (!result.isNewEntry) { |
| if (resultPair.first->m_type == node->m_type) { |
| resultPair.first->setIsOverriddenByDuplicate(); |
| resultPair.first = node; |
| } else { |
| if (resultPair.second) |
| resultPair.second->setIsOverriddenByDuplicate(); |
| resultPair.second = node; |
| } |
| } |
| } |
| |
| // Iterate over the remaining properties in the list. |
| for (; p; p = p->m_next) { |
| PropertyNode* node = p->m_node; |
| dst = node->isInstanceClassProperty() ? prototype : dstOrConstructor; |
| |
| if (p->isComputedClassField()) |
| emitSaveComputedFieldName(generator, *p->m_node); |
| |
| if (p->m_node->type() & (PropertyNode::PrivateGetter | PropertyNode::PrivateSetter)) |
| continue; |
| |
| if (p->isInstanceClassField()) { |
| ASSERT(instanceFieldLocations); |
| ASSERT(node->m_type & PropertyNode::Constant); |
| instanceFieldLocations->append(p->position()); |
| continue; |
| } |
| |
| if (p->isStaticClassField()) { |
| ASSERT(staticFieldLocations); |
| staticFieldLocations->append(p->position()); |
| continue; |
| } |
| |
| // Handle regular values. |
| if (node->m_type & PropertyNode::Constant) { |
| emitPutConstantProperty(generator, dst, *node); |
| continue; |
| } else if (node->m_type & PropertyNode::Spread) { |
| generator.emitNode(dst, node->m_assign); |
| continue; |
| } |
| |
| RefPtr<RegisterID> value = generator.emitNode(node->m_assign); |
| bool needsSuperBinding = node->needsSuperBinding(); |
| if (needsSuperBinding) |
| emitPutHomeObject(generator, value.get(), dst); |
| |
| unsigned attributes = node->isClassProperty() ? (PropertyAttribute::Accessor | PropertyAttribute::DontEnum) : static_cast<unsigned>(PropertyAttribute::Accessor); |
| |
| ASSERT(node->m_type & (PropertyNode::Getter | PropertyNode::Setter)); |
| |
| // This is a get/set property which may be overridden by a computed property or spread later. |
| if (canOverrideProperties) { |
| // Computed accessors. |
| if (node->m_type & PropertyNode::Computed) { |
| RefPtr<RegisterID> propertyName = generator.emitNode(node->m_expression); |
| if (generator.shouldSetFunctionName(node->m_assign)) { |
| propertyName = generator.emitToPropertyKey(generator.newTemporary(), propertyName.get()); |
| generator.emitSetFunctionName(value.get(), propertyName.get()); |
| } |
| if (node->m_type & PropertyNode::Getter) |
| generator.emitPutGetterByVal(dst, propertyName.get(), attributes, value.get()); |
| else |
| generator.emitPutSetterByVal(dst, propertyName.get(), attributes, value.get()); |
| continue; |
| } |
| |
| if (node->m_type & PropertyNode::Getter) |
| generator.emitPutGetterById(dst, *node->name(), attributes, value.get()); |
| else |
| generator.emitPutSetterById(dst, *node->name(), attributes, value.get()); |
| continue; |
| } |
| |
| // This is a get/set property pair. |
| GetterSetterMap& map = node->isStaticClassProperty() ? staticMap : instanceMap; |
| GetterSetterMap::iterator it = map.find(node->name()->impl()); |
| ASSERT(it != map.end()); |
| GetterSetterPair& pair = it->value; |
| |
| // Was this already generated as a part of its partner? |
| if (pair.second == node || node->isOverriddenByDuplicate()) |
| continue; |
| |
| // Generate the paired node now. |
| RefPtr<RegisterID> getterReg; |
| RefPtr<RegisterID> setterReg; |
| RegisterID* secondReg = nullptr; |
| |
| if (node->m_type & PropertyNode::Getter) { |
| getterReg = value; |
| if (pair.second) { |
| ASSERT(pair.second->m_type & PropertyNode::Setter); |
| setterReg = generator.emitNode(pair.second->m_assign); |
| secondReg = setterReg.get(); |
| } else { |
| setterReg = generator.newTemporary(); |
| generator.emitLoad(setterReg.get(), jsUndefined()); |
| } |
| } else { |
| ASSERT(node->m_type & PropertyNode::Setter); |
| setterReg = value; |
| if (pair.second) { |
| ASSERT(pair.second->m_type & PropertyNode::Getter); |
| getterReg = generator.emitNode(pair.second->m_assign); |
| secondReg = getterReg.get(); |
| } else { |
| getterReg = generator.newTemporary(); |
| generator.emitLoad(getterReg.get(), jsUndefined()); |
| } |
| } |
| |
| ASSERT(!pair.second || needsSuperBinding == pair.second->needsSuperBinding()); |
| if (needsSuperBinding && pair.second) |
| emitPutHomeObject(generator, secondReg, dst); |
| |
| generator.emitPutGetterSetter(dst, *node->name(), attributes, getterReg.get(), setterReg.get()); |
| } |
| } |
| |
| return dstOrConstructor; |
| } |
| |
| void PropertyListNode::emitPutConstantProperty(BytecodeGenerator& generator, RegisterID* newObj, PropertyNode& node) |
| { |
| // Private fields are handled in a synthetic classFieldInitializer function, not here. |
| ASSERT(!(node.type() & PropertyNode::PrivateField)); |
| |
| if (PropertyNode::isUnderscoreProtoSetter(generator.vm(), node)) { |
| RefPtr<RegisterID> prototype = generator.emitNode(node.m_assign); |
| generator.emitDirectSetPrototypeOf<InvalidPrototypeMode::Ignore>(newObj, prototype.get(), m_position, m_position, m_position); |
| return; |
| } |
| |
| bool shouldSetFunctionName = generator.shouldSetFunctionName(node.m_assign); |
| |
| RefPtr<RegisterID> propertyName; |
| if (!node.name()) { |
| propertyName = generator.newTemporary(); |
| if (shouldSetFunctionName) |
| generator.emitToPropertyKey(propertyName.get(), generator.emitNode(node.m_expression)); |
| else |
| generator.emitNode(propertyName.get(), node.m_expression); |
| } |
| |
| RefPtr<RegisterID> value = generator.emitNode(node.m_assign); |
| if (node.needsSuperBinding()) |
| emitPutHomeObject(generator, value.get(), newObj); |
| |
| if (node.isClassProperty()) { |
| ASSERT(node.needsSuperBinding()); |
| ASSERT(!(node.type() & PropertyNode::PrivateSetter)); |
| ASSERT(!(node.type() & PropertyNode::PrivateGetter)); |
| |
| if (node.type() & PropertyNode::PrivateMethod) { |
| Variable var = generator.variable(*node.name()); |
| generator.emitPutToScope(generator.scopeRegister(), var, value.get(), DoNotThrowIfNotFound, InitializationMode::ConstInitialization); |
| return; |
| } |
| |
| RefPtr<RegisterID> propertyNameRegister; |
| if (node.name()) |
| propertyName = generator.emitLoad(nullptr, *node.name()); |
| |
| if (shouldSetFunctionName) |
| generator.emitSetFunctionName(value.get(), propertyName.get()); |
| generator.emitCallDefineProperty(newObj, propertyName.get(), value.get(), nullptr, nullptr, BytecodeGenerator::PropertyConfigurable | BytecodeGenerator::PropertyWritable, m_position); |
| return; |
| } |
| |
| if (const auto* identifier = node.name()) { |
| ASSERT(!propertyName); |
| std::optional<uint32_t> optionalIndex = parseIndex(*identifier); |
| if (!optionalIndex) { |
| generator.emitDirectPutById(newObj, *identifier, value.get()); |
| return; |
| } |
| |
| propertyName = generator.emitLoad(nullptr, jsNumber(optionalIndex.value())); |
| generator.emitDirectPutByVal(newObj, propertyName.get(), value.get()); |
| return; |
| } |
| |
| if (shouldSetFunctionName) |
| generator.emitSetFunctionName(value.get(), propertyName.get()); |
| generator.emitDirectPutByVal(newObj, propertyName.get(), value.get()); |
| } |
| |
| void PropertyListNode::emitSaveComputedFieldName(BytecodeGenerator& generator, PropertyNode& node) |
| { |
| ASSERT(node.isComputedClassField()); |
| |
| // The 'name' refers to a synthetic private name in the class scope, where the property key is saved for later use. |
| const Identifier& description = *node.name(); |
| Variable var = generator.variable(description); |
| ASSERT(!var.local()); |
| |
| RefPtr<RegisterID> propertyExpr = generator.emitNode(node.m_expression); |
| RefPtr<RegisterID> propertyName = generator.emitToPropertyKey(generator.newTemporary(), propertyExpr.get()); |
| |
| if (node.isStaticClassField()) { |
| Ref<Label> validPropertyNameLabel = generator.newLabel(); |
| RefPtr<RegisterID> prototypeString = generator.emitLoad(nullptr, JSValue(generator.addStringConstant(generator.propertyNames().prototype))); |
| generator.emitJumpIfFalse(generator.emitBinaryOp<OpStricteq>(generator.newTemporary(), prototypeString.get(), propertyName.get(), OperandTypes(ResultType::stringType(), ResultType::stringType())), validPropertyNameLabel.get()); |
| generator.emitThrowTypeError("Cannot declare a static field named 'prototype'"_s); |
| generator.emitLabel(validPropertyNameLabel.get()); |
| } |
| |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| generator.emitPutToScope(scope.get(), var, propertyName.get(), ThrowIfNotFound, InitializationMode::ConstInitialization); |
| } |
| |
| // ------------------------------ BracketAccessorNode -------------------------------- |
| |
| static bool isNonIndexStringElement(ExpressionNode& element) |
| { |
| return element.isString() && !parseIndex(static_cast<StringNode&>(element).value()); |
| } |
| |
| RegisterID* BracketAccessorNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_base->isSuperNode()) { |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| RefPtr<RegisterID> superBase = emitSuperBaseForCallee(generator); |
| |
| if (isNonIndexStringElement(*m_subscript)) { |
| const Identifier& id = static_cast<StringNode*>(m_subscript)->value(); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitGetById(finalDest.get(), superBase.get(), thisValue.get(), id); |
| } else { |
| RefPtr<RegisterID> subscript = generator.emitNodeForProperty(m_subscript); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitGetByVal(finalDest.get(), superBase.get(), thisValue.get(), subscript.get()); |
| } |
| |
| generator.emitProfileType(finalDest.get(), divotStart(), divotEnd()); |
| return finalDest.get(); |
| } |
| |
| RegisterID* ret; |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| |
| bool subscriptIsNonIndexString = isNonIndexStringElement(*m_subscript); |
| RefPtr<RegisterID> base = subscriptIsNonIndexString |
| ? generator.emitNode(m_base) |
| : generator.emitNodeForLeftHandSide(m_base, m_subscriptHasAssignments, m_subscript->isPure(generator)); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| |
| if (subscriptIsNonIndexString) { |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| ret = generator.emitGetById(finalDest.get(), base.get(), static_cast<StringNode*>(m_subscript)->value()); |
| } else { |
| RegisterID* property = generator.emitNodeForProperty(m_subscript); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| ret = generator.emitGetByVal(finalDest.get(), base.get(), property); |
| } |
| |
| generator.emitProfileType(finalDest.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| // ------------------------------ DotAccessorNode -------------------------------- |
| |
| RegisterID* DotAccessorNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| bool baseIsSuper = m_base->isSuperNode(); |
| |
| RefPtr<RegisterID> base; |
| if (baseIsSuper) |
| base = emitSuperBaseForCallee(generator); |
| else { |
| base = generator.emitNode(m_base); |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| } |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RegisterID* ret = emitGetPropertyValue(generator, finalDest.get(), base.get()); |
| |
| generator.emitProfileType(finalDest.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| RegisterID* BaseDotNode::emitGetPropertyValue(BytecodeGenerator& generator, RegisterID* dst, RegisterID* base, RefPtr<RegisterID>& thisValue) |
| { |
| if (isPrivateMember()) { |
| auto identifierName = identifier(); |
| auto privateTraits = generator.getPrivateTraits(identifierName); |
| if (privateTraits.isMethod()) { |
| Variable var = generator.variable(identifierName); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base, privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| return generator.emitGetFromScope(dst, scope.get(), var, ThrowIfNotFound); |
| } |
| |
| if (privateTraits.isGetter()) { |
| Variable var = generator.variable(identifierName); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base, privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> getterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().getPrivateName()); |
| CallArguments args(generator, nullptr); |
| generator.move(args.thisRegister(), base); |
| return generator.emitCall(dst, getterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| } |
| |
| if (privateTraits.isSetter()) { |
| // We need to perform brand check to follow the spec |
| Variable var = generator.variable(identifierName); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base, privateBrandSymbol.get(), privateTraits.isStatic()); |
| generator.emitThrowTypeError("Trying to access an undefined private getter"_s); |
| return dst; |
| } |
| |
| ASSERT(privateTraits.isField()); |
| Variable var = generator.variable(m_ident); |
| ASSERT_WITH_MESSAGE(!var.local(), "Private Field names must be stored in captured variables"); |
| |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, DoNotThrowIfNotFound); |
| return generator.emitGetPrivateName(dst, base, privateName.get()); |
| } |
| |
| if (m_base->isSuperNode()) { |
| if (!thisValue) |
| thisValue = generator.ensureThis(); |
| return generator.emitGetById(dst, base, thisValue.get(), m_ident); |
| } |
| |
| return generator.emitGetById(dst, base, m_ident); |
| } |
| |
| RegisterID* BaseDotNode::emitGetPropertyValue(BytecodeGenerator& generator, RegisterID* dst, RegisterID* base) |
| { |
| RefPtr<RegisterID> thisValue; |
| return emitGetPropertyValue(generator, dst, base, thisValue); |
| } |
| |
| RegisterID* BaseDotNode::emitPutProperty(BytecodeGenerator& generator, RegisterID* base, RegisterID* value, RefPtr<RegisterID>& thisValue) |
| { |
| if (isPrivateMember()) { |
| auto identifierName = identifier(); |
| auto privateTraits = generator.getPrivateTraits(identifierName); |
| if (privateTraits.isSetter()) { |
| Variable var = generator.variable(identifierName); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base, privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> setterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().setPrivateName()); |
| CallArguments args(generator, nullptr, 1); |
| generator.move(args.thisRegister(), base); |
| generator.move(args.argumentRegister(0), value); |
| generator.emitCall(generator.newTemporary(), setterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| |
| return value; |
| } |
| |
| if (privateTraits.isGetter() || privateTraits.isMethod()) { |
| Variable var = generator.variable(identifierName); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base, privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| generator.emitThrowTypeError("Trying to access an undefined private setter"_s); |
| return value; |
| } |
| |
| ASSERT(privateTraits.isField()); |
| Variable var = generator.variable(m_ident); |
| ASSERT_WITH_MESSAGE(!var.local(), "Private Field names must be stored in captured variables"); |
| |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, DoNotThrowIfNotFound); |
| return generator.emitPrivateFieldPut(base, privateName.get(), value); |
| } |
| |
| if (m_base->isSuperNode()) { |
| if (!thisValue) |
| thisValue = generator.ensureThis(); |
| return generator.emitPutById(base, thisValue.get(), m_ident, value); |
| } |
| |
| return generator.emitPutById(base, m_ident, value); |
| } |
| |
| RegisterID* BaseDotNode::emitPutProperty(BytecodeGenerator& generator, RegisterID* base, RegisterID* value) |
| { |
| RefPtr<RegisterID> thisValue; |
| return emitPutProperty(generator, base, value, thisValue); |
| } |
| |
| // ------------------------------ ArgumentListNode ----------------------------- |
| |
| RegisterID* ArgumentListNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_expr); |
| return generator.emitNode(dst, m_expr); |
| } |
| |
| // ------------------------------ NewExprNode ---------------------------------- |
| |
| RegisterID* NewExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ExpectedFunction expectedFunction; |
| if (m_expr->isResolveNode()) |
| expectedFunction = generator.expectedFunctionForIdentifier(static_cast<ResolveNode*>(m_expr)->identifier()); |
| else |
| expectedFunction = NoExpectedFunction; |
| |
| RefPtr<RegisterID> func = nullptr; |
| if (m_args && m_args->hasAssignments()) |
| func = generator.newTemporary(); |
| func = generator.emitNode(func.get(), m_expr); |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, func.get()); |
| CallArguments callArguments(generator, m_args); |
| return generator.emitConstruct(returnValue.get(), func.get(), func.get(), expectedFunction, callArguments, divot(), divotStart(), divotEnd()); |
| } |
| |
| CallArguments::CallArguments(BytecodeGenerator& generator, ArgumentsNode* argumentsNode, unsigned additionalArguments) |
| : m_argumentsNode(argumentsNode) |
| , m_padding(0) |
| { |
| size_t argumentCountIncludingThis = 1 + additionalArguments; // 'this' register. |
| if (argumentsNode) { |
| for (ArgumentListNode* node = argumentsNode->m_listNode; node; node = node->m_next) |
| ++argumentCountIncludingThis; |
| } |
| |
| m_argv.grow(argumentCountIncludingThis); |
| for (int i = argumentCountIncludingThis - 1; i >= 0; --i) { |
| m_argv[i] = generator.newTemporary(); |
| ASSERT(static_cast<size_t>(i) == m_argv.size() - 1 || m_argv[i]->index() == m_argv[i + 1]->index() - 1); |
| } |
| |
| // We need to ensure that the frame size is stack-aligned |
| while ((CallFrame::headerSizeInRegisters + m_argv.size()) % stackAlignmentRegisters()) { |
| m_argv.insert(0, generator.newTemporary()); |
| m_padding++; |
| } |
| |
| while (stackOffset() % stackAlignmentRegisters()) { |
| m_argv.insert(0, generator.newTemporary()); |
| m_padding++; |
| } |
| } |
| |
| // ------------------------------ EvalFunctionCallNode ---------------------------------- |
| |
| RegisterID* EvalFunctionCallNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| // We need try to load 'this' before call eval in constructor, because 'this' can created by 'super' in some of the arrow function |
| // var A = class A { |
| // constructor () { this.id = 'A'; } |
| // } |
| // |
| // var B = class B extend A { |
| // constructor () { |
| // var arrow = () => super(); |
| // arrow(); |
| // eval("this.id = 'B'"); |
| // } |
| // } |
| if (generator.constructorKind() == ConstructorKind::Extends && generator.needsToUpdateArrowFunctionContext() && generator.isThisUsedInInnerArrowFunction()) |
| generator.emitLoadThisFromArrowFunctionLexicalEnvironment(); |
| |
| Variable var = generator.variable(generator.propertyNames().eval); |
| RefPtr<RegisterID> local = var.local(); |
| RefPtr<RegisterID> func; |
| if (local) { |
| generator.emitTDZCheckIfNecessary(var, local.get(), nullptr); |
| func = generator.move(generator.tempDestination(dst), local.get()); |
| } else |
| func = generator.newTemporary(); |
| CallArguments callArguments(generator, m_args); |
| |
| if (local) |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| else { |
| JSTextPosition newDivot = divotStart() + 4; |
| generator.emitExpressionInfo(newDivot, divotStart(), newDivot); |
| generator.move( |
| callArguments.thisRegister(), |
| generator.emitResolveScope(callArguments.thisRegister(), var)); |
| generator.emitGetFromScope(func.get(), callArguments.thisRegister(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, func.get(), nullptr); |
| } |
| |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, func.get()); |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(func.get()); |
| |
| return generator.emitCallEval(returnValue.get(), func.get(), callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| } |
| |
| // ------------------------------ FunctionCallValueNode ---------------------------------- |
| |
| RegisterID* FunctionCallValueNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_expr->isSuperNode()) { |
| RefPtr<RegisterID> func = emitGetSuperFunctionForConstruct(generator); |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, func.get()); |
| CallArguments callArguments(generator, m_args); |
| |
| ASSERT(generator.isConstructor() || generator.derivedContextType() == DerivedContextType::DerivedConstructorContext); |
| ASSERT(generator.constructorKind() == ConstructorKind::Extends || generator.derivedContextType() == DerivedContextType::DerivedConstructorContext); |
| RegisterID* ret = generator.emitConstruct(returnValue.get(), func.get(), generator.newTarget(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd()); |
| |
| bool isConstructorKindDerived = generator.constructorKind() == ConstructorKind::Extends; |
| bool doWeUseArrowFunctionInConstructor = isConstructorKindDerived && generator.needsToUpdateArrowFunctionContext(); |
| |
| if (generator.isDerivedConstructorContext() || (doWeUseArrowFunctionInConstructor && generator.isSuperCallUsedInInnerArrowFunction())) |
| generator.emitLoadThisFromArrowFunctionLexicalEnvironment(); |
| |
| Ref<Label> thisIsEmptyLabel = generator.newLabel(); |
| generator.emitJumpIfTrue(generator.emitIsEmpty(generator.newTemporary(), generator.thisRegister()), thisIsEmptyLabel.get()); |
| generator.emitThrowReferenceError("'super()' can't be called more than once in a constructor."_s); |
| generator.emitLabel(thisIsEmptyLabel.get()); |
| |
| generator.move(generator.thisRegister(), ret); |
| |
| if (generator.isDerivedConstructorContext() || doWeUseArrowFunctionInConstructor) |
| generator.emitPutThisToArrowFunctionContextScope(); |
| |
| // Initialize instance fields after super-call. |
| if (generator.privateBrandRequirement() == PrivateBrandRequirement::Needed) |
| generator.emitInstallPrivateBrand(generator.thisRegister()); |
| |
| if (generator.needsClassFieldInitializer() == NeedsClassFieldInitializer::Yes) { |
| ASSERT(generator.isConstructor() || generator.isDerivedConstructorContext()); |
| func = generator.emitLoadDerivedConstructor(); |
| generator.emitInstanceFieldInitializationIfNeeded(generator.thisRegister(), func.get(), divot(), divotStart(), divotEnd()); |
| } |
| return ret; |
| } |
| |
| RefPtr<RegisterID> func = nullptr; |
| if (m_args && m_args->hasAssignments()) |
| func = generator.newTemporary(); |
| func = generator.emitNode(func.get(), m_expr); |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, func.get()); |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(func.get()); |
| |
| CallArguments callArguments(generator, m_args); |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| RegisterID* ret = generator.emitCallInTailPosition(returnValue.get(), func.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| // ------------------------------ FunctionCallResolveNode ---------------------------------- |
| |
| RegisterID* FunctionCallResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!ASSERT_ENABLED) { |
| if (UNLIKELY(m_ident == generator.vm().propertyNames->builtinNames().assertPrivateName())) |
| return generator.move(dst, generator.emitLoad(nullptr, jsUndefined())); |
| } |
| |
| ExpectedFunction expectedFunction = generator.expectedFunctionForIdentifier(m_ident); |
| |
| Variable var = generator.variable(m_ident); |
| RefPtr<RegisterID> local = var.local(); |
| RefPtr<RegisterID> func; |
| if (local) { |
| generator.emitTDZCheckIfNecessary(var, local.get(), nullptr); |
| func = generator.move(generator.tempDestination(dst), local.get()); |
| } else |
| func = generator.newTemporary(); |
| CallArguments callArguments(generator, m_args); |
| |
| if (local) { |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| // This passes NoExpectedFunction because we expect that if the function is in a |
| // local variable, then it's not one of our built-in constructors. |
| expectedFunction = NoExpectedFunction; |
| } else { |
| JSTextPosition newDivot = divotStart() + m_ident.length(); |
| generator.emitExpressionInfo(newDivot, divotStart(), newDivot); |
| generator.move( |
| callArguments.thisRegister(), |
| generator.emitResolveScope(callArguments.thisRegister(), var)); |
| generator.emitGetFromScope(func.get(), callArguments.thisRegister(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, func.get(), nullptr); |
| } |
| |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, func.get()); |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(func.get()); |
| |
| RegisterID* ret = generator.emitCallInTailPosition(returnValue.get(), func.get(), expectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| // ------------------------------ BytecodeIntrinsicNode ---------------------------------- |
| |
| RegisterID* BytecodeIntrinsicNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_entry.type() == BytecodeIntrinsicRegistry::Type::Emitter) |
| return (this->*m_entry.emitter())(generator, dst); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.moveLinkTimeConstant(dst, m_entry.linkTimeConstant()); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getByIdDirect(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| ASSERT(node->m_expr->isString()); |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| ASSERT(!node->m_next); |
| return generator.emitDirectGetById(generator.finalDestination(dst), base.get(), ident); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getByIdDirectPrivate(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| ASSERT(node->m_expr->isString()); |
| SymbolImpl* symbol = generator.vm().propertyNames->builtinNames().lookUpPrivateName(static_cast<StringNode*>(node->m_expr)->value()); |
| ASSERT(symbol); |
| ASSERT(!node->m_next); |
| return generator.emitDirectGetById(generator.finalDestination(dst), base.get(), generator.parserArena().identifierArena().makeIdentifier(generator.vm(), symbol)); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getPrototypeOf(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| return generator.emitGetPrototypeOf(generator.finalDestination(dst), value.get()); |
| } |
| |
| static JSPromise::Field promiseInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_promiseFieldFlags) |
| return JSPromise::Field::Flags; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_promiseFieldReactionsOrResult) |
| return JSPromise::Field::ReactionsOrResult; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSPromise::Field::Flags; |
| } |
| |
| static JSGenerator::Field generatorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldState) |
| return JSGenerator::Field::State; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldNext) |
| return JSGenerator::Field::Next; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldThis) |
| return JSGenerator::Field::This; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldFrame) |
| return JSGenerator::Field::Frame; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSGenerator::Field::State; |
| } |
| |
| static JSAsyncGenerator::Field asyncGeneratorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldState) |
| return JSAsyncGenerator::Field::State; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldNext) |
| return JSAsyncGenerator::Field::Next; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldThis) |
| return JSAsyncGenerator::Field::This; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_generatorFieldFrame) |
| return JSAsyncGenerator::Field::Frame; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_asyncGeneratorFieldSuspendReason) |
| return JSAsyncGenerator::Field::SuspendReason; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_asyncGeneratorFieldQueueFirst) |
| return JSAsyncGenerator::Field::QueueFirst; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_asyncGeneratorFieldQueueLast) |
| return JSAsyncGenerator::Field::QueueLast; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSAsyncGenerator::Field::State; |
| } |
| |
| static AbstractModuleRecord::Field abstractModuleRecordInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_abstractModuleRecordFieldState) |
| return AbstractModuleRecord::Field::State; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return AbstractModuleRecord::Field::State; |
| } |
| |
| static JSArrayIterator::Field arrayIteratorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_arrayIteratorFieldIndex) |
| return JSArrayIterator::Field::Index; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_arrayIteratorFieldIteratedObject) |
| return JSArrayIterator::Field::IteratedObject; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_arrayIteratorFieldKind) |
| return JSArrayIterator::Field::Kind; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSArrayIterator::Field::Index; |
| } |
| |
| static JSStringIterator::Field stringIteratorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_stringIteratorFieldIndex) |
| return JSStringIterator::Field::Index; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_stringIteratorFieldIteratedString) |
| return JSStringIterator::Field::IteratedString; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSStringIterator::Field::Index; |
| } |
| |
| static JSMapIterator::Field mapIteratorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_mapIteratorFieldMapBucket) |
| return JSMapIterator::Field::MapBucket; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_mapIteratorFieldKind) |
| return JSMapIterator::Field::Kind; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSMapIterator::Field::MapBucket; |
| } |
| |
| static JSSetIterator::Field setIteratorInternalFieldIndex(BytecodeIntrinsicNode* node) |
| { |
| ASSERT(node->entry().type() == BytecodeIntrinsicRegistry::Type::Emitter); |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_setIteratorFieldSetBucket) |
| return JSSetIterator::Field::SetBucket; |
| if (node->entry().emitter() == &BytecodeIntrinsicNode::emit_intrinsic_setIteratorFieldKind) |
| return JSSetIterator::Field::Kind; |
| RELEASE_ASSERT_NOT_REACHED(); |
| return JSSetIterator::Field::SetBucket; |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getPromiseInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(promiseInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSPromise::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getGeneratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(generatorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSGenerator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getAsyncGeneratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(asyncGeneratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSAsyncGenerator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getAbstractModuleRecordInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(abstractModuleRecordInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < AbstractModuleRecord::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getArrayIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(arrayIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSArrayIterator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getStringIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(stringIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSStringIterator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getMapIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(mapIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSMapIterator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_getSetIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(setIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSSetIterator::numberOfInternalFields); |
| ASSERT(!node->m_next); |
| |
| return generator.emitGetInternalField(generator.finalDestination(dst), base.get(), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_argument(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| ASSERT(node->m_expr->isNumber()); |
| double value = static_cast<NumberNode*>(node->m_expr)->value(); |
| int32_t index = static_cast<int32_t>(value); |
| ASSERT(value == index); |
| ASSERT(index >= 0); |
| ASSERT(!node->m_next); |
| |
| // The body functions of generator and async have different mechanism for arguments. |
| ASSERT(generator.parseMode() != SourceParseMode::GeneratorBodyMode); |
| ASSERT(!isAsyncFunctionBodyParseMode(generator.parseMode())); |
| |
| return generator.emitGetArgument(generator.finalDestination(dst), index); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_argumentCount(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(!m_args->m_listNode); |
| |
| return generator.emitArgumentCount(generator.finalDestination(dst)); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_arrayPush(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| RefPtr<RegisterID> length = generator.emitDirectGetById(generator.newTemporary(), base.get(), generator.propertyNames().length); |
| return generator.move(dst, generator.emitDirectPutByVal(base.get(), length.get(), value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putByIdDirect(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| ASSERT(node->m_expr->isString()); |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitDirectPutById(base.get(), ident, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putByIdDirectPrivate(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| ASSERT(node->m_expr->isString()); |
| SymbolImpl* symbol = generator.vm().propertyNames->builtinNames().lookUpPrivateName(static_cast<StringNode*>(node->m_expr)->value()); |
| ASSERT(symbol); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitDirectPutById(base.get(), generator.parserArena().identifierArena().makeIdentifier(generator.vm(), symbol), value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putByValDirect(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RefPtr<RegisterID> index = generator.emitNodeForProperty(node); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitDirectPutByVal(base.get(), index.get(), value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putPromiseInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(promiseInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSPromise::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putGeneratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(generatorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSGenerator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putAsyncGeneratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(asyncGeneratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSAsyncGenerator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putArrayIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(arrayIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSArrayIterator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putStringIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(stringIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSStringIterator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putMapIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(mapIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSMapIterator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_putSetIteratorInternalField(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| RELEASE_ASSERT(node->m_expr->isBytecodeIntrinsicNode()); |
| unsigned index = static_cast<unsigned>(setIteratorInternalFieldIndex(static_cast<BytecodeIntrinsicNode*>(node->m_expr))); |
| ASSERT(index < JSSetIterator::numberOfInternalFields); |
| node = node->m_next; |
| RefPtr<RegisterID> value = generator.emitNode(node); |
| |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitPutInternalField(base.get(), index, value.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_tailCallForwardArguments(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> function = generator.emitNode(node); |
| node = node->m_next; |
| RefPtr<RegisterID> thisRegister = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| |
| RefPtr<RegisterID> finalDst = generator.finalDestination(dst); |
| return generator.emitCallForwardArgumentsInTailPosition(finalDst.get(), function.get(), thisRegister.get(), generator.newTemporary(), 0, divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_throwTypeError(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| ASSERT(!node->m_next); |
| if (node->m_expr->isString()) { |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| generator.emitThrowTypeError(ident); |
| } else { |
| RefPtr<RegisterID> message = generator.emitNode(node); |
| generator.emitThrowStaticError(ErrorTypeWithExtension::TypeError, message.get()); |
| } |
| return dst; |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_throwRangeError(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| ASSERT(!node->m_next); |
| if (node->m_expr->isString()) { |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| generator.emitThrowRangeError(ident); |
| } else { |
| RefPtr<RegisterID> message = generator.emitNode(node); |
| generator.emitThrowStaticError(ErrorTypeWithExtension::RangeError, message.get()); |
| } |
| |
| return dst; |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_throwOutOfMemoryError(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(!m_args->m_listNode); |
| |
| generator.emitThrowOutOfMemoryError(); |
| return dst; |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_tryGetById(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| |
| ASSERT(node->m_expr->isString()); |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| ASSERT(!node->m_next); |
| |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| return generator.emitTryGetById(finalDest.get(), base.get(), ident); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_tryGetByIdWithWellKnownSymbol(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> base = generator.emitNode(node); |
| node = node->m_next; |
| |
| ASSERT(node->m_expr->isString()); |
| SymbolImpl* symbol = generator.vm().propertyNames->builtinNames().lookUpWellKnownSymbol(static_cast<StringNode*>(node->m_expr)->value()); |
| RELEASE_ASSERT(symbol); |
| ASSERT(!node->m_next); |
| |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| return generator.emitTryGetById(finalDest.get(), base.get(), generator.parserArena().identifierArena().makeIdentifier(generator.vm(), symbol)); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_toNumber(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> src = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitToNumber(generator.tempDestination(dst), src.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_toString(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> src = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitToString(generator.tempDestination(dst), src.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_toPropertyKey(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> src = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| |
| return generator.move(dst, generator.emitToPropertyKey(generator.tempDestination(dst), src.get())); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_toObject(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> src = generator.emitNode(node); |
| node = node->m_next; |
| |
| RefPtr<RegisterID> temp = generator.tempDestination(dst); |
| if (node) { |
| ASSERT(node->m_expr->isString()); |
| const Identifier& message = static_cast<StringNode*>(node->m_expr)->value(); |
| ASSERT(!node->m_next); |
| return generator.move(dst, generator.emitToObject(temp.get(), src.get(), message)); |
| } |
| return generator.move(dst, generator.emitToObject(temp.get(), src.get(), generator.vm().propertyNames->emptyIdentifier)); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_idWithProfile(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> idValue = generator.newTemporary(); |
| generator.emitNode(idValue.get(), node); |
| SpeculatedType speculation = SpecNone; |
| while (node->m_next) { |
| node = node->m_next; |
| ASSERT(node->m_expr->isString()); |
| const Identifier& ident = static_cast<StringNode*>(node->m_expr)->value(); |
| speculation |= speculationFromString(ident.utf8().data()); |
| } |
| |
| return generator.move(dst, generator.emitIdWithProfile(idValue.get(), speculation)); |
| } |
| |
| #define CREATE_INTRINSIC_FOR_BRAND_CHECK(lowerName, upperName) \ |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_##lowerName(JSC::BytecodeGenerator& generator, JSC::RegisterID* dst) \ |
| { \ |
| ArgumentListNode* node = m_args->m_listNode; \ |
| RefPtr<RegisterID> src = generator.emitNode(node); \ |
| ASSERT(!node->m_next); \ |
| return generator.move(dst, generator.emit##upperName(generator.tempDestination(dst), src.get())); \ |
| } |
| |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isObject, IsObject) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isCallable, IsCallable) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isConstructor, IsConstructor) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isJSArray, IsJSArray) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isProxyObject, IsProxyObject) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isDerivedArray, IsDerivedArray) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isGenerator, IsGenerator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isAsyncGenerator, IsAsyncGenerator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isPromise, IsPromise) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isRegExpObject, IsRegExpObject) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isMap, IsMap) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isSet, IsSet) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isShadowRealm, IsShadowRealm) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isStringIterator, IsStringIterator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isArrayIterator, IsArrayIterator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isMapIterator, IsMapIterator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isSetIterator, IsSetIterator) |
| CREATE_INTRINSIC_FOR_BRAND_CHECK(isUndefinedOrNull, IsUndefinedOrNull) |
| |
| #undef CREATE_INTRINSIC_FOR_BRAND_CHECK |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_newArrayWithSize(JSC::BytecodeGenerator& generator, JSC::RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> size = generator.emitNode(node); |
| ASSERT(!node->m_next); |
| |
| RefPtr<RegisterID> finalDestination = generator.finalDestination(dst); |
| generator.emitNewArrayWithSize(finalDestination.get(), size.get()); |
| return finalDestination.get(); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_createPromise(JSC::BytecodeGenerator& generator, JSC::RegisterID* dst) |
| { |
| ArgumentListNode* node = m_args->m_listNode; |
| RefPtr<RegisterID> newTarget = generator.emitNode(node); |
| node = node->m_next; |
| bool isInternalPromise = static_cast<BooleanNode*>(node->m_expr)->value(); |
| ASSERT(!node->m_next); |
| |
| return generator.emitCreatePromise(generator.finalDestination(dst), newTarget.get(), isInternalPromise); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_newPromise(JSC::BytecodeGenerator& generator, JSC::RegisterID* dst) |
| { |
| ASSERT(!m_args->m_listNode); |
| RefPtr<RegisterID> finalDestination = generator.finalDestination(dst); |
| bool isInternalPromise = false; |
| generator.emitNewPromise(finalDestination.get(), isInternalPromise); |
| return finalDestination.get(); |
| } |
| |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_createArgumentsButterfly(JSC::BytecodeGenerator& generator, JSC::RegisterID* dst) |
| { |
| ASSERT(!m_args->m_listNode); |
| return generator.emitCreateArgumentsButterfly(generator.finalDestination(dst)); |
| } |
| |
| #define JSC_DECLARE_BYTECODE_INTRINSIC_CONSTANT_GENERATORS(name) \ |
| RegisterID* BytecodeIntrinsicNode::emit_intrinsic_##name(BytecodeGenerator& generator, RegisterID* dst) \ |
| { \ |
| ASSERT(!m_args); \ |
| ASSERT(type() == Type::Constant); \ |
| if (dst == generator.ignoredResult()) \ |
| return nullptr; \ |
| return generator.emitLoad(dst, generator.vm().bytecodeIntrinsicRegistry().name##Value(generator)); \ |
| } |
| JSC_COMMON_BYTECODE_INTRINSIC_CONSTANTS_EACH_NAME(JSC_DECLARE_BYTECODE_INTRINSIC_CONSTANT_GENERATORS) |
| #undef JSC_DECLARE_BYTECODE_INTRINSIC_CONSTANT_GENERATORS |
| |
| // ------------------------------ FunctionCallBracketNode ---------------------------------- |
| |
| RegisterID* FunctionCallBracketNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> function = generator.tempDestination(dst); |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, function.get()); |
| bool baseIsSuper = m_base->isSuperNode(); |
| bool subscriptIsNonIndexString = isNonIndexStringElement(*m_subscript); |
| |
| RefPtr<RegisterID> base; |
| if (baseIsSuper) |
| base = emitSuperBaseForCallee(generator); |
| else { |
| if (subscriptIsNonIndexString) |
| base = generator.emitNode(m_base); |
| else |
| base = generator.emitNodeForLeftHandSide(m_base, m_subscriptHasAssignments, m_subscript->isPure(generator)); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| } |
| |
| RefPtr<RegisterID> thisRegister; |
| if (baseIsSuper) { |
| // Note that we only need to do this once because we either have a non-TDZ this or we throw. Once we have a non-TDZ this, we can't change its value back to TDZ. |
| thisRegister = generator.ensureThis(); |
| } |
| if (subscriptIsNonIndexString) { |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| if (baseIsSuper) |
| generator.emitGetById(function.get(), base.get(), thisRegister.get(), static_cast<StringNode*>(m_subscript)->value()); |
| else |
| generator.emitGetById(function.get(), base.get(), static_cast<StringNode*>(m_subscript)->value()); |
| } else { |
| RefPtr<RegisterID> property = generator.emitNodeForProperty(m_subscript); |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| if (baseIsSuper) |
| generator.emitGetByVal(function.get(), base.get(), thisRegister.get(), property.get()); |
| else |
| generator.emitGetByVal(function.get(), base.get(), property.get()); |
| } |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(function.get()); |
| |
| CallArguments callArguments(generator, m_args); |
| if (baseIsSuper) { |
| generator.emitTDZCheck(generator.thisRegister()); |
| generator.move(callArguments.thisRegister(), thisRegister.get()); |
| } else |
| generator.move(callArguments.thisRegister(), base.get()); |
| RegisterID* ret = generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| // ------------------------------ FunctionCallDotNode ---------------------------------- |
| |
| RegisterID* FunctionCallDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> function = generator.tempDestination(dst); |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst, function.get()); |
| CallArguments callArguments(generator, m_args); |
| bool baseIsSuper = m_base->isSuperNode(); |
| if (baseIsSuper) |
| generator.move(callArguments.thisRegister(), generator.ensureThis()); |
| else { |
| generator.emitNode(callArguments.thisRegister(), m_base); |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(callArguments.thisRegister()); |
| } |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| |
| RefPtr<RegisterID> base = baseIsSuper ? emitSuperBaseForCallee(generator) : callArguments.thisRegister(); |
| emitGetPropertyValue(generator, function.get(), base.get()); |
| |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(function.get()); |
| |
| RegisterID* ret = generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return ret; |
| } |
| |
| static constexpr size_t maxDistanceToInnermostCallOrApply = 2; |
| |
| RegisterID* CallFunctionCallDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst); |
| RefPtr<RegisterID> base = generator.emitNode(m_base); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| |
| RefPtr<RegisterID> function; |
| auto makeFunction = [&] { |
| if (m_base->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| function = generator.emitGetById(generator.tempDestination(dst), base.get(), thisValue.get(), generator.propertyNames().builtinNames().callPublicName()); |
| } else |
| function = generator.emitGetById(generator.tempDestination(dst), base.get(), generator.propertyNames().builtinNames().callPublicName()); |
| |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(function.get()); |
| }; |
| |
| bool emitCallCheck = !generator.isBuiltinFunction(); |
| if (m_distanceToInnermostCallOrApply > maxDistanceToInnermostCallOrApply && emitCallCheck) { |
| makeFunction(); |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.move(dst, returnValue.get()); |
| return returnValue.get(); |
| } |
| |
| Ref<Label> realCall = generator.newLabel(); |
| Ref<Label> end = generator.newLabel(); |
| |
| if (emitCallCheck) { |
| makeFunction(); |
| generator.emitJumpIfNotFunctionCall(function.get(), realCall.get()); |
| } |
| { |
| if (m_args->m_listNode && m_args->m_listNode->m_expr && m_args->m_listNode->m_expr->isSpreadExpression()) { |
| SpreadExpressionNode* spread = static_cast<SpreadExpressionNode*>(m_args->m_listNode->m_expr); |
| ExpressionNode* subject = spread->expression(); |
| RefPtr<RegisterID> argumentsRegister; |
| argumentsRegister = generator.emitNode(subject); |
| generator.emitExpressionInfo(spread->divot(), spread->divotStart(), spread->divotEnd()); |
| RefPtr<RegisterID> thisRegister = generator.emitGetByVal(generator.newTemporary(), argumentsRegister.get(), generator.emitLoad(nullptr, jsNumber(0))); |
| generator.emitCallVarargsInTailPosition(returnValue.get(), base.get(), thisRegister.get(), argumentsRegister.get(), generator.newTemporary(), 1, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } else if (m_args->m_listNode && m_args->m_listNode->m_expr) { |
| ArgumentListNode* oldList = m_args->m_listNode; |
| m_args->m_listNode = m_args->m_listNode->m_next; |
| |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.emitNode(callArguments.thisRegister(), oldList->m_expr); |
| generator.emitCallInTailPosition(returnValue.get(), realFunction.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| m_args->m_listNode = oldList; |
| } else { |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| generator.emitCallInTailPosition(returnValue.get(), realFunction.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| } |
| if (emitCallCheck) { |
| generator.emitJump(end.get()); |
| generator.emitLabel(realCall.get()); |
| { |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| generator.emitLabel(end.get()); |
| } |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return returnValue.get(); |
| } |
| |
| RegisterID* HasOwnPropertyFunctionCallDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst); |
| RefPtr<RegisterID> base = generator.emitNode(m_base); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| |
| RefPtr<RegisterID> function = generator.emitGetById(generator.newTemporary(), base.get(), generator.propertyNames().hasOwnProperty); |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(function.get()); |
| |
| RELEASE_ASSERT(m_args->m_listNode && m_args->m_listNode->m_expr && !m_args->m_listNode->m_next); |
| ExpressionNode* argument = m_args->m_listNode->m_expr; |
| RELEASE_ASSERT(argument->isResolveNode()); |
| ForInContext* context = nullptr; |
| Variable argumentVariable = generator.variable(static_cast<ResolveNode*>(argument)->identifier()); |
| if (argumentVariable.isLocal()) { |
| RegisterID* property = argumentVariable.local(); |
| context = generator.findForInContext(property); |
| } |
| |
| auto canUseFastHasOwnProperty = [&] { |
| if (!context) |
| return false; |
| if (!context->baseVariable()) |
| return false; |
| if (m_base->isResolveNode()) |
| return generator.variable(static_cast<ResolveNode*>(m_base)->identifier()) == context->baseVariable().value(); |
| if (m_base->isThisNode()) { |
| // After generator.ensureThis (which must be invoked in |base|'s materialization), we can ensure that |this| is in local this-register. |
| ASSERT(base); |
| return generator.variable(generator.propertyNames().builtinNames().thisPrivateName(), ThisResolutionType::Local) == context->baseVariable().value(); |
| } |
| return false; |
| }; |
| |
| if (canUseFastHasOwnProperty()) { |
| // It is possible that base register is variable and each for-in body replaces JS object in the base register with a different one. |
| // Even though, this is OK since HasOwnStructureProperty will reject the replaced JS object. |
| Ref<Label> realCall = generator.newLabel(); |
| Ref<Label> end = generator.newLabel(); |
| |
| unsigned branchInsnOffset = generator.emitWideJumpIfNotFunctionHasOwnProperty(function.get(), realCall.get()); |
| generator.emitEnumeratorHasOwnProperty(returnValue.get(), base.get(), context->mode(), generator.emitNode(argument), context->propertyOffset(), context->enumerator()); |
| generator.emitJump(end.get()); |
| |
| generator.emitLabel(realCall.get()); |
| { |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| |
| generator.emitLabel(end.get()); |
| |
| generator.recordHasOwnPropertyInForInLoop(*context, branchInsnOffset, realCall); |
| } else { |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return returnValue.get(); |
| } |
| |
| static bool areTrivialApplyArguments(ArgumentsNode* args) |
| { |
| return !args->m_listNode || !args->m_listNode->m_expr || !args->m_listNode->m_next |
| || (!args->m_listNode->m_next->m_next && args->m_listNode->m_next->m_expr->isSimpleArray()); |
| } |
| |
| RegisterID* ApplyFunctionCallDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| // A few simple cases can be trivially handled as ordinary function calls. |
| // function.apply(), function.apply(arg) -> identical to function.call |
| // function.apply(thisArg, [arg0, arg1, ...]) -> can be trivially coerced into function.call(thisArg, arg0, arg1, ...) and saves object allocation |
| bool mayBeCall = areTrivialApplyArguments(m_args); |
| |
| RefPtr<RegisterID> returnValue = generator.finalDestination(dst); |
| RefPtr<RegisterID> base = generator.emitNode(m_base); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(base.get()); |
| |
| RefPtr<RegisterID> function; |
| auto makeFunction = [&] { |
| if (m_base->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| function = generator.emitGetById(generator.tempDestination(dst), base.get(), thisValue.get(), generator.propertyNames().builtinNames().applyPublicName()); |
| } else |
| function = generator.emitGetById(generator.tempDestination(dst), base.get(), generator.propertyNames().builtinNames().applyPublicName()); |
| |
| if (isOptionalChainBase()) |
| generator.emitOptionalCheck(function.get()); |
| }; |
| |
| bool emitCallCheck = !generator.isBuiltinFunction(); |
| if (m_distanceToInnermostCallOrApply > maxDistanceToInnermostCallOrApply && emitCallCheck) { |
| makeFunction(); |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.move(dst, returnValue.get()); |
| return returnValue.get(); |
| } |
| |
| Ref<Label> realCall = generator.newLabel(); |
| Ref<Label> end = generator.newLabel(); |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| if (emitCallCheck) { |
| makeFunction(); |
| ASSERT(!m_base->isResolveNode() || static_cast<ResolveNode*>(m_base)->identifier() != "Reflect"_s); |
| generator.emitJumpIfNotFunctionApply(function.get(), realCall.get()); |
| } |
| if (mayBeCall) { |
| if (m_args->m_listNode && m_args->m_listNode->m_expr) { |
| ArgumentListNode* oldList = m_args->m_listNode; |
| if (m_args->m_listNode->m_expr->isSpreadExpression()) { |
| SpreadExpressionNode* spread = static_cast<SpreadExpressionNode*>(m_args->m_listNode->m_expr); |
| RefPtr<RegisterID> realFunction = generator.move(generator.newTemporary(), base.get()); |
| RefPtr<RegisterID> index = generator.emitLoad(generator.newTemporary(), jsNumber(0)); |
| RefPtr<RegisterID> thisRegister = generator.emitLoad(generator.newTemporary(), jsUndefined()); |
| RefPtr<RegisterID> argumentsRegister = generator.emitLoad(generator.newTemporary(), jsUndefined()); |
| |
| auto extractor = scopedLambda<void(BytecodeGenerator&, RegisterID*)>([&thisRegister, &argumentsRegister, &index](BytecodeGenerator& generator, RegisterID* value) |
| { |
| Ref<Label> haveThis = generator.newLabel(); |
| Ref<Label> end = generator.newLabel(); |
| generator.emitJumpIfFalse(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), index.get(), generator.emitLoad(nullptr, jsNumber(0))), haveThis.get()); |
| generator.move(thisRegister.get(), value); |
| generator.emitLoad(index.get(), jsNumber(1)); |
| generator.emitJump(end.get()); |
| generator.emitLabel(haveThis.get()); |
| generator.emitJumpIfFalse(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), index.get(), generator.emitLoad(nullptr, jsNumber(1))), end.get()); |
| generator.move(argumentsRegister.get(), value); |
| generator.emitLoad(index.get(), jsNumber(2)); |
| generator.emitLabel(end.get()); |
| }); |
| generator.emitEnumeration(this, spread->expression(), extractor); |
| generator.emitCallVarargsInTailPosition(returnValue.get(), realFunction.get(), thisRegister.get(), argumentsRegister.get(), generator.newTemporary(), 0, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } else if (m_args->m_listNode->m_next) { |
| ASSERT(m_args->m_listNode->m_next->m_expr->isSimpleArray()); |
| ASSERT(!m_args->m_listNode->m_next->m_next); |
| m_args->m_listNode = static_cast<ArrayNode*>(m_args->m_listNode->m_next->m_expr)->toArgumentList(generator.parserArena(), 0, 0); |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.emitNode(callArguments.thisRegister(), oldList->m_expr); |
| generator.emitCallInTailPosition(returnValue.get(), realFunction.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } else { |
| m_args->m_listNode = m_args->m_listNode->m_next; |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.emitNode(callArguments.thisRegister(), oldList->m_expr); |
| generator.emitCallInTailPosition(returnValue.get(), realFunction.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| m_args->m_listNode = oldList; |
| } else { |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.emitLoad(callArguments.thisRegister(), jsUndefined()); |
| generator.emitCallInTailPosition(returnValue.get(), realFunction.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| } else { |
| ASSERT(m_args->m_listNode && m_args->m_listNode->m_next); |
| RefPtr<RegisterID> realFunction = generator.move(generator.tempDestination(dst), base.get()); |
| RefPtr<RegisterID> thisRegister = generator.emitNode(m_args->m_listNode->m_expr); |
| RefPtr<RegisterID> argsRegister; |
| ArgumentListNode* args = m_args->m_listNode->m_next; |
| argsRegister = generator.emitNode(args->m_expr); |
| |
| // Function.prototype.apply ignores extra arguments, but we still |
| // need to evaluate them for side effects. |
| while ((args = args->m_next)) |
| generator.emitNode(args->m_expr); |
| |
| generator.emitCallVarargsInTailPosition(returnValue.get(), realFunction.get(), thisRegister.get(), argsRegister.get(), generator.newTemporary(), 0, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| } |
| if (emitCallCheck) { |
| generator.emitJump(end.get()); |
| generator.emitLabel(realCall.get()); |
| CallArguments callArguments(generator, m_args); |
| generator.move(callArguments.thisRegister(), base.get()); |
| generator.emitCallInTailPosition(returnValue.get(), function.get(), NoExpectedFunction, callArguments, divot(), divotStart(), divotEnd(), DebuggableCall::Yes); |
| generator.emitLabel(end.get()); |
| } |
| generator.emitProfileType(returnValue.get(), divotStart(), divotEnd()); |
| return returnValue.get(); |
| } |
| |
| // ------------------------------ PostfixNode ---------------------------------- |
| |
| static RegisterID* emitIncOrDec(BytecodeGenerator& generator, RegisterID* srcDst, Operator oper) |
| { |
| return (oper == Operator::PlusPlus) ? generator.emitInc(srcDst) : generator.emitDec(srcDst); |
| } |
| |
| static RegisterID* emitPostIncOrDec(BytecodeGenerator& generator, RegisterID* dst, RegisterID* srcDst, Operator oper) |
| { |
| if (dst == srcDst) |
| return generator.emitToNumeric(generator.finalDestination(dst), srcDst); |
| RefPtr<RegisterID> tmp = generator.emitToNumeric(generator.newTemporary(), srcDst); |
| RefPtr<RegisterID> result = generator.tempDestination(srcDst); |
| generator.move(result.get(), tmp.get()); |
| emitIncOrDec(generator, result.get(), oper); |
| generator.move(srcDst, result.get()); |
| return generator.move(dst, tmp.get()); |
| } |
| |
| RegisterID* PostfixNode::emitResolve(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return PrefixNode::emitResolve(generator, dst); |
| |
| ASSERT(m_expr->isResolveNode()); |
| ResolveNode* resolve = static_cast<ResolveNode*>(m_expr); |
| const Identifier& ident = resolve->identifier(); |
| |
| Variable var = generator.variable(ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| RefPtr<RegisterID> localReg = local; |
| if (var.isReadOnly()) { |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| localReg = generator.move(generator.tempDestination(dst), local); |
| } |
| RefPtr<RegisterID> oldValue = emitPostIncOrDec(generator, generator.finalDestination(dst), localReg.get(), m_operator); |
| generator.emitProfileType(localReg.get(), var, divotStart(), divotEnd()); |
| return oldValue.get(); |
| } |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> value = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, value.get(), nullptr); |
| if (var.isReadOnly()) { |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(var); |
| if (threwException) |
| return value.get(); |
| } |
| RefPtr<RegisterID> oldValue = emitPostIncOrDec(generator, generator.finalDestination(dst), value.get(), m_operator); |
| if (!var.isReadOnly()) { |
| generator.emitPutToScope(scope.get(), var, value.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| generator.emitProfileType(value.get(), var, divotStart(), divotEnd()); |
| } |
| |
| return oldValue.get(); |
| } |
| |
| RegisterID* PostfixNode::emitBracket(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return PrefixNode::emitBracket(generator, dst); |
| |
| ASSERT(m_expr->isBracketAccessorNode()); |
| BracketAccessorNode* bracketAccessor = static_cast<BracketAccessorNode*>(m_expr); |
| ExpressionNode* baseNode = bracketAccessor->base(); |
| ExpressionNode* subscript = bracketAccessor->subscript(); |
| |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(baseNode, bracketAccessor->subscriptHasAssignments(), subscript->isPure(generator)); |
| RefPtr<RegisterID> property = generator.emitNodeForProperty(subscript); |
| |
| generator.emitExpressionInfo(bracketAccessor->divot(), bracketAccessor->divotStart(), bracketAccessor->divotEnd()); |
| RefPtr<RegisterID> value; |
| RefPtr<RegisterID> thisValue; |
| if (baseNode->isSuperNode()) { |
| thisValue = generator.ensureThis(); |
| value = generator.emitGetByVal(generator.newTemporary(), base.get(), thisValue.get(), property.get()); |
| } else |
| value = generator.emitGetByVal(generator.newTemporary(), base.get(), property.get()); |
| RegisterID* oldValue = emitPostIncOrDec(generator, generator.tempDestination(dst), value.get(), m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (baseNode->isSuperNode()) |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), value.get()); |
| else |
| generator.emitPutByVal(base.get(), property.get(), value.get()); |
| generator.emitProfileType(value.get(), divotStart(), divotEnd()); |
| return generator.move(dst, oldValue); |
| } |
| |
| RegisterID* PostfixNode::emitDot(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| return PrefixNode::emitDot(generator, dst); |
| |
| ASSERT(m_expr->isDotAccessorNode()); |
| DotAccessorNode* dotAccessor = static_cast<DotAccessorNode*>(m_expr); |
| ExpressionNode* baseNode = dotAccessor->base(); |
| bool baseIsSuper = baseNode->isSuperNode(); |
| const Identifier& ident = dotAccessor->identifier(); |
| |
| RefPtr<RegisterID> base = generator.emitNode(baseNode); |
| |
| generator.emitExpressionInfo(dotAccessor->divot(), dotAccessor->divotStart(), dotAccessor->divotEnd()); |
| |
| if (dotAccessor->isPrivateMember()) { |
| ASSERT(!baseIsSuper); |
| auto privateTraits = generator.getPrivateTraits(ident); |
| |
| if (privateTraits.isField()) { |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, DoNotThrowIfNotFound); |
| |
| RefPtr<RegisterID> value = generator.emitGetPrivateName(generator.newTemporary(), base.get(), privateName.get()); |
| RefPtr<RegisterID> oldValue = emitPostIncOrDec(generator, generator.tempDestination(dst), value.get(), m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitPrivateFieldPut(base.get(), privateName.get(), value.get()); |
| generator.emitProfileType(value.get(), divotStart(), divotEnd()); |
| return generator.move(dst, oldValue.get()); |
| } |
| |
| if (privateTraits.isMethod()) { |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base.get(), privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitThrowTypeError("Trying to access an undefined private setter"_s); |
| return generator.tempDestination(dst); |
| } |
| |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base.get(), privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| RefPtr<RegisterID> value; |
| if (privateTraits.isGetter()) { |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> getterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().getPrivateName()); |
| CallArguments args(generator, nullptr); |
| generator.move(args.thisRegister(), base.get()); |
| value = generator.emitCall(generator.newTemporary(), getterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| } else { |
| generator.emitThrowTypeError("Trying to access an undefined private getter"_s); |
| return generator.tempDestination(dst); |
| } |
| |
| RefPtr<RegisterID> oldValue = emitPostIncOrDec(generator, generator.tempDestination(dst), value.get(), m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| |
| if (privateTraits.isSetter()) { |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> setterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().setPrivateName()); |
| CallArguments args(generator, nullptr, 1); |
| generator.move(args.thisRegister(), base.get()); |
| generator.move(args.argumentRegister(0), value.get()); |
| generator.emitCall(generator.newTemporary(), setterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| generator.emitProfileType(value.get(), divotStart(), divotEnd()); |
| return generator.move(dst, oldValue.get()); |
| } |
| |
| generator.emitThrowTypeError("Trying to access an undefined private getter"_s); |
| return generator.move(dst, oldValue.get()); |
| } |
| |
| RefPtr<RegisterID> value; |
| RefPtr<RegisterID> thisValue; |
| if (baseIsSuper) { |
| thisValue = generator.ensureThis(); |
| value = generator.emitGetById(generator.newTemporary(), base.get(), thisValue.get(), ident); |
| } else |
| value = generator.emitGetById(generator.newTemporary(), base.get(), ident); |
| RegisterID* oldValue = emitPostIncOrDec(generator, generator.tempDestination(dst), value.get(), m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (baseIsSuper) |
| generator.emitPutById(base.get(), thisValue.get(), ident, value.get()); |
| else |
| generator.emitPutById(base.get(), ident, value.get()); |
| generator.emitProfileType(value.get(), divotStart(), divotEnd()); |
| return generator.move(dst, oldValue); |
| } |
| |
| RegisterID* PostfixNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_expr->isResolveNode()) |
| return emitResolve(generator, dst); |
| |
| if (m_expr->isBracketAccessorNode()) |
| return emitBracket(generator, dst); |
| |
| if (m_expr->isDotAccessorNode()) |
| return emitDot(generator, dst); |
| |
| ASSERT(m_expr->isFunctionCall()); |
| return emitThrowReferenceError(generator, m_operator == Operator::PlusPlus |
| ? "Postfix ++ operator applied to value that is not a reference."_s |
| : "Postfix -- operator applied to value that is not a reference."_s, |
| dst); |
| } |
| |
| // ------------------------------ DeleteResolveNode ----------------------------------- |
| |
| RegisterID* DeleteResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| Variable var = generator.variable(m_ident); |
| if (var.local()) { |
| generator.emitTDZCheckIfNecessary(var, var.local(), nullptr); |
| return generator.emitLoad(generator.finalDestination(dst), false); |
| } |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> base = generator.emitResolveScope(dst, var); |
| generator.emitTDZCheckIfNecessary(var, nullptr, base.get()); |
| return generator.emitDeleteById(generator.finalDestination(dst, base.get()), base.get(), m_ident); |
| } |
| |
| // ------------------------------ DeleteBracketNode ----------------------------------- |
| |
| RegisterID* DeleteBracketNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| RefPtr<RegisterID> r0 = generator.emitNode(m_base); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(r0.get()); |
| |
| RefPtr<RegisterID> r1 = generator.emitNode(m_subscript); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (m_base->isSuperNode()) |
| return emitThrowReferenceError(generator, "Cannot delete a super property"_s, dst); |
| return generator.emitDeleteByVal(finalDest.get(), r0.get(), r1.get()); |
| } |
| |
| // ------------------------------ DeleteDotNode ----------------------------------- |
| |
| RegisterID* DeleteDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| RefPtr<RegisterID> r0 = generator.emitNode(m_base); |
| |
| if (m_base->isOptionalChainBase()) |
| generator.emitOptionalCheck(r0.get()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (m_base->isSuperNode()) |
| return emitThrowReferenceError(generator, "Cannot delete a super property"_s, dst); |
| return generator.emitDeleteById(finalDest.get(), r0.get(), m_ident); |
| } |
| |
| // ------------------------------ DeleteValueNode ----------------------------------- |
| |
| RegisterID* DeleteValueNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| generator.emitNode(generator.ignoredResult(), m_expr); |
| |
| // delete on a non-location expression ignores the value and returns true |
| return generator.emitLoad(generator.finalDestination(dst), true); |
| } |
| |
| // ------------------------------ VoidNode ------------------------------------- |
| |
| RegisterID* VoidNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) { |
| generator.emitNode(generator.ignoredResult(), m_expr); |
| return nullptr; |
| } |
| RefPtr<RegisterID> r0 = generator.emitNode(m_expr); |
| return generator.emitLoad(dst, jsUndefined()); |
| } |
| |
| // ------------------------------ TypeOfResolveNode ----------------------------------- |
| |
| RegisterID* TypeOfResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| Variable var = generator.variable(m_ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.emitTypeOf(generator.finalDestination(dst), local); |
| } |
| |
| RefPtr<RegisterID> scope = generator.emitResolveScope(dst, var); |
| RefPtr<RegisterID> value = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, DoNotThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, value.get(), nullptr); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.emitTypeOf(generator.finalDestination(dst, scope.get()), value.get()); |
| } |
| |
| // ------------------------------ TypeOfValueNode ----------------------------------- |
| |
| RegisterID* TypeOfValueNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) { |
| generator.emitNode(generator.ignoredResult(), m_expr); |
| return nullptr; |
| } |
| RefPtr<RegisterID> src = generator.emitNode(m_expr); |
| return generator.emitTypeOf(generator.finalDestination(dst), src.get()); |
| } |
| |
| // ------------------------------ PrefixNode ---------------------------------- |
| |
| RegisterID* PrefixNode::emitResolve(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_expr->isResolveNode()); |
| ResolveNode* resolve = static_cast<ResolveNode*>(m_expr); |
| const Identifier& ident = resolve->identifier(); |
| |
| Variable var = generator.variable(ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| RefPtr<RegisterID> localReg = local; |
| if (var.isReadOnly()) { |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| localReg = generator.move(generator.tempDestination(dst), localReg.get()); |
| } else if (generator.shouldEmitTypeProfilerHooks()) { |
| RefPtr<RegisterID> tempDst = generator.tempDestination(dst); |
| generator.move(tempDst.get(), localReg.get()); |
| emitIncOrDec(generator, tempDst.get(), m_operator); |
| generator.move(localReg.get(), tempDst.get()); |
| generator.emitProfileType(localReg.get(), var, divotStart(), divotEnd()); |
| return generator.move(dst, tempDst.get()); |
| } |
| emitIncOrDec(generator, localReg.get(), m_operator); |
| return generator.move(dst, localReg.get()); |
| } |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(dst, var); |
| RefPtr<RegisterID> value = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, value.get(), nullptr); |
| if (var.isReadOnly()) { |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(var); |
| if (threwException) |
| return value.get(); |
| } |
| |
| emitIncOrDec(generator, value.get(), m_operator); |
| if (!var.isReadOnly()) { |
| generator.emitPutToScope(scope.get(), var, value.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| generator.emitProfileType(value.get(), var, divotStart(), divotEnd()); |
| } |
| return generator.move(dst, value.get()); |
| } |
| |
| RegisterID* PrefixNode::emitBracket(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_expr->isBracketAccessorNode()); |
| BracketAccessorNode* bracketAccessor = static_cast<BracketAccessorNode*>(m_expr); |
| ExpressionNode* baseNode = bracketAccessor->base(); |
| ExpressionNode* subscript = bracketAccessor->subscript(); |
| |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(baseNode, bracketAccessor->subscriptHasAssignments(), subscript->isPure(generator)); |
| RefPtr<RegisterID> property = generator.emitNodeForProperty(subscript); |
| RefPtr<RegisterID> propDst = generator.tempDestination(dst); |
| |
| generator.emitExpressionInfo(bracketAccessor->divot(), bracketAccessor->divotStart(), bracketAccessor->divotEnd()); |
| RegisterID* value; |
| RefPtr<RegisterID> thisValue; |
| if (baseNode->isSuperNode()) { |
| thisValue = generator.ensureThis(); |
| value = generator.emitGetByVal(propDst.get(), base.get(), thisValue.get(), property.get()); |
| } else |
| value = generator.emitGetByVal(propDst.get(), base.get(), property.get()); |
| emitIncOrDec(generator, value, m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (baseNode->isSuperNode()) |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), value); |
| else |
| generator.emitPutByVal(base.get(), property.get(), value); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| RegisterID* PrefixNode::emitDot(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_expr->isDotAccessorNode()); |
| DotAccessorNode* dotAccessor = static_cast<DotAccessorNode*>(m_expr); |
| ExpressionNode* baseNode = dotAccessor->base(); |
| const Identifier& ident = dotAccessor->identifier(); |
| |
| RefPtr<RegisterID> base = generator.emitNode(baseNode); |
| RefPtr<RegisterID> propDst = generator.tempDestination(dst); |
| |
| generator.emitExpressionInfo(dotAccessor->divot(), dotAccessor->divotStart(), dotAccessor->divotEnd()); |
| RegisterID* value; |
| if (dotAccessor->isPrivateMember()) { |
| auto privateTraits = generator.getPrivateTraits(ident); |
| if (privateTraits.isField()) { |
| ASSERT(!baseNode->isSuperNode()); |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, DoNotThrowIfNotFound); |
| |
| value = generator.emitGetPrivateName(propDst.get(), base.get(), privateName.get()); |
| emitIncOrDec(generator, value, m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitPrivateFieldPut(base.get(), privateName.get(), value); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| if (privateTraits.isMethod()) { |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base.get(), privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitThrowTypeError("Trying to access an undefined private setter"_s); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| Variable var = generator.variable(ident); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| RefPtr<RegisterID> privateBrandSymbol = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| generator.emitCheckPrivateBrand(base.get(), privateBrandSymbol.get(), privateTraits.isStatic()); |
| |
| if (privateTraits.isGetter()) { |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> getterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().getPrivateName()); |
| CallArguments args(generator, nullptr); |
| generator.move(args.thisRegister(), base.get()); |
| value = generator.emitCall(propDst.get(), getterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| } else { |
| generator.emitThrowTypeError("Trying to access an undefined private getter"_s); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| emitIncOrDec(generator, value, m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| |
| if (privateTraits.isSetter()) { |
| RefPtr<RegisterID> getterSetterObj = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| RefPtr<RegisterID> setterFunction = generator.emitDirectGetById(generator.newTemporary(), getterSetterObj.get(), generator.propertyNames().builtinNames().setPrivateName()); |
| CallArguments args(generator, nullptr, 1); |
| generator.move(args.thisRegister(), base.get()); |
| generator.move(args.argumentRegister(0), value); |
| generator.emitCall(generator.newTemporary(), setterFunction.get(), NoExpectedFunction, args, m_position, m_position, m_position, DebuggableCall::Yes); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| generator.emitThrowTypeError("Trying to access an undefined private getter"_s); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| RefPtr<RegisterID> thisValue; |
| if (baseNode->isSuperNode()) { |
| thisValue = generator.ensureThis(); |
| value = generator.emitGetById(propDst.get(), base.get(), thisValue.get(), ident); |
| } else |
| value = generator.emitGetById(propDst.get(), base.get(), ident); |
| emitIncOrDec(generator, value, m_operator); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (baseNode->isSuperNode()) |
| generator.emitPutById(base.get(), thisValue.get(), ident, value); |
| else |
| generator.emitPutById(base.get(), ident, value); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| return generator.move(dst, propDst.get()); |
| } |
| |
| RegisterID* PrefixNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_expr->isResolveNode()) |
| return emitResolve(generator, dst); |
| |
| if (m_expr->isBracketAccessorNode()) |
| return emitBracket(generator, dst); |
| |
| if (m_expr->isDotAccessorNode()) |
| return emitDot(generator, dst); |
| |
| ASSERT(m_expr->isFunctionCall()); |
| return emitThrowReferenceError(generator, m_operator == Operator::PlusPlus |
| ? "Prefix ++ operator applied to value that is not a reference."_s |
| : "Prefix -- operator applied to value that is not a reference."_s, |
| dst); |
| } |
| |
| // ------------------------------ Unary Operation Nodes ----------------------------------- |
| |
| RegisterID* UnaryOpNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> src = generator.emitNode(m_expr); |
| generator.emitExpressionInfo(position(), position(), position()); |
| return generator.emitUnaryOp(opcodeID(), generator.finalDestination(dst), src.get(), m_expr->resultDescriptor()); |
| } |
| |
| // ------------------------------ UnaryPlusNode ----------------------------------- |
| |
| RegisterID* UnaryPlusNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(opcodeID() == op_to_number); |
| RefPtr<RegisterID> src = generator.emitNode(expr()); |
| generator.emitExpressionInfo(position(), position(), position()); |
| return generator.emitToNumber(generator.finalDestination(dst), src.get()); |
| } |
| |
| // ------------------------------ LogicalNotNode ----------------------------------- |
| |
| void LogicalNotNode::emitBytecodeInConditionContext(BytecodeGenerator& generator, Label& trueTarget, Label& falseTarget, FallThroughMode fallThroughMode) |
| { |
| if (UNLIKELY(needsDebugHook())) |
| generator.emitDebugHook(this); |
| |
| // Reverse the true and false targets. |
| generator.emitNodeInConditionContext(expr(), falseTarget, trueTarget, invert(fallThroughMode)); |
| } |
| |
| |
| // ------------------------------ Binary Operation Nodes ----------------------------------- |
| |
| // BinaryOpNode::emitStrcat: |
| // |
| // This node generates an op_strcat operation. This opcode can handle concatenation of three or |
| // more values, where we can determine a set of separate op_add operations would be operating on |
| // string values. |
| // |
| // This function expects to be operating on a graph of AST nodes looking something like this: |
| // |
| // (a)... (b) |
| // \ / |
| // (+) (c) |
| // \ / |
| // [d] ((+)) |
| // \ / |
| // [+=] |
| // |
| // The assignment operation is optional, if it exists the register holding the value on the |
| // lefthand side of the assignment should be passing as the optional 'lhs' argument. |
| // |
| // The method should be called on the node at the root of the tree of regular binary add |
| // operations (marked in the diagram with a double set of parentheses). This node must |
| // be performing a string concatenation (determined by statically detecting that at least |
| // one child must be a string). |
| // |
| // Since the minimum number of values being concatenated together is expected to be 3, if |
| // a lhs to a concatenating assignment is not provided then the root add should have at |
| // least one left child that is also an add that can be determined to be operating on strings. |
| // |
| RegisterID* BinaryOpNode::emitStrcat(BytecodeGenerator& generator, RegisterID* dst, RegisterID* lhs, ReadModifyResolveNode* emitExpressionInfoForMe) |
| { |
| ASSERT(isAdd()); |
| ASSERT(resultDescriptor().definitelyIsString()); |
| |
| // Create a list of expressions for all the adds in the tree of nodes we can convert into |
| // a string concatenation. The rightmost node (c) is added first. The rightmost node is |
| // added first, and the leftmost child is never added, so the vector produced for the |
| // example above will be [ c, b ]. |
| Vector<ExpressionNode*, 16> reverseExpressionList; |
| reverseExpressionList.append(m_expr2); |
| |
| // Examine the left child of the add. So long as this is a string add, add its right-child |
| // to the list, and keep processing along the left fork. |
| ExpressionNode* leftMostAddChild = m_expr1; |
| while (leftMostAddChild->isAdd() && leftMostAddChild->resultDescriptor().definitelyIsString()) { |
| reverseExpressionList.append(static_cast<AddNode*>(leftMostAddChild)->m_expr2); |
| leftMostAddChild = static_cast<AddNode*>(leftMostAddChild)->m_expr1; |
| } |
| |
| Vector<RefPtr<RegisterID>, 16> temporaryRegisters; |
| |
| // If there is an assignment, allocate a temporary to hold the lhs after conversion. |
| // We could possibly avoid this (the lhs is converted last anyway, we could let the |
| // op_strcat node handle its conversion if required). |
| if (lhs) |
| temporaryRegisters.append(generator.newTemporary()); |
| |
| // Emit code for the leftmost node ((a) in the example). |
| temporaryRegisters.append(generator.newTemporary()); |
| RegisterID* leftMostAddChildTempRegister = temporaryRegisters.last().get(); |
| generator.emitNode(leftMostAddChildTempRegister, leftMostAddChild); |
| |
| // Note on ordering of conversions: |
| // |
| // We maintain the same ordering of conversions as we would see if the concatenations |
| // was performed as a sequence of adds (otherwise this optimization could change |
| // behaviour should an object have been provided a valueOf or toString method). |
| // |
| // Considering the above example, the sequnce of execution is: |
| // * evaluate operand (a) |
| // * evaluate operand (b) |
| // * convert (a) to primitive <- (this would be triggered by the first add) |
| // * convert (b) to primitive <- (ditto) |
| // * evaluate operand (c) |
| // * convert (c) to primitive <- (this would be triggered by the second add) |
| // And optionally, if there is an assignment: |
| // * convert (d) to primitive <- (this would be triggered by the assigning addition) |
| // |
| // As such we do not plant an op to convert the leftmost child now. Instead, use |
| // 'leftMostAddChildTempRegister' as a flag to trigger generation of the conversion |
| // once the second node has been generated. However, if the leftmost child is an |
| // immediate we can trivially determine that no conversion will be required. |
| // If this is the case |
| if (leftMostAddChild->isString()) |
| leftMostAddChildTempRegister = nullptr; |
| |
| while (reverseExpressionList.size()) { |
| ExpressionNode* node = reverseExpressionList.last(); |
| reverseExpressionList.removeLast(); |
| |
| // Emit the code for the current node. |
| temporaryRegisters.append(generator.newTemporary()); |
| generator.emitNode(temporaryRegisters.last().get(), node); |
| |
| // On the first iteration of this loop, when we first reach this point we have just |
| // generated the second node, which means it is time to convert the leftmost operand. |
| if (leftMostAddChildTempRegister) { |
| generator.emitToPrimitive(leftMostAddChildTempRegister, leftMostAddChildTempRegister); |
| leftMostAddChildTempRegister = nullptr; // Only do this once. |
| } |
| // Plant a conversion for this node, if necessary. |
| if (!node->isString()) |
| generator.emitToPrimitive(temporaryRegisters.last().get(), temporaryRegisters.last().get()); |
| } |
| ASSERT(temporaryRegisters.size() >= 3); |
| |
| // Certain read-modify nodes require expression info to be emitted *after* m_right has been generated. |
| // If this is required the node is passed as 'emitExpressionInfoForMe'; do so now. |
| if (emitExpressionInfoForMe) |
| generator.emitExpressionInfo(emitExpressionInfoForMe->divot(), emitExpressionInfoForMe->divotStart(), emitExpressionInfoForMe->divotEnd()); |
| // If there is an assignment convert the lhs now. This will also copy lhs to |
| // the temporary register we allocated for it. |
| if (lhs) |
| generator.emitToPrimitive(temporaryRegisters[0].get(), lhs); |
| |
| return generator.emitStrcat(generator.finalDestination(dst, temporaryRegisters[0].get()), temporaryRegisters[0].get(), temporaryRegisters.size()); |
| } |
| |
| void BinaryOpNode::emitBytecodeInConditionContext(BytecodeGenerator& generator, Label& trueTarget, Label& falseTarget, FallThroughMode fallThroughMode) |
| { |
| TriState branchCondition; |
| ExpressionNode* branchExpression; |
| tryFoldToBranch(generator, branchCondition, branchExpression); |
| |
| if (UNLIKELY(needsDebugHook())) { |
| if (branchCondition != TriState::Indeterminate) |
| generator.emitDebugHook(this); |
| } |
| |
| if (branchCondition == TriState::Indeterminate) |
| ExpressionNode::emitBytecodeInConditionContext(generator, trueTarget, falseTarget, fallThroughMode); |
| else if (branchCondition == TriState::True) |
| generator.emitNodeInConditionContext(branchExpression, trueTarget, falseTarget, fallThroughMode); |
| else |
| generator.emitNodeInConditionContext(branchExpression, falseTarget, trueTarget, invert(fallThroughMode)); |
| } |
| |
| static inline bool canFoldToBranch(OpcodeID opcodeID, ExpressionNode* branchExpression, JSValue constant) |
| { |
| ResultType expressionType = branchExpression->resultDescriptor(); |
| |
| if (expressionType.definitelyIsBoolean() && constant.isBoolean()) |
| return true; |
| else if (expressionType.definitelyIsBoolean() && constant.isInt32() && (constant.asInt32() == 0 || constant.asInt32() == 1)) |
| return opcodeID == op_eq || opcodeID == op_neq; // Strict equality is false in the case of type mismatch. |
| else if (expressionType.isInt32() && constant.isInt32() && constant.asInt32() == 0) |
| return true; |
| |
| return false; |
| } |
| |
| void BinaryOpNode::tryFoldToBranch(BytecodeGenerator& generator, TriState& branchCondition, ExpressionNode*& branchExpression) |
| { |
| branchCondition = TriState::Indeterminate; |
| branchExpression = nullptr; |
| |
| ConstantNode* constant = nullptr; |
| if (m_expr1->isConstant()) { |
| constant = static_cast<ConstantNode*>(m_expr1); |
| branchExpression = m_expr2; |
| } else if (m_expr2->isConstant()) { |
| constant = static_cast<ConstantNode*>(m_expr2); |
| branchExpression = m_expr1; |
| } |
| |
| if (!constant) |
| return; |
| ASSERT(branchExpression); |
| |
| OpcodeID opcodeID = this->opcodeID(); |
| JSValue value = constant->jsValue(generator); |
| if (UNLIKELY(!value)) |
| return; |
| bool canFoldToBranch = JSC::canFoldToBranch(opcodeID, branchExpression, value); |
| if (!canFoldToBranch) |
| return; |
| |
| if (opcodeID == op_eq || opcodeID == op_stricteq) |
| branchCondition = triState(value.pureToBoolean() != TriState::False); |
| else if (opcodeID == op_neq || opcodeID == op_nstricteq) |
| branchCondition = triState(value.pureToBoolean() == TriState::False); |
| } |
| |
| RegisterID* BinaryOpNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| OpcodeID opcodeID = this->opcodeID(); |
| |
| if (opcodeID == op_less || opcodeID == op_lesseq || opcodeID == op_greater || opcodeID == op_greatereq) { |
| auto isUInt32 = [&] (ExpressionNode* node) -> std::optional<UInt32Result> { |
| if (node->isBinaryOpNode() && static_cast<BinaryOpNode*>(node)->opcodeID() == op_urshift) |
| return UInt32Result::UInt32; |
| if (node->isNumber() && static_cast<NumberNode*>(node)->isIntegerNode()) { |
| auto value = jsNumber(static_cast<NumberNode*>(node)->value()); |
| if (value.isInt32() && value.asInt32() >= 0) |
| return UInt32Result::Constant; |
| } |
| return std::nullopt; |
| }; |
| auto leftResult = isUInt32(m_expr1); |
| auto rightResult = isUInt32(m_expr2); |
| if ((leftResult && rightResult) && (leftResult.value() == UInt32Result::UInt32 || rightResult.value() == UInt32Result::UInt32)) { |
| auto* left = m_expr1; |
| auto* right = m_expr2; |
| if (left->isBinaryOpNode()) { |
| ASSERT(static_cast<BinaryOpNode*>(left)->opcodeID() == op_urshift); |
| static_cast<BinaryOpNode*>(left)->m_shouldToUnsignedResult = false; |
| } |
| if (right->isBinaryOpNode()) { |
| ASSERT(static_cast<BinaryOpNode*>(right)->opcodeID() == op_urshift); |
| static_cast<BinaryOpNode*>(right)->m_shouldToUnsignedResult = false; |
| } |
| RefPtr<RegisterID> src1 = generator.emitNodeForLeftHandSide(left, m_rightHasAssignments, right->isPure(generator)); |
| RefPtr<RegisterID> src2 = generator.emitNode(right); |
| generator.emitExpressionInfo(position(), position(), position()); |
| |
| // Since the both sides only accept Int32, replacing operands is not observable to users. |
| bool replaceOperands = false; |
| OpcodeID resultOp = opcodeID; |
| switch (opcodeID) { |
| case op_less: |
| resultOp = op_below; |
| break; |
| case op_lesseq: |
| resultOp = op_beloweq; |
| break; |
| case op_greater: |
| resultOp = op_below; |
| replaceOperands = true; |
| break; |
| case op_greatereq: |
| resultOp = op_beloweq; |
| replaceOperands = true; |
| break; |
| default: |
| RELEASE_ASSERT_NOT_REACHED(); |
| } |
| OperandTypes operandTypes(left->resultDescriptor(), right->resultDescriptor()); |
| if (replaceOperands) { |
| std::swap(src1, src2); |
| operandTypes = OperandTypes(right->resultDescriptor(), left->resultDescriptor()); |
| } |
| return generator.emitBinaryOp(resultOp, generator.finalDestination(dst, src1.get()), src1.get(), src2.get(), operandTypes); |
| } |
| } |
| |
| if (opcodeID == op_add && m_expr1->isAdd() && m_expr1->resultDescriptor().definitelyIsString()) { |
| generator.emitExpressionInfo(position(), position(), position()); |
| return emitStrcat(generator, dst); |
| } |
| |
| if (opcodeID == op_neq) { |
| if (m_expr1->isNull() || m_expr2->isNull()) { |
| RefPtr<RegisterID> src = generator.tempDestination(dst); |
| generator.emitNode(src.get(), m_expr1->isNull() ? m_expr2 : m_expr1); |
| return generator.emitUnaryOp<OpNeqNull>(generator.finalDestination(dst, src.get()), src.get()); |
| } |
| } |
| |
| ExpressionNode* left = m_expr1; |
| ExpressionNode* right = m_expr2; |
| if (opcodeID == op_neq || opcodeID == op_nstricteq) { |
| if (left->isString()) |
| std::swap(left, right); |
| } |
| |
| RefPtr<RegisterID> src1 = generator.emitNodeForLeftHandSide(left, m_rightHasAssignments, right->isPure(generator)); |
| bool wasTypeof = generator.lastOpcodeID() == op_typeof; |
| RefPtr<RegisterID> src2 = generator.emitNode(right); |
| generator.emitExpressionInfo(position(), position(), position()); |
| if (wasTypeof && (opcodeID == op_neq || opcodeID == op_nstricteq)) { |
| RefPtr<RegisterID> tmp = generator.tempDestination(dst); |
| if (opcodeID == op_neq) |
| generator.emitEqualityOp<OpEq>(generator.finalDestination(tmp.get(), src1.get()), src1.get(), src2.get()); |
| else if (opcodeID == op_nstricteq) |
| generator.emitEqualityOp<OpStricteq>(generator.finalDestination(tmp.get(), src1.get()), src1.get(), src2.get()); |
| else |
| RELEASE_ASSERT_NOT_REACHED(); |
| return generator.emitUnaryOp<OpNot>(generator.finalDestination(dst, tmp.get()), tmp.get()); |
| } |
| RegisterID* result = generator.emitBinaryOp(opcodeID, generator.finalDestination(dst, src1.get()), src1.get(), src2.get(), OperandTypes(left->resultDescriptor(), right->resultDescriptor())); |
| if (m_shouldToUnsignedResult) { |
| if (opcodeID == op_urshift && dst != generator.ignoredResult()) |
| return generator.emitUnaryOp<OpUnsigned>(result, result); |
| } |
| return result; |
| } |
| |
| RegisterID* EqualNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_expr1->isNull() || m_expr2->isNull()) { |
| RefPtr<RegisterID> src = generator.tempDestination(dst); |
| generator.emitNode(src.get(), m_expr1->isNull() ? m_expr2 : m_expr1); |
| return generator.emitUnaryOp<OpEqNull>(generator.finalDestination(dst, src.get()), src.get()); |
| } |
| |
| ExpressionNode* left = m_expr1; |
| ExpressionNode* right = m_expr2; |
| if (left->isString()) |
| std::swap(left, right); |
| |
| RefPtr<RegisterID> src1 = generator.emitNodeForLeftHandSide(left, m_rightHasAssignments, m_expr2->isPure(generator)); |
| RefPtr<RegisterID> src2 = generator.emitNode(right); |
| return generator.emitEqualityOp<OpEq>(generator.finalDestination(dst, src1.get()), src1.get(), src2.get()); |
| } |
| |
| RegisterID* StrictEqualNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ExpressionNode* left = m_expr1; |
| ExpressionNode* right = m_expr2; |
| if (left->isString()) |
| std::swap(left, right); |
| |
| RefPtr<RegisterID> src1 = generator.emitNodeForLeftHandSide(left, m_rightHasAssignments, m_expr2->isPure(generator)); |
| RefPtr<RegisterID> src2 = generator.emitNode(right); |
| return generator.emitEqualityOp<OpStricteq>(generator.finalDestination(dst, src1.get()), src1.get(), src2.get()); |
| } |
| |
| RegisterID* ThrowableBinaryOpNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> src1 = generator.emitNodeForLeftHandSide(m_expr1, m_rightHasAssignments, m_expr2->isPure(generator)); |
| RefPtr<RegisterID> src2 = generator.emitNode(m_expr2); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| return generator.emitBinaryOp(opcodeID(), generator.finalDestination(dst, src1.get()), src1.get(), src2.get(), OperandTypes(m_expr1->resultDescriptor(), m_expr2->resultDescriptor())); |
| } |
| |
| RegisterID* InstanceOfNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> hasInstanceValue = generator.newTemporary(); |
| RefPtr<RegisterID> isObject = generator.newTemporary(); |
| RefPtr<RegisterID> isCustom = generator.newTemporary(); |
| RefPtr<RegisterID> prototype = generator.newTemporary(); |
| RefPtr<RegisterID> value = generator.emitNodeForLeftHandSide(m_expr1, m_rightHasAssignments, m_expr2->isPure(generator)); |
| RefPtr<RegisterID> constructor = generator.emitNode(m_expr2); |
| RefPtr<RegisterID> dstReg = generator.finalDestination(dst, value.get()); |
| Ref<Label> custom = generator.newLabel(); |
| Ref<Label> done = generator.newLabel(); |
| Ref<Label> typeError = generator.newLabel(); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitIsObject(isObject.get(), constructor.get()); |
| generator.emitJumpIfFalse(isObject.get(), typeError.get()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitGetById(hasInstanceValue.get(), constructor.get(), generator.vm().propertyNames->hasInstanceSymbol); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitOverridesHasInstance(isCustom.get(), constructor.get(), hasInstanceValue.get()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitJumpIfTrue(isCustom.get(), custom.get()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitGetById(prototype.get(), constructor.get(), generator.vm().propertyNames->prototype); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitInstanceOf(dstReg.get(), value.get(), prototype.get()); |
| |
| generator.emitJump(done.get()); |
| |
| generator.emitLabel(typeError.get()); |
| generator.emitThrowTypeError("Right hand side of instanceof is not an object"_s); |
| |
| generator.emitLabel(custom.get()); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitInstanceOfCustom(dstReg.get(), value.get(), constructor.get(), hasInstanceValue.get()); |
| |
| generator.emitLabel(done.get()); |
| |
| return dstReg.get(); |
| } |
| |
| // ------------------------------ InNode ---------------------------- |
| |
| RegisterID* InNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (m_expr1->isPrivateIdentifier()) { |
| RefPtr<RegisterID> base = generator.emitNode(m_expr2); |
| |
| auto identifier = static_cast<PrivateIdentifierNode*>(m_expr1)->value(); |
| auto privateTraits = generator.getPrivateTraits(identifier); |
| Variable var = generator.variable(identifier); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| ASSERT(scope); // Private names are always captured. |
| |
| if (privateTraits.isField()) { |
| RefPtr<RegisterID> privateName = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, DoNotThrowIfNotFound); |
| return generator.emitHasPrivateName(generator.finalDestination(dst, base.get()), base.get(), privateName.get()); |
| } |
| |
| ASSERT(privateTraits.isPrivateMethodOrAccessor()); |
| RefPtr<RegisterID> privateBrand = generator.emitGetPrivateBrand(generator.newTemporary(), scope.get(), privateTraits.isStatic()); |
| return generator.emitHasPrivateBrand(generator.finalDestination(dst, base.get()), base.get(), privateBrand.get(), privateTraits.isStatic()); |
| } |
| |
| if (isNonIndexStringElement(*m_expr1)) { |
| RefPtr<RegisterID> base = generator.emitNode(m_expr2); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| return generator.emitInById(generator.finalDestination(dst, base.get()), base.get(), static_cast<StringNode*>(m_expr1)->value()); |
| } |
| |
| RefPtr<RegisterID> key = generator.emitNodeForLeftHandSide(m_expr1, m_rightHasAssignments, m_expr2->isPure(generator)); |
| RefPtr<RegisterID> base = generator.emitNode(m_expr2); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| return generator.emitInByVal(generator.finalDestination(dst, key.get()), key.get(), base.get()); |
| } |
| |
| |
| // ------------------------------ LogicalOpNode ---------------------------- |
| |
| RegisterID* LogicalOpNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> temp = generator.tempDestination(dst); |
| Ref<Label> target = generator.newLabel(); |
| |
| generator.emitNode(temp.get(), m_expr1); |
| if (m_operator == LogicalOperator::And) |
| generator.emitJumpIfFalse(temp.get(), target.get()); |
| else |
| generator.emitJumpIfTrue(temp.get(), target.get()); |
| generator.emitNodeInTailPosition(temp.get(), m_expr2); |
| generator.emitLabel(target.get()); |
| |
| return generator.move(dst, temp.get()); |
| } |
| |
| void LogicalOpNode::emitBytecodeInConditionContext(BytecodeGenerator& generator, Label& trueTarget, Label& falseTarget, FallThroughMode fallThroughMode) |
| { |
| if (UNLIKELY(needsDebugHook())) |
| generator.emitDebugHook(this); |
| |
| Ref<Label> afterExpr1 = generator.newLabel(); |
| if (m_operator == LogicalOperator::And) |
| generator.emitNodeInConditionContext(m_expr1, afterExpr1.get(), falseTarget, FallThroughMeansTrue); |
| else |
| generator.emitNodeInConditionContext(m_expr1, trueTarget, afterExpr1.get(), FallThroughMeansFalse); |
| generator.emitLabel(afterExpr1.get()); |
| |
| generator.emitNodeInConditionContext(m_expr2, trueTarget, falseTarget, fallThroughMode); |
| } |
| |
| // ------------------------------ CoalesceNode ---------------------------- |
| |
| RegisterID* CoalesceNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> temp = generator.tempDestination(dst); |
| Ref<Label> endLabel = generator.newLabel(); |
| |
| if (m_hasAbsorbedOptionalChain) |
| generator.pushOptionalChainTarget(); |
| generator.emitNode(temp.get(), m_expr1); |
| generator.emitJumpIfFalse(generator.emitIsUndefinedOrNull(generator.newTemporary(), temp.get()), endLabel.get()); |
| |
| if (m_hasAbsorbedOptionalChain) |
| generator.popOptionalChainTarget(); |
| generator.emitNodeInTailPosition(temp.get(), m_expr2); |
| |
| generator.emitLabel(endLabel.get()); |
| return generator.move(dst, temp.get()); |
| } |
| |
| // ------------------------------ OptionalChainNode ---------------------------- |
| |
| RegisterID* OptionalChainNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> finalDest = generator.finalDestination(dst); |
| |
| if (m_isOutermost) |
| generator.pushOptionalChainTarget(); |
| generator.emitNodeInTailPosition(finalDest.get(), m_expr); |
| if (m_isOutermost) |
| generator.popOptionalChainTarget(finalDest.get(), m_expr->isDeleteNode()); |
| |
| return finalDest.get(); |
| } |
| |
| // ------------------------------ ConditionalNode ------------------------------ |
| |
| RegisterID* ConditionalNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> newDst = generator.finalDestination(dst); |
| Ref<Label> beforeElse = generator.newLabel(); |
| Ref<Label> afterElse = generator.newLabel(); |
| |
| Ref<Label> beforeThen = generator.newLabel(); |
| generator.emitNodeInConditionContext(m_logical, beforeThen.get(), beforeElse.get(), FallThroughMeansTrue); |
| generator.emitLabel(beforeThen.get()); |
| |
| generator.emitProfileControlFlow(m_expr1->startOffset()); |
| generator.emitNodeInTailPosition(newDst.get(), m_expr1); |
| generator.emitJump(afterElse.get()); |
| |
| generator.emitLabel(beforeElse.get()); |
| generator.emitProfileControlFlow(m_expr1->endOffset() + 1); |
| generator.emitNodeInTailPosition(newDst.get(), m_expr2); |
| |
| generator.emitLabel(afterElse.get()); |
| |
| generator.emitProfileControlFlow(m_expr2->endOffset() + 1); |
| |
| return newDst.get(); |
| } |
| |
| // ------------------------------ ReadModifyResolveNode ----------------------------------- |
| |
| // FIXME: should this be moved to be a method on BytecodeGenerator? |
| static ALWAYS_INLINE RegisterID* emitReadModifyAssignment(BytecodeGenerator& generator, RegisterID* dst, RegisterID* src1, ExpressionNode* m_right, Operator oper, OperandTypes types, ReadModifyResolveNode* emitExpressionInfoForMe = nullptr, Variable* emitReadOnlyExceptionIfNeededForMe = nullptr) |
| { |
| OpcodeID opcodeID; |
| switch (oper) { |
| case Operator::MultEq: |
| opcodeID = op_mul; |
| break; |
| case Operator::DivEq: |
| opcodeID = op_div; |
| break; |
| case Operator::PlusEq: |
| if (m_right->isAdd() && m_right->resultDescriptor().definitelyIsString()) { |
| RegisterID* result = static_cast<AddNode*>(m_right)->emitStrcat(generator, dst, src1, emitExpressionInfoForMe); |
| if (emitReadOnlyExceptionIfNeededForMe) |
| generator.emitReadOnlyExceptionIfNeeded(*emitReadOnlyExceptionIfNeededForMe); |
| return result; |
| } |
| |
| opcodeID = op_add; |
| break; |
| case Operator::MinusEq: |
| opcodeID = op_sub; |
| break; |
| case Operator::LShift: |
| opcodeID = op_lshift; |
| break; |
| case Operator::RShift: |
| opcodeID = op_rshift; |
| break; |
| case Operator::URShift: |
| opcodeID = op_urshift; |
| break; |
| case Operator::BitAndEq: |
| opcodeID = op_bitand; |
| break; |
| case Operator::BitXOrEq: |
| opcodeID = op_bitxor; |
| break; |
| case Operator::BitOrEq: |
| opcodeID = op_bitor; |
| break; |
| case Operator::ModEq: |
| opcodeID = op_mod; |
| break; |
| case Operator::PowEq: |
| opcodeID = op_pow; |
| break; |
| default: |
| RELEASE_ASSERT_NOT_REACHED(); |
| return dst; |
| } |
| |
| RegisterID* src2 = generator.emitNode(m_right); |
| |
| if (emitReadOnlyExceptionIfNeededForMe) { |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(*emitReadOnlyExceptionIfNeededForMe); |
| if (threwException) |
| return src2; |
| } |
| |
| // Certain read-modify nodes require expression info to be emitted *after* m_right has been generated. |
| // If this is required the node is passed as 'emitExpressionInfoForMe'; do so now. |
| if (emitExpressionInfoForMe) |
| generator.emitExpressionInfo(emitExpressionInfoForMe->divot(), emitExpressionInfoForMe->divotStart(), emitExpressionInfoForMe->divotEnd()); |
| |
| RegisterID* result = generator.emitBinaryOp(opcodeID, dst, src1, src2, types); |
| if (oper == Operator::URShift) |
| return generator.emitUnaryOp<OpUnsigned>(result, result); |
| return result; |
| } |
| |
| RegisterID* ReadModifyResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| JSTextPosition newDivot = divotStart() + m_ident.length(); |
| Variable var = generator.variable(m_ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| if (var.isReadOnly()) { |
| RegisterID* result = emitReadModifyAssignment(generator, generator.finalDestination(dst), local, m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor()), nullptr, &var); |
| generator.emitProfileType(result, divotStart(), divotEnd()); |
| return result; |
| } |
| |
| if (generator.leftHandSideNeedsCopy(m_rightHasAssignments, m_right->isPure(generator))) { |
| RefPtr<RegisterID> result = generator.newTemporary(); |
| generator.move(result.get(), local); |
| emitReadModifyAssignment(generator, result.get(), result.get(), m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor())); |
| generator.move(local, result.get()); |
| generator.emitProfileType(local, divotStart(), divotEnd()); |
| return generator.move(dst, result.get()); |
| } |
| |
| RegisterID* result = emitReadModifyAssignment(generator, local, local, m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor())); |
| generator.emitProfileType(result, divotStart(), divotEnd()); |
| return generator.move(dst, result); |
| } |
| |
| generator.emitExpressionInfo(newDivot, divotStart(), newDivot); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> value = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, value.get(), nullptr); |
| RefPtr<RegisterID> result = emitReadModifyAssignment(generator, generator.finalDestination(dst, value.get()), value.get(), m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor()), this, var.isReadOnly() ? &var : nullptr); |
| RegisterID* returnResult = result.get(); |
| if (!var.isReadOnly()) { |
| returnResult = generator.emitPutToScope(scope.get(), var, result.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| generator.emitProfileType(result.get(), var, divotStart(), divotEnd()); |
| } |
| return returnResult; |
| } |
| |
| // ------------------------------ ShortCircuitReadModifyResolveNode ----------------------------------- |
| |
| static ALWAYS_INLINE void emitShortCircuitAssignment(BytecodeGenerator& generator, RegisterID* value, Operator oper, Label& afterAssignment) |
| { |
| switch (oper) { |
| case Operator::CoalesceEq: |
| generator.emitJumpIfFalse(generator.emitIsUndefinedOrNull(generator.newTemporary(), value), afterAssignment); |
| break; |
| |
| case Operator::OrEq: |
| generator.emitJumpIfTrue(value, afterAssignment); |
| break; |
| |
| case Operator::AndEq: |
| generator.emitJumpIfFalse(value, afterAssignment); |
| break; |
| |
| default: |
| RELEASE_ASSERT_NOT_REACHED(); |
| break; |
| } |
| } |
| |
| RegisterID* ShortCircuitReadModifyResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| JSTextPosition newDivot = divotStart() + m_ident.length(); |
| |
| Variable var = generator.variable(m_ident); |
| bool isReadOnly = var.isReadOnly(); |
| |
| if (RefPtr<RegisterID> local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local.get(), nullptr); |
| |
| if (isReadOnly) { |
| RefPtr<RegisterID> result = local; |
| |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, result.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(result.get(), m_right); // Execute side effects first. |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(var); |
| |
| if (!threwException) |
| generator.emitProfileType(result.get(), divotStart(), divotEnd()); |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(dst, result.get()); |
| } |
| |
| if (generator.leftHandSideNeedsCopy(m_rightHasAssignments, m_right->isPure(generator))) { |
| RefPtr<RegisterID> result = generator.tempDestination(dst); |
| generator.move(result.get(), local.get()); |
| |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, result.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(result.get(), m_right); |
| generator.move(local.get(), result.get()); |
| generator.emitProfileType(result.get(), var, divotStart(), divotEnd()); |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(dst, result.get()); |
| } |
| |
| RefPtr<RegisterID> result = local; |
| |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, result.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(result.get(), m_right); |
| generator.emitProfileType(result.get(), var, divotStart(), divotEnd()); |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(dst, result.get()); |
| } |
| |
| generator.emitExpressionInfo(newDivot, divotStart(), newDivot); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| |
| RefPtr<RegisterID> uncheckedResult = generator.newTemporary(); |
| |
| generator.emitGetFromScope(uncheckedResult.get(), scope.get(), var, ThrowIfNotFound); |
| generator.emitTDZCheckIfNecessary(var, uncheckedResult.get(), nullptr); |
| |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, uncheckedResult.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(uncheckedResult.get(), m_right); // Execute side effects first. |
| |
| bool threwException = isReadOnly ? generator.emitReadOnlyExceptionIfNeeded(var) : false; |
| |
| if (!threwException) |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| |
| if (!isReadOnly) { |
| generator.emitPutToScope(scope.get(), var, uncheckedResult.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| generator.emitProfileType(uncheckedResult.get(), var, divotStart(), divotEnd()); |
| } |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(generator.finalDestination(dst, uncheckedResult.get()), uncheckedResult.get()); |
| } |
| |
| // ------------------------------ AssignResolveNode ----------------------------------- |
| |
| static InitializationMode initializationModeForAssignmentContext(AssignmentContext assignmentContext) |
| { |
| switch (assignmentContext) { |
| case AssignmentContext::DeclarationStatement: |
| return InitializationMode::Initialization; |
| case AssignmentContext::ConstDeclarationStatement: |
| return InitializationMode::ConstInitialization; |
| case AssignmentContext::AssignmentExpression: |
| return InitializationMode::NotInitialization; |
| } |
| |
| ASSERT_NOT_REACHED(); |
| return InitializationMode::NotInitialization; |
| } |
| |
| RegisterID* AssignResolveNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| Variable var = generator.variable(m_ident); |
| bool isReadOnly = var.isReadOnly() && m_assignmentContext != AssignmentContext::ConstDeclarationStatement; |
| if (RegisterID* local = var.local()) { |
| RegisterID* result = nullptr; |
| if (m_assignmentContext == AssignmentContext::AssignmentExpression) |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| |
| if (isReadOnly) { |
| result = generator.emitNode(dst, m_right); // Execute side effects first. |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| generator.emitProfileType(result, var, divotStart(), divotEnd()); |
| } else if (var.isSpecial()) { |
| RefPtr<RegisterID> tempDst = generator.tempDestination(dst); |
| generator.emitNode(tempDst.get(), m_right); |
| generator.move(local, tempDst.get()); |
| generator.emitProfileType(local, var, divotStart(), divotEnd()); |
| result = generator.move(dst, tempDst.get()); |
| } else { |
| RegisterID* right = generator.emitNode(local, m_right); |
| generator.emitProfileType(right, var, divotStart(), divotEnd()); |
| result = generator.move(dst, right); |
| } |
| |
| if (m_assignmentContext == AssignmentContext::DeclarationStatement || m_assignmentContext == AssignmentContext::ConstDeclarationStatement) |
| generator.liftTDZCheckIfPossible(var); |
| return result; |
| } |
| |
| if (generator.ecmaMode().isStrict()) |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| if (m_assignmentContext == AssignmentContext::AssignmentExpression) |
| generator.emitTDZCheckIfNecessary(var, nullptr, scope.get()); |
| if (dst == generator.ignoredResult()) |
| dst = nullptr; |
| RefPtr<RegisterID> result = generator.emitNode(dst, m_right); // Execute side effects first. |
| if (isReadOnly) { |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(var); |
| if (threwException) |
| return result.get(); |
| } |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RegisterID* returnResult = result.get(); |
| if (!isReadOnly) { |
| returnResult = generator.emitPutToScope(scope.get(), var, result.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, initializationModeForAssignmentContext(m_assignmentContext)); |
| generator.emitProfileType(result.get(), var, divotStart(), divotEnd()); |
| } |
| |
| if (m_assignmentContext == AssignmentContext::DeclarationStatement || m_assignmentContext == AssignmentContext::ConstDeclarationStatement) |
| generator.liftTDZCheckIfPossible(var); |
| return returnResult; |
| } |
| |
| // ------------------------------ AssignDotNode ----------------------------------- |
| |
| RegisterID* AssignDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_rightHasAssignments, m_right->isPure(generator)); |
| RefPtr<RegisterID> value = generator.destinationForAssignResult(dst); |
| RefPtr<RegisterID> result = generator.emitNode(value.get(), m_right); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> forwardResult = (dst == generator.ignoredResult()) ? result.get() : generator.move(generator.tempDestination(result.get()), result.get()); |
| emitPutProperty(generator, base.get(), forwardResult.get()); |
| generator.emitProfileType(forwardResult.get(), divotStart(), divotEnd()); |
| return generator.move(dst, forwardResult.get()); |
| } |
| |
| // ------------------------------ ReadModifyDotNode ----------------------------------- |
| |
| RegisterID* ReadModifyDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_rightHasAssignments, m_right->isPure(generator)); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| RefPtr<RegisterID> thisValue; |
| RefPtr<RegisterID> value = emitGetPropertyValue(generator, generator.tempDestination(dst), base.get(), thisValue); |
| |
| RegisterID* updatedValue = emitReadModifyAssignment(generator, generator.finalDestination(dst, value.get()), value.get(), m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor())); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> ret = emitPutProperty(generator, base.get(), updatedValue, thisValue); |
| generator.emitProfileType(updatedValue, divotStart(), divotEnd()); |
| return ret.get(); |
| } |
| |
| // ------------------------------ ShortCircuitReadModifyDotNode ----------------------------------- |
| |
| RegisterID* ShortCircuitReadModifyDotNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_rightHasAssignments, m_right->isPure(generator)); |
| RefPtr<RegisterID> thisValue; |
| |
| RefPtr<RegisterID> result = generator.tempDestination(dst); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| emitGetPropertyValue(generator, result.get(), base.get(), thisValue); |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, result.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(result.get(), m_right); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| emitPutProperty(generator, base.get(), result.get(), thisValue); |
| generator.emitProfileType(result.get(), divotStart(), divotEnd()); |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(dst, result.get()); |
| } |
| |
| // ------------------------------ AssignErrorNode ----------------------------------- |
| |
| RegisterID* AssignErrorNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| return emitThrowReferenceError(generator, "Left side of assignment is not a reference."_s, dst); |
| } |
| |
| // ------------------------------ AssignBracketNode ----------------------------------- |
| |
| RegisterID* AssignBracketNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_subscriptHasAssignments || m_rightHasAssignments, m_subscript->isPure(generator) && m_right->isPure(generator)); |
| RefPtr<RegisterID> property = generator.emitNodeForLeftHandSideForProperty(m_subscript, m_rightHasAssignments, m_right->isPure(generator)); |
| RefPtr<RegisterID> value = generator.destinationForAssignResult(dst); |
| RefPtr<RegisterID> result = generator.emitNode(value.get(), m_right); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| RegisterID* forwardResult = (dst == generator.ignoredResult()) ? result.get() : generator.move(generator.tempDestination(result.get()), result.get()); |
| |
| if (isNonIndexStringElement(*m_subscript)) { |
| if (m_base->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| generator.emitPutById(base.get(), thisValue.get(), static_cast<StringNode*>(m_subscript)->value(), forwardResult); |
| } else |
| generator.emitPutById(base.get(), static_cast<StringNode*>(m_subscript)->value(), forwardResult); |
| } else { |
| if (m_base->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), forwardResult); |
| } else |
| generator.emitPutByVal(base.get(), property.get(), forwardResult); |
| } |
| |
| generator.emitProfileType(forwardResult, divotStart(), divotEnd()); |
| return generator.move(dst, forwardResult); |
| } |
| |
| // ------------------------------ ReadModifyBracketNode ----------------------------------- |
| |
| RegisterID* ReadModifyBracketNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_subscriptHasAssignments || m_rightHasAssignments, m_subscript->isPure(generator) && m_right->isPure(generator)); |
| RefPtr<RegisterID> property = generator.emitNodeForLeftHandSideForProperty(m_subscript, m_rightHasAssignments, m_right->isPure(generator)); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| RefPtr<RegisterID> value; |
| RefPtr<RegisterID> thisValue; |
| if (m_base->isSuperNode()) { |
| thisValue = generator.ensureThis(); |
| value = generator.emitGetByVal(generator.tempDestination(dst), base.get(), thisValue.get(), property.get()); |
| } else |
| value = generator.emitGetByVal(generator.tempDestination(dst), base.get(), property.get()); |
| RegisterID* updatedValue = emitReadModifyAssignment(generator, generator.finalDestination(dst, value.get()), value.get(), m_right, m_operator, OperandTypes(ResultType::unknownType(), m_right->resultDescriptor())); |
| |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (m_base->isSuperNode()) |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), updatedValue); |
| else |
| generator.emitPutByVal(base.get(), property.get(), updatedValue); |
| generator.emitProfileType(updatedValue, divotStart(), divotEnd()); |
| |
| return updatedValue; |
| } |
| |
| // ------------------------------ ShortCircuitReadModifyBracketNode ----------------------------------- |
| |
| RegisterID* ShortCircuitReadModifyBracketNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(m_base, m_subscriptHasAssignments || m_rightHasAssignments, m_subscript->isPure(generator) && m_right->isPure(generator)); |
| RefPtr<RegisterID> property = generator.emitNodeForLeftHandSideForProperty(m_subscript, m_rightHasAssignments, m_right->isPure(generator)); |
| RefPtr<RegisterID> thisValue; |
| |
| RefPtr<RegisterID> result = generator.tempDestination(dst); |
| |
| generator.emitExpressionInfo(subexpressionDivot(), subexpressionStart(), subexpressionEnd()); |
| if (m_base->isSuperNode()) { |
| thisValue = generator.ensureThis(); |
| generator.emitGetByVal(result.get(), base.get(), thisValue.get(), property.get()); |
| } else |
| generator.emitGetByVal(result.get(), base.get(), property.get()); |
| |
| Ref<Label> afterAssignment = generator.newLabel(); |
| emitShortCircuitAssignment(generator, result.get(), m_operator, afterAssignment.get()); |
| |
| generator.emitNode(result.get(), m_right); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (m_base->isSuperNode()) |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), result.get()); |
| else |
| generator.emitPutByVal(base.get(), property.get(), result.get()); |
| generator.emitProfileType(result.get(), divotStart(), divotEnd()); |
| |
| generator.emitLabel(afterAssignment.get()); |
| return generator.move(dst, result.get()); |
| } |
| |
| // ------------------------------ CommaNode ------------------------------------ |
| |
| RegisterID* CommaNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| DebugHookType debugHookType = isOnlyChildOfStatement() ? WillExecuteStatement : WillExecuteExpression; |
| |
| CommaNode* node = this; |
| for (; node->next(); node = node->next()) { |
| generator.emitDebugHook(debugHookType, node->m_expr->position()); |
| generator.emitNode(generator.ignoredResult(), node->m_expr); |
| } |
| generator.emitDebugHook(debugHookType, node->m_expr->position()); |
| return generator.emitNodeInTailPosition(dst, node->m_expr); |
| } |
| |
| // ------------------------------ SourceElements ------------------------------- |
| |
| inline void SourceElements::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| StatementNode* lastStatementWithCompletionValue = nullptr; |
| if (generator.shouldBeConcernedWithCompletionValue()) { |
| for (StatementNode* statement = m_head; statement; statement = statement->next()) { |
| if (statement->hasCompletionValue()) |
| lastStatementWithCompletionValue = statement; |
| } |
| } |
| |
| for (StatementNode* statement = m_head; statement; statement = statement->next()) { |
| if (statement == lastStatementWithCompletionValue) |
| generator.emitLoad(dst, jsUndefined()); |
| generator.emitNodeInTailPosition(dst, statement); |
| } |
| } |
| |
| // ------------------------------ BlockNode ------------------------------------ |
| |
| void BlockNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_statements) |
| return; |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::LetConstScope, BytecodeGenerator::TDZCheckOptimization::Optimize, BytecodeGenerator::NestedScopeType::IsNested); |
| m_statements->emitBytecode(generator, dst); |
| generator.popLexicalScope(this); |
| } |
| |
| // ------------------------------ EmptyStatementNode --------------------------- |
| |
| void EmptyStatementNode::emitBytecode(BytecodeGenerator&, RegisterID*) |
| { |
| RELEASE_ASSERT(needsDebugHook()); |
| } |
| |
| // ------------------------------ DebuggerStatementNode --------------------------- |
| |
| void DebuggerStatementNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| generator.emitDebugHook(DidReachDebuggerStatement, position()); |
| } |
| |
| // ------------------------------ ExprStatementNode ---------------------------- |
| |
| void ExprStatementNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_expr); |
| generator.emitNode(dst, m_expr); |
| } |
| |
| // ------------------------------ DeclarationStatement ---------------------------- |
| |
| void DeclarationStatement::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| ASSERT(m_expr); |
| generator.emitNode(m_expr); |
| } |
| |
| // ------------------------------ EmptyVarExpression ---------------------------- |
| |
| RegisterID* EmptyVarExpression::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| // It's safe to return null here because this node will always be a child node of DeclarationStatement which ignores our return value. |
| if (!generator.shouldEmitTypeProfilerHooks()) |
| return nullptr; |
| |
| Variable var = generator.variable(m_ident); |
| if (RegisterID* local = var.local()) |
| generator.emitProfileType(local, var, position(), position() + m_ident.length()); |
| else { |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> value = generator.emitGetFromScope(generator.newTemporary(), scope.get(), var, DoNotThrowIfNotFound); |
| generator.emitProfileType(value.get(), var, position(), position() + m_ident.length()); |
| } |
| |
| return nullptr; |
| } |
| |
| // ------------------------------ EmptyLetExpression ---------------------------- |
| |
| RegisterID* EmptyLetExpression::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| // Lexical declarations like 'let' must move undefined into their variables so we don't |
| // get TDZ errors for situations like this: `let x; x;` |
| Variable var = generator.variable(m_ident); |
| if (RegisterID* local = var.local()) { |
| generator.emitLoad(local, jsUndefined()); |
| generator.emitProfileType(local, var, position(), position() + m_ident.length()); |
| } else { |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> value = generator.emitLoad(nullptr, jsUndefined()); |
| generator.emitPutToScope(scope.get(), var, value.get(), generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::Initialization); |
| generator.emitProfileType(value.get(), var, position(), position() + m_ident.length()); |
| } |
| |
| generator.liftTDZCheckIfPossible(var); |
| |
| // It's safe to return null here because this node will always be a child node of DeclarationStatement which ignores our return value. |
| return nullptr; |
| } |
| |
| // ------------------------------ IfElseNode --------------------------------------- |
| |
| static inline StatementNode* singleStatement(StatementNode* statementNode) |
| { |
| if (statementNode->isBlock()) |
| return static_cast<BlockNode*>(statementNode)->singleStatement(); |
| return statementNode; |
| } |
| |
| bool IfElseNode::tryFoldBreakAndContinue(BytecodeGenerator& generator, StatementNode* ifBlock, |
| Label*& trueTarget, FallThroughMode& fallThroughMode) |
| { |
| StatementNode* singleStatement = JSC::singleStatement(ifBlock); |
| if (!singleStatement) |
| return false; |
| |
| if (singleStatement->isBreak()) { |
| BreakNode* breakNode = static_cast<BreakNode*>(singleStatement); |
| Label* target = breakNode->trivialTarget(generator); |
| if (!target) |
| return false; |
| trueTarget = target; |
| fallThroughMode = FallThroughMeansFalse; |
| return true; |
| } |
| |
| if (singleStatement->isContinue()) { |
| ContinueNode* continueNode = static_cast<ContinueNode*>(singleStatement); |
| Label* target = continueNode->trivialTarget(generator); |
| if (!target) |
| return false; |
| trueTarget = target; |
| fallThroughMode = FallThroughMeansFalse; |
| return true; |
| } |
| |
| return false; |
| } |
| |
| void IfElseNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (generator.shouldBeConcernedWithCompletionValue()) { |
| if (m_ifBlock->hasEarlyBreakOrContinue() || (m_elseBlock && m_elseBlock->hasEarlyBreakOrContinue())) |
| generator.emitLoad(dst, jsUndefined()); |
| } |
| |
| Ref<Label> beforeThen = generator.newLabel(); |
| Ref<Label> beforeElse = generator.newLabel(); |
| Ref<Label> afterElse = generator.newLabel(); |
| |
| Label* trueTarget = beforeThen.ptr(); |
| Label& falseTarget = beforeElse.get(); |
| FallThroughMode fallThroughMode = FallThroughMeansTrue; |
| bool didFoldIfBlock = tryFoldBreakAndContinue(generator, m_ifBlock, trueTarget, fallThroughMode); |
| |
| generator.emitNodeInConditionContext(m_condition, *trueTarget, falseTarget, fallThroughMode); |
| generator.emitLabel(beforeThen.get()); |
| generator.emitProfileControlFlow(m_ifBlock->startOffset()); |
| |
| if (!didFoldIfBlock) { |
| generator.emitNodeInTailPosition(dst, m_ifBlock); |
| if (m_elseBlock) |
| generator.emitJump(afterElse.get()); |
| } |
| |
| generator.emitLabel(beforeElse.get()); |
| |
| if (m_elseBlock) { |
| generator.emitProfileControlFlow(m_ifBlock->endOffset() + (m_ifBlock->isBlock() ? 1 : 0)); |
| generator.emitNodeInTailPosition(dst, m_elseBlock); |
| } |
| |
| generator.emitLabel(afterElse.get()); |
| StatementNode* endingBlock = m_elseBlock ? m_elseBlock : m_ifBlock; |
| generator.emitProfileControlFlow(endingBlock->endOffset() + (endingBlock->isBlock() ? 1 : 0)); |
| } |
| |
| // ------------------------------ DoWhileNode ---------------------------------- |
| |
| void DoWhileNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::Loop); |
| |
| Ref<Label> topOfLoop = generator.newLabel(); |
| generator.emitLabel(topOfLoop.get()); |
| generator.emitLoopHint(); |
| |
| generator.emitNodeInTailPosition(dst, m_statement); |
| |
| generator.emitLabel(*scope->continueTarget()); |
| generator.emitNodeInConditionContext(m_expr, topOfLoop.get(), scope->breakTarget(), FallThroughMeansFalse); |
| |
| generator.emitLabel(scope->breakTarget()); |
| } |
| |
| // ------------------------------ WhileNode ------------------------------------ |
| |
| void WhileNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::Loop); |
| Ref<Label> topOfLoop = generator.newLabel(); |
| |
| generator.emitNodeInConditionContext(m_expr, topOfLoop.get(), scope->breakTarget(), FallThroughMeansTrue); |
| |
| generator.emitLabel(topOfLoop.get()); |
| generator.emitLoopHint(); |
| |
| generator.emitProfileControlFlow(m_statement->startOffset()); |
| generator.emitNodeInTailPosition(dst, m_statement); |
| |
| generator.emitLabel(*scope->continueTarget()); |
| |
| generator.emitNodeInConditionContext(m_expr, topOfLoop.get(), scope->breakTarget(), FallThroughMeansFalse); |
| |
| generator.emitLabel(scope->breakTarget()); |
| |
| generator.emitProfileControlFlow(m_statement->endOffset() + (m_statement->isBlock() ? 1 : 0)); |
| } |
| |
| // ------------------------------ ForNode -------------------------------------- |
| |
| void ForNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::Loop); |
| |
| RegisterID* forLoopSymbolTable = nullptr; |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::LetConstScope, BytecodeGenerator::TDZCheckOptimization::Optimize, BytecodeGenerator::NestedScopeType::IsNested, &forLoopSymbolTable); |
| |
| if (m_expr1) |
| generator.emitNode(generator.ignoredResult(), m_expr1); |
| |
| Ref<Label> topOfLoop = generator.newLabel(); |
| if (m_expr2) |
| generator.emitNodeInConditionContext(m_expr2, topOfLoop.get(), scope->breakTarget(), FallThroughMeansTrue); |
| |
| generator.emitLabel(topOfLoop.get()); |
| generator.emitLoopHint(); |
| generator.emitProfileControlFlow(m_statement->startOffset()); |
| |
| generator.emitNodeInTailPosition(dst, m_statement); |
| |
| generator.emitLabel(*scope->continueTarget()); |
| generator.prepareLexicalScopeForNextForLoopIteration(this, forLoopSymbolTable); |
| if (m_expr3) |
| generator.emitNode(generator.ignoredResult(), m_expr3); |
| |
| if (m_expr2) |
| generator.emitNodeInConditionContext(m_expr2, topOfLoop.get(), scope->breakTarget(), FallThroughMeansFalse); |
| else |
| generator.emitJump(topOfLoop.get()); |
| |
| generator.emitLabel(scope->breakTarget()); |
| generator.popLexicalScope(this); |
| generator.emitProfileControlFlow(m_statement->endOffset() + (m_statement->isBlock() ? 1 : 0)); |
| } |
| |
| // ------------------------------ ForInNode ------------------------------------ |
| |
| RegisterID* ForInNode::tryGetBoundLocal(BytecodeGenerator& generator) |
| { |
| if (m_lexpr->isResolveNode()) { |
| const Identifier& ident = static_cast<ResolveNode*>(m_lexpr)->identifier(); |
| return generator.variable(ident).local(); |
| } |
| |
| if (m_lexpr->isDestructuringNode()) { |
| DestructuringAssignmentNode* assignNode = static_cast<DestructuringAssignmentNode*>(m_lexpr); |
| auto binding = assignNode->bindings(); |
| if (!binding->isBindingNode()) |
| return nullptr; |
| |
| auto simpleBinding = static_cast<BindingNode*>(binding); |
| const Identifier& ident = simpleBinding->boundProperty(); |
| Variable var = generator.variable(ident); |
| if (var.isSpecial()) |
| return nullptr; |
| return var.local(); |
| } |
| |
| return nullptr; |
| } |
| |
| void ForInNode::emitLoopHeader(BytecodeGenerator& generator, RegisterID* propertyName) |
| { |
| auto lambdaEmitResolveVariable = [&] (const Identifier& ident) { |
| Variable var = generator.variable(ident); |
| if (RegisterID* local = var.local()) { |
| if (var.isReadOnly()) |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| generator.move(local, propertyName); |
| } else { |
| if (generator.ecmaMode().isStrict()) |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (var.isReadOnly()) |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitPutToScope(scope.get(), var, propertyName, generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| } |
| generator.emitProfileType(propertyName, var, m_lexpr->position(), m_lexpr->position() + ident.length()); |
| }; |
| |
| if (m_lexpr->isResolveNode()) { |
| const Identifier& ident = static_cast<ResolveNode*>(m_lexpr)->identifier(); |
| lambdaEmitResolveVariable(ident); |
| return; |
| } |
| |
| if (m_lexpr->isAssignResolveNode()) { |
| const Identifier& ident = static_cast<AssignResolveNode*>(m_lexpr)->identifier(); |
| lambdaEmitResolveVariable(ident); |
| return; |
| } |
| |
| if (m_lexpr->isDotAccessorNode()) { |
| DotAccessorNode* assignNode = static_cast<DotAccessorNode*>(m_lexpr); |
| RefPtr<RegisterID> base = generator.emitNode(assignNode->base()); |
| generator.emitExpressionInfo(assignNode->divot(), assignNode->divotStart(), assignNode->divotEnd()); |
| assignNode->emitPutProperty(generator, base.get(), propertyName); |
| generator.emitProfileType(propertyName, assignNode->divotStart(), assignNode->divotEnd()); |
| return; |
| } |
| |
| if (m_lexpr->isBracketAccessorNode()) { |
| BracketAccessorNode* assignNode = static_cast<BracketAccessorNode*>(m_lexpr); |
| RefPtr<RegisterID> base = generator.emitNode(assignNode->base()); |
| RefPtr<RegisterID> subscript = generator.emitNodeForProperty(assignNode->subscript()); |
| generator.emitExpressionInfo(assignNode->divot(), assignNode->divotStart(), assignNode->divotEnd()); |
| if (assignNode->base()->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| generator.emitPutByVal(base.get(), thisValue.get(), subscript.get(), propertyName); |
| } else |
| generator.emitPutByVal(base.get(), subscript.get(), propertyName); |
| generator.emitProfileType(propertyName, assignNode->divotStart(), assignNode->divotEnd()); |
| return; |
| } |
| |
| if (m_lexpr->isDestructuringNode()) { |
| DestructuringAssignmentNode* assignNode = static_cast<DestructuringAssignmentNode*>(m_lexpr); |
| auto binding = assignNode->bindings(); |
| if (!binding->isBindingNode()) { |
| assignNode->bindings()->bindValue(generator, propertyName); |
| return; |
| } |
| |
| auto simpleBinding = static_cast<BindingNode*>(binding); |
| const Identifier& ident = simpleBinding->boundProperty(); |
| Variable var = generator.variable(ident); |
| if (!var.local() || var.isSpecial()) { |
| assignNode->bindings()->bindValue(generator, propertyName); |
| return; |
| } |
| generator.move(var.local(), propertyName); |
| generator.emitProfileType(propertyName, var, simpleBinding->divotStart(), simpleBinding->divotEnd()); |
| return; |
| } |
| |
| RELEASE_ASSERT_NOT_REACHED(); |
| } |
| |
| void ForInNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_lexpr->isAssignResolveNode() && !m_lexpr->isAssignmentLocation()) { |
| emitThrowReferenceError(generator, "Left side of for-in statement is not a reference."_s); |
| return; |
| } |
| |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| RegisterID* forLoopSymbolTable = nullptr; |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::LetConstScope, BytecodeGenerator::TDZCheckOptimization::Optimize, BytecodeGenerator::NestedScopeType::IsNested, &forLoopSymbolTable); |
| |
| if (m_lexpr->isAssignResolveNode()) |
| generator.emitNode(generator.ignoredResult(), m_lexpr); |
| |
| RefPtr<RegisterID> base = generator.newTemporary(); |
| |
| generator.emitNode(base.get(), m_expr); |
| RefPtr<RegisterID> local = this->tryGetBoundLocal(generator); |
| |
| |
| std::optional<Variable> baseVariable; |
| if (m_expr->isResolveNode()) |
| baseVariable = generator.variable(static_cast<ResolveNode*>(m_expr)->identifier()); |
| else if (m_expr->isThisNode()) { |
| // After generator.ensureThis (which must be invoked in |base|'s materialization), we can ensure that |this| is in local this-register. |
| ASSERT(base); |
| baseVariable = generator.variable(generator.propertyNames().builtinNames().thisPrivateName(), ThisResolutionType::Local); |
| } |
| |
| int profilerStartOffset = m_statement->startOffset(); |
| int profilerEndOffset = m_statement->endOffset() + (m_statement->isBlock() ? 1 : 0); |
| |
| { |
| RefPtr<RegisterID> enumerator = generator.newTemporary(); |
| RefPtr<RegisterID> mode = generator.emitLoad(generator.newTemporary(), jsNumber(static_cast<unsigned>(JSPropertyNameEnumerator::InitMode))); |
| RefPtr<RegisterID> index = generator.emitLoad(generator.newTemporary(), jsNumber(0)); |
| RefPtr<RegisterID> propertyName = generator.newTemporary(); |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::Loop); |
| |
| enumerator = generator.emitGetPropertyEnumerator(generator.newTemporary(), base.get()); |
| generator.emitJumpIfEmptyPropertyNameEnumerator(enumerator.get(), scope->breakTarget()); |
| |
| generator.emitLabel(*scope->continueTarget()); |
| generator.emitLoopHint(); |
| generator.prepareLexicalScopeForNextForLoopIteration(this, forLoopSymbolTable); |
| generator.emitDebugHook(m_lexpr); // Pause at the assignment expression for each for..in iteration. |
| |
| // FIXME: We should have a way to see if anyone is actually using the propertyName for something other than a get_by_val. If not, we could eliminate the toString in this opcode. |
| generator.emitEnumeratorNext(propertyName.get(), mode.get(), index.get(), base.get(), enumerator.get()); |
| generator.emitJumpIfSentinelString(propertyName.get(), scope->breakTarget()); |
| |
| this->emitLoopHeader(generator, propertyName.get()); |
| |
| generator.emitProfileControlFlow(profilerStartOffset); |
| |
| generator.pushForInScope(local.get(), propertyName.get(), index.get(), enumerator.get(), mode.get(), baseVariable); |
| generator.emitNode(dst, m_statement); |
| generator.popForInScope(local.get()); |
| |
| generator.emitProfileControlFlow(profilerEndOffset); |
| generator.emitJump(*scope->continueTarget()); |
| |
| generator.emitLabel(scope->breakTarget()); |
| } |
| |
| generator.popLexicalScope(this); |
| generator.emitProfileControlFlow(profilerEndOffset); |
| } |
| |
| // ------------------------------ ForOfNode ------------------------------------ |
| void ForOfNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_lexpr->isAssignmentLocation()) { |
| emitThrowReferenceError(generator, "Left side of for-of statement is not a reference."_s); |
| return; |
| } |
| |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| RegisterID* forLoopSymbolTable = nullptr; |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::LetConstScope, BytecodeGenerator::TDZCheckOptimization::Optimize, BytecodeGenerator::NestedScopeType::IsNested, &forLoopSymbolTable); |
| auto extractor = scopedLambda<void(BytecodeGenerator&, RegisterID*)>([this, dst](BytecodeGenerator& generator, RegisterID* value) |
| { |
| if (m_lexpr->isResolveNode()) { |
| const Identifier& ident = static_cast<ResolveNode*>(m_lexpr)->identifier(); |
| Variable var = generator.variable(ident); |
| if (RegisterID* local = var.local()) { |
| if (var.isReadOnly()) |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| generator.move(local, value); |
| } else { |
| if (generator.ecmaMode().isStrict()) |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| if (var.isReadOnly()) |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitPutToScope(scope.get(), var, value, generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| } |
| generator.emitProfileType(value, var, m_lexpr->position(), m_lexpr->position() + ident.length()); |
| } else if (m_lexpr->isDotAccessorNode()) { |
| DotAccessorNode* assignNode = static_cast<DotAccessorNode*>(m_lexpr); |
| RefPtr<RegisterID> base = generator.emitNode(assignNode->base()); |
| generator.emitExpressionInfo(assignNode->divot(), assignNode->divotStart(), assignNode->divotEnd()); |
| assignNode->emitPutProperty(generator, base.get(), value); |
| generator.emitProfileType(value, assignNode->divotStart(), assignNode->divotEnd()); |
| } else if (m_lexpr->isBracketAccessorNode()) { |
| BracketAccessorNode* assignNode = static_cast<BracketAccessorNode*>(m_lexpr); |
| RefPtr<RegisterID> base = generator.emitNode(assignNode->base()); |
| RegisterID* subscript = generator.emitNodeForProperty(assignNode->subscript()); |
| |
| generator.emitExpressionInfo(assignNode->divot(), assignNode->divotStart(), assignNode->divotEnd()); |
| if (assignNode->base()->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| generator.emitPutByVal(base.get(), thisValue.get(), subscript, value); |
| } else |
| generator.emitPutByVal(base.get(), subscript, value); |
| generator.emitProfileType(value, assignNode->divotStart(), assignNode->divotEnd()); |
| } else { |
| ASSERT(m_lexpr->isDestructuringNode()); |
| DestructuringAssignmentNode* assignNode = static_cast<DestructuringAssignmentNode*>(m_lexpr); |
| assignNode->bindings()->bindValue(generator, value); |
| } |
| generator.emitProfileControlFlow(m_statement->startOffset()); |
| generator.emitNode(dst, m_statement); |
| }); |
| generator.emitEnumeration(this, m_expr, extractor, this, forLoopSymbolTable); |
| generator.popLexicalScope(this); |
| generator.emitProfileControlFlow(m_statement->endOffset() + (m_statement->isBlock() ? 1 : 0)); |
| } |
| |
| // ------------------------------ ContinueNode --------------------------------- |
| |
| Label* ContinueNode::trivialTarget(BytecodeGenerator& generator) |
| { |
| if (generator.shouldEmitDebugHooks()) |
| return nullptr; |
| |
| LabelScope* scope = generator.continueTarget(m_ident); |
| ASSERT(scope); |
| |
| if (generator.labelScopeDepth() != scope->scopeDepth()) |
| return nullptr; |
| |
| return scope->continueTarget(); |
| } |
| |
| void ContinueNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| LabelScope* scope = generator.continueTarget(m_ident); |
| ASSERT(scope); |
| |
| bool hasFinally = generator.emitJumpViaFinallyIfNeeded(scope->scopeDepth(), *scope->continueTarget()); |
| if (!hasFinally) { |
| int lexicalScopeIndex = generator.labelScopeDepthToLexicalScopeIndex(scope->scopeDepth()); |
| generator.restoreScopeRegister(lexicalScopeIndex); |
| generator.emitJump(*scope->continueTarget()); |
| } |
| |
| generator.emitProfileControlFlow(endOffset()); |
| } |
| |
| // ------------------------------ BreakNode ------------------------------------ |
| |
| Label* BreakNode::trivialTarget(BytecodeGenerator& generator) |
| { |
| if (generator.shouldEmitDebugHooks()) |
| return nullptr; |
| |
| LabelScope* scope = generator.breakTarget(m_ident); |
| ASSERT(scope); |
| |
| if (generator.labelScopeDepth() != scope->scopeDepth()) |
| return nullptr; |
| |
| return &scope->breakTarget(); |
| } |
| |
| void BreakNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| LabelScope* scope = generator.breakTarget(m_ident); |
| ASSERT(scope); |
| |
| bool hasFinally = generator.emitJumpViaFinallyIfNeeded(scope->scopeDepth(), scope->breakTarget()); |
| if (!hasFinally) { |
| int lexicalScopeIndex = generator.labelScopeDepthToLexicalScopeIndex(scope->scopeDepth()); |
| generator.restoreScopeRegister(lexicalScopeIndex); |
| generator.emitJump(scope->breakTarget()); |
| } |
| |
| generator.emitProfileControlFlow(endOffset()); |
| } |
| |
| // ------------------------------ ReturnNode ----------------------------------- |
| |
| void ReturnNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(generator.codeType() == FunctionCode); |
| |
| if (dst == generator.ignoredResult()) |
| dst = nullptr; |
| |
| RefPtr<RegisterID> returnRegister = m_value ? generator.emitNodeInTailPosition(dst, m_value) : generator.emitLoad(dst, jsUndefined()); |
| |
| generator.emitProfileType(returnRegister.get(), ProfileTypeBytecodeFunctionReturnStatement, divotStart(), divotEnd()); |
| |
| bool hasFinally = generator.emitReturnViaFinallyIfNeeded(returnRegister.get()); |
| if (!hasFinally) { |
| if (generator.parseMode() == SourceParseMode::AsyncGeneratorBodyMode) { |
| returnRegister = generator.move(generator.newTemporary(), returnRegister.get()); |
| generator.emitAwait(returnRegister.get()); |
| } |
| |
| generator.emitWillLeaveCallFrameDebugHook(); |
| generator.emitReturn(returnRegister.get()); |
| } |
| |
| generator.emitProfileControlFlow(endOffset()); |
| // Emitting an unreachable return here is needed in case this op_profile_control_flow is the |
| // last opcode in a CodeBlock because a CodeBlock's instructions must end with a terminal opcode. |
| if (generator.shouldEmitControlFlowProfilerHooks()) |
| generator.emitReturn(generator.emitLoad(nullptr, jsUndefined())); |
| } |
| |
| // ------------------------------ WithNode ------------------------------------- |
| |
| void WithNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> scope = generator.emitNode(m_expr); |
| generator.emitExpressionInfo(m_divot, m_divot - m_expressionLength, m_divot); |
| generator.emitPushWithScope(scope.get()); |
| if (generator.shouldBeConcernedWithCompletionValue() && m_statement->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| generator.emitNodeInTailPosition(dst, m_statement); |
| generator.emitPopWithScope(); |
| } |
| |
| // ------------------------------ CaseClauseNode -------------------------------- |
| |
| inline void CaseClauseNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| generator.emitProfileControlFlow(m_startOffset); |
| if (!m_statements) |
| return; |
| m_statements->emitBytecode(generator, dst); |
| } |
| |
| // ------------------------------ CaseBlockNode -------------------------------- |
| |
| enum SwitchKind { |
| SwitchUnset = 0, |
| SwitchNumber = 1, |
| SwitchString = 2, |
| SwitchNeither = 3 |
| }; |
| |
| static void processClauseList(ClauseListNode* list, Vector<ExpressionNode*, 8>& 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<NumberNode*>(clauseExpression)->value(); |
| int32_t intVal = static_cast<int32_t>(value); |
| if ((typeForTable & ~SwitchNumber) || (intVal != value)) { |
| typeForTable = SwitchNeither; |
| break; |
| } |
| 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; |
| } |
| auto& value = static_cast<StringNode*>(clauseExpression)->value().string(); |
| if (singleCharacterSwitch &= value.length() == 1) { |
| int32_t intVal = value[0]; |
| if (intVal < min_num) |
| min_num = intVal; |
| if (intVal > max_num) |
| max_num = intVal; |
| } |
| typeForTable = SwitchString; |
| continue; |
| } |
| typeForTable = SwitchNeither; |
| break; |
| } |
| } |
| |
| static inline size_t length(ClauseListNode* list1, ClauseListNode* list2) |
| { |
| size_t length = 0; |
| for (ClauseListNode* node = list1; node; node = node->getNext()) |
| ++length; |
| for (ClauseListNode* node = list2; node; node = node->getNext()) |
| ++length; |
| return length; |
| } |
| |
| SwitchInfo::SwitchType CaseBlockNode::tryTableSwitch(Vector<ExpressionNode*, 8>& literalVector, int32_t& min_num, int32_t& max_num) |
| { |
| if (length(m_list1, m_list2) < s_tableSwitchMinimum) |
| return SwitchInfo::SwitchNone; |
| |
| SwitchKind typeForTable = SwitchUnset; |
| bool singleCharacterSwitch = true; |
| |
| processClauseList(m_list1, literalVector, typeForTable, singleCharacterSwitch, min_num, max_num); |
| processClauseList(m_list2, 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; |
| } |
| |
| void CaseBlockNode::emitBytecodeForBlock(BytecodeGenerator& generator, RegisterID* switchExpression, RegisterID* dst) |
| { |
| Vector<Ref<Label>, 8> labelVector; |
| Vector<ExpressionNode*, 8> literalVector; |
| int32_t min_num = std::numeric_limits<int32_t>::max(); |
| int32_t max_num = std::numeric_limits<int32_t>::min(); |
| SwitchInfo::SwitchType switchType = tryTableSwitch(literalVector, min_num, max_num); |
| |
| Ref<Label> defaultLabel = generator.newLabel(); |
| if (switchType != SwitchInfo::SwitchNone) { |
| // Prepare the various labels |
| for (uint32_t i = 0; i < literalVector.size(); i++) |
| labelVector.append(generator.newLabel()); |
| generator.beginSwitch(switchExpression, switchType); |
| } else { |
| // Setup jumps |
| for (ClauseListNode* list = m_list1; list; list = list->getNext()) { |
| RefPtr<RegisterID> clauseVal = generator.newTemporary(); |
| generator.emitNode(clauseVal.get(), list->getClause()->expr()); |
| Ref<Label> clauseLabel = generator.newLabel(); |
| labelVector.append(clauseLabel); |
| generator.emitJumpIfTrue(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), clauseVal.get(), switchExpression), clauseLabel.get()); |
| } |
| |
| for (ClauseListNode* list = m_list2; list; list = list->getNext()) { |
| RefPtr<RegisterID> clauseVal = generator.newTemporary(); |
| generator.emitNode(clauseVal.get(), list->getClause()->expr()); |
| Ref<Label> clauseLabel = generator.newLabel(); |
| labelVector.append(clauseLabel); |
| generator.emitJumpIfTrue(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), clauseVal.get(), switchExpression), clauseLabel.get()); |
| } |
| generator.emitJump(defaultLabel.get()); |
| } |
| |
| size_t i = 0; |
| for (ClauseListNode* list = m_list1; list; list = list->getNext()) { |
| generator.emitLabel(labelVector[i++].get()); |
| list->getClause()->emitBytecode(generator, dst); |
| } |
| |
| if (m_defaultClause) { |
| generator.emitLabel(defaultLabel.get()); |
| m_defaultClause->emitBytecode(generator, dst); |
| } |
| |
| for (ClauseListNode* list = m_list2; list; list = list->getNext()) { |
| generator.emitLabel(labelVector[i++].get()); |
| list->getClause()->emitBytecode(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, literalVector.data(), defaultLabel.get(), min_num, max_num); |
| } |
| } |
| |
| // ------------------------------ SwitchNode ----------------------------------- |
| |
| void SwitchNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (generator.shouldBeConcernedWithCompletionValue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::Switch); |
| |
| RefPtr<RegisterID> r0 = generator.emitNode(m_expr); |
| |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::LetConstScope, BytecodeGenerator::TDZCheckOptimization::DoNotOptimize, BytecodeGenerator::NestedScopeType::IsNested); |
| m_block->emitBytecodeForBlock(generator, r0.get(), dst); |
| generator.popLexicalScope(this); |
| |
| generator.emitLabel(scope->breakTarget()); |
| generator.emitProfileControlFlow(endOffset()); |
| } |
| |
| // ------------------------------ LabelNode ------------------------------------ |
| |
| void LabelNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(!generator.breakTarget(m_name)); |
| |
| Ref<LabelScope> scope = generator.newLabelScope(LabelScope::NamedLabel, &m_name); |
| generator.emitNodeInTailPosition(dst, m_statement); |
| |
| generator.emitLabel(scope->breakTarget()); |
| } |
| |
| // ------------------------------ ThrowNode ------------------------------------ |
| |
| void ThrowNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (dst == generator.ignoredResult()) |
| dst = nullptr; |
| RefPtr<RegisterID> expr = generator.emitNode(m_expr); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitThrow(expr.get()); |
| |
| generator.emitProfileControlFlow(endOffset()); |
| } |
| |
| // ------------------------------ TryNode -------------------------------------- |
| |
| void TryNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| // NOTE: The catch and finally blocks must be labeled explicitly, so the |
| // optimizer knows they may be jumped to from anywhere. |
| ASSERT(m_catchBlock || m_finallyBlock); |
| |
| RefPtr<RegisterID> tryCatchDst = dst; |
| if (generator.shouldBeConcernedWithCompletionValue()) { |
| if (m_finallyBlock) |
| tryCatchDst = generator.newTemporary(); |
| |
| if (m_finallyBlock || m_tryBlock->hasEarlyBreakOrContinue()) |
| generator.emitLoad(tryCatchDst.get(), jsUndefined()); |
| } |
| |
| RefPtr<Label> catchLabel; |
| RefPtr<Label> catchEndLabel; |
| RefPtr<Label> finallyLabel; |
| RefPtr<Label> finallyEndLabel; |
| std::optional<FinallyContext> finallyContext; |
| |
| if (m_finallyBlock) { |
| finallyLabel = generator.newLabel(); |
| finallyEndLabel = generator.newLabel(); |
| |
| finallyContext.emplace(generator, *finallyLabel); |
| generator.pushFinallyControlFlowScope(finallyContext.value()); |
| } |
| if (m_catchBlock) { |
| catchLabel = generator.newLabel(); |
| catchEndLabel = generator.newLabel(); |
| } |
| |
| Ref<Label> tryLabel = generator.newEmittedLabel(); |
| Label& tryHandlerLabel = m_catchBlock ? *catchLabel : *finallyLabel; |
| HandlerType tryHandlerType = m_catchBlock ? HandlerType::Catch : HandlerType::Finally; |
| TryData* tryData = generator.pushTry(tryLabel.get(), tryHandlerLabel, tryHandlerType); |
| TryData* finallyTryData = nullptr; |
| if (!m_catchBlock && m_finallyBlock) |
| finallyTryData = tryData; |
| |
| generator.emitNode(tryCatchDst.get(), m_tryBlock); |
| |
| if (m_finallyBlock) |
| generator.emitJump(*finallyLabel); |
| else |
| generator.emitJump(*catchEndLabel); |
| |
| Ref<Label> tryEndLabel = generator.newEmittedLabel(); |
| generator.popTry(tryData, tryEndLabel.get()); |
| |
| if (m_catchBlock) { |
| // Uncaught exception path: the catch block. |
| generator.emitLabel(*catchLabel); |
| RefPtr<RegisterID> thrownValueRegister = generator.newTemporary(); |
| RegisterID* completionTypeRegister = m_finallyBlock ? finallyContext->completionTypeRegister() : nullptr; |
| generator.emitOutOfLineCatchHandler(thrownValueRegister.get(), completionTypeRegister, tryData); |
| generator.restoreScopeRegister(); |
| |
| if (m_finallyBlock) { |
| // If the catch block throws an exception and we have a finally block, then the finally |
| // block should "catch" that exception. |
| finallyTryData = generator.pushTry(*catchLabel, *finallyLabel, HandlerType::Finally); |
| } |
| |
| if (m_catchPattern) { |
| generator.emitPushCatchScope(m_lexicalVariables); |
| m_catchPattern->bindValue(generator, thrownValueRegister.get()); |
| } |
| |
| generator.emitProfileControlFlow(m_tryBlock->endOffset() + 1); |
| if (m_finallyBlock) |
| generator.emitNode(tryCatchDst.get(), m_catchBlock); |
| else |
| generator.emitNodeInTailPosition(tryCatchDst.get(), m_catchBlock); |
| generator.emitLoad(thrownValueRegister.get(), jsUndefined()); |
| |
| if (m_catchPattern) |
| generator.emitPopCatchScope(m_lexicalVariables); |
| |
| if (m_finallyBlock) { |
| generator.emitLoad(finallyContext->completionTypeRegister(), CompletionType::Normal); |
| generator.emitJump(*finallyLabel); |
| generator.popTry(finallyTryData, *finallyLabel); |
| } |
| |
| generator.emitLabel(*catchEndLabel); |
| generator.emitProfileControlFlow(m_catchBlock->endOffset() + 1); |
| } |
| |
| if (m_finallyBlock) { |
| generator.popFinallyControlFlowScope(); |
| |
| // Entry to the finally block for CompletionType::Throw to be generated later. |
| generator.emitOutOfLineFinallyHandler(finallyContext->completionValueRegister(), finallyContext->completionTypeRegister(), finallyTryData); |
| |
| // Entry to the finally block for CompletionTypes other than Throw. |
| generator.emitLabel(*finallyLabel); |
| generator.restoreScopeRegister(); |
| |
| int finallyStartOffset = m_catchBlock ? m_catchBlock->endOffset() + 1 : m_tryBlock->endOffset() + 1; |
| |
| // The completion value of a finally block is ignored *just* when it is a normal completion. |
| if (generator.shouldBeConcernedWithCompletionValue()) { |
| ASSERT(dst != tryCatchDst.get()); |
| if (m_finallyBlock->hasEarlyBreakOrContinue()) |
| generator.emitLoad(dst, jsUndefined()); |
| |
| generator.emitProfileControlFlow(finallyStartOffset); |
| generator.emitNodeInTailPosition(dst, m_finallyBlock); |
| |
| generator.move(dst, tryCatchDst.get()); |
| } else { |
| generator.emitProfileControlFlow(finallyStartOffset); |
| generator.emitNodeInTailPosition(m_finallyBlock); |
| } |
| |
| generator.emitFinallyCompletion(finallyContext.value(), *finallyEndLabel); |
| generator.emitLabel(*finallyEndLabel); |
| generator.emitProfileControlFlow(m_finallyBlock->endOffset() + 1); |
| } |
| } |
| |
| // ------------------------------ ScopeNode ----------------------------- |
| |
| inline void ScopeNode::emitStatementsBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!m_statements) |
| return; |
| m_statements->emitBytecode(generator, dst); |
| } |
| |
| static void emitProgramNodeBytecode(BytecodeGenerator& generator, ScopeNode& scopeNode) |
| { |
| generator.emitDebugHook(WillExecuteProgram, scopeNode.startLine(), scopeNode.startStartOffset(), scopeNode.startLineStartOffset()); |
| |
| RefPtr<RegisterID> dstRegister = generator.newTemporary(); |
| generator.emitLoad(dstRegister.get(), jsUndefined()); |
| generator.emitProfileControlFlow(scopeNode.startStartOffset()); |
| scopeNode.emitStatementsBytecode(generator, dstRegister.get()); |
| |
| generator.emitDebugHook(DidExecuteProgram, scopeNode.lastLine(), scopeNode.startOffset(), scopeNode.lineStartOffset()); |
| generator.emitEnd(dstRegister.get()); |
| } |
| |
| // ------------------------------ ProgramNode ----------------------------- |
| |
| void ProgramNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| emitProgramNodeBytecode(generator, *this); |
| } |
| |
| // ------------------------------ ModuleProgramNode -------------------- |
| |
| void ModuleProgramNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| emitProgramNodeBytecode(generator, *this); |
| } |
| |
| // ------------------------------ EvalNode ----------------------------- |
| |
| void EvalNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| generator.emitDebugHook(WillExecuteProgram, startLine(), startStartOffset(), startLineStartOffset()); |
| |
| RefPtr<RegisterID> dstRegister = generator.newTemporary(); |
| generator.emitLoad(dstRegister.get(), jsUndefined()); |
| emitStatementsBytecode(generator, dstRegister.get()); |
| |
| generator.emitDebugHook(DidExecuteProgram, lastLine(), startOffset(), lineStartOffset()); |
| generator.emitEnd(dstRegister.get()); |
| } |
| |
| // ------------------------------ FunctionNode ----------------------------- |
| |
| void FunctionNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| if (generator.shouldEmitTypeProfilerHooks()) { |
| // If the parameter list is non simple one, it is handled in bindValue's code. |
| if (m_parameters->isSimpleParameterList()) { |
| for (size_t i = 0; i < m_parameters->size(); i++) { |
| BindingNode* bindingNode = static_cast<BindingNode*>(m_parameters->at(i).first); |
| RegisterID reg(CallFrame::argumentOffset(i)); |
| generator.emitProfileType(®, ProfileTypeBytecodeFunctionArgument, bindingNode->divotStart(), bindingNode->divotEnd()); |
| } |
| } |
| } |
| |
| generator.emitProfileControlFlow(startStartOffset()); |
| generator.emitDebugHook(DidEnterCallFrame, startLine(), startStartOffset(), startLineStartOffset()); |
| |
| switch (generator.parseMode()) { |
| case SourceParseMode::GeneratorWrapperFunctionMode: |
| case SourceParseMode::GeneratorWrapperMethodMode: |
| case SourceParseMode::AsyncGeneratorWrapperMethodMode: |
| case SourceParseMode::AsyncGeneratorWrapperFunctionMode: { |
| StatementNode* singleStatement = this->singleStatement(); |
| ASSERT(singleStatement->isExprStatement()); |
| ExprStatementNode* exprStatement = static_cast<ExprStatementNode*>(singleStatement); |
| ExpressionNode* expr = exprStatement->expr(); |
| ASSERT(expr->isFuncExprNode()); |
| FuncExprNode* funcExpr = static_cast<FuncExprNode*>(expr); |
| |
| RefPtr<RegisterID> next = generator.newTemporary(); |
| generator.emitNode(next.get(), funcExpr); |
| |
| if (generator.superBinding() == SuperBinding::Needed) { |
| RefPtr<RegisterID> homeObject = emitHomeObjectForCallee(generator); |
| emitPutHomeObject(generator, next.get(), homeObject.get()); |
| } |
| |
| if (isGeneratorWrapperParseMode(generator.parseMode())) |
| generator.emitPutGeneratorFields(next.get()); |
| else { |
| ASSERT(isAsyncGeneratorWrapperParseMode(generator.parseMode())); |
| generator.emitPutAsyncGeneratorFields(next.get()); |
| } |
| |
| ASSERT(startOffset() >= lineStartOffset()); |
| generator.emitDebugHook(WillLeaveCallFrame, lastLine(), startOffset(), lineStartOffset()); |
| generator.emitReturn(generator.generatorRegister()); |
| break; |
| } |
| |
| case SourceParseMode::AsyncFunctionMode: |
| case SourceParseMode::AsyncMethodMode: |
| case SourceParseMode::AsyncArrowFunctionMode: { |
| StatementNode* singleStatement = this->singleStatement(); |
| ASSERT(singleStatement->isExprStatement()); |
| ExprStatementNode* exprStatement = static_cast<ExprStatementNode*>(singleStatement); |
| ExpressionNode* expr = exprStatement->expr(); |
| ASSERT(expr->isFuncExprNode()); |
| FuncExprNode* funcExpr = static_cast<FuncExprNode*>(expr); |
| |
| RefPtr<RegisterID> next = generator.newTemporary(); |
| generator.emitNode(next.get(), funcExpr); |
| |
| if (generator.superBinding() == SuperBinding::Needed || (generator.parseMode() == SourceParseMode::AsyncArrowFunctionMode && generator.isSuperUsedInInnerArrowFunction())) { |
| RefPtr<RegisterID> homeObject = emitHomeObjectForCallee(generator); |
| emitPutHomeObject(generator, next.get(), homeObject.get()); |
| } |
| |
| if (generator.parseMode() == SourceParseMode::AsyncArrowFunctionMode && generator.isThisUsedInInnerArrowFunction()) |
| generator.emitLoadThisFromArrowFunctionLexicalEnvironment(); |
| |
| generator.emitPutGeneratorFields(next.get()); |
| |
| ASSERT(startOffset() >= lineStartOffset()); |
| generator.emitDebugHook(WillLeaveCallFrame, lastLine(), startOffset(), lineStartOffset()); |
| |
| // load and call @asyncFunctionResume |
| RefPtr<RegisterID> asyncFunctionResume = generator.moveLinkTimeConstant(nullptr, LinkTimeConstant::asyncFunctionResume); |
| |
| CallArguments args(generator, nullptr, 4); |
| unsigned argumentCount = 0; |
| generator.emitLoad(args.thisRegister(), jsUndefined()); |
| generator.move(args.argumentRegister(argumentCount++), generator.generatorRegister()); |
| generator.move(args.argumentRegister(argumentCount++), generator.promiseRegister()); |
| generator.emitLoad(args.argumentRegister(argumentCount++), jsUndefined()); |
| generator.emitLoad(args.argumentRegister(argumentCount++), JSGenerator::ResumeMode::NormalMode); |
| // JSTextPosition(int _line, int _offset, int _lineStartOffset) |
| JSTextPosition divot(firstLine(), startOffset(), lineStartOffset()); |
| |
| RefPtr<RegisterID> result = generator.newTemporary(); |
| generator.emitCallInTailPosition(result.get(), asyncFunctionResume.get(), NoExpectedFunction, args, divot, divot, divot, DebuggableCall::No); |
| generator.emitReturn(result.get()); |
| break; |
| } |
| |
| case SourceParseMode::AsyncGeneratorBodyMode: |
| case SourceParseMode::AsyncArrowFunctionBodyMode: |
| case SourceParseMode::AsyncFunctionBodyMode: |
| case SourceParseMode::GeneratorBodyMode: { |
| Ref<Label> generatorBodyLabel = generator.newLabel(); |
| { |
| generator.emitJumpIfTrue(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), generator.generatorResumeModeRegister(), generator.emitLoad(nullptr, JSGenerator::ResumeMode::NormalMode)), generatorBodyLabel.get()); |
| |
| Ref<Label> throwLabel = generator.newLabel(); |
| generator.emitJumpIfTrue(generator.emitEqualityOp<OpStricteq>(generator.newTemporary(), generator.generatorResumeModeRegister(), generator.emitLoad(nullptr, JSGenerator::ResumeMode::ThrowMode)), throwLabel.get()); |
| |
| generator.emitReturn(generator.generatorValueRegister()); |
| |
| generator.emitLabel(throwLabel.get()); |
| generator.emitThrow(generator.generatorValueRegister()); |
| } |
| |
| generator.emitLabel(generatorBodyLabel.get()); |
| |
| emitStatementsBytecode(generator, generator.ignoredResult()); |
| |
| Ref<Label> done = generator.newLabel(); |
| generator.emitLabel(done.get()); |
| generator.emitReturn(generator.emitLoad(nullptr, jsUndefined())); |
| break; |
| } |
| |
| default: { |
| emitStatementsBytecode(generator, generator.ignoredResult()); |
| |
| StatementNode* singleStatement = this->singleStatement(); |
| ReturnNode* returnNode = nullptr; |
| |
| // Check for a return statement at the end of a function composed of a single block. |
| if (singleStatement && singleStatement->isBlock()) { |
| StatementNode* lastStatementInBlock = static_cast<BlockNode*>(singleStatement)->lastStatement(); |
| if (lastStatementInBlock && lastStatementInBlock->isReturnNode()) |
| returnNode = static_cast<ReturnNode*>(lastStatementInBlock); |
| } |
| |
| // If there is no return we must automatically insert one. |
| if (!returnNode) { |
| if (generator.constructorKind() == ConstructorKind::Extends && generator.needsToUpdateArrowFunctionContext() && generator.isSuperCallUsedInInnerArrowFunction()) |
| generator.emitLoadThisFromArrowFunctionLexicalEnvironment(); // Arrow function can invoke 'super' in constructor and before leave constructor we need load 'this' from lexical arrow function environment |
| |
| RegisterID* r0 = nullptr; |
| if (generator.isConstructor() && generator.constructorKind() != ConstructorKind::Naked) |
| r0 = generator.thisRegister(); |
| else |
| r0 = generator.emitLoad(nullptr, jsUndefined()); |
| generator.emitProfileType(r0, ProfileTypeBytecodeFunctionReturnStatement); // Do not emit expression info for this profile because it's not in the user's source code. |
| ASSERT(startOffset() >= lineStartOffset()); |
| generator.emitWillLeaveCallFrameDebugHook(); |
| generator.emitReturn(r0); |
| return; |
| } |
| break; |
| } |
| } |
| } |
| |
| // ------------------------------ FuncDeclNode --------------------------------- |
| |
| void FuncDeclNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| generator.hoistSloppyModeFunctionIfNecessary(metadata()->ident()); |
| } |
| |
| // ------------------------------ FuncExprNode --------------------------------- |
| |
| RegisterID* FuncExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| return generator.emitNewFunctionExpression(generator.finalDestination(dst), this); |
| } |
| |
| // ------------------------------ ArrowFuncExprNode --------------------------------- |
| |
| RegisterID* ArrowFuncExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| return generator.emitNewArrowFunctionExpression(generator.finalDestination(dst), this); |
| } |
| |
| // ------------------------------ MethodDefinitionNode --------------------------------- |
| |
| RegisterID* MethodDefinitionNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| return generator.emitNewMethodDefinition(generator.finalDestination(dst), this); |
| } |
| |
| // ------------------------------ YieldExprNode -------------------------------- |
| |
| RegisterID* YieldExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (!delegate()) { |
| RefPtr<RegisterID> arg = nullptr; |
| if (argument()) { |
| arg = generator.newTemporary(); |
| generator.emitNode(arg.get(), argument()); |
| } else |
| arg = generator.emitLoad(nullptr, jsUndefined()); |
| RefPtr<RegisterID> value = generator.emitYield(arg.get(), JSAsyncGenerator::AsyncGeneratorSuspendReason::Yield); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.move(generator.finalDestination(dst), value.get()); |
| } |
| RefPtr<RegisterID> arg = generator.newTemporary(); |
| generator.emitNode(arg.get(), argument()); |
| RefPtr<RegisterID> value = generator.emitDelegateYield(arg.get(), this); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.move(generator.finalDestination(dst), value.get()); |
| } |
| |
| // ------------------------------ AwaitExprNode -------------------------------- |
| |
| RegisterID* AwaitExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> arg = generator.newTemporary(); |
| generator.emitNode(arg.get(), argument()); |
| RefPtr<RegisterID> value = generator.emitYield(arg.get(), JSAsyncGenerator::AsyncGeneratorSuspendReason::Await); |
| if (dst == generator.ignoredResult()) |
| return nullptr; |
| return generator.move(generator.finalDestination(dst), value.get()); |
| } |
| |
| // ------------------------------ DefineFieldNode --------------------------------- |
| |
| void DefineFieldNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| RefPtr<RegisterID> value = generator.newTemporary(); |
| bool shouldSetFunctionName = false; |
| |
| if (!m_assign) |
| generator.emitLoad(value.get(), jsUndefined()); |
| else { |
| generator.emitNode(value.get(), m_assign); |
| shouldSetFunctionName = generator.shouldSetFunctionName(m_assign); |
| if (m_ident && shouldSetFunctionName && m_type != DefineFieldNode::Type::ComputedName) |
| generator.emitSetFunctionName(value.get(), *m_ident); |
| } |
| |
| switch (m_type) { |
| case DefineFieldNode::Type::Name: { |
| StrictModeScope strictModeScope(generator); |
| if (auto index = parseIndex(*m_ident)) { |
| RefPtr<RegisterID> propertyName = generator.emitLoad(nullptr, jsNumber(index.value())); |
| generator.emitDirectPutByVal(generator.thisRegister(), propertyName.get(), value.get()); |
| } else |
| generator.emitDirectPutById(generator.thisRegister(), *m_ident, value.get()); |
| break; |
| } |
| case DefineFieldNode::Type::PrivateName: { |
| Variable var = generator.variable(*m_ident); |
| ASSERT_WITH_MESSAGE(!var.local(), "Private Field names must be stored in captured variables"); |
| |
| generator.emitExpressionInfo(position(), position(), position() + m_ident->length()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, DoNotThrowIfNotFound); |
| generator.emitDefinePrivateField(generator.thisRegister(), privateName.get(), value.get()); |
| break; |
| } |
| case DefineFieldNode::Type::ComputedName: { |
| // For ComputedNames, the expression has already been evaluated earlier during evaluation of a ClassExprNode. |
| // Here, `m_ident` refers to private symbol ID in a class lexical scope, containing the value already converted to an Expression. |
| Variable var = generator.variable(*m_ident); |
| ASSERT_WITH_MESSAGE(!var.local(), "Computed names must be stored in captured variables"); |
| |
| generator.emitExpressionInfo(position(), position(), position() + 1); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| RefPtr<RegisterID> privateName = generator.newTemporary(); |
| generator.emitGetFromScope(privateName.get(), scope.get(), var, ThrowIfNotFound); |
| if (shouldSetFunctionName) |
| generator.emitSetFunctionName(value.get(), privateName.get()); |
| generator.emitProfileType(privateName.get(), var, m_position, m_position + m_ident->length()); |
| { |
| StrictModeScope strictModeScope(generator); |
| generator.emitDirectPutByVal(generator.thisRegister(), privateName.get(), value.get()); |
| } |
| break; |
| } |
| } |
| } |
| |
| // ------------------------------ ClassDeclNode --------------------------------- |
| |
| void ClassDeclNode::emitBytecode(BytecodeGenerator& generator, RegisterID*) |
| { |
| generator.emitNode(m_classDeclaration); |
| } |
| |
| // ------------------------------ ClassExprNode --------------------------------- |
| |
| RegisterID* ClassExprNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| StrictModeScope strictModeScope(generator); |
| |
| if (!m_name.isNull()) |
| generator.pushClassHeadLexicalScope(m_classHeadEnvironment); |
| |
| // Class heritage must be evaluated outside of private fields access. |
| RefPtr<RegisterID> superclass; |
| if (m_classHeritage) { |
| superclass = generator.newTemporary(); |
| generator.emitNode(superclass.get(), m_classHeritage); |
| } |
| |
| if (m_needsLexicalScope) |
| generator.pushLexicalScope(this, BytecodeGenerator::ScopeType::ClassScope, BytecodeGenerator::TDZCheckOptimization::Optimize, BytecodeGenerator::NestedScopeType::IsNested); |
| |
| bool hasPrivateNames = !!m_lexicalVariables.privateNamesSize(); |
| bool shouldEmitPrivateBrand = m_lexicalVariables.hasInstancePrivateMethodOrAccessor(); |
| bool shouldInstallBrandOnConstructor = m_lexicalVariables.hasStaticPrivateMethodOrAccessor(); |
| if (hasPrivateNames) |
| generator.pushPrivateAccessNames(m_lexicalVariables.privateNameEnvironment()); |
| if (shouldEmitPrivateBrand) |
| generator.emitCreatePrivateBrand(m_position, m_position, m_position); |
| |
| RefPtr<RegisterID> constructor = generator.tempDestination(dst); |
| bool needsHomeObject = false; |
| |
| auto needsClassFieldInitializer = this->hasInstanceFields() ? NeedsClassFieldInitializer::Yes : NeedsClassFieldInitializer::No; |
| auto privateBrandRequirement = shouldEmitPrivateBrand ? PrivateBrandRequirement::Needed : PrivateBrandRequirement::None; |
| if (m_constructorExpression) { |
| ASSERT(m_constructorExpression->isFuncExprNode()); |
| FunctionMetadataNode* metadata = static_cast<FuncExprNode*>(m_constructorExpression)->metadata(); |
| metadata->setEcmaName(ecmaName()); |
| metadata->setClassSource(m_classSource); |
| metadata->setNeedsClassFieldInitializer(needsClassFieldInitializer == NeedsClassFieldInitializer::Yes); |
| metadata->setPrivateBrandRequirement(privateBrandRequirement); |
| constructor = generator.emitNode(constructor.get(), m_constructorExpression); |
| needsHomeObject = m_classHeritage || metadata->superBinding() == SuperBinding::Needed; |
| } else |
| constructor = generator.emitNewDefaultConstructor(constructor.get(), m_classHeritage ? ConstructorKind::Extends : ConstructorKind::Base, m_name, ecmaName(), m_classSource, needsClassFieldInitializer, privateBrandRequirement); |
| |
| const auto& propertyNames = generator.propertyNames(); |
| RefPtr<RegisterID> prototype = generator.emitNewObject(generator.newTemporary()); |
| |
| if (superclass) { |
| RefPtr<RegisterID> protoParent = generator.newTemporary(); |
| generator.emitLoad(protoParent.get(), jsNull()); |
| |
| Ref<Label> superclassIsNullLabel = generator.newLabel(); |
| generator.emitJumpIfTrue(generator.emitIsNull(generator.newTemporary(), superclass.get()), superclassIsNullLabel.get()); |
| |
| Ref<Label> superclassIsConstructorLabel = generator.newLabel(); |
| generator.emitJumpIfTrue(generator.emitIsConstructor(generator.newTemporary(), superclass.get()), superclassIsConstructorLabel.get()); |
| generator.emitExpressionInfo(divot(), divotStart(), divotEnd()); |
| generator.emitThrowTypeError("The superclass is not a constructor."_s); |
| generator.emitLabel(superclassIsConstructorLabel.get()); |
| generator.emitGetById(protoParent.get(), superclass.get(), generator.propertyNames().prototype); |
| |
| generator.emitDirectSetPrototypeOf<InvalidPrototypeMode::Throw>(constructor.get(), superclass.get(), m_position, m_position, m_position); // never actually throws |
| generator.emitLabel(superclassIsNullLabel.get()); |
| generator.emitDirectSetPrototypeOf<InvalidPrototypeMode::Throw>(prototype.get(), protoParent.get(), divot(), divotStart(), divotEnd()); |
| } |
| |
| if (needsHomeObject) |
| emitPutHomeObject(generator, constructor.get(), prototype.get()); |
| |
| RefPtr<RegisterID> constructorNameRegister = generator.emitLoad(nullptr, propertyNames.constructor); |
| generator.emitCallDefineProperty(prototype.get(), constructorNameRegister.get(), constructor.get(), nullptr, nullptr, |
| BytecodeGenerator::PropertyConfigurable | BytecodeGenerator::PropertyWritable, m_position); |
| |
| RefPtr<RegisterID> prototypeNameRegister = generator.emitLoad(nullptr, propertyNames.prototype); |
| generator.emitCallDefineProperty(constructor.get(), prototypeNameRegister.get(), prototype.get(), nullptr, nullptr, 0, m_position); |
| |
| Vector<JSTextPosition> staticFieldLocations; |
| if (m_classElements) { |
| m_classElements->emitDeclarePrivateFieldNames(generator, generator.scopeRegister()); |
| |
| Vector<JSTextPosition> instanceFieldLocations; |
| generator.emitDefineClassElements(m_classElements, constructor.get(), prototype.get(), instanceFieldLocations, staticFieldLocations); |
| if (!instanceFieldLocations.isEmpty()) { |
| RefPtr<RegisterID> instanceFieldInitializer = generator.emitNewClassFieldInitializerFunction(generator.newTemporary(), WTFMove(instanceFieldLocations), m_classHeritage); |
| |
| // FIXME: Skip this if the initializer function isn't going to need a home object (no eval or super properties) |
| // https://bugs.webkit.org/show_bug.cgi?id=196867 |
| emitPutHomeObject(generator, instanceFieldInitializer.get(), prototype.get()); |
| |
| generator.emitDirectPutById(constructor.get(), generator.propertyNames().builtinNames().instanceFieldInitializerPrivateName(), instanceFieldInitializer.get()); |
| } |
| } |
| |
| if (!m_name.isNull()) { |
| Variable classNameVar = generator.variable(m_name); |
| RELEASE_ASSERT(classNameVar.isResolved()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, classNameVar); |
| generator.emitPutToScope(scope.get(), classNameVar, constructor.get(), ThrowIfNotFound, InitializationMode::Initialization); |
| } |
| |
| if (shouldInstallBrandOnConstructor) |
| generator.emitInstallPrivateClassBrand(constructor.get()); |
| |
| if (!staticFieldLocations.isEmpty()) { |
| RefPtr<RegisterID> staticFieldInitializer = generator.emitNewClassFieldInitializerFunction(generator.newTemporary(), WTFMove(staticFieldLocations), m_classHeritage); |
| // FIXME: Skip this if the initializer function isn't going to need a home object (no eval or super properties) |
| // https://bugs.webkit.org/show_bug.cgi?id=196867 |
| emitPutHomeObject(generator, staticFieldInitializer.get(), constructor.get()); |
| |
| CallArguments args(generator, nullptr); |
| generator.move(args.thisRegister(), constructor.get()); |
| generator.emitCall(generator.newTemporary(), staticFieldInitializer.get(), NoExpectedFunction, args, position(), position(), position(), DebuggableCall::No); |
| } |
| |
| if (hasPrivateNames) |
| generator.popPrivateAccessNames(); |
| |
| if (m_needsLexicalScope) |
| generator.popLexicalScope(this); |
| |
| if (!m_name.isNull()) |
| generator.popClassHeadLexicalScope(m_classHeadEnvironment); |
| |
| return generator.move(generator.finalDestination(dst, constructor.get()), constructor.get()); |
| } |
| |
| // ------------------------------ ImportDeclarationNode ----------------------- |
| |
| void ImportDeclarationNode::emitBytecode(BytecodeGenerator&, RegisterID*) |
| { |
| // Do nothing at runtime. |
| } |
| |
| // ------------------------------ ExportAllDeclarationNode -------------------- |
| |
| void ExportAllDeclarationNode::emitBytecode(BytecodeGenerator&, RegisterID*) |
| { |
| // Do nothing at runtime. |
| } |
| |
| // ------------------------------ ExportDefaultDeclarationNode ---------------- |
| |
| void ExportDefaultDeclarationNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_declaration); |
| generator.emitNode(dst, m_declaration); |
| } |
| |
| // ------------------------------ ExportLocalDeclarationNode ------------------ |
| |
| void ExportLocalDeclarationNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| ASSERT(m_declaration); |
| generator.emitNode(dst, m_declaration); |
| } |
| |
| // ------------------------------ ExportNamedDeclarationNode ------------------ |
| |
| void ExportNamedDeclarationNode::emitBytecode(BytecodeGenerator&, RegisterID*) |
| { |
| // Do nothing at runtime. |
| } |
| |
| // ------------------------------ DestructuringAssignmentNode ----------------- |
| RegisterID* DestructuringAssignmentNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| if (RegisterID* result = m_bindings->emitDirectBinding(generator, dst, m_initializer)) |
| return result; |
| RefPtr<RegisterID> initializer = generator.tempDestination(dst); |
| generator.emitNode(initializer.get(), m_initializer); |
| m_bindings->bindValue(generator, initializer.get()); |
| return generator.move(dst, initializer.get()); |
| } |
| |
| static void assignDefaultValueIfUndefined(BytecodeGenerator& generator, RegisterID* maybeUndefined, ExpressionNode* defaultValue) |
| { |
| ASSERT(defaultValue); |
| Ref<Label> isNotUndefined = generator.newLabel(); |
| generator.emitJumpIfFalse(generator.emitIsUndefined(generator.newTemporary(), maybeUndefined), isNotUndefined.get()); |
| generator.emitNode(maybeUndefined, defaultValue); |
| generator.emitLabel(isNotUndefined.get()); |
| } |
| |
| void ArrayPatternNode::bindValue(BytecodeGenerator& generator, RegisterID* rhs) const |
| { |
| RefPtr<RegisterID> iterator = generator.newTemporary(); |
| { |
| generator.emitGetById(iterator.get(), rhs, generator.propertyNames().iteratorSymbol); |
| CallArguments args(generator, nullptr); |
| generator.move(args.thisRegister(), rhs); |
| generator.emitCall(iterator.get(), iterator.get(), NoExpectedFunction, args, divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| } |
| RefPtr<RegisterID> nextMethod = generator.emitGetById(generator.newTemporary(), iterator.get(), generator.propertyNames().next); |
| |
| if (m_targetPatterns.isEmpty()) { |
| generator.emitIteratorGenericClose(iterator.get(), this); |
| return; |
| } |
| |
| RefPtr<RegisterID> done; |
| for (auto& target : m_targetPatterns) { |
| switch (target.bindingType) { |
| case BindingType::Elision: |
| case BindingType::Element: { |
| Ref<Label> iterationSkipped = generator.newLabel(); |
| if (!done) |
| done = generator.newTemporary(); |
| else |
| generator.emitJumpIfTrue(done.get(), iterationSkipped.get()); |
| |
| RefPtr<RegisterID> value = generator.newTemporary(); |
| generator.emitIteratorGenericNext(value.get(), nextMethod.get(), iterator.get(), this); |
| generator.emitGetById(done.get(), value.get(), generator.propertyNames().done); |
| generator.emitJumpIfTrue(done.get(), iterationSkipped.get()); |
| generator.emitGetById(value.get(), value.get(), generator.propertyNames().value); |
| |
| { |
| Ref<Label> valueIsSet = generator.newLabel(); |
| generator.emitJump(valueIsSet.get()); |
| generator.emitLabel(iterationSkipped.get()); |
| generator.emitLoad(value.get(), jsUndefined()); |
| generator.emitLabel(valueIsSet.get()); |
| } |
| |
| if (target.bindingType == BindingType::Element) { |
| if (target.defaultValue) |
| assignDefaultValueIfUndefined(generator, value.get(), target.defaultValue); |
| target.pattern->bindValue(generator, value.get()); |
| } |
| break; |
| } |
| |
| case BindingType::RestElement: { |
| RefPtr<RegisterID> array = generator.emitNewArray(generator.newTemporary(), nullptr, 0, ArrayWithUndecided); |
| |
| Ref<Label> iterationDone = generator.newLabel(); |
| if (!done) |
| done = generator.newTemporary(); |
| else |
| generator.emitJumpIfTrue(done.get(), iterationDone.get()); |
| |
| RefPtr<RegisterID> index = generator.newTemporary(); |
| generator.emitLoad(index.get(), jsNumber(0)); |
| Ref<Label> loopStart = generator.newLabel(); |
| generator.emitLabel(loopStart.get()); |
| |
| RefPtr<RegisterID> value = generator.newTemporary(); |
| generator.emitIteratorGenericNext(value.get(), nextMethod.get(), iterator.get(), this); |
| generator.emitGetById(done.get(), value.get(), generator.propertyNames().done); |
| generator.emitJumpIfTrue(done.get(), iterationDone.get()); |
| generator.emitGetById(value.get(), value.get(), generator.propertyNames().value); |
| |
| generator.emitDirectPutByVal(array.get(), index.get(), value.get()); |
| generator.emitInc(index.get()); |
| generator.emitJump(loopStart.get()); |
| |
| generator.emitLabel(iterationDone.get()); |
| target.pattern->bindValue(generator, array.get()); |
| break; |
| } |
| } |
| } |
| |
| Ref<Label> iteratorClosed = generator.newLabel(); |
| generator.emitJumpIfTrue(done.get(), iteratorClosed.get()); |
| generator.emitIteratorGenericClose(iterator.get(), this); |
| generator.emitLabel(iteratorClosed.get()); |
| } |
| |
| RegisterID* ArrayPatternNode::emitDirectBinding(BytecodeGenerator& generator, RegisterID* dst, ExpressionNode* rhs) |
| { |
| if (!rhs->isSimpleArray()) |
| return nullptr; |
| |
| if (m_targetPatterns.findIf([&] (auto& target) { return target.bindingType == BindingType::RestElement; }) != notFound) |
| return nullptr; |
| |
| ElementNode* elementNodes = static_cast<ArrayNode*>(rhs)->elements(); |
| Vector<ExpressionNode*> elements; |
| for (; elementNodes; elementNodes = elementNodes->next()) { |
| ExpressionNode* value = elementNodes->value(); |
| ASSERT(!value->isSpreadExpression()); |
| elements.append(value); |
| } |
| |
| RefPtr<RegisterID> resultRegister; |
| if (dst != generator.ignoredResult()) |
| resultRegister = generator.emitNewArray(generator.newTemporary(), nullptr, 0, ArrayWithUndecided); |
| if (m_targetPatterns.size() != elements.size()) |
| return nullptr; |
| Vector<RefPtr<RegisterID>> registers; |
| registers.reserveInitialCapacity(m_targetPatterns.size()); |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| registers.uncheckedAppend(generator.newTemporary()); |
| generator.emitNode(registers.last().get(), elements[i]); |
| if (m_targetPatterns[i].defaultValue) |
| assignDefaultValueIfUndefined(generator, registers.last().get(), m_targetPatterns[i].defaultValue); |
| if (resultRegister) { |
| RefPtr<RegisterID> index = generator.emitLoad(nullptr, jsNumber(i)); |
| generator.emitDirectPutByVal(resultRegister.get(), index.get(), registers.last().get()); |
| } |
| } |
| |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| if (m_targetPatterns[i].pattern) |
| m_targetPatterns[i].pattern->bindValue(generator, registers[i].get()); |
| } |
| if (resultRegister) |
| return generator.move(generator.finalDestination(dst, resultRegister.get()), resultRegister.get()); |
| return generator.emitLoad(generator.finalDestination(dst), jsUndefined()); |
| } |
| |
| void ArrayPatternNode::toString(StringBuilder& builder) const |
| { |
| builder.append('['); |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| const auto& target = m_targetPatterns[i]; |
| |
| switch (target.bindingType) { |
| case BindingType::Elision: |
| builder.append(','); |
| break; |
| |
| case BindingType::Element: |
| target.pattern->toString(builder); |
| if (i < m_targetPatterns.size() - 1) |
| builder.append(','); |
| break; |
| |
| case BindingType::RestElement: |
| builder.append("..."); |
| target.pattern->toString(builder); |
| break; |
| } |
| } |
| builder.append(']'); |
| } |
| |
| void ArrayPatternNode::collectBoundIdentifiers(Vector<Identifier>& identifiers) const |
| { |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| if (DestructuringPatternNode* node = m_targetPatterns[i].pattern) |
| node->collectBoundIdentifiers(identifiers); |
| } |
| } |
| |
| void ObjectPatternNode::toString(StringBuilder& builder) const |
| { |
| builder.append('{'); |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| if (m_targetPatterns[i].wasString) |
| builder.appendQuotedJSONString(m_targetPatterns[i].propertyName.string()); |
| else |
| builder.append(m_targetPatterns[i].propertyName.string()); |
| builder.append(':'); |
| m_targetPatterns[i].pattern->toString(builder); |
| if (i < m_targetPatterns.size() - 1) |
| builder.append(','); |
| } |
| builder.append('}'); |
| } |
| |
| void ObjectPatternNode::bindValue(BytecodeGenerator& generator, RegisterID* rhs) const |
| { |
| generator.emitRequireObjectCoercible(rhs, "Right side of assignment cannot be destructured"_s); |
| |
| BytecodeGenerator::PreservedTDZStack preservedTDZStack; |
| generator.preserveTDZStack(preservedTDZStack); |
| |
| { |
| RefPtr<RegisterID> newObject; |
| IdentifierSet excludedSet; |
| std::optional<CallArguments> args; |
| unsigned numberOfComputedProperties = 0; |
| unsigned indexInArguments = 2; |
| if (m_containsRestElement) { |
| if (m_containsComputedProperty) { |
| for (const auto& target : m_targetPatterns) { |
| if (target.bindingType == BindingType::Element) { |
| if (target.propertyExpression) |
| ++numberOfComputedProperties; |
| } |
| } |
| } |
| newObject = generator.newTemporary(); |
| args.emplace(generator, nullptr, indexInArguments + numberOfComputedProperties); |
| } |
| |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) { |
| const auto& target = m_targetPatterns[i]; |
| if (target.bindingType == BindingType::Element) { |
| // If the destructuring becomes get_by_id and mov, then we should store results directly to the local's binding. |
| // From |
| // get_by_id dst:loc10, base:loc9, property:0 |
| // mov dst:loc6, src:loc10 |
| // To |
| // get_by_id dst:loc6, base:loc9, property:0 |
| auto writableDirectBindingIfPossible = [&]() -> RegisterID* { |
| // The following pattern is possible. In that case, after setting |data| local variable, we need to store property name into the set. |
| // So, old property name |data| result must be kept before setting it into |data|. |
| // ({ [data]: data, ...obj } = object); |
| if (m_containsRestElement && m_containsComputedProperty && target.propertyExpression) |
| return nullptr; |
| // default value can include a reference to local variable. So filling value to a local variable can differ result. |
| // We give up fast path if default value includes non constant. |
| // For example, |
| // ({ data = data } = object); |
| if (target.defaultValue && !target.defaultValue->isConstant()) |
| return nullptr; |
| return target.pattern->writableDirectBindingIfPossible(generator); |
| }; |
| |
| auto finishDirectBindingAssignment = [&]() { |
| ASSERT(writableDirectBindingIfPossible()); |
| target.pattern->finishDirectBindingAssignment(generator); |
| }; |
| |
| RefPtr<RegisterID> temp; |
| RegisterID* directBinding = writableDirectBindingIfPossible(); |
| if (directBinding) |
| temp = directBinding; |
| else |
| temp = generator.newTemporary(); |
| |
| if (!target.propertyExpression) { |
| std::optional<uint32_t> optionalIndex = parseIndex(target.propertyName); |
| if (!optionalIndex) |
| generator.emitGetById(temp.get(), rhs, target.propertyName); |
| else { |
| RefPtr<RegisterID> propertyIndex = generator.emitLoad(nullptr, jsNumber(optionalIndex.value())); |
| generator.emitGetByVal(temp.get(), rhs, propertyIndex.get()); |
| } |
| if (m_containsRestElement) |
| excludedSet.add(target.propertyName.impl()); |
| } else { |
| RefPtr<RegisterID> propertyName; |
| if (m_containsRestElement) { |
| propertyName = generator.emitNodeForProperty(args->argumentRegister(indexInArguments++), target.propertyExpression); |
| // ToPropertyKey(Number | String) does not have side-effect. |
| // And @copyDataProperties performs ToPropertyKey internally. |
| // And for Number case, passing it to GetByVal is better for performance. |
| if (!target.propertyExpression->isNumber() || !target.propertyExpression->isString()) |
| propertyName = generator.emitToPropertyKey(propertyName.get(), propertyName.get()); |
| } else |
| propertyName = generator.emitNodeForProperty(target.propertyExpression); |
| generator.emitGetByVal(temp.get(), rhs, propertyName.get()); |
| } |
| |
| if (target.defaultValue) |
| assignDefaultValueIfUndefined(generator, temp.get(), target.defaultValue); |
| if (directBinding) |
| finishDirectBindingAssignment(); |
| else |
| target.pattern->bindValue(generator, temp.get()); |
| } else { |
| ASSERT(target.bindingType == BindingType::RestElement); |
| ASSERT(i == m_targetPatterns.size() - 1); |
| |
| generator.emitNewObject(newObject.get()); |
| |
| // load and call @copyDataProperties |
| RefPtr<RegisterID> copyDataProperties = generator.moveLinkTimeConstant(nullptr, LinkTimeConstant::copyDataProperties); |
| |
| // This must be non-tail-call because @copyDataProperties accesses caller-frame. |
| generator.move(args->thisRegister(), newObject.get()); |
| generator.move(args->argumentRegister(0), rhs); |
| generator.emitLoad(args->argumentRegister(1), WTFMove(excludedSet)); |
| generator.emitCall(generator.newTemporary(), copyDataProperties.get(), NoExpectedFunction, args.value(), divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| target.pattern->bindValue(generator, newObject.get()); |
| } |
| } |
| } |
| |
| generator.restoreTDZStack(preservedTDZStack); |
| } |
| |
| void ObjectPatternNode::collectBoundIdentifiers(Vector<Identifier>& identifiers) const |
| { |
| for (size_t i = 0; i < m_targetPatterns.size(); i++) |
| m_targetPatterns[i].pattern->collectBoundIdentifiers(identifiers); |
| } |
| |
| RegisterID* BindingNode::writableDirectBindingIfPossible(BytecodeGenerator& generator) const |
| { |
| Variable var = generator.variable(m_boundProperty); |
| bool isReadOnly = var.isReadOnly() && m_bindingContext != AssignmentContext::ConstDeclarationStatement; |
| if (RegisterID* local = var.local()) { |
| if (m_bindingContext == AssignmentContext::AssignmentExpression) { |
| if (generator.needsTDZCheck(var)) |
| return nullptr; |
| } |
| if (isReadOnly) |
| return nullptr; |
| return local; |
| } |
| return nullptr; |
| } |
| |
| void BindingNode::finishDirectBindingAssignment(BytecodeGenerator& generator) const |
| { |
| ASSERT(writableDirectBindingIfPossible(generator)); |
| Variable var = generator.variable(m_boundProperty); |
| RegisterID* local = var.local(); |
| generator.emitProfileType(local, var, divotStart(), divotEnd()); |
| if (m_bindingContext == AssignmentContext::DeclarationStatement || m_bindingContext == AssignmentContext::ConstDeclarationStatement) |
| generator.liftTDZCheckIfPossible(var); |
| } |
| |
| void BindingNode::bindValue(BytecodeGenerator& generator, RegisterID* value) const |
| { |
| Variable var = generator.variable(m_boundProperty); |
| bool isReadOnly = var.isReadOnly() && m_bindingContext != AssignmentContext::ConstDeclarationStatement; |
| if (RegisterID* local = var.local()) { |
| if (m_bindingContext == AssignmentContext::AssignmentExpression) |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| if (isReadOnly) { |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| return; |
| } |
| generator.move(local, value); |
| generator.emitProfileType(local, var, divotStart(), divotEnd()); |
| if (m_bindingContext == AssignmentContext::DeclarationStatement || m_bindingContext == AssignmentContext::ConstDeclarationStatement) |
| generator.liftTDZCheckIfPossible(var); |
| return; |
| } |
| if (generator.ecmaMode().isStrict()) |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| if (m_bindingContext == AssignmentContext::AssignmentExpression) |
| generator.emitTDZCheckIfNecessary(var, nullptr, scope.get()); |
| if (isReadOnly) { |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| return; |
| } |
| generator.emitPutToScope(scope.get(), var, value, generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, initializationModeForAssignmentContext(m_bindingContext)); |
| generator.emitProfileType(value, var, divotStart(), divotEnd()); |
| if (m_bindingContext == AssignmentContext::DeclarationStatement || m_bindingContext == AssignmentContext::ConstDeclarationStatement) |
| generator.liftTDZCheckIfPossible(var); |
| return; |
| } |
| |
| void BindingNode::toString(StringBuilder& builder) const |
| { |
| builder.append(m_boundProperty.string()); |
| } |
| |
| void BindingNode::collectBoundIdentifiers(Vector<Identifier>& identifiers) const |
| { |
| identifiers.append(m_boundProperty); |
| } |
| |
| RegisterID* AssignmentElementNode::writableDirectBindingIfPossible(BytecodeGenerator& generator) const |
| { |
| if (!m_assignmentTarget->isResolveNode()) |
| return nullptr; |
| ResolveNode* lhs = static_cast<ResolveNode*>(m_assignmentTarget); |
| Variable var = generator.variable(lhs->identifier()); |
| bool isReadOnly = var.isReadOnly(); |
| if (RegisterID* local = var.local()) { |
| if (generator.needsTDZCheck(var)) |
| return nullptr; |
| if (isReadOnly) |
| return nullptr; |
| return local; |
| } |
| return nullptr; |
| } |
| |
| void AssignmentElementNode::finishDirectBindingAssignment(BytecodeGenerator& generator) const |
| { |
| ASSERT_UNUSED(generator, writableDirectBindingIfPossible(generator)); |
| ResolveNode* lhs = static_cast<ResolveNode*>(m_assignmentTarget); |
| Variable var = generator.variable(lhs->identifier()); |
| RegisterID* local = var.local(); |
| generator.emitProfileType(local, divotStart(), divotEnd()); |
| } |
| |
| void AssignmentElementNode::collectBoundIdentifiers(Vector<Identifier>&) const |
| { |
| } |
| |
| void AssignmentElementNode::bindValue(BytecodeGenerator& generator, RegisterID* value) const |
| { |
| if (m_assignmentTarget->isResolveNode()) { |
| ResolveNode* lhs = static_cast<ResolveNode*>(m_assignmentTarget); |
| Variable var = generator.variable(lhs->identifier()); |
| bool isReadOnly = var.isReadOnly(); |
| if (RegisterID* local = var.local()) { |
| generator.emitTDZCheckIfNecessary(var, local, nullptr); |
| |
| if (isReadOnly) |
| generator.emitReadOnlyExceptionIfNeeded(var); |
| else { |
| generator.move(local, value); |
| generator.emitProfileType(local, divotStart(), divotEnd()); |
| } |
| return; |
| } |
| if (generator.ecmaMode().isStrict()) |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| RefPtr<RegisterID> scope = generator.emitResolveScope(nullptr, var); |
| generator.emitTDZCheckIfNecessary(var, nullptr, scope.get()); |
| if (isReadOnly) { |
| bool threwException = generator.emitReadOnlyExceptionIfNeeded(var); |
| if (threwException) |
| return; |
| } |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| if (!isReadOnly) { |
| generator.emitPutToScope(scope.get(), var, value, generator.ecmaMode().isStrict() ? ThrowIfNotFound : DoNotThrowIfNotFound, InitializationMode::NotInitialization); |
| generator.emitProfileType(value, var, divotStart(), divotEnd()); |
| } |
| } else if (m_assignmentTarget->isDotAccessorNode()) { |
| DotAccessorNode* lhs = static_cast<DotAccessorNode*>(m_assignmentTarget); |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(lhs->base(), true, false); |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| lhs->emitPutProperty(generator, base.get(), value); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| } else if (m_assignmentTarget->isBracketAccessorNode()) { |
| BracketAccessorNode* lhs = static_cast<BracketAccessorNode*>(m_assignmentTarget); |
| RefPtr<RegisterID> base = generator.emitNodeForLeftHandSide(lhs->base(), true, false); |
| RefPtr<RegisterID> property = generator.emitNodeForLeftHandSideForProperty(lhs->subscript(), true, false); |
| generator.emitExpressionInfo(divotEnd(), divotStart(), divotEnd()); |
| if (lhs->base()->isSuperNode()) { |
| RefPtr<RegisterID> thisValue = generator.ensureThis(); |
| generator.emitPutByVal(base.get(), thisValue.get(), property.get(), value); |
| } else |
| generator.emitPutByVal(base.get(), property.get(), value); |
| generator.emitProfileType(value, divotStart(), divotEnd()); |
| } |
| } |
| |
| void AssignmentElementNode::toString(StringBuilder& builder) const |
| { |
| if (m_assignmentTarget->isResolveNode()) |
| builder.append(static_cast<ResolveNode*>(m_assignmentTarget)->identifier().string()); |
| } |
| |
| void RestParameterNode::collectBoundIdentifiers(Vector<Identifier>& identifiers) const |
| { |
| m_pattern->collectBoundIdentifiers(identifiers); |
| } |
| |
| void RestParameterNode::toString(StringBuilder& builder) const |
| { |
| builder.append("..."); |
| m_pattern->toString(builder); |
| } |
| |
| void RestParameterNode::bindValue(BytecodeGenerator&, RegisterID*) const |
| { |
| RELEASE_ASSERT_NOT_REACHED(); |
| } |
| |
| void RestParameterNode::emit(BytecodeGenerator& generator) |
| { |
| RefPtr<RegisterID> temp = generator.newTemporary(); |
| generator.emitRestParameter(temp.get(), m_numParametersToSkip); |
| m_pattern->bindValue(generator, temp.get()); |
| } |
| |
| |
| RegisterID* SpreadExpressionNode::emitBytecode(BytecodeGenerator&, RegisterID*) |
| { |
| RELEASE_ASSERT_NOT_REACHED(); |
| return nullptr; |
| } |
| |
| RegisterID* ObjectSpreadExpressionNode::emitBytecode(BytecodeGenerator& generator, RegisterID* dst) |
| { |
| RefPtr<RegisterID> src = generator.newTemporary(); |
| generator.emitNode(src.get(), m_expression); |
| |
| RefPtr<RegisterID> copyDataProperties = generator.moveLinkTimeConstant(nullptr, LinkTimeConstant::copyDataProperties); |
| |
| CallArguments args(generator, nullptr, 1); |
| generator.move(args.thisRegister(), dst); |
| generator.move(args.argumentRegister(0), src.get()); |
| |
| // This must be non-tail-call because @copyDataProperties accesses caller-frame. |
| generator.emitCall(generator.newTemporary(), copyDataProperties.get(), NoExpectedFunction, args, divot(), divotStart(), divotEnd(), DebuggableCall::No); |
| |
| return dst; |
| } |
| |
| } // namespace JSC |