You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
* Ensure object creation specifies the realm
"Realm" is an ECMAScript concept best explained in
https://html.spec.whatwg.org/multipage/webappapis.html#realms-and-their-counterparts
Newly created JS objects must be associated with a Realm; while older
specs didn't do this explicitly, best practice is to be explicit about
this, especially for steps running "in parallel", or in algorithms
separate from method steps. Do so!
This also adds lint tests to try and catch future violations. Note
that dictionaries (e.g. MLOperatorDescriptor) are Infra "ordered maps"
it the body of spec algorithms, not JS objects, so they don't have a
realm. Conversion to a JS object when returning a dictionary to script
is handled by WebIDL bindings logic.
Also note that DOMExceptions, either thrown or as promise rejection
values, are not given a realm. This is a known issue across all web
specs and is tracked in whatwg/webidl#135.
Resolveswebmachinelearning#793.
* Don't double-init realm; and don't need realm for dicts
* Variable name improvement from @fdwr
Copy file name to clipboardexpand all lines: index.bs
+36-30
Original file line number
Diff line number
Diff line change
@@ -825,14 +825,14 @@ The <dfn dfn-for=MLContextOptions dfn-type=dict-member>powerPreference</dfn> opt
825
825
<summary>
826
826
To <dfn>create a context</dfn> given [=realm=] |realm| and |options| (a {{GPUDevice}} or {{MLContextOptions}}), run these steps:
827
827
</summary>
828
-
1. Let |context| be a new {{MLContext}}object with |realm|.
828
+
1. Let |context| be a new {{MLContext}}in |realm|.
829
829
1. If |options| is a {{GPUDevice}} object:
830
830
1. Set |context|.{{MLContext/[[contextType]]}} to "[=context type/webgpu=]".
831
831
1. Set |context|.{{MLContext/[[deviceType]]}} to {{MLDeviceType/"gpu"}}.
832
832
1. Set |context|.{{MLContext/[[powerPreference]]}} to {{MLPowerPreference/"default"}}.
833
833
1. Otherwise:
834
834
1. Set |context|.{{MLContext/[[contextType]]}} to "[=context type/default=]".
835
-
1. Set |context|.{{MLContext/[[lost]]}} to [=a new promise=].
835
+
1. Set |context|.{{MLContext/[[lost]]}} to [=a new promise=] in |realm|.
836
836
1. If |options|["{{MLContextOptions/deviceType}}"][=map/exists=], then set |context|.{{MLContext/[[deviceType]]}} to |options|["{{MLContextOptions/deviceType}}"].
837
837
1. Otherwise, set |context|.{{MLContext/[[deviceType]]}} to {{MLDeviceType/"cpu"}}.
838
838
1. If |options|["{{MLContextOptions/powerPreference}}"][=map/exists=], then set |context|.{{MLContext/[[powerPreference]]}} to |options|["{{MLContextOptions/powerPreference}}"].
@@ -846,9 +846,9 @@ The <dfn dfn-for=MLContextOptions dfn-type=dict-member>powerPreference</dfn> opt
846
846
The <dfn method for=ML>createContext(|options|)</dfn> steps are:
847
847
</summary>
848
848
1. Let |global| be [=this=]'s [=relevant global object=].
849
-
1. If |global|'s [=associated Document=] is not [=allowed to use=] the [=webnn-feature|webnn=] feature, return [=a new promise=][=rejected=] with a "{{SecurityError}}" {{DOMException}}.
850
849
1. Let |realm| be [=this=]'s [=relevant realm=].
851
-
1. Let |promise| be [=a new promise=].
850
+
1. If |global|'s [=associated Document=] is not [=allowed to use=] the [=webnn-feature|webnn=] feature, return [=a new promise=] in |realm| [=rejected=] with a "{{SecurityError}}" {{DOMException}}.
851
+
1. Let |promise| be [=a new promise=] in |realm|.
852
852
1. Run the following steps [=in parallel=].
853
853
1. Let |context| be the result of [=creating a context=] given |realm| and |options|. If that returns failure, then [=queue an ML task=] with |global| to [=reject=] |promise| with a "{{NotSupportedError}}" {{DOMException}} and abort these steps.
854
854
1. [=Queue an ML task=] with |global| to [=resolve=] |promise| with |context|.
@@ -860,9 +860,9 @@ The <dfn dfn-for=MLContextOptions dfn-type=dict-member>powerPreference</dfn> opt
860
860
The <dfn method for=ML>createContext(|gpuDevice|)</dfn> method steps are:
861
861
</summary>
862
862
1. Let |global| be [=this=]'s [=relevant global object=].
863
-
1. If |global|'s [=associated Document=] is not [=allowed to use=] the [=webnn-feature|webnn=] feature, return [=a new promise=][=rejected=] with a "{{SecurityError}}" {{DOMException}}.
864
863
1. Let |realm| be [=this=]'s [=relevant realm=].
865
-
1. Let |promise| be [=a new promise=].
864
+
1. If |global|'s [=associated Document=] is not [=allowed to use=] the [=webnn-feature|webnn=] feature, return [=a new promise=] in |realm| [=rejected=] with a "{{SecurityError}}" {{DOMException}}.
865
+
1. Let |promise| be [=a new promise=] in |realm|.
866
866
1. Run the following steps [=in parallel=].
867
867
1. Let |context| be the result of [=creating a context=] given |realm| and |gpuDevice|. If that returns failure, then [=queue an ML task=] with |global| to [=reject=] |promise| with a "{{NotSupportedError}}" {{DOMException}} and abort these steps.
868
868
1. [=Queue an ML task=] with |global| to [=resolve=] |promise| with |context|.
@@ -1044,7 +1044,7 @@ Note: `dispatch()` itself provides no signal that graph execution has completed.
1044
1044
'C': outputTensorC
1045
1045
};
1046
1046
context.dispatch(graph, inputs, outputs);
1047
-
1047
+
1048
1048
// 6. Read back the computed result.
1049
1049
const result = await context.readTensor(outputTensorC);
1050
1050
console.log('Output value:', new Float32Array(result)); // [1, 1, 1, 1]
@@ -1068,9 +1068,10 @@ Creates an {{MLTensor}} associated with this {{MLContext}}.
1068
1068
The <dfn method for=MLContext>createTensor(|descriptor|)</dfn> method steps are:
1069
1069
</summary>
1070
1070
1. Let |global| be [=this=]'s [=relevant global object=].
1071
-
1. If [=this=][=MLContext/is lost=], then return [=a new promise=][=rejected=] with an "{{InvalidStateError}}" {{DOMException}}.
1071
+
1. Let |realm| be [=this=]'s [=relevant realm=].
1072
+
1. If [=this=][=MLContext/is lost=], then return [=a new promise=] in |realm| [=rejected=] with an "{{InvalidStateError}}" {{DOMException}}.
1072
1073
1. Let |tensor| be the result of [=creating an MLTensor=] given [=this=], and |descriptor|.
1073
-
1. Let |promise| be [=a new promise=].
1074
+
1. Let |promise| be [=a new promise=] in |realm|.
1074
1075
1. Enqueue the following steps to [=this=].{{MLContext/[[timeline]]}}:
1075
1076
1. Run these steps, but [=/abort when=][=this=][=MLContext/is lost=]:
1076
1077
1. Create |tensor|.{{MLTensor/[[data]]}} given |descriptor| and initialize all bytes to zeros.
@@ -1097,10 +1098,10 @@ Reads back the {{MLTensor/[[data]]}} of an {{MLTensor}} from the {{MLContext}}.{
1097
1098
</summary>
1098
1099
1. Let |global| be [=this=]'s [=relevant global object=].
1099
1100
1. Let |realm| be [=this=]'s [=relevant realm=].
1100
-
1. If |tensor|.{{MLTensor/[[context]]}} is not [=this=], then return [=a new promise=][=rejected=] with a {{TypeError}}.
1101
-
1. If |tensor|.{{MLTensor/[[isDestroyed]]}} is true, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1102
-
1. If |tensor|.{{MLTensor/[[descriptor]]}}.{{MLTensorDescriptor/readable}} is false, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1103
-
1. Let |promise| be [=a new promise=].
1101
+
1. If |tensor|.{{MLTensor/[[context]]}} is not [=this=], then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1102
+
1. If |tensor|.{{MLTensor/[[isDestroyed]]}} is true, then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1103
+
1. If |tensor|.{{MLTensor/[[descriptor]]}}.{{MLTensorDescriptor/readable}} is false, then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1104
+
1. Let |promise| be [=a new promise=] in |realm|.
1104
1105
1. Enqueue the following steps to |tensor|.{{MLTensor/[[context]]}}.{{MLContext/[[timeline]]}}:
1105
1106
1. Run these steps, but [=/abort when=][=this=][=MLContext/is lost=]:
1106
1107
1. Let |bytes| be a [=/byte sequence=] containing a copy of |tensor|.{{MLTensor/[[data]]}}.
@@ -1127,11 +1128,12 @@ Bring-your-own-buffer variant of {{MLContext/readTensor(tensor)}}. Reads back th
1127
1128
The <dfn method for=MLContext>readTensor(|tensor|, |outputData|)</dfn> method steps are:
1128
1129
</summary>
1129
1130
1. Let |global| be [=this=]'s [=relevant global object=].
1130
-
1. If |tensor|.{{MLTensor/[[context]]}} is not [=this=], then return [=a new promise=][=rejected=] with a {{TypeError}}.
1131
-
1. If |tensor|.{{MLTensor/[[isDestroyed]]}} is true, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1132
-
1. If |tensor|.{{MLTensor/[[descriptor]]}}.{{MLTensorDescriptor/readable}} is false, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1133
-
1. If [=validating buffer with descriptor=] given |outputData| and |tensor|.{{MLTensor/[[descriptor]]}} returns false, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1134
-
1. Let |promise| be [=a new promise=].
1131
+
1. Let |realm| be [=this=]'s [=relevant realm=].
1132
+
1. If |tensor|.{{MLTensor/[[context]]}} is not [=this=], then return [=a new promise=] in |realm| [=rejected=] with a {{TypeError}}.
1133
+
1. If |tensor|.{{MLTensor/[[isDestroyed]]}} is true, then return [=a new promise=] in |realm| [=rejected=] with a {{TypeError}}.
1134
+
1. If |tensor|.{{MLTensor/[[descriptor]]}}.{{MLTensorDescriptor/readable}} is false, then return [=a new promise=] in |realm| [=rejected=] with a {{TypeError}}.
1135
+
1. If [=validating buffer with descriptor=] given |outputData| and |tensor|.{{MLTensor/[[descriptor]]}} returns false, then return [=a new promise=] in |realm| [=rejected=] with a {{TypeError}}.
1136
+
1. Let |promise| be [=a new promise=] in |realm|.
1135
1137
1. Enqueue the following steps to |tensor|.{{MLTensor/[[context]]}}.{{MLContext/[[timeline]]}}:
1136
1138
1. Run these steps, but [=/abort when=][=this=][=MLContext/is lost=]:
1137
1139
1. Let |bytes| be a [=/byte sequence=] containing a copy of |tensor|.{{MLTensor/[[data]]}}.
@@ -1498,7 +1500,8 @@ The {{MLOperand}} objects are created by the methods of {{MLGraphBuilder}}, inte
1498
1500
<summary>
1499
1501
To <dfn>create an MLOperand</dfn> given {{MLGraphBuilder}} |builder| and {{MLOperandDescriptor}} |desc|, run the following steps:
1500
1502
</summary>
1501
-
1. Let |operand| be a new {{MLOperand}}.
1503
+
1. Let |realm| be |builder|'s [=relevant realm=].
1504
+
1. Let |operand| be a new {{MLOperand}} in |realm|.
1502
1505
1. Set |operand|.{{MLOperand/[[builder]]}} to |builder|.
1503
1506
1. Set |operand|.{{MLOperand/[[descriptor]]}} to |desc|.
1504
1507
1. Return |operand|.
@@ -1508,8 +1511,10 @@ The {{MLOperand}} objects are created by the methods of {{MLGraphBuilder}}, inte
1508
1511
<summary>
1509
1512
To <dfn>copy an MLOperand</dfn> given {{MLOperand}} |operand|, run the following steps:
1510
1513
</summary>
1511
-
1. Let |result| be a new {{MLOperand}}.
1512
-
1. Set |result|.{{MLOperand/[[builder]]}} to |operand|.{{MLOperand/[[builder]]}}.
1514
+
1. Let |builder| be |operand|.{{MLOperand/[[builder]]}}.
1515
+
1. Let |realm| be |builder|'s [=relevant realm=].
1516
+
1. Let |result| be a new {{MLOperand}} in |realm|.
1517
+
1. Set |result|.{{MLOperand/[[builder]]}} to |builder|.
1513
1518
1. Set |result|.{{MLOperand/[[descriptor]]}} to |operand|.{{MLOperand/[[descriptor]]}}.
1514
1519
1. If |operand|.{{MLOperand/[[name]]}}[=map/exists=], then set |result|.{{MLOperand/[[name]]}} to |operand|.{{MLOperand/[[name]]}}.
1515
1520
1. Return |result|.
@@ -1607,7 +1612,8 @@ An {{MLTensor}} is created by its associated {{MLContext}}.
1607
1612
<summary>
1608
1613
To <dfn>create an MLTensor</dfn> given {{MLContext}} |context| and {{MLTensorDescriptor}} |descriptor|, run the following steps:
1609
1614
</summary>
1610
-
1. Let |tensor| be a new {{MLTensor}}.
1615
+
1. Let |realm| be |context|'s [=relevant realm=].
1616
+
1. Let |tensor| be a new {{MLTensor}} in |realm|.
1611
1617
1. Set |tensor|.{{MLTensor/[[context]]}} to |context|.
1612
1618
1. Set |tensor|.{{MLTensor/[[descriptor]]}} to |descriptor|.
1613
1619
1. Set |tensor|.{{MLTensor/[[isDestroyed]]}} to false.
@@ -1796,12 +1802,13 @@ Build a composed graph up to a given output operand into a computational graph a
1796
1802
<summary>
1797
1803
The <dfn method for=MLGraphBuilder>build(|outputs|)</dfn> method steps are:
1798
1804
</summary>
1799
-
1. If [=this=][=MLGraphBuilder/can not build=], then return [=a new promise=][=rejected=] with an "{{InvalidStateError}}" {{DOMException}}.
1800
-
1. If |outputs| is empty, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1805
+
1. Let |realm| be [=this=]'s [=relevant realm=].
1806
+
1. If [=this=][=MLGraphBuilder/can not build=], then return [=a new promise=] in |realm| [=rejected=] with an "{{InvalidStateError}}" {{DOMException}}.
1807
+
1. If |outputs| is empty, then return [=a new promise=] in |realm| [=rejected=] with a {{TypeError}}.
1801
1808
1. [=map/For each=] |name| → |operand| of |outputs|:
1802
-
1. If |name| is empty, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1803
-
1. If [=MLGraphBuilder/validating operand=] given [=this=] and |operand| returns false, then return [=a new promise=][=rejected=] with a {{TypeError}}.
1804
-
1. If |operand| is in [=this=]'s [=MLGraphBuilder/graph=]'s [=computational graph/inputs=] or [=computational graph/constants=], then return [=a new promise=][=rejected=] with a {{TypeError}}.
1809
+
1. If |name| is empty, then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1810
+
1. If [=MLGraphBuilder/validating operand=] given [=this=] and |operand| returns false, then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1811
+
1. If |operand| is in [=this=]'s [=MLGraphBuilder/graph=]'s [=computational graph/inputs=] or [=computational graph/constants=], then return [=a new promise=]in |realm| [=rejected=] with a {{TypeError}}.
1805
1812
1. Let |operands| be a new empty [=/set=].
1806
1813
1. Let |operators| be a new empty [=/set=].
1807
1814
1. Let |inputs| be a new empty [=/set=].
@@ -1814,16 +1821,15 @@ Build a composed graph up to a given output operand into a computational graph a
1814
1821
1. [=list/For each=] |input| of |operand|.{{MLOperand/[[operator]]}}'s [=operator/inputs=]:
1815
1822
1. [=queue/Enqueue=] |input| to |queue|.
1816
1823
1. Let |global| be [=this=]'s [=relevant global object=].
1817
-
1. Let |realm| be [=this=]'s [=relevant realm=].
1818
-
1. Let |graph| be a new {{MLGraph}} with |realm|.
1824
+
1. Let |graph| be a new {{MLGraph}} in |realm|.
1819
1825
1. Set |graph|.{{MLGraph/[[context]]}} to [=this=].{{MLGraphBuilder/[[context]]}}.
1820
1826
1. Set |graph|.{{MLGraph/[[isDestroyed]]}} to false.
1821
1827
1. [=set/For each=] |operand| in |inputs|:
1822
1828
1. Set |graph|.{{MLGraph/[[inputDescriptors]]}}[|operand|.{{MLOperand/[[name]]}}] to |operand|.{{MLOperand/[[descriptor]]}}.
1823
1829
1. [=map/For each=] |name| → |operand| of |outputs|:
1824
1830
1. Set |graph|.{{MLGraph/[[outputDescriptors]]}}[|name|] to |operand|.{{MLOperand/[[descriptor]]}}.
1825
1831
1. Set [=this=].{{MLGraphBuilder/[[hasBuilt]]}} to true.
1826
-
1. Let |promise| be [=a new promise=].
1832
+
1. Let |promise| be [=a new promise=] in |realm|.
1827
1833
1. Run the following steps [=in parallel=]:
1828
1834
1. Run these steps, but [=/abort when=] |graph|.{{MLGraph/[[context]]}}[=MLContext/is lost=]:
1829
1835
1. Let |graphImpl| be the result of converting [=this=]'s [=MLGraphBuilder/graph=] with |operands|, |operators|, |inputs|, and |outputs|'s [=map/values=] into an [=implementation-defined=] format which can be interpreted by the underlying platform.
0 commit comments