PromiseOperations.js [plain text]
@globalPrivate
function newPromiseReaction(promiseOrCapability, onFulfilled, onRejected)
{
"use strict";
return {
@promiseOrCapability: promiseOrCapability,
@onFulfilled: onFulfilled,
@onRejected: onRejected,
@next: @undefined,
};
}
@globalPrivate
function newPromiseCapabilitySlow(constructor)
{
var promiseCapability = {
@resolve: @undefined,
@reject: @undefined,
@promise: @undefined,
};
var promise = new constructor((resolve, reject) => {
if (promiseCapability.@resolve !== @undefined)
@throwTypeError("resolve function is already set");
if (promiseCapability.@reject !== @undefined)
@throwTypeError("reject function is already set");
promiseCapability.@resolve = resolve;
promiseCapability.@reject = reject;
});
if (!@isCallable(promiseCapability.@resolve))
@throwTypeError("executor did not take a resolve function");
if (!@isCallable(promiseCapability.@reject))
@throwTypeError("executor did not take a reject function");
promiseCapability.@promise = promise;
return promiseCapability;
}
@globalPrivate
function newPromiseCapability(constructor)
{
"use strict";
if (constructor === @Promise) {
var promise = @newPromise();
var capturedPromise = promise;
function @resolve(resolution) {
return @resolvePromiseWithFirstResolvingFunctionCallCheck(capturedPromise, resolution);
}
function @reject(reason) {
return @rejectPromiseWithFirstResolvingFunctionCallCheck(capturedPromise, reason);
}
return { @resolve, @reject, @promise: promise };
}
return @newPromiseCapabilitySlow(constructor);
}
@globalPrivate
function promiseResolve(constructor, value)
{
if (@isPromise(value) && value.constructor === constructor)
return value;
if (constructor === @Promise) {
var promise = @newPromise();
@resolvePromiseWithFirstResolvingFunctionCallCheck(promise, value);
return promise;
}
return @promiseResolveSlow(constructor, value);
}
@globalPrivate
function promiseResolveSlow(constructor, value)
{
@assert(constructor !== @Promise);
var promiseCapability = @newPromiseCapabilitySlow(constructor);
promiseCapability.@resolve.@call(@undefined, value);
return promiseCapability.@promise;
}
@globalPrivate
function promiseRejectSlow(constructor, reason)
{
@assert(constructor !== @Promise);
var promiseCapability = @newPromiseCapabilitySlow(constructor);
promiseCapability.@reject.@call(@undefined, reason);
return promiseCapability.@promise;
}
@globalPrivate
function newHandledRejectedPromise(error)
{
"use strict";
var promise = @newPromise();
@rejectPromiseWithFirstResolvingFunctionCallCheck(promise, error);
@putPromiseInternalField(promise, @promiseFieldFlags, @getPromiseInternalField(promise, @promiseFieldFlags) | @promiseFlagsIsHandled);
return promise;
}
@globalPrivate
function triggerPromiseReactions(state, reactions, argument)
{
"use strict";
var previous = @undefined;
var current = reactions;
while (current) {
var next = current.@next;
current.@next = previous;
previous = current;
current = next;
}
reactions = previous;
current = reactions;
while (current) {
@enqueueJob(@promiseReactionJob, state, current, argument);
current = current.@next;
}
}
@globalPrivate
function resolvePromise(promise, resolution)
{
"use strict";
@assert(@isPromise(promise));
if (resolution === promise)
return @rejectPromise(promise, @makeTypeError("Cannot resolve a promise with itself"));
if (!@isObject(resolution))
return @fulfillPromise(promise, resolution);
var then;
try {
then = resolution.then;
} catch (error) {
return @rejectPromise(promise, error);
}
if (@isPromise(resolution) && then === @defaultPromiseThen) {
@enqueueJob(@promiseResolveThenableJobFast, resolution, promise);
return;
}
if (!@isCallable(then))
return @fulfillPromise(promise, resolution);
@enqueueJob(@promiseResolveThenableJob, resolution, then, @createResolvingFunctions(promise));
}
@globalPrivate
function rejectPromise(promise, reason)
{
"use strict";
@assert(@isPromise(promise));
@assert((@getPromiseInternalField(promise, @promiseFieldFlags) & @promiseStateMask) == @promiseStatePending);
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
var reactions = @getPromiseInternalField(promise, @promiseFieldReactionsOrResult);
@putPromiseInternalField(promise, @promiseFieldReactionsOrResult, reason);
@putPromiseInternalField(promise, @promiseFieldFlags, flags | @promiseStateRejected);
@InspectorInstrumentation.promiseRejected(promise, reason, reactions);
if (!(flags & @promiseFlagsIsHandled))
@hostPromiseRejectionTracker(promise, @promiseRejectionReject);
@triggerPromiseReactions(@promiseStateRejected, reactions, reason);
}
@globalPrivate
function fulfillPromise(promise, value)
{
"use strict";
@assert(@isPromise(promise));
@assert((@getPromiseInternalField(promise, @promiseFieldFlags) & @promiseStateMask) == @promiseStatePending);
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
var reactions = @getPromiseInternalField(promise, @promiseFieldReactionsOrResult);
@putPromiseInternalField(promise, @promiseFieldReactionsOrResult, value);
@putPromiseInternalField(promise, @promiseFieldFlags, flags | @promiseStateFulfilled);
@InspectorInstrumentation.promiseFulfilled(promise, value, reactions);
@triggerPromiseReactions(@promiseStateFulfilled, reactions, value);
}
@globalPrivate
function resolvePromiseWithFirstResolvingFunctionCallCheck(promise, value)
{
@assert(@isPromise(promise));
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
if (flags & @promiseFlagsIsFirstResolvingFunctionCalled)
return;
@putPromiseInternalField(promise, @promiseFieldFlags, flags | @promiseFlagsIsFirstResolvingFunctionCalled);
return @resolvePromise(promise, value);
}
@globalPrivate
function fulfillPromiseWithFirstResolvingFunctionCallCheck(promise, value)
{
@assert(@isPromise(promise));
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
if (flags & @promiseFlagsIsFirstResolvingFunctionCalled)
return;
@putPromiseInternalField(promise, @promiseFieldFlags, flags | @promiseFlagsIsFirstResolvingFunctionCalled);
return @fulfillPromise(promise, value);
}
@globalPrivate
function rejectPromiseWithFirstResolvingFunctionCallCheck(promise, reason)
{
@assert(@isPromise(promise));
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
if (flags & @promiseFlagsIsFirstResolvingFunctionCalled)
return;
@putPromiseInternalField(promise, @promiseFieldFlags, flags | @promiseFlagsIsFirstResolvingFunctionCalled);
return @rejectPromise(promise, reason);
}
@globalPrivate
function createResolvingFunctions(promise)
{
"use strict";
@assert(@isPromise(promise));
var alreadyResolved = false;
var resolve = (0, (resolution) => {
if (alreadyResolved)
return @undefined;
alreadyResolved = true;
return @resolvePromise(promise, resolution);
});
var reject = (0, (reason) => {
if (alreadyResolved)
return @undefined;
alreadyResolved = true;
return @rejectPromise(promise, reason);
});
return { @resolve: resolve, @reject: reject };
}
@globalPrivate
function promiseReactionJobWithoutPromise(handler, argument)
{
"use strict";
try {
handler(argument);
} catch {
}
}
@globalPrivate
function resolveWithoutPromise(resolution, onFulfilled, onRejected)
{
"use strict";
if (!@isObject(resolution)) {
@fulfillWithoutPromise(resolution, onFulfilled, onRejected);
return;
}
var then;
try {
then = resolution.then;
} catch (error) {
@rejectWithoutPromise(error, onFulfilled, onRejected);
return;
}
if (@isPromise(resolution) && then === @defaultPromiseThen) {
@enqueueJob(@promiseResolveThenableJobWithoutPromiseFast, resolution, onFulfilled, onRejected);
return;
}
if (!@isCallable(then)) {
@fulfillWithoutPromise(resolution, onFulfilled, onRejected);
return;
}
@enqueueJob(@promiseResolveThenableJob, resolution, then, @createResolvingFunctionsWithoutPromise(onFulfilled, onRejected));
}
@globalPrivate
function rejectWithoutPromise(reason, onFulfilled, onRejected)
{
"use strict";
@enqueueJob(@promiseReactionJobWithoutPromise, onRejected, reason);
}
@globalPrivate
function fulfillWithoutPromise(value, onFulfilled, onRejected)
{
"use strict";
@enqueueJob(@promiseReactionJobWithoutPromise, onFulfilled, value);
}
@globalPrivate
function createResolvingFunctionsWithoutPromise(onFulfilled, onRejected)
{
"use strict";
var alreadyResolved = false;
var resolve = (0, (resolution) => {
if (alreadyResolved)
return @undefined;
alreadyResolved = true;
@resolveWithoutPromise(resolution, onFulfilled, onRejected);
});
var reject = (0, (reason) => {
if (alreadyResolved)
return @undefined;
alreadyResolved = true;
@rejectWithoutPromise(reason, onFulfilled, onRejected);
});
return { @resolve: resolve, @reject: reject };
}
@globalPrivate
function promiseReactionJob(state, reaction, argument)
{
"use strict";
var promiseOrCapability = reaction.@promiseOrCapability;
if (@isUndefinedOrNull(reaction.@onRejected)) {
@assert(@isUndefinedOrNull(reaction.@onFulfilled));
try {
@assert(@isPromise(promiseOrCapability));
if (state === @promiseStateFulfilled)
@resolvePromise(promiseOrCapability, argument);
else
@rejectPromise(promiseOrCapability, argument);
} catch {
}
return;
}
var handler = (state === @promiseStateFulfilled) ? reaction.@onFulfilled: reaction.@onRejected;
if (!promiseOrCapability) {
@promiseReactionJobWithoutPromise(handler, argument);
return;
}
var result;
try {
result = handler(argument);
} catch (error) {
if (@isPromise(promiseOrCapability)) {
@rejectPromise(promiseOrCapability, error);
return;
}
promiseOrCapability.@reject.@call(@undefined, error);
return;
}
if (@isPromise(promiseOrCapability)) {
@resolvePromise(promiseOrCapability, result);
return;
}
promiseOrCapability.@resolve.@call(@undefined, result);
}
@globalPrivate
function promiseResolveThenableJobFast(thenable, promiseToResolve)
{
"use strict";
@assert(@isPromise(thenable));
@assert(@isPromise(promiseToResolve));
var constructor = @speciesConstructor(thenable, @Promise);
if (constructor !== @Promise && constructor !== @InternalPromise) {
@promiseResolveThenableJobWithDerivedPromise(thenable, constructor, @createResolvingFunctions(promiseToResolve));
return;
}
var flags = @getPromiseInternalField(thenable, @promiseFieldFlags);
var state = flags & @promiseStateMask;
var reaction = @newPromiseReaction(promiseToResolve, @undefined, @undefined);
if (state === @promiseStatePending) {
reaction.@next = @getPromiseInternalField(thenable, @promiseFieldReactionsOrResult);
@putPromiseInternalField(thenable, @promiseFieldReactionsOrResult, reaction);
} else {
if (state === @promiseStateRejected && !(flags & @promiseFlagsIsHandled))
@hostPromiseRejectionTracker(thenable, @promiseRejectionHandle);
@enqueueJob(@promiseReactionJob, state, reaction, @getPromiseInternalField(thenable, @promiseFieldReactionsOrResult));
}
@putPromiseInternalField(thenable, @promiseFieldFlags, @getPromiseInternalField(thenable, @promiseFieldFlags) | @promiseFlagsIsHandled);
}
@globalPrivate
function promiseResolveThenableJobWithoutPromiseFast(thenable, onFulfilled, onRejected)
{
"use strict";
@assert(@isPromise(thenable));
var constructor = @speciesConstructor(thenable, @Promise);
if (constructor !== @Promise && constructor !== @InternalPromise) {
@promiseResolveThenableJobWithDerivedPromise(thenable, constructor, @createResolvingFunctionsWithoutPromise(onFulfilled, onRejected));
return;
}
var flags = @getPromiseInternalField(thenable, @promiseFieldFlags);
var state = flags & @promiseStateMask;
if (state === @promiseStatePending) {
var reaction = @newPromiseReaction(@undefined, onFulfilled, onRejected);
reaction.@next = @getPromiseInternalField(thenable, @promiseFieldReactionsOrResult);
@putPromiseInternalField(thenable, @promiseFieldReactionsOrResult, reaction);
} else {
var result = @getPromiseInternalField(thenable, @promiseFieldReactionsOrResult);
if (state === @promiseStateRejected) {
if (!(flags & @promiseFlagsIsHandled))
@hostPromiseRejectionTracker(thenable, @promiseRejectionHandle);
@rejectWithoutPromise(result, onFulfilled, onRejected);
} else
@fulfillWithoutPromise(result, onFulfilled, onRejected);
}
@putPromiseInternalField(thenable, @promiseFieldFlags, @getPromiseInternalField(thenable, @promiseFieldFlags) | @promiseFlagsIsHandled);
}
@globalPrivate
function promiseResolveThenableJob(thenable, then, resolvingFunctions)
{
"use strict";
try {
return then.@call(thenable, resolvingFunctions.@resolve, resolvingFunctions.@reject);
} catch (error) {
return resolvingFunctions.@reject.@call(@undefined, error);
}
}
@globalPrivate
function promiseResolveThenableJobWithDerivedPromise(thenable, constructor, resolvingFunctions)
{
"use strict";
try {
var promiseOrCapability = @newPromiseCapabilitySlow(constructor);
@performPromiseThen(thenable, resolvingFunctions.@resolve, resolvingFunctions.@reject, promiseOrCapability);
return promiseOrCapability.@promise;
} catch (error) {
return resolvingFunctions.@reject.@call(@undefined, error);
}
}
@globalPrivate
function performPromiseThen(promise, onFulfilled, onRejected, promiseOrCapability)
{
"use strict";
if (!@isCallable(onFulfilled))
onFulfilled = function (argument) { return argument; };
if (!@isCallable(onRejected))
onRejected = function (argument) { throw argument; };
var reaction = @newPromiseReaction(promiseOrCapability, onFulfilled, onRejected);
var flags = @getPromiseInternalField(promise, @promiseFieldFlags);
var state = flags & @promiseStateMask;
if (state === @promiseStatePending) {
reaction.@next = @getPromiseInternalField(promise, @promiseFieldReactionsOrResult);
@putPromiseInternalField(promise, @promiseFieldReactionsOrResult, reaction);
} else {
if (state === @promiseStateRejected && !(flags & @promiseFlagsIsHandled))
@hostPromiseRejectionTracker(promise, @promiseRejectionHandle);
@enqueueJob(@promiseReactionJob, state, reaction, @getPromiseInternalField(promise, @promiseFieldReactionsOrResult));
}
@putPromiseInternalField(promise, @promiseFieldFlags, @getPromiseInternalField(promise, @promiseFieldFlags) | @promiseFlagsIsHandled);
}