
com.oracle.truffle.js.nodes.control.AsyncGeneratorYieldNode Maven / Gradle / Ivy
/*
* Copyright (c) 2018, 2020, Oracle and/or its affiliates. All rights reserved.
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
*
* The Universal Permissive License (UPL), Version 1.0
*
* Subject to the condition set forth below, permission is hereby granted to any
* person obtaining a copy of this software, associated documentation and/or
* data (collectively the "Software"), free of charge and under any and all
* copyright rights in the Software, and any and all patent rights owned or
* freely licensable by each licensor hereunder covering either (i) the
* unmodified Software as contributed to or provided by such licensor, or (ii)
* the Larger Works (as defined below), to deal in both
*
* (a) the Software, and
*
* (b) any piece of software and/or hardware listed in the lrgrwrks.txt file if
* one is included with the Software each a "Larger Work" to which the Software
* is contributed by such licensors),
*
* without restriction, including without limitation the rights to copy, create
* derivative works of, display, perform, and distribute the Software and make,
* use, sell, offer for sale, import, export, have made, and have sold the
* Software and the Larger Work(s), and to sublicense the foregoing rights on
* either these or other terms.
*
* This license is subject to the following condition:
*
* The above copyright notice and either this complete permission notice or at a
* minimum a reference to the UPL must be included in all copies or substantial
* portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
* SOFTWARE.
*/
package com.oracle.truffle.js.nodes.control;
import com.oracle.truffle.api.frame.VirtualFrame;
import com.oracle.truffle.api.instrumentation.Tag;
import com.oracle.truffle.api.object.DynamicObject;
import com.oracle.truffle.js.nodes.JavaScriptNode;
import com.oracle.truffle.js.nodes.access.GetIteratorNode;
import com.oracle.truffle.js.nodes.access.GetMethodNode;
import com.oracle.truffle.js.nodes.access.IteratorCompleteNode;
import com.oracle.truffle.js.nodes.access.IteratorNextNode;
import com.oracle.truffle.js.nodes.access.IteratorValueNode;
import com.oracle.truffle.js.nodes.access.JSReadFrameSlotNode;
import com.oracle.truffle.js.nodes.access.WriteNode;
import com.oracle.truffle.js.nodes.control.ReturnNode.FrameReturnNode;
import com.oracle.truffle.js.nodes.control.YieldNode.ExceptionYieldResultNode;
import com.oracle.truffle.js.nodes.control.YieldNode.YieldResultNode;
import com.oracle.truffle.js.nodes.function.JSFunctionCallNode;
import com.oracle.truffle.js.runtime.Errors;
import com.oracle.truffle.js.runtime.GraalJSException;
import com.oracle.truffle.js.runtime.JSArguments;
import com.oracle.truffle.js.runtime.JSContext;
import com.oracle.truffle.js.runtime.JSRuntime;
import com.oracle.truffle.js.runtime.UserScriptException;
import com.oracle.truffle.js.runtime.objects.Completion;
import com.oracle.truffle.js.runtime.objects.IteratorRecord;
import com.oracle.truffle.js.runtime.objects.Undefined;
import java.util.Set;
public class AsyncGeneratorYieldNode extends AwaitNode {
@Child protected ReturnNode returnNode;
@Child private YieldResultNode generatorYieldNode;
protected AsyncGeneratorYieldNode(JSContext context, JavaScriptNode expression, JSReadFrameSlotNode readAsyncContextNode, JSReadFrameSlotNode readYieldResultNode, ReturnNode returnNode) {
super(context, expression, readAsyncContextNode, readYieldResultNode);
this.returnNode = returnNode;
this.generatorYieldNode = new ExceptionYieldResultNode();
}
public static AsyncGeneratorYieldNode createYield(JSContext context, JavaScriptNode expression, JSReadFrameSlotNode readAsyncContextNode, JSReadFrameSlotNode readAsyncResultNode,
ReturnNode returnNode) {
return new AsyncGeneratorYieldNode(context, expression, readAsyncContextNode, readAsyncResultNode, returnNode);
}
public static AsyncGeneratorYieldNode createYieldStar(JSContext context, JavaScriptNode expression, JSReadFrameSlotNode readAsyncContextNode, JSReadFrameSlotNode readAsyncResultNode,
ReturnNode returnNode, JavaScriptNode readTemp, WriteNode writeTemp) {
return new AsyncGeneratorYieldStarNode(context, expression, readAsyncContextNode, readAsyncResultNode, returnNode, readTemp, writeTemp);
}
@Override
public Object resume(VirtualFrame frame) {
int state = getStateAsInt(frame);
// 0 .. execute expression and await
// 1 .. resume await and yield
// 2 .. resume yield (and await if return)
// 3 .. resume await and return
final int awaitValue = 1;
final int suspendedYield = 2;
final int awaitResumptionValue = 3;
if (state == 0) {
Object value = expression.execute(frame);
setState(frame, awaitValue);
return suspendAwait(frame, value);
} else if (state == awaitValue) {
Object awaited = resumeAwait(frame);
setState(frame, suspendedYield);
return suspendYield(frame, awaited);
} else {
assert state >= suspendedYield;
setState(frame, 0);
if (state == suspendedYield) {
Completion completion = resumeYield(frame);
if (completion.isNormal()) {
return completion.getValue();
} else if (completion.isThrow()) {
throw UserScriptException.create(completion.getValue(), this, context.getContextOptions().getStackTraceLimit());
} else {
assert completion.isReturn();
// Let awaited be Await(resumptionValue.[[Value]]).
setState(frame, awaitResumptionValue);
return suspendAwait(frame, completion.getValue());
}
} else {
assert state == awaitResumptionValue;
// If awaited.[[Type]] is throw return Completion(awaited).
Object awaited = resumeAwait(frame);
// Assert: awaited.[[Type]] is normal.
return returnValue(frame, awaited);
}
}
}
protected final Object suspendYield(VirtualFrame frame, Object awaited) {
return generatorYieldNode.generatorYield(frame, awaited);
}
protected final Completion resumeYield(VirtualFrame frame) {
return (Completion) readAsyncResultNode.execute(frame);
}
protected final Object returnValue(VirtualFrame frame, Object value) {
assert getStateAsInt(frame) == 0;
if (returnNode instanceof FrameReturnNode) {
((WriteNode) returnNode.expression).executeWrite(frame, value);
}
throw new ReturnException(value);
}
@Override
protected JavaScriptNode copyUninitialized(Set> materializedTags) {
return createYield(context, cloneUninitialized(expression, materializedTags), cloneUninitialized(readAsyncContextNode, materializedTags),
cloneUninitialized(readAsyncResultNode, materializedTags), cloneUninitialized(returnNode, materializedTags));
}
}
class AsyncGeneratorYieldStarNode extends AsyncGeneratorYieldNode {
@Child private JavaScriptNode readIteratorTemp;
@Child private WriteNode writeIteratorTemp;
@Child private GetIteratorNode getIteratorNode;
@Child private IteratorNextNode iteratorNextNode;
@Child private IteratorCompleteNode iteratorCompleteNode;
@Child private IteratorValueNode iteratorValueNode;
@Child private GetMethodNode getThrowMethodNode;
@Child private GetMethodNode getReturnMethodNode;
@Child private JSFunctionCallNode callThrowNode;
@Child private JSFunctionCallNode callReturnNode;
protected AsyncGeneratorYieldStarNode(JSContext context, JavaScriptNode expression, JSReadFrameSlotNode readAsyncContextNode, JSReadFrameSlotNode readYieldResultNode,
ReturnNode returnNode, JavaScriptNode readTemp, WriteNode writeTemp) {
super(context, expression, readAsyncContextNode, readYieldResultNode, returnNode);
this.readIteratorTemp = readTemp;
this.writeIteratorTemp = writeTemp;
this.getIteratorNode = GetIteratorNode.createAsync(context, null);
this.iteratorNextNode = IteratorNextNode.create();
this.iteratorCompleteNode = IteratorCompleteNode.create(context);
this.iteratorValueNode = IteratorValueNode.create(context, null);
this.getThrowMethodNode = GetMethodNode.create(context, null, "throw");
this.getReturnMethodNode = GetMethodNode.create(context, null, "return");
this.callThrowNode = JSFunctionCallNode.createCall();
this.callReturnNode = JSFunctionCallNode.createCall();
}
@Override
public Object resume(VirtualFrame frame) {
int state = getStateAsInt(frame);
final int loopBegin = 1;
final int normalOrThrowAwaitInnerResult = 2;
final int returnAwaitInnerReturnResult = 3;
final int asyncGeneratorYieldInnerResult = 4;
final int asyncGeneratorYieldInnerResultSuspendedYield = 5;
final int asyncGeneratorYieldInnerResultReturn = 6;
final int returnAwaitReceivedValue = 7;
final int throwAwaitReturnResult = 8;
IteratorRecord iteratorRecord;
if (state == 0) {
iteratorRecord = getIteratorNode.execute(expression.execute(frame));
writeIteratorTemp.executeWrite(frame, iteratorRecord);
state = loopBegin;
} else {
iteratorRecord = (IteratorRecord) readIteratorTemp.execute(frame);
}
DynamicObject iterator = iteratorRecord.getIterator();
Completion received = Completion.forNormal(Undefined.instance);
for (;;) {
switch (state) {
case loopBegin: {
if (received.isNormal()) {
DynamicObject innerResult = iteratorNextNode.execute(iteratorRecord, received.getValue());
awaitWithNext(frame, innerResult, normalOrThrowAwaitInnerResult);
} else if (received.isThrow()) {
Object throwMethod = getThrowMethodNode.executeWithTarget(iterator);
if (throwMethod != Undefined.instance) {
Object innerResult = callThrowMethod(throwMethod, iterator, received.getValue());
awaitWithNext(frame, innerResult, normalOrThrowAwaitInnerResult);
/*
* NOTE: Exceptions from the inner iterator throw method are propagated.
* Normal completions from an inner throw method are processed similarly
* to an inner next.
*/
} else {
/*
* NOTE: If iterator does not have a throw method, this throw is going
* to terminate the yield* loop. But first we need to give iterator a
* chance to clean up.
*/
// AsyncIteratorClose
Object returnMethod = getReturnMethodNode.executeWithTarget(iterator);
error: if (returnMethod != Undefined.instance) {
Object returnResult;
try {
returnResult = callReturnNode.executeCall(JSArguments.createZeroArg(iterator, returnMethod));
} catch (GraalJSException e) {
// swallow inner error
break error;
}
awaitWithNext(frame, returnResult, throwAwaitReturnResult);
}
throw Errors.createTypeErrorYieldStarThrowMethodMissing(this);
}
} else {
assert received.isReturn();
Object returnMethod = getReturnMethodNode.executeWithTarget(iterator);
if (returnMethod != Undefined.instance) {
Object innerReturnResult = callReturnMethod(returnMethod, iterator, received.getValue());
awaitWithNext(frame, innerReturnResult, returnAwaitInnerReturnResult);
} else {
awaitWithNext(frame, received.getValue(), returnAwaitReceivedValue);
}
}
break;
}
// received.[[Type]] is normal or throw
case normalOrThrowAwaitInnerResult: {
Object awaited = resumeAwait(frame);
DynamicObject innerResult = checkcastIterResult(awaited);
boolean done = iteratorCompleteNode.execute(innerResult);
if (done) {
reset(frame);
return iteratorValueNode.execute(innerResult);
}
Object iteratorValue = iteratorValueNode.execute(innerResult);
awaitWithNext(frame, iteratorValue, asyncGeneratorYieldInnerResult);
break;
}
// received.[[Type]] is return
case returnAwaitInnerReturnResult: {
Object awaited = resumeAwait(frame);
DynamicObject innerReturnResult = checkcastIterResult(awaited);
boolean done = iteratorCompleteNode.execute(innerReturnResult);
if (done) {
reset(frame);
return returnValue(frame, iteratorValueNode.execute(innerReturnResult));
}
Object iteratorValue = iteratorValueNode.execute(innerReturnResult);
awaitWithNext(frame, iteratorValue, asyncGeneratorYieldInnerResult);
break;
}
// received.[[Type]] is normal, throw, or return
// AsyncGeneratorYield, then repeat
case asyncGeneratorYieldInnerResult: {
Object awaited = resumeAwait(frame);
yieldWithNext(frame, awaited, asyncGeneratorYieldInnerResultSuspendedYield);
break;
}
case asyncGeneratorYieldInnerResultSuspendedYield: {
Completion resumptionValue = resumeYield(frame);
if (!resumptionValue.isReturn()) {
received = resumptionValue;
state = loopBegin; // repeat
break;
} else {
assert resumptionValue.isReturn();
awaitWithNext(frame, resumptionValue.getValue(), asyncGeneratorYieldInnerResultReturn);
break;
}
}
case asyncGeneratorYieldInnerResultReturn: {
Completion returnValue = resumeYield(frame);
if (returnValue.isNormal()) {
received = Completion.forReturn(returnValue.getValue());
} else {
assert returnValue.isThrow();
received = returnValue;
}
state = loopBegin; // repeat
break;
}
// received.[[Type]] is return, return method is undefined
case returnAwaitReceivedValue: {
Object awaited = resumeAwait(frame);
reset(frame);
return returnValue(frame, awaited);
}
// received.[[Type]] is throw, throw method is undefined
case throwAwaitReturnResult: {
// AsyncIteratorClose: handle Await(innerResult) throw completion.
resumeAwait(frame);
throw Errors.createTypeErrorYieldStarThrowMethodMissing(this);
}
default:
throw Errors.shouldNotReachHere();
}
// Either suspend and resume into another state or repeat from the beginning.
assert state == loopBegin;
}
}
private void awaitWithNext(VirtualFrame frame, Object value, int nextState) {
setState(frame, nextState);
suspendAwait(frame, value);
}
private Object yieldWithNext(VirtualFrame frame, Object value, int nextState) {
setState(frame, nextState);
return suspendYield(frame, value);
}
private void reset(VirtualFrame frame) {
setState(frame, 0);
writeIteratorTemp.executeWrite(frame, Undefined.instance);
}
private Object callThrowMethod(Object throwMethod, DynamicObject iterator, Object received) {
return callThrowNode.executeCall(JSArguments.createOneArg(iterator, throwMethod, received));
}
private Object callReturnMethod(Object returnMethod, DynamicObject iterator, Object received) {
return callReturnNode.executeCall(JSArguments.createOneArg(iterator, returnMethod, received));
}
private DynamicObject checkcastIterResult(Object iterResult) {
if (!JSRuntime.isObject(iterResult)) {
throw Errors.createTypeErrorIterResultNotAnObject(iterResult, this);
}
return (DynamicObject) iterResult;
}
@Override
protected JavaScriptNode copyUninitialized(Set> materializedTags) {
return createYieldStar(context, cloneUninitialized(expression, materializedTags), cloneUninitialized(readAsyncContextNode, materializedTags),
cloneUninitialized(readAsyncResultNode, materializedTags), cloneUninitialized(returnNode, materializedTags),
cloneUninitialized(readIteratorTemp, materializedTags), (WriteNode) cloneUninitialized((JavaScriptNode) writeIteratorTemp, materializedTags));
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy