Commit 92db2105 authored by rossberg@chromium.org's avatar rossberg@chromium.org

Object.deliverChangeRecords should remove the observer from activeObservers

To preserve ordering guarantees during end-of-turn delivery, Object.deliverChangeRecords needs to remove the delivered-to observer from the list of active observers.

The added test demonstrates this behavior.

Review URL: https://codereview.chromium.org/11410046
Patch from Adam Klein <adamk@chromium.org>.

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@12951 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 36c3d015
...@@ -35,7 +35,7 @@ if (IS_UNDEFINED(observationState.observerInfoMap)) { ...@@ -35,7 +35,7 @@ if (IS_UNDEFINED(observationState.observerInfoMap)) {
observationState.observerInfoMap = %CreateObjectHashTable(); observationState.observerInfoMap = %CreateObjectHashTable();
observationState.objectInfoMap = %CreateObjectHashTable(); observationState.objectInfoMap = %CreateObjectHashTable();
observationState.notifierTargetMap = %CreateObjectHashTable(); observationState.notifierTargetMap = %CreateObjectHashTable();
observationState.activeObservers = new InternalArray; observationState.pendingObservers = new InternalArray;
observationState.observerPriority = 0; observationState.observerPriority = 0;
} }
...@@ -119,7 +119,7 @@ function EnqueueChangeRecord(changeRecord, observers) { ...@@ -119,7 +119,7 @@ function EnqueueChangeRecord(changeRecord, observers) {
for (var i = 0; i < observers.length; i++) { for (var i = 0; i < observers.length; i++) {
var observer = observers[i]; var observer = observers[i];
var observerInfo = observerInfoMap.get(observer); var observerInfo = observerInfoMap.get(observer);
observationState.activeObservers[observerInfo.priority] = observer; observationState.pendingObservers[observerInfo.priority] = observer;
%SetObserverDeliveryPending(); %SetObserverDeliveryPending();
if (IS_NULL(observerInfo.pendingChangeRecords)) { if (IS_NULL(observerInfo.pendingChangeRecords)) {
observerInfo.pendingChangeRecords = new InternalArray(changeRecord); observerInfo.pendingChangeRecords = new InternalArray(changeRecord);
...@@ -202,6 +202,7 @@ function DeliverChangeRecordsForObserver(observer) { ...@@ -202,6 +202,7 @@ function DeliverChangeRecordsForObserver(observer) {
return; return;
observerInfo.pendingChangeRecords = null; observerInfo.pendingChangeRecords = null;
delete observationState.pendingObservers[observerInfo.priority];
var delivered = []; var delivered = [];
%MoveArrayContents(pendingChangeRecords, delivered); %MoveArrayContents(pendingChangeRecords, delivered);
try { try {
...@@ -217,11 +218,11 @@ function ObjectDeliverChangeRecords(callback) { ...@@ -217,11 +218,11 @@ function ObjectDeliverChangeRecords(callback) {
} }
function DeliverChangeRecords() { function DeliverChangeRecords() {
while (observationState.activeObservers.length) { while (observationState.pendingObservers.length) {
var activeObservers = observationState.activeObservers; var pendingObservers = observationState.pendingObservers;
observationState.activeObservers = new InternalArray; observationState.pendingObservers = new InternalArray;
for (var i in activeObservers) { for (var i in pendingObservers) {
DeliverChangeRecordsForObserver(activeObservers[i]); DeliverChangeRecordsForObserver(pendingObservers[i]);
} }
} }
} }
......
...@@ -166,6 +166,30 @@ TEST(DeliveryOrderingReentrant) { ...@@ -166,6 +166,30 @@ TEST(DeliveryOrderingReentrant) {
CHECK_EQ(2, CompileRun("ordering[1]")->Int32Value()); CHECK_EQ(2, CompileRun("ordering[1]")->Int32Value());
} }
TEST(DeliveryOrderingDeliverChangeRecords) {
HarmonyIsolate isolate;
HandleScope scope;
LocalContext context;
CompileRun(
"var obj = {};"
"var ordering = [];"
"function observer1() { ordering.push(1); if (!obj.b) obj.b = true };"
"function observer2() { ordering.push(2); };"
"Object.observe(obj, observer1);"
"Object.observe(obj, observer2);"
"obj.a = 1;"
"Object.deliverChangeRecords(observer2);");
CHECK_EQ(4, CompileRun("ordering.length")->Int32Value());
// First, observer2 is called due to deliverChangeRecords
CHECK_EQ(2, CompileRun("ordering[0]")->Int32Value());
// Then, observer1 is called when the stack unwinds
CHECK_EQ(1, CompileRun("ordering[1]")->Int32Value());
// observer1's mutation causes both 1 and 2 to be reactivated,
// with 1 having priority.
CHECK_EQ(1, CompileRun("ordering[2]")->Int32Value());
CHECK_EQ(2, CompileRun("ordering[3]")->Int32Value());
}
TEST(ObjectHashTableGrowth) { TEST(ObjectHashTableGrowth) {
HarmonyIsolate isolate; HarmonyIsolate isolate;
HandleScope scope; HandleScope scope;
......
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