1// Copyright 2012 the V8 project authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
5"use strict";
6
7// Overview:
8//
9// This file contains all of the routing and accounting for Object.observe.
10// User code will interact with these mechanisms via the Object.observe APIs
11// and, as a side effect of mutation objects which are observed. The V8 runtime
12// (both C++ and JS) will interact with these mechanisms primarily by enqueuing
13// proper change records for objects which were mutated. The Object.observe
14// routing and accounting consists primarily of three participants
15//
16// 1) ObjectInfo. This represents the observed state of a given object. It
17//    records what callbacks are observing the object, with what options, and
18//    what "change types" are in progress on the object (i.e. via
19//    notifier.performChange).
20//
21// 2) CallbackInfo. This represents a callback used for observation. It holds
22//    the records which must be delivered to the callback, as well as the global
23//    priority of the callback (which determines delivery order between
24//    callbacks).
25//
26// 3) observationState.pendingObservers. This is the set of observers which
27//    have change records which must be delivered. During "normal" delivery
28//    (i.e. not Object.deliverChangeRecords), this is the mechanism by which
29//    callbacks are invoked in the proper order until there are no more
30//    change records pending to a callback.
31//
32// Note that in order to reduce allocation and processing costs, the
33// implementation of (1) and (2) have "optimized" states which represent
34// common cases which can be handled more efficiently.
35
36var observationState;
37
38function GetObservationStateJS() {
39  if (IS_UNDEFINED(observationState))
40    observationState = %GetObservationState();
41
42  if (IS_UNDEFINED(observationState.callbackInfoMap)) {
43    observationState.callbackInfoMap = %ObservationWeakMapCreate();
44    observationState.objectInfoMap = %ObservationWeakMapCreate();
45    observationState.notifierObjectInfoMap = %ObservationWeakMapCreate();
46    observationState.pendingObservers = null;
47    observationState.nextCallbackPriority = 0;
48  }
49
50  return observationState;
51}
52
53function GetWeakMapWrapper() {
54  function MapWrapper(map) {
55    this.map_ = map;
56  };
57
58  MapWrapper.prototype = {
59    __proto__: null,
60    get: function(key) {
61      return %WeakCollectionGet(this.map_, key);
62    },
63    set: function(key, value) {
64      %WeakCollectionSet(this.map_, key, value);
65    },
66    has: function(key) {
67      return !IS_UNDEFINED(this.get(key));
68    }
69  };
70
71  return MapWrapper;
72}
73
74var contextMaps;
75
76function GetContextMaps() {
77  if (IS_UNDEFINED(contextMaps)) {
78    var map = GetWeakMapWrapper();
79    var observationState = GetObservationStateJS();
80    contextMaps = {
81      callbackInfoMap: new map(observationState.callbackInfoMap),
82      objectInfoMap: new map(observationState.objectInfoMap),
83      notifierObjectInfoMap: new map(observationState.notifierObjectInfoMap)
84    };
85  }
86
87  return contextMaps;
88}
89
90function GetCallbackInfoMap() {
91  return GetContextMaps().callbackInfoMap;
92}
93
94function GetObjectInfoMap() {
95  return GetContextMaps().objectInfoMap;
96}
97
98function GetNotifierObjectInfoMap() {
99  return GetContextMaps().notifierObjectInfoMap;
100}
101
102function GetPendingObservers() {
103  return GetObservationStateJS().pendingObservers;
104}
105
106function SetPendingObservers(pendingObservers) {
107  GetObservationStateJS().pendingObservers = pendingObservers;
108}
109
110function GetNextCallbackPriority() {
111  return GetObservationStateJS().nextCallbackPriority++;
112}
113
114function nullProtoObject() {
115  return { __proto__: null };
116}
117
118function TypeMapCreate() {
119  return nullProtoObject();
120}
121
122function TypeMapAddType(typeMap, type, ignoreDuplicate) {
123  typeMap[type] = ignoreDuplicate ? 1 : (typeMap[type] || 0) + 1;
124}
125
126function TypeMapRemoveType(typeMap, type) {
127  typeMap[type]--;
128}
129
130function TypeMapCreateFromList(typeList, length) {
131  var typeMap = TypeMapCreate();
132  for (var i = 0; i < length; i++) {
133    TypeMapAddType(typeMap, typeList[i], true);
134  }
135  return typeMap;
136}
137
138function TypeMapHasType(typeMap, type) {
139  return !!typeMap[type];
140}
141
142function TypeMapIsDisjointFrom(typeMap1, typeMap2) {
143  if (!typeMap1 || !typeMap2)
144    return true;
145
146  for (var type in typeMap1) {
147    if (TypeMapHasType(typeMap1, type) && TypeMapHasType(typeMap2, type))
148      return false;
149  }
150
151  return true;
152}
153
154var defaultAcceptTypes = (function() {
155  var defaultTypes = [
156    'add',
157    'update',
158    'delete',
159    'setPrototype',
160    'reconfigure',
161    'preventExtensions'
162  ];
163  return TypeMapCreateFromList(defaultTypes, defaultTypes.length);
164})();
165
166// An Observer is a registration to observe an object by a callback with
167// a given set of accept types. If the set of accept types is the default
168// set for Object.observe, the observer is represented as a direct reference
169// to the callback. An observer never changes its accept types and thus never
170// needs to "normalize".
171function ObserverCreate(callback, acceptList) {
172  if (IS_UNDEFINED(acceptList))
173    return callback;
174  var observer = nullProtoObject();
175  observer.callback = callback;
176  observer.accept = acceptList;
177  return observer;
178}
179
180function ObserverGetCallback(observer) {
181  return IS_SPEC_FUNCTION(observer) ? observer : observer.callback;
182}
183
184function ObserverGetAcceptTypes(observer) {
185  return IS_SPEC_FUNCTION(observer) ? defaultAcceptTypes : observer.accept;
186}
187
188function ObserverIsActive(observer, objectInfo) {
189  return TypeMapIsDisjointFrom(ObjectInfoGetPerformingTypes(objectInfo),
190                               ObserverGetAcceptTypes(observer));
191}
192
193function ObjectInfoGetOrCreate(object) {
194  var objectInfo = ObjectInfoGet(object);
195  if (IS_UNDEFINED(objectInfo)) {
196    if (!%IsJSProxy(object))
197      %SetIsObserved(object);
198
199    objectInfo = {
200      object: object,
201      changeObservers: null,
202      notifier: null,
203      performing: null,
204      performingCount: 0,
205    };
206    GetObjectInfoMap().set(object, objectInfo);
207  }
208  return objectInfo;
209}
210
211function ObjectInfoGet(object) {
212  return GetObjectInfoMap().get(object);
213}
214
215function ObjectInfoGetFromNotifier(notifier) {
216  return GetNotifierObjectInfoMap().get(notifier);
217}
218
219function ObjectInfoGetNotifier(objectInfo) {
220  if (IS_NULL(objectInfo.notifier)) {
221    objectInfo.notifier = { __proto__: notifierPrototype };
222    GetNotifierObjectInfoMap().set(objectInfo.notifier, objectInfo);
223  }
224
225  return objectInfo.notifier;
226}
227
228function ObjectInfoGetObject(objectInfo) {
229  return objectInfo.object;
230}
231
232function ChangeObserversIsOptimized(changeObservers) {
233  return typeof changeObservers === 'function' ||
234         typeof changeObservers.callback === 'function';
235}
236
237// The set of observers on an object is called 'changeObservers'. The first
238// observer is referenced directly via objectInfo.changeObservers. When a second
239// is added, changeObservers "normalizes" to become a mapping of callback
240// priority -> observer and is then stored on objectInfo.changeObservers.
241function ObjectInfoNormalizeChangeObservers(objectInfo) {
242  if (ChangeObserversIsOptimized(objectInfo.changeObservers)) {
243    var observer = objectInfo.changeObservers;
244    var callback = ObserverGetCallback(observer);
245    var callbackInfo = CallbackInfoGet(callback);
246    var priority = CallbackInfoGetPriority(callbackInfo);
247    objectInfo.changeObservers = nullProtoObject();
248    objectInfo.changeObservers[priority] = observer;
249  }
250}
251
252function ObjectInfoAddObserver(objectInfo, callback, acceptList) {
253  var callbackInfo = CallbackInfoGetOrCreate(callback);
254  var observer = ObserverCreate(callback, acceptList);
255
256  if (!objectInfo.changeObservers) {
257    objectInfo.changeObservers = observer;
258    return;
259  }
260
261  ObjectInfoNormalizeChangeObservers(objectInfo);
262  var priority = CallbackInfoGetPriority(callbackInfo);
263  objectInfo.changeObservers[priority] = observer;
264}
265
266function ObjectInfoRemoveObserver(objectInfo, callback) {
267  if (!objectInfo.changeObservers)
268    return;
269
270  if (ChangeObserversIsOptimized(objectInfo.changeObservers)) {
271    if (callback === ObserverGetCallback(objectInfo.changeObservers))
272      objectInfo.changeObservers = null;
273    return;
274  }
275
276  var callbackInfo = CallbackInfoGet(callback);
277  var priority = CallbackInfoGetPriority(callbackInfo);
278  objectInfo.changeObservers[priority] = null;
279}
280
281function ObjectInfoHasActiveObservers(objectInfo) {
282  if (IS_UNDEFINED(objectInfo) || !objectInfo.changeObservers)
283    return false;
284
285  if (ChangeObserversIsOptimized(objectInfo.changeObservers))
286    return ObserverIsActive(objectInfo.changeObservers, objectInfo);
287
288  for (var priority in objectInfo.changeObservers) {
289    var observer = objectInfo.changeObservers[priority];
290    if (!IS_NULL(observer) && ObserverIsActive(observer, objectInfo))
291      return true;
292  }
293
294  return false;
295}
296
297function ObjectInfoAddPerformingType(objectInfo, type) {
298  objectInfo.performing = objectInfo.performing || TypeMapCreate();
299  TypeMapAddType(objectInfo.performing, type);
300  objectInfo.performingCount++;
301}
302
303function ObjectInfoRemovePerformingType(objectInfo, type) {
304  objectInfo.performingCount--;
305  TypeMapRemoveType(objectInfo.performing, type);
306}
307
308function ObjectInfoGetPerformingTypes(objectInfo) {
309  return objectInfo.performingCount > 0 ? objectInfo.performing : null;
310}
311
312function ConvertAcceptListToTypeMap(arg) {
313  // We use undefined as a sentinel for the default accept list.
314  if (IS_UNDEFINED(arg))
315    return arg;
316
317  if (!IS_SPEC_OBJECT(arg))
318    throw MakeTypeError("observe_accept_invalid");
319
320  var len = ToInteger(arg.length);
321  if (len < 0) len = 0;
322
323  return TypeMapCreateFromList(arg, len);
324}
325
326// CallbackInfo's optimized state is just a number which represents its global
327// priority. When a change record must be enqueued for the callback, it
328// normalizes. When delivery clears any pending change records, it re-optimizes.
329function CallbackInfoGet(callback) {
330  return GetCallbackInfoMap().get(callback);
331}
332
333function CallbackInfoGetOrCreate(callback) {
334  var callbackInfo = GetCallbackInfoMap().get(callback);
335  if (!IS_UNDEFINED(callbackInfo))
336    return callbackInfo;
337
338  var priority =  GetNextCallbackPriority();
339  GetCallbackInfoMap().set(callback, priority);
340  return priority;
341}
342
343function CallbackInfoGetPriority(callbackInfo) {
344  if (IS_NUMBER(callbackInfo))
345    return callbackInfo;
346  else
347    return callbackInfo.priority;
348}
349
350function CallbackInfoNormalize(callback) {
351  var callbackInfo = GetCallbackInfoMap().get(callback);
352  if (IS_NUMBER(callbackInfo)) {
353    var priority = callbackInfo;
354    callbackInfo = new InternalArray;
355    callbackInfo.priority = priority;
356    GetCallbackInfoMap().set(callback, callbackInfo);
357  }
358  return callbackInfo;
359}
360
361function ObjectObserve(object, callback, acceptList) {
362  if (!IS_SPEC_OBJECT(object))
363    throw MakeTypeError("observe_non_object", ["observe"]);
364  if (%IsJSGlobalProxy(object))
365    throw MakeTypeError("observe_global_proxy", ["observe"]);
366  if (!IS_SPEC_FUNCTION(callback))
367    throw MakeTypeError("observe_non_function", ["observe"]);
368  if (ObjectIsFrozen(callback))
369    throw MakeTypeError("observe_callback_frozen");
370
371  var objectObserveFn = %GetObjectContextObjectObserve(object);
372  return objectObserveFn(object, callback, acceptList);
373}
374
375function NativeObjectObserve(object, callback, acceptList) {
376  var objectInfo = ObjectInfoGetOrCreate(object);
377  var typeList = ConvertAcceptListToTypeMap(acceptList);
378  ObjectInfoAddObserver(objectInfo, callback, typeList);
379  return object;
380}
381
382function ObjectUnobserve(object, callback) {
383  if (!IS_SPEC_OBJECT(object))
384    throw MakeTypeError("observe_non_object", ["unobserve"]);
385  if (%IsJSGlobalProxy(object))
386    throw MakeTypeError("observe_global_proxy", ["unobserve"]);
387  if (!IS_SPEC_FUNCTION(callback))
388    throw MakeTypeError("observe_non_function", ["unobserve"]);
389
390  var objectInfo = ObjectInfoGet(object);
391  if (IS_UNDEFINED(objectInfo))
392    return object;
393
394  ObjectInfoRemoveObserver(objectInfo, callback);
395  return object;
396}
397
398function ArrayObserve(object, callback) {
399  return ObjectObserve(object, callback, ['add',
400                                          'update',
401                                          'delete',
402                                          'splice']);
403}
404
405function ArrayUnobserve(object, callback) {
406  return ObjectUnobserve(object, callback);
407}
408
409function ObserverEnqueueIfActive(observer, objectInfo, changeRecord) {
410  if (!ObserverIsActive(observer, objectInfo) ||
411      !TypeMapHasType(ObserverGetAcceptTypes(observer), changeRecord.type)) {
412    return;
413  }
414
415  var callback = ObserverGetCallback(observer);
416  if (!%ObserverObjectAndRecordHaveSameOrigin(callback, changeRecord.object,
417                                              changeRecord)) {
418    return;
419  }
420
421  var callbackInfo = CallbackInfoNormalize(callback);
422  if (IS_NULL(GetPendingObservers())) {
423    SetPendingObservers(nullProtoObject());
424    %EnqueueMicrotask(ObserveMicrotaskRunner);
425  }
426  GetPendingObservers()[callbackInfo.priority] = callback;
427  callbackInfo.push(changeRecord);
428}
429
430function ObjectInfoEnqueueExternalChangeRecord(objectInfo, changeRecord, type) {
431  if (!ObjectInfoHasActiveObservers(objectInfo))
432    return;
433
434  var hasType = !IS_UNDEFINED(type);
435  var newRecord = hasType ?
436      { object: ObjectInfoGetObject(objectInfo), type: type } :
437      { object: ObjectInfoGetObject(objectInfo) };
438
439  for (var prop in changeRecord) {
440    if (prop === 'object' || (hasType && prop === 'type')) continue;
441    %DefineOrRedefineDataProperty(newRecord, prop, changeRecord[prop],
442        READ_ONLY + DONT_DELETE);
443  }
444  ObjectFreezeJS(newRecord);
445
446  ObjectInfoEnqueueInternalChangeRecord(objectInfo, newRecord);
447}
448
449function ObjectInfoEnqueueInternalChangeRecord(objectInfo, changeRecord) {
450  // TODO(rossberg): adjust once there is a story for symbols vs proxies.
451  if (IS_SYMBOL(changeRecord.name)) return;
452
453  if (ChangeObserversIsOptimized(objectInfo.changeObservers)) {
454    var observer = objectInfo.changeObservers;
455    ObserverEnqueueIfActive(observer, objectInfo, changeRecord);
456    return;
457  }
458
459  for (var priority in objectInfo.changeObservers) {
460    var observer = objectInfo.changeObservers[priority];
461    if (IS_NULL(observer))
462      continue;
463    ObserverEnqueueIfActive(observer, objectInfo, changeRecord);
464  }
465}
466
467function BeginPerformSplice(array) {
468  var objectInfo = ObjectInfoGet(array);
469  if (!IS_UNDEFINED(objectInfo))
470    ObjectInfoAddPerformingType(objectInfo, 'splice');
471}
472
473function EndPerformSplice(array) {
474  var objectInfo = ObjectInfoGet(array);
475  if (!IS_UNDEFINED(objectInfo))
476    ObjectInfoRemovePerformingType(objectInfo, 'splice');
477}
478
479function EnqueueSpliceRecord(array, index, removed, addedCount) {
480  var objectInfo = ObjectInfoGet(array);
481  if (!ObjectInfoHasActiveObservers(objectInfo))
482    return;
483
484  var changeRecord = {
485    type: 'splice',
486    object: array,
487    index: index,
488    removed: removed,
489    addedCount: addedCount
490  };
491
492  ObjectFreezeJS(changeRecord);
493  ObjectFreezeJS(changeRecord.removed);
494  ObjectInfoEnqueueInternalChangeRecord(objectInfo, changeRecord);
495}
496
497function NotifyChange(type, object, name, oldValue) {
498  var objectInfo = ObjectInfoGet(object);
499  if (!ObjectInfoHasActiveObservers(objectInfo))
500    return;
501
502  var changeRecord;
503  if (arguments.length == 2) {
504    changeRecord = { type: type, object: object };
505  } else if (arguments.length == 3) {
506    changeRecord = { type: type, object: object, name: name };
507  } else {
508    changeRecord = {
509      type: type,
510      object: object,
511      name: name,
512      oldValue: oldValue
513    };
514  }
515
516  ObjectFreezeJS(changeRecord);
517  ObjectInfoEnqueueInternalChangeRecord(objectInfo, changeRecord);
518}
519
520var notifierPrototype = {};
521
522function ObjectNotifierNotify(changeRecord) {
523  if (!IS_SPEC_OBJECT(this))
524    throw MakeTypeError("called_on_non_object", ["notify"]);
525
526  var objectInfo = ObjectInfoGetFromNotifier(this);
527  if (IS_UNDEFINED(objectInfo))
528    throw MakeTypeError("observe_notify_non_notifier");
529  if (!IS_STRING(changeRecord.type))
530    throw MakeTypeError("observe_type_non_string");
531
532  ObjectInfoEnqueueExternalChangeRecord(objectInfo, changeRecord);
533}
534
535function ObjectNotifierPerformChange(changeType, changeFn) {
536  if (!IS_SPEC_OBJECT(this))
537    throw MakeTypeError("called_on_non_object", ["performChange"]);
538
539  var objectInfo = ObjectInfoGetFromNotifier(this);
540  if (IS_UNDEFINED(objectInfo))
541    throw MakeTypeError("observe_notify_non_notifier");
542  if (!IS_STRING(changeType))
543    throw MakeTypeError("observe_perform_non_string");
544  if (!IS_SPEC_FUNCTION(changeFn))
545    throw MakeTypeError("observe_perform_non_function");
546
547  var performChangeFn = %GetObjectContextNotifierPerformChange(objectInfo);
548  performChangeFn(objectInfo, changeType, changeFn);
549}
550
551function NativeObjectNotifierPerformChange(objectInfo, changeType, changeFn) {
552  ObjectInfoAddPerformingType(objectInfo, changeType);
553
554  var changeRecord;
555  try {
556    changeRecord = %_CallFunction(UNDEFINED, changeFn);
557  } finally {
558    ObjectInfoRemovePerformingType(objectInfo, changeType);
559  }
560
561  if (IS_SPEC_OBJECT(changeRecord))
562    ObjectInfoEnqueueExternalChangeRecord(objectInfo, changeRecord, changeType);
563}
564
565function ObjectGetNotifier(object) {
566  if (!IS_SPEC_OBJECT(object))
567    throw MakeTypeError("observe_non_object", ["getNotifier"]);
568  if (%IsJSGlobalProxy(object))
569    throw MakeTypeError("observe_global_proxy", ["getNotifier"]);
570
571  if (ObjectIsFrozen(object)) return null;
572
573  if (!%ObjectWasCreatedInCurrentOrigin(object)) return null;
574
575  var getNotifierFn = %GetObjectContextObjectGetNotifier(object);
576  return getNotifierFn(object);
577}
578
579function NativeObjectGetNotifier(object) {
580  var objectInfo = ObjectInfoGetOrCreate(object);
581  return ObjectInfoGetNotifier(objectInfo);
582}
583
584function CallbackDeliverPending(callback) {
585  var callbackInfo = GetCallbackInfoMap().get(callback);
586  if (IS_UNDEFINED(callbackInfo) || IS_NUMBER(callbackInfo))
587    return false;
588
589  // Clear the pending change records from callback and return it to its
590  // "optimized" state.
591  var priority = callbackInfo.priority;
592  GetCallbackInfoMap().set(callback, priority);
593
594  if (GetPendingObservers())
595    delete GetPendingObservers()[priority];
596
597  var delivered = [];
598  %MoveArrayContents(callbackInfo, delivered);
599
600  try {
601    %_CallFunction(UNDEFINED, delivered, callback);
602  } catch (ex) {}  // TODO(rossberg): perhaps log uncaught exceptions.
603  return true;
604}
605
606function ObjectDeliverChangeRecords(callback) {
607  if (!IS_SPEC_FUNCTION(callback))
608    throw MakeTypeError("observe_non_function", ["deliverChangeRecords"]);
609
610  while (CallbackDeliverPending(callback)) {}
611}
612
613function ObserveMicrotaskRunner() {
614  var pendingObservers = GetPendingObservers();
615  if (pendingObservers) {
616    SetPendingObservers(null);
617    for (var i in pendingObservers) {
618      CallbackDeliverPending(pendingObservers[i]);
619    }
620  }
621}
622
623function SetupObjectObserve() {
624  %CheckIsBootstrapping();
625  InstallFunctions($Object, DONT_ENUM, $Array(
626    "deliverChangeRecords", ObjectDeliverChangeRecords,
627    "getNotifier", ObjectGetNotifier,
628    "observe", ObjectObserve,
629    "unobserve", ObjectUnobserve
630  ));
631  InstallFunctions($Array, DONT_ENUM, $Array(
632    "observe", ArrayObserve,
633    "unobserve", ArrayUnobserve
634  ));
635  InstallFunctions(notifierPrototype, DONT_ENUM, $Array(
636    "notify", ObjectNotifierNotify,
637    "performChange", ObjectNotifierPerformChange
638  ));
639}
640
641SetupObjectObserve();
642