Commit 91daa127 authored by adamk@chromium.org's avatar adamk@chromium.org

Revert "Implement Object.getNotifier(obj).performChange()" (r14696)

Reverts r14696 because it caused debug assertion failures when running
test/mjsunit/harmony/object-observe.js

TBR=rossberg

Review URL: https://codereview.chromium.org/15203002

git-svn-id: http://v8.googlecode.com/svn/branches/bleeding_edge@14697 ce2b1a6d-e550-0410-aec6-3dcde31c8c00
parent 07a54cd0
...@@ -98,10 +98,7 @@ var kMessages = { ...@@ -98,10 +98,7 @@ var kMessages = {
observe_non_object: ["Object.", "%0", " cannot ", "%0", " non-object"], observe_non_object: ["Object.", "%0", " cannot ", "%0", " non-object"],
observe_non_function: ["Object.", "%0", " cannot deliver to non-function"], observe_non_function: ["Object.", "%0", " cannot deliver to non-function"],
observe_callback_frozen: ["Object.observe cannot deliver to a frozen function object"], observe_callback_frozen: ["Object.observe cannot deliver to a frozen function object"],
observe_invalid_accept: ["Object.observe accept must be an array of strings."],
observe_type_non_string: ["Invalid changeRecord with non-string 'type' property"], observe_type_non_string: ["Invalid changeRecord with non-string 'type' property"],
observe_perform_non_string: ["Invalid non-string changeType"],
observe_perform_non_function: ["Cannot perform non-function"],
observe_notify_non_notifier: ["notify called on non-notifier object"], observe_notify_non_notifier: ["notify called on non-notifier object"],
proto_poison_pill: ["Generic use of __proto__ accessor not allowed"], proto_poison_pill: ["Generic use of __proto__ accessor not allowed"],
parameterless_typed_array_constr: parameterless_typed_array_constr:
......
...@@ -66,147 +66,18 @@ function CreateObjectInfo(object) { ...@@ -66,147 +66,18 @@ function CreateObjectInfo(object) {
var info = { var info = {
changeObservers: new InternalArray, changeObservers: new InternalArray,
notifier: null, notifier: null,
inactiveObservers: new InternalArray,
performing: { __proto__: null },
performingCount: 0,
}; };
objectInfoMap.set(object, info); objectInfoMap.set(object, info);
return info; return info;
} }
var defaultAcceptTypes = { function ObjectObserve(object, callback) {
__proto__: null,
'new': true,
'updated': true,
'deleted': true,
'prototype': true,
'reconfigured': true
};
function CreateObserver(callback, accept) {
var observer = {
__proto__: null,
callback: callback,
accept: defaultAcceptTypes
};
if (IS_UNDEFINED(accept))
return observer;
var acceptMap = { __proto__: null };
for (var i = 0; i < accept.length; i++)
acceptMap[accept[i]] = true;
observer.accept = acceptMap;
return observer;
}
function ObserverIsActive(observer, objectInfo) {
if (objectInfo.performingCount === 0)
return true;
var performing = objectInfo.performing;
for (var type in performing) {
if (performing[type] > 0 && observer.accept[type])
return false;
}
return true;
}
function ObserverIsInactive(observer, objectInfo) {
return !ObserverIsActive(observer, objectInfo);
}
function RemoveNullElements(from) {
var i = 0;
var j = 0;
for (; i < from.length; i++) {
if (from[i] === null)
continue;
if (j < i)
from[j] = from[i];
j++;
}
if (i !== j)
from.length = from.length - (i - j);
}
function RepartitionObservers(conditionFn, from, to, objectInfo) {
var anyRemoved = false;
for (var i = 0; i < from.length; i++) {
var observer = from[i];
if (conditionFn(observer, objectInfo)) {
anyRemoved = true;
from[i] = null;
to.push(observer);
}
}
if (anyRemoved)
RemoveNullElements(from);
}
function BeginPerformChange(objectInfo, type) {
objectInfo.performing[type] = (objectInfo.performing[type] || 0) + 1;
objectInfo.performingCount++;
RepartitionObservers(ObserverIsInactive,
objectInfo.changeObservers,
objectInfo.inactiveObservers,
objectInfo);
}
function EndPerformChange(objectInfo, type) {
objectInfo.performing[type]--;
objectInfo.performingCount--;
RepartitionObservers(ObserverIsActive,
objectInfo.inactiveObservers,
objectInfo.changeObservers,
objectInfo);
}
function ensureObserverRemoved(objectInfo, callback) {
function remove(observerList) {
for (var i = 0; i < observerList.length; i++) {
if (observerList[i].callback === callback) {
observerList.splice(i, 1);
return true;
}
}
return false;
}
if (!remove(objectInfo.changeObservers))
remove(objectInfo.inactiveObservers);
}
function AcceptArgIsValid(arg) {
if (IS_UNDEFINED(arg))
return true;
if (!IS_SPEC_OBJECT(arg) ||
!IS_NUMBER(arg.length) ||
arg.length < 0)
return false;
var length = arg.length;
for (var i = 0; i < length; i++) {
if (!IS_STRING(arg[i]))
return false;
}
return true;
}
function ObjectObserve(object, callback, accept) {
if (!IS_SPEC_OBJECT(object)) if (!IS_SPEC_OBJECT(object))
throw MakeTypeError("observe_non_object", ["observe"]); throw MakeTypeError("observe_non_object", ["observe"]);
if (!IS_SPEC_FUNCTION(callback)) if (!IS_SPEC_FUNCTION(callback))
throw MakeTypeError("observe_non_function", ["observe"]); throw MakeTypeError("observe_non_function", ["observe"]);
if (ObjectIsFrozen(callback)) if (ObjectIsFrozen(callback))
throw MakeTypeError("observe_callback_frozen"); throw MakeTypeError("observe_callback_frozen");
if (!AcceptArgIsValid(accept))
throw MakeTypeError("observe_accept_invalid");
if (!observerInfoMap.has(callback)) { if (!observerInfoMap.has(callback)) {
observerInfoMap.set(callback, { observerInfoMap.set(callback, {
...@@ -219,13 +90,8 @@ function ObjectObserve(object, callback, accept) { ...@@ -219,13 +90,8 @@ function ObjectObserve(object, callback, accept) {
if (IS_UNDEFINED(objectInfo)) objectInfo = CreateObjectInfo(object); if (IS_UNDEFINED(objectInfo)) objectInfo = CreateObjectInfo(object);
%SetIsObserved(object, true); %SetIsObserved(object, true);
ensureObserverRemoved(objectInfo, callback); var changeObservers = objectInfo.changeObservers;
if (changeObservers.indexOf(callback) < 0) changeObservers.push(callback);
var observer = CreateObserver(callback, accept);
if (ObserverIsActive(observer, objectInfo))
objectInfo.changeObservers.push(observer);
else
objectInfo.inactiveObservers.push(observer);
return object; return object;
} }
...@@ -240,11 +106,11 @@ function ObjectUnobserve(object, callback) { ...@@ -240,11 +106,11 @@ function ObjectUnobserve(object, callback) {
if (IS_UNDEFINED(objectInfo)) if (IS_UNDEFINED(objectInfo))
return object; return object;
ensureObserverRemoved(objectInfo, callback); var changeObservers = objectInfo.changeObservers;
var index = changeObservers.indexOf(callback);
if (objectInfo.changeObservers.length === 0 && if (index >= 0) {
objectInfo.inactiveObservers.length === 0) { changeObservers.splice(index, 1);
%SetIsObserved(object, false); if (changeObservers.length === 0) %SetIsObserved(object, false);
} }
return object; return object;
...@@ -256,12 +122,8 @@ function EnqueueChangeRecord(changeRecord, observers) { ...@@ -256,12 +122,8 @@ 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];
if (IS_UNDEFINED(observer.accept[changeRecord.type])) var observerInfo = observerInfoMap.get(observer);
continue; observationState.pendingObservers[observerInfo.priority] = observer;
var callback = observer.callback;
var observerInfo = observerInfoMap.get(callback);
observationState.pendingObservers[observerInfo.priority] = callback;
%SetObserverDeliveryPending(); %SetObserverDeliveryPending();
if (IS_NULL(observerInfo.pendingChangeRecords)) { if (IS_NULL(observerInfo.pendingChangeRecords)) {
observerInfo.pendingChangeRecords = new InternalArray(changeRecord); observerInfo.pendingChangeRecords = new InternalArray(changeRecord);
...@@ -273,9 +135,6 @@ function EnqueueChangeRecord(changeRecord, observers) { ...@@ -273,9 +135,6 @@ function EnqueueChangeRecord(changeRecord, observers) {
function NotifyChange(type, object, name, oldValue) { function NotifyChange(type, object, name, oldValue) {
var objectInfo = objectInfoMap.get(object); var objectInfo = objectInfoMap.get(object);
if (objectInfo.changeObservers.length === 0)
return;
var changeRecord = (arguments.length < 4) ? var changeRecord = (arguments.length < 4) ?
{ type: type, object: object, name: name } : { type: type, object: object, name: name } :
{ type: type, object: object, name: name, oldValue: oldValue }; { type: type, object: object, name: name, oldValue: oldValue };
...@@ -314,36 +173,6 @@ function ObjectNotifierNotify(changeRecord) { ...@@ -314,36 +173,6 @@ function ObjectNotifierNotify(changeRecord) {
EnqueueChangeRecord(newRecord, objectInfo.changeObservers); EnqueueChangeRecord(newRecord, objectInfo.changeObservers);
} }
function ObjectNotifierPerformChange(changeType, changeFn, receiver) {
if (!IS_SPEC_OBJECT(this))
throw MakeTypeError("called_on_non_object", ["performChange"]);
var target = notifierTargetMap.get(this);
if (IS_UNDEFINED(target))
throw MakeTypeError("observe_notify_non_notifier");
if (!IS_STRING(changeType))
throw MakeTypeError("observe_perform_non_string");
if (!IS_SPEC_FUNCTION(changeFn))
throw MakeTypeError("observe_perform_non_function");
if (IS_NULL_OR_UNDEFINED(receiver)) {
receiver = %GetDefaultReceiver(changeFn) || receiver;
} else if (!IS_SPEC_OBJECT(receiver) && %IsClassicModeFunction(changeFn)) {
receiver = ToObject(receiver);
}
var objectInfo = objectInfoMap.get(target);
if (IS_UNDEFINED(objectInfo))
return;
BeginPerformChange(objectInfo, changeType);
try {
%_CallFunction(receiver, changeFn);
} finally {
EndPerformChange(objectInfo, changeType);
}
}
function ObjectGetNotifier(object) { function ObjectGetNotifier(object) {
if (!IS_SPEC_OBJECT(object)) if (!IS_SPEC_OBJECT(object))
throw MakeTypeError("observe_non_object", ["getNotifier"]); throw MakeTypeError("observe_non_object", ["getNotifier"]);
...@@ -406,8 +235,7 @@ function SetupObjectObserve() { ...@@ -406,8 +235,7 @@ function SetupObjectObserve() {
"unobserve", ObjectUnobserve "unobserve", ObjectUnobserve
)); ));
InstallFunctions(notifierPrototype, DONT_ENUM, $Array( InstallFunctions(notifierPrototype, DONT_ENUM, $Array(
"notify", ObjectNotifierNotify, "notify", ObjectNotifierNotify
"performChange", ObjectNotifierPerformChange
)); ));
} }
......
This diff is collapsed.
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