Commit 1d4fe002 authored by gsathya's avatar gsathya Committed by Commit bot

Promises: Lazily create arrays to store resolve, reject callbacks

For the common use case of having a single resolve or reject callback,
the callbacks are stored directly. Only when an additional callback is
registered, we create an array to store these callbacks.

There are 3 possible states for the resolve, reject symbols when we add
a new callback --
1) UNDEFINED -- This is the zero state where there is no callback
registered. When we see this state, we directly attach the callbacks to
the symbol.
2) !IS_ARRAY -- There is a single callback directly attached to the
symbols. We need to create a new array to store additional callbacks.
3) IS_ARRAY -- There are multiple callbacks already registered,
therefore we can just push the new callback to the existing array.

Also, this change creates a new symbol for storing the deferred objects.
Previously the deferred objects were stored in the callback arrays, but
since we no longer create arrays for the initial case, we need this new
symbol. The cctest has been updated to account for this new symbol.

This patch results in a 19% improvement(over 5 runs) in the bluebird benchmark.

BUG=v8:5046

Review-Url: https://codereview.chromium.org/2007803002
Cr-Commit-Position: refs/heads/master@{#36536}
parent ffdd76e6
...@@ -176,6 +176,7 @@ ...@@ -176,6 +176,7 @@
V(promise_combined_deferred_symbol) \ V(promise_combined_deferred_symbol) \
V(promise_debug_marker_symbol) \ V(promise_debug_marker_symbol) \
V(promise_has_handler_symbol) \ V(promise_has_handler_symbol) \
V(promise_deferred_reactions_symbol) \
V(promise_fulfill_reactions_symbol) \ V(promise_fulfill_reactions_symbol) \
V(promise_reject_reactions_symbol) \ V(promise_reject_reactions_symbol) \
V(promise_raw_symbol) \ V(promise_raw_symbol) \
......
...@@ -21,6 +21,8 @@ var promiseRejectReactionsSymbol = ...@@ -21,6 +21,8 @@ var promiseRejectReactionsSymbol =
utils.ImportNow("promise_reject_reactions_symbol"); utils.ImportNow("promise_reject_reactions_symbol");
var promiseFulfillReactionsSymbol = var promiseFulfillReactionsSymbol =
utils.ImportNow("promise_fulfill_reactions_symbol"); utils.ImportNow("promise_fulfill_reactions_symbol");
var promiseDeferredReactionsSymbol =
utils.ImportNow("promise_deferred_reactions_symbol");
var promiseRawSymbol = utils.ImportNow("promise_raw_symbol"); var promiseRawSymbol = utils.ImportNow("promise_raw_symbol");
var promiseStateSymbol = utils.ImportNow("promise_state_symbol"); var promiseStateSymbol = utils.ImportNow("promise_state_symbol");
var promiseResultSymbol = utils.ImportNow("promise_result_symbol"); var promiseResultSymbol = utils.ImportNow("promise_result_symbol");
...@@ -98,11 +100,22 @@ var GlobalPromise = function Promise(executor) { ...@@ -98,11 +100,22 @@ var GlobalPromise = function Promise(executor) {
// Core functionality. // Core functionality.
function PromiseSet(promise, status, value, onResolve, onReject) { function PromiseSet(promise, status, value) {
SET_PRIVATE(promise, promiseStateSymbol, status); SET_PRIVATE(promise, promiseStateSymbol, status);
SET_PRIVATE(promise, promiseResultSymbol, value); SET_PRIVATE(promise, promiseResultSymbol, value);
SET_PRIVATE(promise, promiseFulfillReactionsSymbol, onResolve);
SET_PRIVATE(promise, promiseRejectReactionsSymbol, onReject); // There are 3 possible states for the resolve, reject symbols when we add
// a new callback --
// 1) UNDEFINED -- This is the zero state where there is no callback
// registered. When we see this state, we directly attach the callbacks to
// the symbol.
// 2) !IS_ARRAY -- There is a single callback directly attached to the
// symbols. We need to create a new array to store additional callbacks.
// 3) IS_ARRAY -- There are multiple callbacks already registered,
// therefore we can just push the new callback to the existing array.
SET_PRIVATE(promise, promiseFulfillReactionsSymbol, UNDEFINED);
SET_PRIVATE(promise, promiseRejectReactionsSymbol, UNDEFINED);
SET_PRIVATE(promise, promiseDeferredReactionsSymbol, UNDEFINED);
return promise; return promise;
} }
...@@ -115,13 +128,17 @@ function PromiseCreateAndSet(status, value) { ...@@ -115,13 +128,17 @@ function PromiseCreateAndSet(status, value) {
function PromiseInit(promise) { function PromiseInit(promise) {
return PromiseSet( return PromiseSet(
promise, kPending, UNDEFINED, new InternalArray, new InternalArray) promise, kPending, UNDEFINED, UNDEFINED, UNDEFINED, UNDEFINED);
} }
function FulfillPromise(promise, status, value, promiseQueue) { function FulfillPromise(promise, status, value, promiseQueue) {
if (GET_PRIVATE(promise, promiseStateSymbol) === kPending) { if (GET_PRIVATE(promise, promiseStateSymbol) === kPending) {
var tasks = GET_PRIVATE(promise, promiseQueue); var tasks = GET_PRIVATE(promise, promiseQueue);
if (tasks.length) PromiseEnqueue(value, tasks, status); if (!IS_UNDEFINED(tasks)) {
var tasks = GET_PRIVATE(promise, promiseQueue);
var deferreds = GET_PRIVATE(promise, promiseDeferredReactionsSymbol);
PromiseEnqueue(value, tasks, deferreds, status);
}
PromiseSet(promise, status, value); PromiseSet(promise, status, value);
} }
} }
...@@ -139,14 +156,18 @@ function PromiseHandle(value, handler, deferred) { ...@@ -139,14 +156,18 @@ function PromiseHandle(value, handler, deferred) {
} }
} }
function PromiseEnqueue(value, tasks, status) { function PromiseEnqueue(value, tasks, deferreds, status) {
var id, name, instrumenting = DEBUG_IS_ACTIVE; var id, name, instrumenting = DEBUG_IS_ACTIVE;
%EnqueueMicrotask(function() { %EnqueueMicrotask(function() {
if (instrumenting) { if (instrumenting) {
%DebugAsyncTaskEvent({ type: "willHandle", id: id, name: name }); %DebugAsyncTaskEvent({ type: "willHandle", id: id, name: name });
} }
for (var i = 0; i < tasks.length; i += 2) { if (IS_ARRAY(tasks)) {
PromiseHandle(value, tasks[i], tasks[i + 1]) for (var i = 0; i < tasks.length; i += 1) {
PromiseHandle(value, tasks[i], deferreds[i]);
}
} else {
PromiseHandle(value, tasks, deferreds);
} }
if (instrumenting) { if (instrumenting) {
%DebugAsyncTaskEvent({ type: "didHandle", id: id, name: name }); %DebugAsyncTaskEvent({ type: "didHandle", id: id, name: name });
...@@ -159,6 +180,36 @@ function PromiseEnqueue(value, tasks, status) { ...@@ -159,6 +180,36 @@ function PromiseEnqueue(value, tasks, status) {
} }
} }
function PromiseAttachCallbacks(promise, deferred, onResolve, onReject) {
var maybeResolveCallbacks =
GET_PRIVATE(promise, promiseFulfillReactionsSymbol);
if (IS_UNDEFINED(maybeResolveCallbacks)) {
SET_PRIVATE(promise, promiseFulfillReactionsSymbol, onResolve);
SET_PRIVATE(promise, promiseRejectReactionsSymbol, onReject);
SET_PRIVATE(promise, promiseDeferredReactionsSymbol, deferred);
} else if (!IS_ARRAY(maybeResolveCallbacks)) {
var resolveCallbacks = new InternalArray();
var rejectCallbacks = new InternalArray();
var deferreds = new InternalArray();
resolveCallbacks.push(maybeResolveCallbacks);
rejectCallbacks.push(GET_PRIVATE(promise, promiseRejectReactionsSymbol));
deferreds.push(GET_PRIVATE(promise, promiseDeferredReactionsSymbol));
resolveCallbacks.push(onResolve);
rejectCallbacks.push(onReject);
deferreds.push(deferred);
SET_PRIVATE(promise, promiseFulfillReactionsSymbol, resolveCallbacks);
SET_PRIVATE(promise, promiseRejectReactionsSymbol, rejectCallbacks);
SET_PRIVATE(promise, promiseDeferredReactionsSymbol, deferreds);
} else {
maybeResolveCallbacks.push(onResolve);
GET_PRIVATE(promise, promiseRejectReactionsSymbol).push(onReject);
GET_PRIVATE(promise, promiseDeferredReactionsSymbol).push(deferred);
}
}
function PromiseIdResolveHandler(x) { return x } function PromiseIdResolveHandler(x) { return x }
function PromiseIdRejectHandler(r) { throw r } function PromiseIdRejectHandler(r) { throw r }
...@@ -319,14 +370,11 @@ function PromiseThen(onResolve, onReject) { ...@@ -319,14 +370,11 @@ function PromiseThen(onResolve, onReject) {
var deferred = NewPromiseCapability(constructor); var deferred = NewPromiseCapability(constructor);
switch (status) { switch (status) {
case kPending: case kPending:
GET_PRIVATE(this, promiseFulfillReactionsSymbol).push(onResolve, PromiseAttachCallbacks(this, deferred, onResolve, onReject);
deferred);
GET_PRIVATE(this, promiseRejectReactionsSymbol).push(onReject, deferred);
break; break;
case kFulfilled: case kFulfilled:
PromiseEnqueue(GET_PRIVATE(this, promiseResultSymbol), PromiseEnqueue(GET_PRIVATE(this, promiseResultSymbol),
[onResolve, deferred], onResolve, deferred, kFulfilled);
kFulfilled);
break; break;
case kRejected: case kRejected:
if (!HAS_DEFINED_PRIVATE(this, promiseHasHandlerSymbol)) { if (!HAS_DEFINED_PRIVATE(this, promiseHasHandlerSymbol)) {
...@@ -335,8 +383,7 @@ function PromiseThen(onResolve, onReject) { ...@@ -335,8 +383,7 @@ function PromiseThen(onResolve, onReject) {
%PromiseRevokeReject(this); %PromiseRevokeReject(this);
} }
PromiseEnqueue(GET_PRIVATE(this, promiseResultSymbol), PromiseEnqueue(GET_PRIVATE(this, promiseResultSymbol),
[onReject, deferred], onReject, deferred, kRejected);
kRejected);
break; break;
} }
// Mark this promise as having handler. // Mark this promise as having handler.
...@@ -445,20 +492,30 @@ function PromiseRace(iterable) { ...@@ -445,20 +492,30 @@ function PromiseRace(iterable) {
// Utility for debugger // Utility for debugger
function PromiseHasUserDefinedRejectHandlerCheck(handler, deferred) {
if (handler !== PromiseIdRejectHandler) {
var combinedDeferred = GET_PRIVATE(handler, promiseCombinedDeferredSymbol);
if (IS_UNDEFINED(combinedDeferred)) return true;
if (PromiseHasUserDefinedRejectHandlerRecursive(combinedDeferred.promise)) {
return true;
}
} else if (PromiseHasUserDefinedRejectHandlerRecursive(deferred.promise)) {
return true;
}
return false;
}
function PromiseHasUserDefinedRejectHandlerRecursive(promise) { function PromiseHasUserDefinedRejectHandlerRecursive(promise) {
var queue = GET_PRIVATE(promise, promiseRejectReactionsSymbol); var queue = GET_PRIVATE(promise, promiseRejectReactionsSymbol);
var deferreds = GET_PRIVATE(promise, promiseDeferredReactionsSymbol);
if (IS_UNDEFINED(queue)) return false; if (IS_UNDEFINED(queue)) return false;
for (var i = 0; i < queue.length; i += 2) { if (!IS_ARRAY(queue)) {
var handler = queue[i]; return PromiseHasUserDefinedRejectHandlerCheck(queue, deferreds);
if (handler !== PromiseIdRejectHandler) { } else {
var deferred = GET_PRIVATE(handler, promiseCombinedDeferredSymbol); for (var i = 0; i < queue.length; i += 1) {
if (IS_UNDEFINED(deferred)) return true; if (PromiseHasUserDefinedRejectHandlerCheck(queue[i], deferreds[i])) {
if (PromiseHasUserDefinedRejectHandlerRecursive(deferred.promise)) {
return true; return true;
} }
} else if (PromiseHasUserDefinedRejectHandlerRecursive(
queue[i + 1].promise)) {
return true;
} }
} }
return false; return false;
......
...@@ -1100,7 +1100,7 @@ TEST(SubclassPromiseBuiltin) { ...@@ -1100,7 +1100,7 @@ TEST(SubclassPromiseBuiltin) {
CcTest::InitializeVM(); CcTest::InitializeVM();
v8::HandleScope scope(CcTest::isolate()); v8::HandleScope scope(CcTest::isolate());
const int first_field = 4; const int first_field = 5;
TestSubclassBuiltin("A1", JS_PROMISE_TYPE, "Promise", TestSubclassBuiltin("A1", JS_PROMISE_TYPE, "Promise",
"function(resolve, reject) { resolve('ok'); }", "function(resolve, reject) { resolve('ok'); }",
first_field); first_field);
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment