From cdf551f46bfa6cfdc1ec5b7be93b08928f252ee7 Mon Sep 17 00:00:00 2001 From: Ashley Claymore Date: Mon, 6 Feb 2023 11:04:42 +0000 Subject: [PATCH 1/5] Normative: Permit Symbols as WeakMap keys - Proposal: https://github.com/tc39/proposal-symbols-as-weakmap-keys - Also allows Symbols in WeakSet, WeakRef, and FinalizationRegistry - Adds new AO 'CanBeHeldWeakly' - Registered Symbols can not be held weakly Closes #1194 Co-authored-by: Daniel Ehrenberg Co-authored-by: Leo Balter Co-authored-by: Mathieu Hofman <86499+mhofman@users.noreply.github.com> Co-authored-by: Richard Gibson Co-authored-by: Jordan Harband Co-authored-by: Shu-yu Guo Co-authored-by: Michael Dyck Co-authored-by: Michael Ficarra --- spec.html | 139 ++++++++++++++++++++++++++++++++++-------------------- 1 file changed, 89 insertions(+), 50 deletions(-) diff --git a/spec.html b/spec.html index 2b7abc1690..df1a161a2d 100644 --- a/spec.html +++ b/spec.html @@ -1086,8 +1086,8 @@

Value Notation

Identity

In this specification, the word “is” is used to compare two values through equality, as in “If _bool_ is *true*, then”. For the purposes of this equality comparison, all values within this specification fall into one of the following two categories.

-

Values without identity are equal to other values without identity if all of their innate characteristics are the same — characteristics such as the magnitude of an integer or the length of a sequence. Because of this, values without identity may be manifest anywhere simply by fully describing their characteristics. It is not meaningful to change the characteristics of a value that does not have identity. Examples of values without identity include, but are not limited to: Booleans; mathematical values and extended mathematical values; Numbers; BigInts; *null*; *undefined*; sequences, including Strings, ECMAScript source text, surrogate pairs, Directive Prologues, etc; UTF-16 code units; Unicode code points; enums; abstract operations, including syntax-directed operations, host hooks, etc; and ordered pairs. The preceding list is exhaustive for ECMAScript language values.

-

In contrast, each value with identity is unique and therefore only equal to itself. Values with identity are like values without identity but with an additional unguessable, unchangeable, universally-unique characteristic called identity. References to existing values with identity cannot be manifest simply by describing them, as the identity itself is indescribable; instead, references to these values must be explicitly passed from one place to another. Some values with identity are mutable and therefore can have their characteristics (except their identity) changed in-place, causing all holders of the value to observe the new characteristics. Examples of values with identity include, but are not limited to: Objects, including function objects, exotic objects, etc; any kind of Records, including Property Descriptors, PrivateElements, etc; symbols; Parse Nodes; Lists; Sets and Relations; Abstract Closures; Data Blocks; Private Names; execution contexts and execution context stacks; agent signifiers; and WaiterLists. The preceding list is exhaustive for ECMAScript language values.

+

Values without identity are equal to other values without identity if all of their innate characteristics are the same — characteristics such as the magnitude of an integer or the length of a sequence. Because of this, values without identity may be manifest anywhere simply by fully describing their characteristics. It is not meaningful to change the characteristics of a value that does not have identity. Examples of values without identity include, but are not limited to: Booleans; mathematical values and extended mathematical values; Numbers; BigInts; *null*; *undefined*; sequences, including Strings, ECMAScript source text, surrogate pairs, Directive Prologues, etc; UTF-16 code units; Unicode code points; Symbols in the GlobalSymbolRegistry; enums; abstract operations, including syntax-directed operations, host hooks, etc; and ordered pairs. The preceding list is exhaustive for ECMAScript language values.

+

In contrast, each value with identity is unique and therefore only equal to itself. Values with identity are like values without identity but with an additional unguessable, unchangeable, universally-unique characteristic called identity. References to existing values with identity cannot be manifest simply by describing them, as the identity itself is indescribable; instead, references to these values must be explicitly passed from one place to another. Some values with identity are mutable and therefore can have their characteristics (except their identity) changed in-place, causing all holders of the value to observe the new characteristics. Examples of values with identity include, but are not limited to: Objects, including function objects, exotic objects, etc; any kind of Records, including Property Descriptors, PrivateElements, etc; Symbols not in the GlobalSymbolRegistry; Parse Nodes; Lists; Sets and Relations; Abstract Closures; Data Blocks; Private Names; execution contexts and execution context stacks; agent signifiers; and WaiterLists. The preceding list is exhaustive for ECMAScript language values.

A value without identity is never equal to a value with identity.

@@ -12014,8 +12014,8 @@

Agents

[[KeptAlive]] - a List of Objects - Initially a new empty List, representing the list of objects to be kept alive until the end of the current Job + a List of either Objects or Symbols + Initially a new empty List, representing the list of objects and/or symbols to be kept alive until the end of the current Job @@ -12130,22 +12130,22 @@

Forward Progress

-

Processing Model of WeakRef and FinalizationRegistry Objects

+

Processing Model of WeakRef and FinalizationRegistry Targets

Objectives

-

This specification does not make any guarantees that any object will be garbage collected. Objects which are not live may be released after long periods of time, or never at all. For this reason, this specification uses the term "may" when describing behaviour triggered by garbage collection.

+

This specification does not make any guarantees that any object or symbol will be garbage collected. Objects or symbols which are not live may be released after long periods of time, or never at all. For this reason, this specification uses the term "may" when describing behaviour triggered by garbage collection.

The semantics of WeakRefs and FinalizationRegistrys is based on two operations which happen at particular points in time:

  • - When `WeakRef.prototype.deref` is called, the referent (if *undefined* is not returned) is kept alive so that subsequent, synchronous accesses also return the object. This list is reset when synchronous work is done using the ClearKeptObjects abstract operation. + When `WeakRef.prototype.deref` is called, the referent (if *undefined* is not returned) is kept alive so that subsequent, synchronous accesses also return the same value. This list is reset when synchronous work is done using the ClearKeptObjects abstract operation.
  • - When an object which is registered with a FinalizationRegistry becomes unreachable, a call of the FinalizationRegistry's cleanup callback may eventually be made, after synchronous ECMAScript execution completes. The FinalizationRegistry cleanup is performed with the CleanupFinalizationRegistry abstract operation. + When an object or symbol which is registered with a FinalizationRegistry becomes unreachable, a call of the FinalizationRegistry's cleanup callback may eventually be made, after synchronous ECMAScript execution completes. The FinalizationRegistry cleanup is performed with the CleanupFinalizationRegistry abstract operation.
@@ -12157,33 +12157,33 @@

Objectives

Liveness

-

For some set of objects _S_, a hypothetical WeakRef-oblivious execution with respect to _S_ is an execution whereby the abstract operation WeakRefDeref of a WeakRef whose referent is an element of _S_ always returns *undefined*.

+

For some set of objects and/or symbols _S_ a hypothetical WeakRef-oblivious execution with respect to _S_ is an execution whereby the abstract operation WeakRefDeref of a WeakRef whose referent is an element of _S_ always returns *undefined*.

- WeakRef-obliviousness, together with liveness, capture two notions. One, that a WeakRef itself does not keep an object alive. Two, that cycles in liveness does not imply that an object is live. To be concrete, if determining _obj_'s liveness depends on determining the liveness of another WeakRef referent, _obj2_, _obj2_'s liveness cannot assume _obj_'s liveness, which would be circular reasoning. + WeakRef-obliviousness, together with liveness, capture two notions. One, that a WeakRef itself does not keep its referent alive. Two, that cycles in liveness does not imply that a value is live. To be concrete, if determining _v_'s liveness depends on determining the liveness of a WeakRef referent, _r_, _r_'s liveness cannot assume _v_'s liveness, which would be circular reasoning. - WeakRef-obliviousness is defined on sets of objects instead of individual objects to account for cycles. If it were defined on individual objects, then an object in a cycle will be considered live even though its Object value is only observed via WeakRefs of other objects in the cycle. + WeakRef-obliviousness is defined on sets of objects or symbols instead of individual values to account for cycles. If it were defined on individual values, then a WeakRef referent in a cycle will be considered live even though its identity is only observed via other WeakRef referents in the cycle. - Colloquially, we say that an individual object is live if every set of objects containing it is live. + Colloquially, we say that an individual object or symbol is live if every set of objects containing it is live. -

At any point during evaluation, a set of objects _S_ is considered live if either of the following conditions is met:

+

At any point during evaluation, a set of objects and/or symbols _S_ is considered live if either of the following conditions is met:

  • Any element in _S_ is included in any agent's [[KeptAlive]] List.
  • - There exists a valid future hypothetical WeakRef-oblivious execution with respect to _S_ that observes the Object value of any object in _S_. + There exists a valid future hypothetical WeakRef-oblivious execution with respect to _S_ that observes any value in _S_.
- The second condition above intends to capture the intuition that an object is live if its identity is observable via non-WeakRef means. An object's identity may be observed by observing a strict equality comparison between objects or observing the object being used as key in a Map. + The second condition above intends to capture the intuition that a value is live if its identity is observable via non-WeakRef means. A value's identity may be observed by observing a strict equality comparison or observing the value being used as key in a Map. -

Presence of an object in a field, an internal slot, or a property does not imply that the object is live. For example if the object in question is never passed back to the program, then it cannot be observed.

+

Presence of an object or a symbol in a field, an internal slot, or a property does not imply that the value is live. For example if the value in question is never passed back to the program, then it cannot be observed.

This is the case for keys in a WeakMap, members of a WeakSet, as well as the [[WeakRefTarget]] and [[UnregisterToken]] fields of a FinalizationRegistry Cell record.

@@ -12197,20 +12197,20 @@

Liveness

Execution

-

At any time, if a set of objects _S_ is not live, an ECMAScript implementation may perform the following steps atomically:

+

At any time, if a set of objects and/or symbols _S_ is not live, an ECMAScript implementation may perform the following steps atomically:

- 1. For each element _obj_ of _S_, do - 1. For each WeakRef _ref_ such that _ref_.[[WeakRefTarget]] is _obj_, do + 1. For each element _value_ of _S_, do + 1. For each WeakRef _ref_ such that _ref_.[[WeakRefTarget]] is _value_, do 1. Set _ref_.[[WeakRefTarget]] to ~empty~. - 1. For each FinalizationRegistry _fg_ such that _fg_.[[Cells]] contains a Record _cell_ such that _cell_.[[WeakRefTarget]] is _obj_, do + 1. For each FinalizationRegistry _fg_ such that _fg_.[[Cells]] contains a Record _cell_ such that _cell_.[[WeakRefTarget]] is _value_, do 1. Set _cell_.[[WeakRefTarget]] to ~empty~. 1. Optionally, perform HostEnqueueFinalizationRegistryCleanupJob(_fg_). - 1. For each WeakMap _map_ such that _map_.[[WeakMapData]] contains a Record _r_ such that _r_.[[Key]] is _obj_, do + 1. For each WeakMap _map_ such that _map_.[[WeakMapData]] contains a Record _r_ such that _r_.[[Key]] is _value_, do 1. Set _r_.[[Key]] to ~empty~. 1. Set _r_.[[Value]] to ~empty~. - 1. For each WeakSet _set_ such that _set_.[[WeakSetData]] contains _obj_, do - 1. Replace the element of _set_.[[WeakSetData]] whose value is _obj_ with an element whose value is ~empty~. + 1. For each WeakSet _set_ such that _set_.[[WeakSetData]] contains _value_, do + 1. Replace the element of _set_.[[WeakSetData]] whose value is _value_ with an element whose value is ~empty~. @@ -12223,8 +12223,8 @@

Execution

Because calling HostEnqueueFinalizationRegistryCleanupJob is optional, registered objects in a FinalizationRegistry do not necessarily hold that FinalizationRegistry live. Implementations may omit FinalizationRegistry callbacks for any reason, e.g., if the FinalizationRegistry itself becomes dead, or if the application is shutting down.

-

Implementations are not obligated to empty WeakRefs for maximal sets of non-live objects.

-

If an implementation chooses a non-live set _S_ in which to empty WeakRefs, this definition requires that it empties WeakRefs for all objects in _S_ simultaneously. In other words, it is not conformant for an implementation to empty a WeakRef pointing to an object _obj_ without emptying out other WeakRefs that, if not emptied, could result in an execution that observes the Object value of _obj_.

+

Implementations are not obligated to empty WeakRefs for maximal sets of non-live objects or symbols.

+

If an implementation chooses a non-live set _S_ in which to empty WeakRefs, this definition requires that it empties WeakRefs for all values in _S_ simultaneously. In other words, it is not conformant for an implementation to empty a WeakRef pointing to a value _v_ without emptying out other WeakRefs that, if not emptied, could result in an execution that observes the value of _v_.

@@ -12266,18 +12266,18 @@

ClearKeptObjects ( ): ~unused~

AddToKeptObjects ( - _object_: an Object, + _value_: an Object or a Symbol, ): ~unused~

1. Let _agentRecord_ be the surrounding agent's Agent Record. - 1. Append _object_ to _agentRecord_.[[KeptAlive]]. + 1. Append _value_ to _agentRecord_.[[KeptAlive]]. 1. Return ~unused~. - When the abstract operation AddToKeptObjects is called with a target object reference, it adds the target to a list that will point strongly at the target until ClearKeptObjects is called. + When the abstract operation AddToKeptObjects is called with a target object or symbol, it adds the target to a list that will point strongly at the target until ClearKeptObjects is called.
@@ -12299,6 +12299,26 @@

1. Return ~unused~. + + +

+ CanBeHeldWeakly ( + _v_: an ECMAScript language value, + ): a Boolean +

+
+
description
+
It returns *true* if and only if _v_ is considered to have an identity suitable for a weak reference. i.e. Is _v_ a valid candidate key of a WeakMap, or entry in a WeakSet, or target of a WeakRef or FinalizationRegistry.
+
+ + 1. If _v_ is an Object, return *true*. + 1. If _v_ is a Symbol and KeyForSymbol(_v_) is *undefined*, return *true*. + 1. Return *false*. + + +

Symbols in the GlobalSymbolRegistry () are not considered suitable for use as a WeakMap key because they can be described by the String used to register them, and therefore do not have identity. Well-known symbols () are likely to never be collected, but are nonetheless treated as suitable for weak reference because they are limited to elements of a fixed size list and therefore manageable by a variety of implementation approaches. However, any values associated to a well-known symbol in a live WeakMap is unlikely to be collected and could “leak” memory resources in many implementations.

+
+
@@ -30330,10 +30350,7 @@

Symbol.keyFor ( _sym_ )

This function performs the following steps when called:

1. If _sym_ is not a Symbol, throw a *TypeError* exception. - 1. For each element _e_ of the GlobalSymbolRegistry List (see ), do - 1. If SameValue(_e_.[[Symbol]], _sym_) is *true*, return _e_.[[Key]]. - 1. Assert: GlobalSymbolRegistry does not currently contain an entry for _sym_. - 1. Return *undefined*. + 1. Return KeyForSymbol(_sym_).
@@ -30490,6 +30507,28 @@

Symbol.prototype [ @@toStringTag ]

Properties of Symbol Instances

Symbol instances are ordinary objects that inherit properties from the Symbol prototype object. Symbol instances have a [[SymbolData]] internal slot. The [[SymbolData]] internal slot is the Symbol value represented by this Symbol object.

+ + +

Abstract Operations for Symbols

+ + +

+ KeyForSymbol ( + _sym_: a Symbol, + ): a String or *undefined* +

+
+
description
+
If _sym_ is in the GlobalSymbolRegistry (see ) the String used to register _sym_ will be returned.
+
+ + 1. For each element _e_ of the GlobalSymbolRegistry List, do + 1. If SameValue(_e_.[[Symbol]], _sym_) is *true*, return _e_.[[Key]]. + 1. Assert: GlobalSymbolRegistry does not currently contain an entry for _sym_. + 1. Return *undefined*. + +
+
@@ -40719,11 +40758,11 @@

%SetIteratorPrototype% [ @@toStringTag ]

WeakMap Objects

-

WeakMaps are collections of key/value pairs where the keys are objects and values may be arbitrary ECMAScript language values. A WeakMap may be queried to see if it contains a key/value pair with a specific key, but no mechanism is provided for enumerating the objects it holds as keys. In certain conditions, objects which are not live are removed as WeakMap keys, as described in .

+

WeakMaps are collections of key/value pairs where the keys are objects and/or symbols and values may be arbitrary ECMAScript language values. A WeakMap may be queried to see if it contains a key/value pair with a specific key, but no mechanism is provided for enumerating the values it holds as keys. In certain conditions, values which are not live are removed as WeakMap keys, as described in .

An implementation may impose an arbitrarily determined latency between the time a key/value pair of a WeakMap becomes inaccessible and the time when the key/value pair is removed from the WeakMap. If this latency was observable to ECMAScript program, it would be a source of indeterminacy that could impact program execution. For that reason, an ECMAScript implementation must not provide any means to observe a key of a WeakMap that does not require the observer to present the observed key.

WeakMaps must be implemented using either hash tables or other mechanisms that, on average, provide access times that are sublinear on the number of key/value pairs in the collection. The data structure used in this specification is only intended to describe the required observable semantics of WeakMaps. It is not intended to be a viable implementation model.

-

WeakMap and WeakSets are intended to provide mechanisms for dynamically associating state with an object in a manner that does not “leak” memory resources if, in the absence of the WeakMap or WeakSet, the object otherwise became inaccessible and subject to resource reclamation by the implementation's garbage collection mechanisms. This characteristic can be achieved by using an inverted per-object mapping of weak map instances to keys. Alternatively each weak map may internally store its key to value mappings but this approach requires coordination between the WeakMap or WeakSet implementation and the garbage collector. The following references describe mechanism that may be useful to implementations of WeakMap and WeakSets:

+

WeakMap and WeakSet are intended to provide mechanisms for dynamically associating state with an object or symbol in a manner that does not “leak” memory resources if, in the absence of the WeakMap or WeakSet instance, the object or symbol otherwise became inaccessible and subject to resource reclamation by the implementation's garbage collection mechanisms. This characteristic can be achieved by using an inverted per-object/symbol mapping of WeakMap or WeakSet instances to keys. Alternatively, each WeakMap or WeakSet instance may internally store its key and value data, but this approach requires coordination between the WeakMap or WeakSet implementation and the garbage collector. The following references describe mechanism that may be useful to implementations of WeakMap and WeakSet:

