Source code
Revision control
Copy as Markdown
Other Tools
/* This Source Code Form is subject to the terms of the Mozilla Public
* License, v. 2.0. If a copy of the MPL was not distributed with this
#include "frontend/UsingEmitter.h"
#include "frontend/BytecodeEmitter.h"
#include "frontend/EmitterScope.h"
#include "frontend/IfEmitter.h"
#include "frontend/TryEmitter.h"
#include "frontend/WhileEmitter.h"
#include "vm/CompletionKind.h"
using namespace js;
using namespace js::frontend;
UsingEmitter::UsingEmitter(BytecodeEmitter* bce) : bce_(bce) {}
bool UsingEmitter::emitTakeDisposeCapability() {
if (!bce_->emit1(JSOp::TakeDisposeCapability)) {
// [stack] RESOURCES-OR-UNDEF
return false;
}
if (!bce_->emit1(JSOp::IsNullOrUndefined)) {
// [stack] RESOURCES-OR-UNDEF IS-UNDEF
return false;
}
InternalIfEmitter ifUndefined(bce_);
if (!ifUndefined.emitThenElse()) {
// [stack] UNDEFINED
return false;
}
if (!bce_->emit1(JSOp::Zero)) {
// [stack] UNDEFINED 0
return false;
}
if (!ifUndefined.emitElse()) {
// [stack] RESOURCES
return false;
}
if (!bce_->emit1(JSOp::Dup)) {
// [stack] RESOURCES RESOURCES
return false;
}
if (!bce_->emitAtomOp(JSOp::GetProp,
TaggedParserAtomIndex::WellKnown::length())) {
// [stack] RESOURCES COUNT
return false;
}
if (!ifUndefined.emitEnd()) {
// [stack] RESOURCES COUNT
return false;
}
return true;
}
bool UsingEmitter::emitThrowIfException() {
// [stack] EXC THROWING
InternalIfEmitter ifThrow(bce_);
if (!ifThrow.emitThenElse()) {
// [stack] EXC
return false;
}
if (!bce_->emit1(JSOp::Throw)) {
// [stack]
return false;
}
if (!ifThrow.emitElse()) {
// [stack] EXC
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack]
return false;
}
if (!ifThrow.emitEnd()) {
// [stack]
return false;
}
return true;
}
// Explicit Resource Management Proposal
// DisposeResources ( disposeCapability, completion )
//
// This implementation of DisposeResources is designed for using and await using
// syntax and covers the complete algorithm as defined in the spec for
// both sync and async disposals as necessary in bytecode.
bool UsingEmitter::emitDisposeLoop(EmitterScope& es,
CompletionKind initialCompletion) {
MOZ_ASSERT(initialCompletion != CompletionKind::Return);
// [stack] # if CompletionKind::Throw
// [stack] EXC
// [stack] # otherwise (CompletionKind::Normal)
// [stack]
if (hasAwaitUsing_) {
// Awaits can cause suspension of the current frame and
// the erasure of the frame's return value, thus we preserve
// the frame's return value on the value stack.
if (!bce_->emit1(JSOp::GetRval)) {
// [stack] EXC? RVAL
return false;
}
// Step 1. Let needsAwait be false.
if (!bce_->emit1(JSOp::False)) {
// [stack] EXC? RVAL NEEDS-AWAIT
return false;
}
// Step 2. Let hasAwaited be false.
if (!bce_->emit1(JSOp::False)) {
// [stack] EXC? RVAL NEEDS-AWAIT HAS-AWAITED
return false;
}
}
// corresponds to completion parameter
if (initialCompletion == CompletionKind::Throw) {
if (!bce_->emit1(JSOp::True)) {
// [stack] EXC RVAL? NEEDS-AWAIT? HAS-AWAITED? THROWING
return false;
}
if (hasAwaitUsing_) {
// [stack] EXC RVAL NEEDS-AWAIT HAS-AWAITED THROWING
if (!bce_->emitPickN(4)) {
// [stack] RVAL NEEDS-AWAIT HAS-AWAITED THROWING EXC
return false;
}
} else {
// [stack] EXC THROWING
if (!bce_->emit1(JSOp::Swap)) {
// [stack] THROWING EXC
return false;
}
}
} else {
if (!bce_->emit1(JSOp::False)) {
// [stack] RVAL? NEEDS-AWAIT? HAS-AWAITED? THROWING
return false;
}
if (!bce_->emit1(JSOp::Undefined)) {
// [stack] RVAL? NEEDS-AWAIT? HAS-AWAITED? THROWING UNDEF
return false;
}
}
// [stack] ...
// For the purpose of readbility RVAL has been omitted from
// the stack comments below and is assumed to be present,
// we mention it again below at the completion steps when we
// use it.
// We do the iteration in reverse order as per spec,
// there can be the case when count is 0 and hence index
// below becomes -1 but the loop condition will ensure
// no code is executed in that case.
// Step 6. Set disposeCapability.[[DisposableResourceStack]] to a new empty
// List.
if (!emitTakeDisposeCapability()) {
// [stack] ... RESOURCES COUNT
return false;
}
if (!bce_->emit1(JSOp::Dec)) {
// [stack] ... RESOURCES INDEX
return false;
}
InternalWhileEmitter wh(bce_);
// Step 3. For each element resource of
// disposeCapability.[[DisposableResourceStack]], in reverse list order, do
if (!wh.emitCond()) {
// [stack] ... RESOURCES INDEX
return false;
}
if (!bce_->emit1(JSOp::Dup)) {
// [stack] ... RESOURCES INDEX INDEX
return false;
}
if (!bce_->emit1(JSOp::Zero)) {
// [stack] ... RESOURCES INDEX INDEX 0
return false;
}
if (!bce_->emit1(JSOp::Ge)) {
// [stack] ... RESOURCES INDEX BOOL
return false;
}
if (!wh.emitBody()) {
// [stack] ... RESOURCES INDEX
return false;
}
if (!bce_->emit1(JSOp::Dup2)) {
// [stack] ... RESOURCES INDEX RESOURCES INDEX
return false;
}
if (!bce_->emit1(JSOp::GetElem)) {
// [stack] ... RESOURCES INDEX RESOURCE
return false;
}
// Step 3.a. Let value be resource.[[ResourceValue]].
// Step 3.b. Let hint be resource.[[Hint]].
// Step 3.c. Let method be resource.[[DisposeMethod]].
// TODO: Defer property accesses until they are actually
if (!bce_->emit1(JSOp::Dup)) {
// [stack] ... RESOURCES INDEX RESOURCE RESOURCE
return false;
}
if (!bce_->emitAtomOp(JSOp::GetProp,
TaggedParserAtomIndex::WellKnown::hint())) {
// [stack] ... RESOURCES INDEX RESOURCE HINT
return false;
}
if (!bce_->emitDupAt(1)) {
// [stack] ... RESOURCES INDEX RESOURCE HINT RESOURCE
return false;
}
if (!bce_->emitAtomOp(JSOp::GetProp,
TaggedParserAtomIndex::WellKnown::method())) {
// [stack] ... RESOURCES INDEX RESOURCE HINT METHOD
return false;
}
if (!bce_->emitPickN(2)) {
// [stack] ... RESOURCES INDEX HINT METHOD RESOURCE
return false;
}
if (!bce_->emitAtomOp(JSOp::GetProp,
TaggedParserAtomIndex::WellKnown::value())) {
// [stack] ... RESOURCES INDEX HINT METHOD VALUE
return false;
}
if (hasAwaitUsing_) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... HINT METHOD VALUE
// Step 3.d. If hint is sync-dispose and needsAwait is true and hasAwaited
// is false, then
if (!bce_->emitDupAt(2)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... HINT METHOD VALUE HINT
return false;
}
// [stack] NEEDS-AWAIT HAS-AWAITED ... HINT
static_assert(uint8_t(UsingHint::Sync) == 0, "Sync hint must be 0");
static_assert(uint8_t(UsingHint::Async) == 1, "Async hint must be 1");
if (!bce_->emit1(JSOp::Not)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... IS-SYNC
return false;
}
if (!bce_->emitDupAt(9)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... IS-SYNC NEEDS-AWAIT
return false;
}
if (!bce_->emitDupAt(9)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... IS-SYNC NEEDS-AWAIT HAS-AWAITED
return false;
}
// [stack] ... IS-SYNC NEEDS-AWAIT HAS-AWAITED
if (!bce_->emit1(JSOp::Not)) {
// [stack] ... IS-SYNC NEEDS-AWAIT (!HAS-AWAITED)
return false;
}
// The use of BitAnd is a simple optimisation to avoid having
// jumps if we were to implement this using && operator. The value
// IS-SYNC is integer 0 or 1 (see static_assert above) and
// NEEDS-AWAIT and HAS-AWAITED are boolean values. thus
// the result of the operation is either 0 or 1 which is
// truthy value that can be consumed by the IfEmitter.
if (!bce_->emit1(JSOp::BitAnd)) {
// [stack] ... IS-SYNC (NEEDS-AWAIT & !HAS-AWAITED)
return false;
}
if (!bce_->emit1(JSOp::BitAnd)) {
// [stack] ... (IS-SYNC & NEEDS-AWAIT & !HAS-AWAITED)
return false;
}
// [stack] NEEDS-AWAIT HAS-AWAITED ... UNDEF-AWAIT-NEEDED
InternalIfEmitter ifNeedsSyncDisposeUndefinedAwaited(bce_);
if (!ifNeedsSyncDisposeUndefinedAwaited.emitThen()) {
// [stack] NEEDS-AWAIT HAS-AWAITED ...
return false;
}
// Step 3.d.i. Perform ! Await(undefined).
if (!bce_->emit1(JSOp::Undefined)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... UNDEF
return false;
}
if (!bce_->emitAwaitInScope(es)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESOLVED
return false;
}
// Step 3.d.ii. Set needsAwait to false.
if (!bce_->emitPickN(9)) {
// [stack] HAS-AWAITED ... RESOLVED NEEDS-AWAIT
return false;
}
if (!bce_->emitPopN(2)) {
// [stack] HAS-AWAITED ...
return false;
}
if (!bce_->emit1(JSOp::False)) {
// [stack] HAS-AWAITED ... NEEDS-AWAIT
return false;
}
if (!bce_->emitUnpickN(8)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ...
return false;
}
if (!ifNeedsSyncDisposeUndefinedAwaited.emitEnd()) {
// [stack] NEEDS-AWAIT HAS-AWAITED ...
return false;
}
}
// [stack] ... HINT METHOD VALUE
// Step 3.e. If method is not undefined, then
if (!bce_->emitDupAt(1)) {
// [stack] ... HINT METHOD VALUE METHOD
return false;
}
if (!bce_->emit1(JSOp::IsNullOrUndefined)) {
// [stack] ... HINT METHOD VALUE METHOD IS-UNDEF
return false;
}
InternalIfEmitter ifMethodNotUndefined(bce_);
if (!ifMethodNotUndefined.emitThenElse(IfEmitter::ConditionKind::Negative)) {
// [stack] ... HINT METHOD VALUE METHOD
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] ... HINT METHOD VALUE
return false;
}
TryEmitter tryCall(bce_, TryEmitter::Kind::TryCatch,
TryEmitter::ControlKind::NonSyntactic);
if (!tryCall.emitTry()) {
// [stack] ... HINT METHOD VALUE
return false;
}
if (!bce_->emit1(JSOp::Dup2)) {
// [stack] ... HINT METHOD VALUE METHOD VALUE
return false;
}
// Step 3.e.i. Let result be Completion(Call(method, value)).
if (!bce_->emitCall(JSOp::Call, 0)) {
// [stack] ... HINT METHOD VALUE RESULT
return false;
}
if (hasAwaitUsing_) {
// Step 3.e.ii. If result is a normal completion and hint is async-dispose,
// then
if (!bce_->emitDupAt(3)) {
// [stack] ... HINT METHOD VALUE RESULT HINT
return false;
}
// Hint value is either 0 or 1, which can be consumed by the IfEmitter,
// see static_assert above.
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESULT IS-ASYNC
InternalIfEmitter ifAsyncDispose(bce_);
if (!ifAsyncDispose.emitThen()) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESULT
return false;
}
// Step 3.e.ii.2. Set hasAwaited to true. (reordered)
if (!bce_->emitPickN(8)) {
// [stack] NEEDS-AWAIT ... RESULT HAS-AWAITED
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] NEEDS-AWAIT ... RESULT
return false;
}
if (!bce_->emit1(JSOp::True)) {
// [stack] NEEDS-AWAIT ... RESULT HAS-AWAITED
return false;
}
if (!bce_->emitUnpickN(8)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESULT
return false;
}
// Step 3.e.ii.1. Set result to Completion(Await(result.[[Value]])).
if (!bce_->emitAwaitInScope(es)) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESOLVED
return false;
}
if (!ifAsyncDispose.emitEnd()) {
// [stack] NEEDS-AWAIT HAS-AWAITED ... RESULT
return false;
}
}
// [stack] ... THROWING EXC RESOURCES INDEX HINT METHOD VALUE RESULT
if (!bce_->emit1(JSOp::Pop)) {
// [stack] ... THROWING EXC RESOURCES INDEX HINT METHOD VALUE
return false;
}
// Step 3.e.iii. If result is a throw completion, then
if (!tryCall.emitCatch()) {
// [stack] ... THROWING EXC RESOURCES INDEX HINT METHOD VALUE EXC2
return false;
}
if (!bce_->emitPickN(6)) {
// [stack] .. THROWING RESOURCES INDEX HINT METHOD VALUE EXC2 EXC
return false;
}
if (initialCompletion == CompletionKind::Throw &&
bce_->sc->isSuspendableContext() &&
bce_->sc->asSuspendableContext()->isGenerator()) {
// [stack] ... EXC2 EXC
// Generator closure is implemented by throwing a magic value
// thus when we have a throw completion we must check whether
// the pending exception is a generator closing exception and overwrite
// it with the normal exception here or else we will end up exposing
// the magic value to user program.
if (!bce_->emit1(JSOp::IsGenClosing)) {
// [stack] ... EXC2 EXC GEN-CLOSING
return false;
}
if (!bce_->emit1(JSOp::Not)) {
// [stack] ... EXC2 EXC !GEN-CLOSING
return false;
}
if (!bce_->emitPickN(8)) {
// [stack] ... EXC2 EXC (!GEN-CLOSING) THROWING
return false;
}
if (!bce_->emit1(JSOp::BitAnd)) {
// [stack] ... EXC2 EXC (!GEN-CLOSING & THROWING)
return false;
}
} else {
if (!bce_->emitPickN(7)) {
// [stack] ... RESOURCES INDEX HINT METHOD VALUE EXC2 EXC THROWING
return false;
}
}
// [stack] NEEDS-AWAIT? HAS-AWAITED? ... EXC2 EXC THROWING
InternalIfEmitter ifException(bce_);
// Step 3.e.iii.1. If completion is a throw completion, then
if (!ifException.emitThenElse()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? ... EXC2 EXC
return false;
}
// Step 3.e.iii.1.a-f
if (!bce_->emit1(JSOp::CreateSuppressedError)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? ... SUPPRESSED
return false;
}
if (!bce_->emitUnpickN(5)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? SUPPRESSED ...
return false;
}
if (!bce_->emit1(JSOp::True)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? SUPPRESSED ... THROWING
return false;
}
if (!bce_->emitUnpickN(6)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING SUPPRESSED ...
return false;
}
// Step 3.e.iii.2. Else,
// Step 3.e.iii.2.a. Set completion to result.
if (!ifException.emitElse()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? ... EXC2 EXC
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? ... EXC2
return false;
}
if (!bce_->emitUnpickN(5)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? EXC2 ...
return false;
}
if (!bce_->emit1(JSOp::True)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? EXC2 ... THROWING
return false;
}
if (!bce_->emitUnpickN(6)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC2 ...
return false;
}
if (!ifException.emitEnd()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC ...
return false;
}
if (!tryCall.emitEnd()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC ...
return false;
}
// [stack] ... THROWING EXC RESOURCES INDEX HINT METHOD VALUE
if (!bce_->emitPopN(3)) {
// [stack] ... THROWING EXC RESOURCES INDEX
return false;
}
// Step 3.f. Else,
// Step 3.f.i. Assert: hint is async-dispose.
// (implicit)
if (!ifMethodNotUndefined.emitElse()) {
// [stack] ... THROWING EXC RESOURCES INDEX HINT METHOD VALUE METHOD
return false;
}
if (!bce_->emitPopN(4)) {
// [stack] ... THROWING EXC RESOURCES INDEX
return false;
}
if (hasAwaitUsing_) {
// [stack] NEEDS-AWAIT HAS-AWAITED THROWING EXC RESOURCES INDEX
// Step 3.f.ii. Set needsAwait to true.
if (!bce_->emitPickN(5)) {
// [stack] HAS-AWAITED THROWING EXC RESOURCES INDEX NEEDS-AWAIT
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] HAS-AWAITED THROWING EXC RESOURCES INDEX
return false;
}
if (!bce_->emit1(JSOp::True)) {
// [stack] HAS-AWAITED THROWING EXC RESOURCES INDEX NEEDS-AWAIT
return false;
}
if (!bce_->emitUnpickN(5)) {
// [stack] NEEDS-AWAIT HAS-AWAITED THROWING EXC RESOURCES INDEX
return false;
}
}
if (!ifMethodNotUndefined.emitEnd()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC RESOURCES INDEX
return false;
}
if (!bce_->emit1(JSOp::Dec)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC RESOURCES INDEX
return false;
}
if (!wh.emitEnd()) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC RESOURCES INDEX
return false;
}
if (!bce_->emitPopN(2)) {
// [stack] NEEDS-AWAIT? HAS-AWAITED? THROWING EXC
return false;
}
if (hasAwaitUsing_) {
// Step 4. If needsAwait is true and hasAwaited is false, then
if (!bce_->emitPickN(3)) {
// [stack] HAS-AWAITED THROWING EXC NEEDS-AWAIT
return false;
}
if (!bce_->emitPickN(3)) {
// [stack] THROWING EXC NEEDS-AWAIT HAS-AWAITED
return false;
}
if (!bce_->emit1(JSOp::Not)) {
// [stack] THROWING EXC NEEDS-AWAIT (!HAS-AWAITED)
return false;
}
if (!bce_->emit1(JSOp::BitAnd)) {
// [stack] THROWING EXC (NEEDS-AWAIT & !HAS-AWAITED)
return false;
}
InternalIfEmitter ifNeedsUndefinedAwait(bce_);
if (!ifNeedsUndefinedAwait.emitThen()) {
// [stack] THROWING EXC
return false;
}
if (!bce_->emit1(JSOp::Undefined)) {
// [stack] THROWING EXC UNDEF
return false;
}
if (!bce_->emitAwaitInScope(es)) {
// [stack] THROWING EXC RESOLVED
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] THROWING EXC
return false;
}
if (!ifNeedsUndefinedAwait.emitEnd()) {
// [stack] THROWING EXC
return false;
}
}
// Step 7. Return ? completion.
if (!bce_->emit1(JSOp::Swap)) {
// [stack] EXC THROWING
return false;
}
if (hasAwaitUsing_) {
// [stack] RVAL EXC THROWING
if (!bce_->emitPickN(2)) {
// [stack] EXC THROWING RVAL
return false;
}
if (!bce_->emit1(JSOp::SetRval)) {
// [stack] EXC THROWING
return false;
}
}
return true;
}
bool UsingEmitter::prepareForDisposableScopeBody() {
tryEmitter_.emplace(bce_, TryEmitter::Kind::TryFinally,
TryEmitter::ControlKind::NonSyntactic);
return tryEmitter_->emitTry();
}
// Explicit Resource Management Proposal
// GetDisposeMethod ( V, hint )
// Steps 1.a-1.b.i., 2-3.
bool UsingEmitter::emitGetDisposeMethod(UsingHint hint) {
// [stack] VAL
// Step 1. If hint is async-dispose, then
if (hint == UsingHint::Async) {
// Step 1.a. Let method be ? GetMethod(V, @@asyncDispose).
if (!bce_->emit1(JSOp::Dup)) {
// [stack] VAL VAL
return false;
}
if (!bce_->emit1(JSOp::Dup)) {
// [stack] VAL VAL VAL
return false;
}
if (!bce_->emit2(JSOp::Symbol, uint8_t(JS::SymbolCode::asyncDispose))) {
// [stack] VAL VAL VAL SYMBOL
return false;
}
if (!bce_->emit1(JSOp::GetElem)) {
// [stack] VAL VAL ASYNC-DISPOSE
return false;
}
// Step 1.b. If method is undefined, then
// GetMethod returns undefined if the function is null but
// since we do not do the conversion here we check for
// null or undefined here.
if (!bce_->emit1(JSOp::IsNullOrUndefined)) {
// [stack] VAL VAL ASYNC-DISPOSE IS-NULL-OR-UNDEF
return false;
}
InternalIfEmitter ifAsyncDisposeNullOrUndefined(bce_);
if (!ifAsyncDisposeNullOrUndefined.emitThenElse()) {
// [stack] VAL VAL ASYNC-DISPOSE
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] VAL VAL
return false;
}
if (!bce_->emit1(JSOp::Dup)) {
// [stack] VAL VAL VAL
return false;
}
if (!bce_->emit2(JSOp::Symbol, uint8_t(JS::SymbolCode::dispose))) {
// [stack] VAL VAL VAL SYMBOL
return false;
}
// Step 1.b.i. Set method to ? GetMethod(V, @@dispose).
if (!bce_->emit1(JSOp::GetElem)) {
// [stack] VAL VAL DISPOSE
return false;
}
if (!bce_->emit1(JSOp::True)) {
// [stack] VAL VAL DISPOSE NEEDS-CLOSURE
return false;
}
if (!ifAsyncDisposeNullOrUndefined.emitElse()) {
// [stack] VAL VAL ASYNC-DISPOSE
return false;
}
if (!bce_->emit1(JSOp::False)) {
// [stack] VAL VAL ASYNC-DISPOSE NEEDS-CLOSURE
return false;
}
if (!ifAsyncDisposeNullOrUndefined.emitEnd()) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE
return false;
}
} else {
MOZ_ASSERT(hint == UsingHint::Sync);
// Step 2. Else,
// Step 2.a. Let method be ? GetMethod(V, @@dispose).
if (!bce_->emit1(JSOp::Dup)) {
// [stack] VAL VAL
return false;
}
if (!bce_->emit1(JSOp::Dup)) {
// [stack] VAL VAL VAL
return false;
}
if (!bce_->emit2(JSOp::Symbol, uint8_t(JS::SymbolCode::dispose))) {
// [stack] VAL VAL VAL SYMBOL
return false;
}
if (!bce_->emit1(JSOp::GetElem)) {
// [stack] VAL VAL DISPOSE
return false;
}
if (!bce_->emit1(JSOp::False)) {
// [stack] VAL VAL DISPOSE NEEDS-CLOSURE
return false;
}
}
if (!bce_->emitDupAt(1)) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE METHOD
return false;
}
// According to spec GetMethod throws TypeError if the method is not callable
// and returns undefined if the value is either undefined or null.
// but the caller of this function, emitCreateDisposableResource is
// supposed to throw TypeError as well if returned value is undefined,
// thus we combine the steps here.
if (!bce_->emitCheckIsCallable()) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE METHOD IS-CALLABLE
return false;
}
InternalIfEmitter ifMethodNotCallable(bce_);
if (!ifMethodNotCallable.emitThen(IfEmitter::ConditionKind::Negative)) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE METHOD
return false;
}
if (!bce_->emit2(JSOp::ThrowMsg, uint8_t(ThrowMsgKind::DisposeNotCallable))) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE METHOD
return false;
}
if (!ifMethodNotCallable.emitEnd()) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE METHOD
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE
return false;
}
return true;
}
// Explicit Resource Management Proposal
// CreateDisposableResource ( V, hint [ , method ] )
bool UsingEmitter::emitCreateDisposableResource(UsingHint hint) {
// [stack] VAL
// Step 1. If method is not present, then (implicit)
// Step 1.a. If V is either null or undefined, then
if (!bce_->emit1(JSOp::IsNullOrUndefined)) {
// [stack] VAL IS-NULL-OR-UNDEF
return false;
}
InternalIfEmitter ifNullUndefined(bce_);
if (!ifNullUndefined.emitThenElse()) {
// [stack] VAL
return false;
}
// Step 1.a.i. Set V to undefined.
if (!bce_->emit1(JSOp::Undefined)) {
// [stack] VAL UNDEF
return false;
}
// Step 1.a.ii. Set method to undefined.
if (!bce_->emit1(JSOp::Undefined)) {
// [stack] VAL UNDEF UNDEF
return false;
}
if (!bce_->emit1(JSOp::False)) {
// [stack] VAL UNDEF UNDEF NEEDS-CLOSURE
return false;
}
// Step 1.b. Else,
if (!ifNullUndefined.emitElse()) {
// [stack] VAL
return false;
}
// Step 1.b.i. If V is not an Object, throw a TypeError exception.
if (!bce_->emitCheckIsObj(CheckIsObjectKind::Disposable)) {
// [stack] VAL
return false;
}
// Step 1.b.ii. Set method to ? GetDisposeMethod(V, hint).
// Step 1.b.iii. If method is undefined, throw a TypeError exception.
if (!emitGetDisposeMethod(hint)) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE
return false;
}
if (!ifNullUndefined.emitEnd()) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE
return false;
}
return true;
}
// Explicit Resource Management Proposal
// 7.5.4 AddDisposableResource ( disposeCapability, V, hint [ , method ] )
// Steps 1, 3-4.
bool UsingEmitter::prepareForAssignment(UsingHint hint) {
MOZ_ASSERT(bce_->innermostEmitterScope()->hasDisposables());
if (hint == UsingHint::Async) {
hasAwaitUsing_ = true;
}
// [stack] VAL
// Step 1. If method is not present, then (implicit)
// Step 1.a. If V is either null or undefined and hint is sync-dispose, return
// unused.
if (hint == UsingHint::Sync) {
if (!bce_->emit1(JSOp::IsNullOrUndefined)) {
// [stack] VAL IS-NULL-OR-UNDEF
return false;
}
if (!bce_->emit1(JSOp::Not)) {
// [stack] VAL !IS-NULL-OR-UNDEF
return false;
}
} else {
MOZ_ASSERT(hint == UsingHint::Async);
if (!bce_->emit1(JSOp::True)) {
// [stack] VAL TRUE
return false;
}
}
// [stack] VAL SHOULD-CREATE-RESOURCE
InternalIfEmitter ifCreateResource(bce_);
if (!ifCreateResource.emitThen()) {
// [stack] VAL
return false;
}
// Step 1.c. Let resource be ? CreateDisposableResource(V, hint).
if (!emitCreateDisposableResource(hint)) {
// [stack] VAL VAL METHOD NEEDS-CLOSURE
return false;
}
// Step 3. Append resource to disposeCapability.[[DisposableResourceStack]].
if (!bce_->emit2(JSOp::AddDisposable, uint8_t(hint))) {
// [stack] VAL
return false;
}
if (!ifCreateResource.emitEnd()) {
// [stack] VAL
return false;
}
// Step 4. Return unused.
return true;
}
bool UsingEmitter::prepareForForOfLoopIteration() {
EmitterScope* es = bce_->innermostEmitterScopeNoCheck();
MOZ_ASSERT(es->hasDisposables());
if (!emitDisposeLoop(*es)) {
// [stack] EXC THROWING
return false;
}
return emitThrowIfException();
}
bool UsingEmitter::prepareForForOfIteratorCloseOnThrow() {
EmitterScope* es = bce_->innermostEmitterScopeNoCheck();
MOZ_ASSERT(es->hasDisposables());
// [stack] EXC STACK
if (!bce_->emit1(JSOp::Swap)) {
// [stack] STACK EXC
return false;
}
if (!emitDisposeLoop(*es, CompletionKind::Throw)) {
// [stack] STACK EXC THROWING
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] STACK EXC
return false;
}
return bce_->emit1(JSOp::Swap);
// [stack] EXC STACK
}
bool UsingEmitter::emitNonLocalJump(EmitterScope* present) {
MOZ_ASSERT(present->hasDisposables());
if (!emitDisposeLoop(*present)) {
// [stack] EXC THROWING
return false;
}
return emitThrowIfException();
}
bool UsingEmitter::emitEnd() {
EmitterScope* es = bce_->innermostEmitterScopeNoCheck();
MOZ_ASSERT(es->hasDisposables());
MOZ_ASSERT(tryEmitter_.isSome());
// Given that we are using NonSyntactic TryEmitter we do
// not have fallthrough behaviour in the normal completion case
// see comment on controlInfo_ in TryEmitter.h
if (!emitDisposeLoop(*es)) {
// [stack] EXC THROWING
return false;
}
if (!emitThrowIfException()) {
// [stack]
return false;
}
#ifdef DEBUG
// We want to ensure that we have EXC and STACK on the stack
// and not RESUME_INDEX, non-existence of control info
// confirms the same.
MOZ_ASSERT(!tryEmitter_->hasControlInfo());
#endif
if (!tryEmitter_->emitFinally()) {
// [stack] EXC STACK THROWING
return false;
}
if (!bce_->emitPickN(2)) {
// [stack] STACK THROWING EXC
return false;
}
if (!emitDisposeLoop(*es, CompletionKind::Throw)) {
// [stack] STACK THROWING EXC THROWING
return false;
}
if (!bce_->emit1(JSOp::Pop)) {
// [stack] STACK THROWING EXC
return false;
}
if (!bce_->emitUnpickN(2)) {
// [stack] EXC STACK THROWING
return false;
}
if (!tryEmitter_->emitEnd()) {
// [stack]
return false;
}
return true;
}