Chromium Code Reviews
chromiumcodereview-hr@appspot.gserviceaccount.com (chromiumcodereview-hr) | Please choose your nickname with Settings | Help | Chromium Project | Gerrit Changes | Sign out
(55)

Side by Side Diff: src/object-observe.js

Issue 148153010: Synchronize with r15701. (Closed) Base URL: https://v8.googlecode.com/svn/branches/experimental/a64
Patch Set: Created 6 years, 10 months ago
Use n/p to move between diff chunks; N/P to move between comments. Draft comments are only viewable by you.
Jump to:
View unified diff | Download patch | Annotate | Revision Log
« no previous file with comments | « src/mksnapshot.cc ('k') | src/objects.h » ('j') | no next file with comments »
Toggle Intra-line Diffs ('i') | Expand Comments ('e') | Collapse Comments ('c') | Show Comments Hide Comments ('s')
OLDNEW
1 // Copyright 2012 the V8 project authors. All rights reserved. 1 // Copyright 2012 the V8 project authors. All rights reserved.
2 // Redistribution and use in source and binary forms, with or without 2 // Redistribution and use in source and binary forms, with or without
3 // modification, are permitted provided that the following conditions are 3 // modification, are permitted provided that the following conditions are
4 // met: 4 // met:
5 // 5 //
6 // * Redistributions of source code must retain the above copyright 6 // * Redistributions of source code must retain the above copyright
7 // notice, this list of conditions and the following disclaimer. 7 // notice, this list of conditions and the following disclaimer.
8 // * Redistributions in binary form must reproduce the above 8 // * Redistributions in binary form must reproduce the above
9 // copyright notice, this list of conditions and the following 9 // copyright notice, this list of conditions and the following
10 // disclaimer in the documentation and/or other materials provided 10 // disclaimer in the documentation and/or other materials provided
(...skipping 10 matching lines...) Expand all
21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT 21 // SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, 22 // LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY 23 // DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT 24 // THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE 25 // (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. 26 // OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
27 27
28 "use strict"; 28 "use strict";
29 29
30 var observationState = %GetObservationState(); 30 var observationState = %GetObservationState();
31 if (IS_UNDEFINED(observationState.observerInfoMap)) { 31 if (IS_UNDEFINED(observationState.callbackInfoMap)) {
32 observationState.observerInfoMap = %ObservationWeakMapCreate(); 32 observationState.callbackInfoMap = %ObservationWeakMapCreate();
33 observationState.objectInfoMap = %ObservationWeakMapCreate(); 33 observationState.objectInfoMap = %ObservationWeakMapCreate();
34 observationState.notifierTargetMap = %ObservationWeakMapCreate(); 34 observationState.notifierTargetMap = %ObservationWeakMapCreate();
35 observationState.pendingObservers = new InternalArray; 35 observationState.pendingObservers = new InternalArray;
36 observationState.observerPriority = 0; 36 observationState.nextCallbackPriority = 0;
37 } 37 }
38 38
39 function ObservationWeakMap(map) { 39 function ObservationWeakMap(map) {
40 this.map_ = map; 40 this.map_ = map;
41 } 41 }
42 42
43 ObservationWeakMap.prototype = { 43 ObservationWeakMap.prototype = {
44 get: function(key) { 44 get: function(key) {
45 key = %UnwrapGlobalProxy(key); 45 key = %UnwrapGlobalProxy(key);
46 if (!IS_SPEC_OBJECT(key)) return void 0; 46 if (!IS_SPEC_OBJECT(key)) return void 0;
47 return %WeakMapGet(this.map_, key); 47 return %WeakMapGet(this.map_, key);
48 }, 48 },
49 set: function(key, value) { 49 set: function(key, value) {
50 key = %UnwrapGlobalProxy(key); 50 key = %UnwrapGlobalProxy(key);
51 if (!IS_SPEC_OBJECT(key)) return void 0; 51 if (!IS_SPEC_OBJECT(key)) return void 0;
52 %WeakMapSet(this.map_, key, value); 52 %WeakMapSet(this.map_, key, value);
53 }, 53 },
54 has: function(key) { 54 has: function(key) {
55 return !IS_UNDEFINED(this.get(key)); 55 return !IS_UNDEFINED(this.get(key));
56 } 56 }
57 }; 57 };
58 58
59 var observerInfoMap = 59 var callbackInfoMap =
60 new ObservationWeakMap(observationState.observerInfoMap); 60 new ObservationWeakMap(observationState.callbackInfoMap);
61 var objectInfoMap = new ObservationWeakMap(observationState.objectInfoMap); 61 var objectInfoMap = new ObservationWeakMap(observationState.objectInfoMap);
62 var notifierTargetMap = 62 var notifierTargetMap =
63 new ObservationWeakMap(observationState.notifierTargetMap); 63 new ObservationWeakMap(observationState.notifierTargetMap);
64 64
65 function CreateObjectInfo(object) { 65 function CreateObjectInfo(object) {
66 var info = { 66 var info = {
67 changeObservers: new InternalArray, 67 changeObservers: new InternalArray,
68 notifier: null, 68 notifier: null,
69 inactiveObservers: new InternalArray, 69 inactiveObservers: new InternalArray,
70 performing: { __proto__: null }, 70 performing: { __proto__: null },
(...skipping 120 matching lines...) Expand 10 before | Expand all | Expand 10 after
191 return false; 191 return false;
192 192
193 var length = arg.length; 193 var length = arg.length;
194 for (var i = 0; i < length; i++) { 194 for (var i = 0; i < length; i++) {
195 if (!IS_STRING(arg[i])) 195 if (!IS_STRING(arg[i]))
196 return false; 196 return false;
197 } 197 }
198 return true; 198 return true;
199 } 199 }
200 200
201 function EnsureCallbackPriority(callback) {
202 if (!callbackInfoMap.has(callback))
203 callbackInfoMap.set(callback, observationState.nextCallbackPriority++);
204 }
205
206 function NormalizeCallbackInfo(callback) {
207 var callbackInfo = callbackInfoMap.get(callback);
208 if (IS_NUMBER(callbackInfo)) {
209 var priority = callbackInfo;
210 callbackInfo = new InternalArray;
211 callbackInfo.priority = priority;
212 callbackInfoMap.set(callback, callbackInfo);
213 }
214 return callbackInfo;
215 }
216
201 function ObjectObserve(object, callback, accept) { 217 function ObjectObserve(object, callback, accept) {
202 if (!IS_SPEC_OBJECT(object)) 218 if (!IS_SPEC_OBJECT(object))
203 throw MakeTypeError("observe_non_object", ["observe"]); 219 throw MakeTypeError("observe_non_object", ["observe"]);
204 if (!IS_SPEC_FUNCTION(callback)) 220 if (!IS_SPEC_FUNCTION(callback))
205 throw MakeTypeError("observe_non_function", ["observe"]); 221 throw MakeTypeError("observe_non_function", ["observe"]);
206 if (ObjectIsFrozen(callback)) 222 if (ObjectIsFrozen(callback))
207 throw MakeTypeError("observe_callback_frozen"); 223 throw MakeTypeError("observe_callback_frozen");
208 if (!AcceptArgIsValid(accept)) 224 if (!AcceptArgIsValid(accept))
209 throw MakeTypeError("observe_accept_invalid"); 225 throw MakeTypeError("observe_accept_invalid");
210 226
211 if (!observerInfoMap.has(callback)) { 227 EnsureCallbackPriority(callback);
212 observerInfoMap.set(callback, {
213 pendingChangeRecords: null,
214 priority: observationState.observerPriority++,
215 });
216 }
217 228
218 var objectInfo = objectInfoMap.get(object); 229 var objectInfo = objectInfoMap.get(object);
219 if (IS_UNDEFINED(objectInfo)) objectInfo = CreateObjectInfo(object); 230 if (IS_UNDEFINED(objectInfo)) {
220 %SetIsObserved(object, true); 231 objectInfo = CreateObjectInfo(object);
232 %SetIsObserved(object);
233 }
221 234
222 EnsureObserverRemoved(objectInfo, callback); 235 EnsureObserverRemoved(objectInfo, callback);
223 236
224 var observer = CreateObserver(callback, accept); 237 var observer = CreateObserver(callback, accept);
225 if (ObserverIsActive(observer, objectInfo)) 238 if (ObserverIsActive(observer, objectInfo))
226 objectInfo.changeObservers.push(observer); 239 objectInfo.changeObservers.push(observer);
227 else 240 else
228 objectInfo.inactiveObservers.push(observer); 241 objectInfo.inactiveObservers.push(observer);
229 242
230 return object; 243 return object;
231 } 244 }
232 245
233 function ObjectUnobserve(object, callback) { 246 function ObjectUnobserve(object, callback) {
234 if (!IS_SPEC_OBJECT(object)) 247 if (!IS_SPEC_OBJECT(object))
235 throw MakeTypeError("observe_non_object", ["unobserve"]); 248 throw MakeTypeError("observe_non_object", ["unobserve"]);
236 if (!IS_SPEC_FUNCTION(callback)) 249 if (!IS_SPEC_FUNCTION(callback))
237 throw MakeTypeError("observe_non_function", ["unobserve"]); 250 throw MakeTypeError("observe_non_function", ["unobserve"]);
238 251
239 var objectInfo = objectInfoMap.get(object); 252 var objectInfo = objectInfoMap.get(object);
240 if (IS_UNDEFINED(objectInfo)) 253 if (IS_UNDEFINED(objectInfo))
241 return object; 254 return object;
242 255
243 EnsureObserverRemoved(objectInfo, callback); 256 EnsureObserverRemoved(objectInfo, callback);
244
245 if (objectInfo.changeObservers.length === 0 &&
246 objectInfo.inactiveObservers.length === 0) {
247 %SetIsObserved(object, false);
248 }
249
250 return object; 257 return object;
251 } 258 }
252 259
253 function ArrayObserve(object, callback) { 260 function ArrayObserve(object, callback) {
254 return ObjectObserve(object, callback, ['new', 261 return ObjectObserve(object, callback, ['new',
255 'updated', 262 'updated',
256 'deleted', 263 'deleted',
257 'splice']); 264 'splice']);
258 } 265 }
259 266
260 function ArrayUnobserve(object, callback) { 267 function ArrayUnobserve(object, callback) {
261 return ObjectUnobserve(object, callback); 268 return ObjectUnobserve(object, callback);
262 } 269 }
263 270
271 function EnqueueToCallback(callback, changeRecord) {
272 var callbackInfo = NormalizeCallbackInfo(callback);
273 observationState.pendingObservers[callbackInfo.priority] = callback;
274 callbackInfo.push(changeRecord);
275 %SetObserverDeliveryPending();
276 }
277
264 function EnqueueChangeRecord(changeRecord, observers) { 278 function EnqueueChangeRecord(changeRecord, observers) {
265 // TODO(rossberg): adjust once there is a story for symbols vs proxies. 279 // TODO(rossberg): adjust once there is a story for symbols vs proxies.
266 if (IS_SYMBOL(changeRecord.name)) return; 280 if (IS_SYMBOL(changeRecord.name)) return;
267 281
268 for (var i = 0; i < observers.length; i++) { 282 for (var i = 0; i < observers.length; i++) {
269 var observer = observers[i]; 283 var observer = observers[i];
270 if (IS_UNDEFINED(observer.accept[changeRecord.type])) 284 if (IS_UNDEFINED(observer.accept[changeRecord.type]))
271 continue; 285 continue;
272 286
273 var callback = observer.callback; 287 EnqueueToCallback(observer.callback, changeRecord);
274 var observerInfo = observerInfoMap.get(callback);
275 observationState.pendingObservers[observerInfo.priority] = callback;
276 %SetObserverDeliveryPending();
277 if (IS_NULL(observerInfo.pendingChangeRecords)) {
278 observerInfo.pendingChangeRecords = new InternalArray(changeRecord);
279 } else {
280 observerInfo.pendingChangeRecords.push(changeRecord);
281 }
282 } 288 }
283 } 289 }
284 290
285 function BeginPerformSplice(array) { 291 function BeginPerformSplice(array) {
286 var objectInfo = objectInfoMap.get(array); 292 var objectInfo = objectInfoMap.get(array);
287 if (!IS_UNDEFINED(objectInfo)) 293 if (!IS_UNDEFINED(objectInfo))
288 BeginPerformChange(objectInfo, 'splice'); 294 BeginPerformChange(objectInfo, 'splice');
289 } 295 }
290 296
291 function EndPerformSplice(array) { 297 function EndPerformSplice(array) {
(...skipping 99 matching lines...) Expand 10 before | Expand all | Expand 10 after
391 if (IS_UNDEFINED(objectInfo)) objectInfo = CreateObjectInfo(object); 397 if (IS_UNDEFINED(objectInfo)) objectInfo = CreateObjectInfo(object);
392 398
393 if (IS_NULL(objectInfo.notifier)) { 399 if (IS_NULL(objectInfo.notifier)) {
394 objectInfo.notifier = { __proto__: notifierPrototype }; 400 objectInfo.notifier = { __proto__: notifierPrototype };
395 notifierTargetMap.set(objectInfo.notifier, object); 401 notifierTargetMap.set(objectInfo.notifier, object);
396 } 402 }
397 403
398 return objectInfo.notifier; 404 return objectInfo.notifier;
399 } 405 }
400 406
401 function DeliverChangeRecordsForObserver(observer) { 407 function CallbackDeliverPending(callback) {
402 var observerInfo = observerInfoMap.get(observer); 408 var callbackInfo = callbackInfoMap.get(callback);
403 if (IS_UNDEFINED(observerInfo)) 409 if (IS_UNDEFINED(callbackInfo) || IS_NUMBER(callbackInfo))
404 return false; 410 return false;
405 411
406 var pendingChangeRecords = observerInfo.pendingChangeRecords; 412 // Clear the pending change records from callback and return it to its
407 if (IS_NULL(pendingChangeRecords)) 413 // "optimized" state.
408 return false; 414 var priority = callbackInfo.priority;
415 callbackInfoMap.set(callback, priority);
409 416
410 observerInfo.pendingChangeRecords = null; 417 delete observationState.pendingObservers[priority];
411 delete observationState.pendingObservers[observerInfo.priority];
412 var delivered = []; 418 var delivered = [];
413 %MoveArrayContents(pendingChangeRecords, delivered); 419 %MoveArrayContents(callbackInfo, delivered);
420
414 try { 421 try {
415 %Call(void 0, delivered, observer); 422 %Call(void 0, delivered, callback);
416 } catch (ex) {} 423 } catch (ex) {}
417 return true; 424 return true;
418 } 425 }
419 426
420 function ObjectDeliverChangeRecords(callback) { 427 function ObjectDeliverChangeRecords(callback) {
421 if (!IS_SPEC_FUNCTION(callback)) 428 if (!IS_SPEC_FUNCTION(callback))
422 throw MakeTypeError("observe_non_function", ["deliverChangeRecords"]); 429 throw MakeTypeError("observe_non_function", ["deliverChangeRecords"]);
423 430
424 while (DeliverChangeRecordsForObserver(callback)) {} 431 while (CallbackDeliverPending(callback)) {}
425 } 432 }
426 433
427 function DeliverChangeRecords() { 434 function DeliverChangeRecords() {
428 while (observationState.pendingObservers.length) { 435 while (observationState.pendingObservers.length) {
429 var pendingObservers = observationState.pendingObservers; 436 var pendingObservers = observationState.pendingObservers;
430 observationState.pendingObservers = new InternalArray; 437 observationState.pendingObservers = new InternalArray;
431 for (var i in pendingObservers) { 438 for (var i in pendingObservers) {
432 DeliverChangeRecordsForObserver(pendingObservers[i]); 439 CallbackDeliverPending(pendingObservers[i]);
433 } 440 }
434 } 441 }
435 } 442 }
436 443
437 function SetupObjectObserve() { 444 function SetupObjectObserve() {
438 %CheckIsBootstrapping(); 445 %CheckIsBootstrapping();
439 InstallFunctions($Object, DONT_ENUM, $Array( 446 InstallFunctions($Object, DONT_ENUM, $Array(
440 "deliverChangeRecords", ObjectDeliverChangeRecords, 447 "deliverChangeRecords", ObjectDeliverChangeRecords,
441 "getNotifier", ObjectGetNotifier, 448 "getNotifier", ObjectGetNotifier,
442 "observe", ObjectObserve, 449 "observe", ObjectObserve,
443 "unobserve", ObjectUnobserve 450 "unobserve", ObjectUnobserve
444 )); 451 ));
445 InstallFunctions($Array, DONT_ENUM, $Array( 452 InstallFunctions($Array, DONT_ENUM, $Array(
446 "observe", ArrayObserve, 453 "observe", ArrayObserve,
447 "unobserve", ArrayUnobserve 454 "unobserve", ArrayUnobserve
448 )); 455 ));
449 InstallFunctions(notifierPrototype, DONT_ENUM, $Array( 456 InstallFunctions(notifierPrototype, DONT_ENUM, $Array(
450 "notify", ObjectNotifierNotify, 457 "notify", ObjectNotifierNotify,
451 "performChange", ObjectNotifierPerformChange 458 "performChange", ObjectNotifierPerformChange
452 )); 459 ));
453 } 460 }
454 461
455 SetupObjectObserve(); 462 SetupObjectObserve();
OLDNEW
« no previous file with comments | « src/mksnapshot.cc ('k') | src/objects.h » ('j') | no next file with comments »

Powered by Google App Engine
This is Rietveld 408576698