Barry Hayes. 1997. Ephemerons: a new finalization mechanism. In Proceedings of the 12th ACM SIGPLAN conference on Object-oriented programming, systems, languages, and applications (OOPSLA '97), A. Michael Berman (Ed.). ACM, New York, NY, USA, 176-183, http://doi.acm.org/10.1145/263698.263733.

Alexandra Barros, Roberto Ierusalimschy, Eliminating Cycles in Weak Tables. Journal of Universal Computer Science - J.UCS, vol. 14, no. 21, pp. 3481-3497, 2008, http://www.jucs.org/jucs_14_21/eliminating_cycles_in_weak

@@ -40794,7 +40833,7 @@

WeakMap.prototype.delete ( _key_ )

1. Let _M_ be the *this* value. 1. Perform ? RequireInternalSlot(_M_, [[WeakMapData]]). 1. Let _entries_ be _M_.[[WeakMapData]]. - 1. If _key_ is not an Object, return *false*. + 1. If CanBeHeldWeakly(_key_) is *false*, return *false*. 1. For each Record { [[Key]], [[Value]] } _p_ of _entries_, do 1. If _p_.[[Key]] is not ~empty~ and SameValue(_p_.[[Key]], _key_) is *true*, then 1. Set _p_.[[Key]] to ~empty~. @@ -40814,7 +40853,7 @@

WeakMap.prototype.get ( _key_ )

1. Let _M_ be the *this* value. 1. Perform ? RequireInternalSlot(_M_, [[WeakMapData]]). 1. Let _entries_ be _M_.[[WeakMapData]]. - 1. If _key_ is not an Object, return *undefined*. + 1. If CanBeHeldWeakly(_key_) is *false*, return *undefined*. 1. For each Record { [[Key]], [[Value]] } _p_ of _entries_, do 1. If _p_.[[Key]] is not ~empty~ and SameValue(_p_.[[Key]], _key_) is *true*, return _p_.[[Value]]. 1. Return *undefined*. @@ -40828,7 +40867,7 @@

WeakMap.prototype.has ( _key_ )

1. Let _M_ be the *this* value. 1. Perform ? RequireInternalSlot(_M_, [[WeakMapData]]). 1. Let _entries_ be _M_.[[WeakMapData]]. - 1. If _key_ is not an Object, return *false*. + 1. If CanBeHeldWeakly(_key_) is *false*, return *false*. 1. For each Record { [[Key]], [[Value]] } _p_ of _entries_, do 1. If _p_.[[Key]] is not ~empty~ and SameValue(_p_.[[Key]], _key_) is *true*, return *true*. 1. Return *false*. @@ -40842,7 +40881,7 @@

WeakMap.prototype.set ( _key_, _value_ )

1. Let _M_ be the *this* value. 1. Perform ? RequireInternalSlot(_M_, [[WeakMapData]]). 1. Let _entries_ be _M_.[[WeakMapData]]. - 1. If _key_ is not an Object, throw a *TypeError* exception. + 1. If CanBeHeldWeakly(_key_) is *false*, throw a *TypeError* exception. 1. For each Record { [[Key]], [[Value]] } _p_ of _entries_, do 1. If _p_.[[Key]] is not ~empty~ and SameValue(_p_.[[Key]], _key_) is *true*, then 1. Set _p_.[[Value]] to _value_. @@ -40868,8 +40907,8 @@

Properties of WeakMap Instances

WeakSet Objects

-

WeakSets are collections of objects. A distinct object may only occur once as an element of a WeakSet's collection. A WeakSet may be queried to see if it contains a specific object, but no mechanism is provided for enumerating the objects it holds. In certain conditions, objects which are not live are removed as WeakSet elements, as described in .

-

An implementation may impose an arbitrarily determined latency between the time an object contained in a WeakSet becomes inaccessible and the time when the object is removed from the WeakSet. If this latency was observable to ECMAScript program, it would be a source of indeterminacy that could impact program execution. For that reason, an ECMAScript implementation must not provide any means to determine if a WeakSet contains a particular object that does not require the observer to present the observed object.

+

WeakSets are collections of objects and/or symbols. A distinct object or symbol may only occur once as an element of a WeakSet's collection. A WeakSet may be queried to see if it contains a specific value, but no mechanism is provided for enumerating the values it holds. In certain conditions, values which are not live are removed as WeakSet elements, as described in .

+

An implementation may impose an arbitrarily determined latency between the time a value contained in a WeakSet becomes inaccessible and the time when the value is removed from the WeakSet. If this latency was observable to ECMAScript program, it would be a source of indeterminacy that could impact program execution. For that reason, an ECMAScript implementation must not provide any means to determine if a WeakSet contains a particular value that does not require the observer to present the observed value.

WeakSets must be implemented using either hash tables or other mechanisms that, on average, provide access times that are sublinear on the number of elements in the collection. The data structure used in this specification is only intended to describe the required observable semantics of WeakSets. It is not intended to be a viable implementation model.

See the NOTE in .

@@ -40938,7 +40977,7 @@

WeakSet.prototype.add ( _value_ )

1. Let _S_ be the *this* value. 1. Perform ? RequireInternalSlot(_S_, [[WeakSetData]]). - 1. If _value_ is not an Object, throw a *TypeError* exception. + 1. If CanBeHeldWeakly(_value_) is *false*, throw a *TypeError* exception. 1. Let _entries_ be _S_.[[WeakSetData]]. 1. For each element _e_ of _entries_, do 1. If _e_ is not ~empty~ and SameValue(_e_, _value_) is *true*, then @@ -40959,7 +40998,7 @@

WeakSet.prototype.delete ( _value_ )

1. Let _S_ be the *this* value. 1. Perform ? RequireInternalSlot(_S_, [[WeakSetData]]). - 1. If _value_ is not an Object, return *false*. + 1. If CanBeHeldWeakly(_value_) is *false*, return *false*. 1. Let _entries_ be _S_.[[WeakSetData]]. 1. For each element _e_ of _entries_, do 1. If _e_ is not ~empty~ and SameValue(_e_, _value_) is *true*, then @@ -40979,7 +41018,7 @@

WeakSet.prototype.has ( _value_ )

1. Let _S_ be the *this* value. 1. Perform ? RequireInternalSlot(_S_, [[WeakSetData]]). 1. Let _entries_ be _S_.[[WeakSetData]]. - 1. If _value_ is not an Object, return *false*. + 1. If CanBeHeldWeakly(_value_) is *false*, return *false*. 1. For each element _e_ of _entries_, do 1. If _e_ is not ~empty~ and SameValue(_e_, _value_) is *true*, return *true*. 1. Return *false*. @@ -43061,7 +43100,7 @@

Managing Memory

WeakRef Objects

-

A WeakRef is an object that is used to refer to a target object without preserving it from garbage collection. WeakRefs can be dereferenced to allow access to the target object, if the target object hasn't been reclaimed by garbage collection.

+

A WeakRef is an object that is used to refer to a target object or symbol without preserving it from garbage collection. WeakRefs can be dereferenced to allow access to the target value, if the target hasn't been reclaimed by garbage collection.

The WeakRef Constructor

@@ -43087,7 +43126,7 @@

WeakRef ( _target_ )

This function performs the following steps when called:

1. If NewTarget is *undefined*, throw a *TypeError* exception. - 1. If _target_ is not an Object, throw a *TypeError* exception. + 1. If CanBeHeldWeakly(_target_) is *false*, throw a *TypeError* exception. 1. Let _weakRef_ be ? OrdinaryCreateFromConstructor(NewTarget, *"%WeakRef.prototype%"*, « [[WeakRefTarget]] »). 1. Perform AddToKeptObjects(_target_). 1. Set _weakRef_.[[WeakRefTarget]] to _target_. @@ -43143,7 +43182,7 @@

WeakRef.prototype.deref ( )

-

If the WeakRef returns a _target_ Object that is not *undefined*, then this _target_ object should not be garbage collected until the current execution of ECMAScript code has completed. The AddToKeptObjects operation makes sure read consistency is maintained.

+

If the WeakRef returns a _target_ value that is not *undefined*, then this _target_ value should not be garbage collected until the current execution of ECMAScript code has completed. The AddToKeptObjects operation makes sure read consistency is maintained.


             let target = { foo() {} };
@@ -43199,7 +43238,7 @@ 

Properties of WeakRef Instances

FinalizationRegistry Objects

-

A FinalizationRegistry is an object that manages registration and unregistration of cleanup operations that are performed when target objects are garbage collected.

+

A FinalizationRegistry is an object that manages registration and unregistration of cleanup operations that are performed when target objects and symbols are garbage collected.

The FinalizationRegistry Constructor

@@ -43278,9 +43317,9 @@

FinalizationRegistry.prototype.register ( _target_, _heldValue_ [ , _unregis 1. Let _finalizationRegistry_ be the *this* value. 1. Perform ? RequireInternalSlot(_finalizationRegistry_, [[Cells]]). - 1. If _target_ is not an Object, throw a *TypeError* exception. + 1. If CanBeHeldWeakly(_target_) is *false*, throw a *TypeError* exception. 1. If SameValue(_target_, _heldValue_) is *true*, throw a *TypeError* exception. - 1. If _unregisterToken_ is not an Object, then + 1. If CanBeHeldWeakly(_unregisterToken_) is *false*, then 1. If _unregisterToken_ is not *undefined*, throw a *TypeError* exception. 1. Set _unregisterToken_ to ~empty~. 1. Let _cell_ be the Record { [[WeakRefTarget]]: _target_, [[HeldValue]]: _heldValue_, [[UnregisterToken]]: _unregisterToken_ }. @@ -43299,7 +43338,7 @@

FinalizationRegistry.prototype.unregister ( _unregisterToken_ )

1. Let _finalizationRegistry_ be the *this* value. 1. Perform ? RequireInternalSlot(_finalizationRegistry_, [[Cells]]). - 1. If _unregisterToken_ is not an Object, throw a *TypeError* exception. + 1. If CanBeHeldWeakly(_unregisterToken_) is *false*, throw a *TypeError* exception. 1. Let _removed_ be *false*. 1. For each Record { [[WeakRefTarget]], [[HeldValue]], [[UnregisterToken]] } _cell_ of _finalizationRegistry_.[[Cells]], do 1. If _cell_.[[UnregisterToken]] is not ~empty~ and SameValue(_cell_.[[UnregisterToken]], _unregisterToken_) is *true*, then From 5abd0ba5b77d7ef6d744e514d41f4a117af23c88 Mon Sep 17 00:00:00 2001 From: Ashley Claymore Date: Fri, 10 Feb 2023 17:41:42 +0000 Subject: [PATCH 2/5] Apply suggestions from code review Co-authored-by: Kevin Gibbons --- spec.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/spec.html b/spec.html index df1a161a2d..82f00ab0e7 100644 --- a/spec.html +++ b/spec.html @@ -12166,7 +12166,7 @@

Liveness

WeakRef-obliviousness is defined on sets of objects or symbols instead of individual values to account for cycles. If it were defined on individual values, then a WeakRef referent in a cycle will be considered live even though its identity is only observed via other WeakRef referents in the cycle. - Colloquially, we say that an individual object or symbol is live if every set of objects containing it is live. + Colloquially, we say that an individual object or symbol is live if every set containing it is live.

At any point during evaluation, a set of objects and/or symbols _S_ is considered live if either of the following conditions is met:

@@ -12176,7 +12176,7 @@

Liveness

Any element in _S_ is included in any agent's [[KeptAlive]] List.
  • - There exists a valid future hypothetical WeakRef-oblivious execution with respect to _S_ that observes any value in _S_. + There exists a valid future hypothetical WeakRef-oblivious execution with respect to _S_ that observes the identity of any value in _S_.
  • From c76f06100cca17a6f17e44abb87fb542dbc04939 Mon Sep 17 00:00:00 2001 From: Ashley Claymore Date: Mon, 13 Feb 2023 17:35:40 +0000 Subject: [PATCH 3/5] Apply suggestions from code review Co-authored-by: Michael Dyck --- spec.html | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/spec.html b/spec.html index 82f00ab0e7..5d23372995 100644 --- a/spec.html +++ b/spec.html @@ -12308,14 +12308,14 @@

    description
    -
    It returns *true* if and only if _v_ is considered to have an identity suitable for a weak reference. i.e. Is _v_ a valid candidate key of a WeakMap, or entry in a WeakSet, or target of a WeakRef or FinalizationRegistry.
    +
    It returns *true* if and only if _v_ is considered to have an identity suitable for a weak reference. This determines whether _v_ can be a key of a WeakMap, or entry in a WeakSet, or target of a WeakRef or FinalizationRegistry.
    1. If _v_ is an Object, return *true*. 1. If _v_ is a Symbol and KeyForSymbol(_v_) is *undefined*, return *true*. 1. Return *false*. - +

    Symbols in the GlobalSymbolRegistry () are not considered suitable for use as a WeakMap key because they can be described by the String used to register them, and therefore do not have identity. Well-known symbols () are likely to never be collected, but are nonetheless treated as suitable for weak reference because they are limited to elements of a fixed size list and therefore manageable by a variety of implementation approaches. However, any values associated to a well-known symbol in a live WeakMap is unlikely to be collected and could “leak” memory resources in many implementations.

    From a03a93916b47a75811421d9a722bea6799e0558a Mon Sep 17 00:00:00 2001 From: Ashley Claymore Date: Sun, 26 Feb 2023 10:03:26 +0000 Subject: [PATCH 4/5] Apply suggestion from code review Co-authored-by: Michael Dyck --- spec.html | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/spec.html b/spec.html index 9763b21ad8..90d628dff9 100644 --- a/spec.html +++ b/spec.html @@ -12313,7 +12313,7 @@

    1. Return *false*. -

    Symbols in the GlobalSymbolRegistry () are not considered suitable for use as a WeakMap key because they can be described by the String used to register them, and therefore do not have identity. Well-known symbols () are likely to never be collected, but are nonetheless treated as suitable for weak reference because they are limited to elements of a fixed size list and therefore manageable by a variety of implementation approaches. However, any values associated to a well-known symbol in a live WeakMap is unlikely to be collected and could “leak” memory resources in many implementations.

    +

    Symbols in the GlobalSymbolRegistry () are not considered suitable for use as a WeakMap key because they can be described by the String used to register them, and therefore do not have identity. Well-known symbols () are likely to never be collected, but are nonetheless treated as suitable for weak reference because they are limited in number and therefore manageable by a variety of implementation approaches. However, any values associated to a well-known symbol in a live WeakMap is unlikely to be collected and could “leak” memory resources in many implementations.

    From 3a6c3f8d458b0fac5232ce0c7aceec368a28e5cf Mon Sep 17 00:00:00 2001 From: Michael Ficarra Date: Tue, 28 Feb 2023 14:19:10 -0700 Subject: [PATCH 5/5] don't mess with symbol identity in symbols-as-weakmap-keys proposal --- spec.html | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/spec.html b/spec.html index 90d628dff9..d73106e676 100644 --- a/spec.html +++ b/spec.html @@ -1086,8 +1086,8 @@

    Value Notation

    Identity

    In this specification, the word “is” is used to compare two values through equality, as in “If _bool_ is *true*, then”. For the purposes of this equality comparison, all values within this specification fall into one of the following two categories.

    -

    Values without identity are equal to other values without identity if all of their innate characteristics are the same — characteristics such as the magnitude of an integer or the length of a sequence. Because of this, values without identity may be manifest anywhere simply by fully describing their characteristics. It is not meaningful to change the characteristics of a value that does not have identity. Examples of values without identity include, but are not limited to: Booleans; mathematical values and extended mathematical values; Numbers; BigInts; *null*; *undefined*; sequences, including Strings, ECMAScript source text, surrogate pairs, Directive Prologues, etc; UTF-16 code units; Unicode code points; Symbols in the GlobalSymbolRegistry; enums; abstract operations, including syntax-directed operations, host hooks, etc; and ordered pairs. The preceding list is exhaustive for ECMAScript language values.

    -

    In contrast, each value with identity is unique and therefore only equal to itself. Values with identity are like values without identity but with an additional unguessable, unchangeable, universally-unique characteristic called identity. References to existing values with identity cannot be manifest simply by describing them, as the identity itself is indescribable; instead, references to these values must be explicitly passed from one place to another. Some values with identity are mutable and therefore can have their characteristics (except their identity) changed in-place, causing all holders of the value to observe the new characteristics. Examples of values with identity include, but are not limited to: Objects, including function objects, exotic objects, etc; any kind of Records, including Property Descriptors, PrivateElements, etc; Symbols not in the GlobalSymbolRegistry; Parse Nodes; Lists; Sets and Relations; Abstract Closures; Data Blocks; Private Names; execution contexts and execution context stacks; agent signifiers; and WaiterLists. The preceding list is exhaustive for ECMAScript language values.

    +

    Values without identity are equal to other values without identity if all of their innate characteristics are the same — characteristics such as the magnitude of an integer or the length of a sequence. Because of this, values without identity may be manifest anywhere simply by fully describing their characteristics. It is not meaningful to change the characteristics of a value that does not have identity. Examples of values without identity include, but are not limited to: Booleans; mathematical values and extended mathematical values; Numbers; BigInts; *null*; *undefined*; sequences, including Strings, ECMAScript source text, surrogate pairs, Directive Prologues, etc; UTF-16 code units; Unicode code points; enums; abstract operations, including syntax-directed operations, host hooks, etc; and ordered pairs. The preceding list is exhaustive for ECMAScript language values.

    +

    In contrast, each value with identity is unique and therefore only equal to itself. Values with identity are like values without identity but with an additional unguessable, unchangeable, universally-unique characteristic called identity. References to existing values with identity cannot be manifest simply by describing them, as the identity itself is indescribable; instead, references to these values must be explicitly passed from one place to another. Some values with identity are mutable and therefore can have their characteristics (except their identity) changed in-place, causing all holders of the value to observe the new characteristics. Examples of values with identity include, but are not limited to: Objects, including function objects, exotic objects, etc; any kind of Records, including Property Descriptors, PrivateElements, etc; Symbols; Parse Nodes; Lists; Sets and Relations; Abstract Closures; Data Blocks; Private Names; execution contexts and execution context stacks; agent signifiers; and WaiterLists. The preceding list is exhaustive for ECMAScript language values.

    A value without identity is never equal to a value with identity.

    @@ -12305,7 +12305,7 @@

    description
    -
    It returns *true* if and only if _v_ is considered to have an identity suitable for a weak reference. This determines whether _v_ can be a key of a WeakMap, or entry in a WeakSet, or target of a WeakRef or FinalizationRegistry.
    +
    It returns *true* if and only if _v_ is suitable for use as a weak reference. Only values that are suitable for use as a weak reference may be a key of a WeakMap, an element of a WeakSet, the target of a WeakRef, or one of the targets of a FinalizationRegistry.
    1. If _v_ is an Object, return *true*. @@ -12313,7 +12313,7 @@

    1. Return *false*. -

    Symbols in the GlobalSymbolRegistry () are not considered suitable for use as a WeakMap key because they can be described by the String used to register them, and therefore do not have identity. Well-known symbols () are likely to never be collected, but are nonetheless treated as suitable for weak reference because they are limited in number and therefore manageable by a variety of implementation approaches. However, any values associated to a well-known symbol in a live WeakMap is unlikely to be collected and could “leak” memory resources in many implementations.

    +

    Values without identity are not suitable for use as a weak reference because they may be manifest at any point without a prior reference. Symbols in the GlobalSymbolRegistry () are not suitable for use as a weak reference because they may be retrieved again at any later point using the String with which they were registered, making them ineligible for collection and therefore eternal. Well-known symbols () are likely to never be collected, but are nonetheless suitable for weak reference because they are limited in number and therefore manageable by a variety of implementation approaches. However, any value associated to a well-known symbol in a live WeakMap is unlikely to be collected and could "leak" memory resources in some implementations.

    @@ -30282,7 +30282,7 @@

    Symbol.for ( _key_ )

    1. Append the Record { [[Key]]: _stringKey_, [[Symbol]]: _newSymbol_ } to the GlobalSymbolRegistry List. 1. Return _newSymbol_.
    -

    The GlobalSymbolRegistry is a List that is globally available. It is shared by all realms. Prior to the evaluation of any ECMAScript code, it is initialized as a new empty List. Elements of the GlobalSymbolRegistry are Records with the structure defined in .

    +

    The GlobalSymbolRegistry is an append-only List that is globally available. It is shared by all realms. Prior to the evaluation of any ECMAScript code, it is initialized as a new empty List. Elements of the GlobalSymbolRegistry are Records with the structure defined in .