Skip to content

Commit 4f14b65

Browse files
authoredOct 5, 2021
Support "effects" in AO structured headers (#362)
1 parent a654534 commit 4f14b65

File tree

13 files changed

+484
-18
lines changed

13 files changed

+484
-18
lines changed
 

‎css/elements.css

+16
Original file line numberDiff line numberDiff line change
@@ -35,6 +35,22 @@ a:hover {
3535
color: #239dee;
3636
}
3737

38+
a.e-user-code {
39+
position: relative;
40+
}
41+
42+
a.e-user-code::before {
43+
display: none;
44+
color: brown;
45+
position: absolute;
46+
bottom: -0.9em;
47+
left: 0;
48+
content: 'UC';
49+
font-style: italic;
50+
font-weight: bold;
51+
font-size: small;
52+
}
53+
3854
code {
3955
font-weight: bold;
4056
font-family: Consolas, Monaco, monospace;

‎js/menu.js

+15-1
Original file line numberDiff line numberDiff line change
@@ -964,6 +964,8 @@ function makeLinkToId(id) {
964964
return (targetSec === 'index' ? './' : targetSec + '.html') + hash;
965965
}
966966

967+
let stylesheetWorkaroundForCanCallUserCodeAnnotation;
968+
967969
function doShortcut(e) {
968970
if (!(e.target instanceof HTMLElement)) {
969971
return;
@@ -973,7 +975,10 @@ function doShortcut(e) {
973975
if (name === 'textarea' || name === 'input' || name === 'select' || target.isContentEditable) {
974976
return;
975977
}
976-
if (e.key === 'm' && !e.altKey && !e.ctrlKey && !e.metaKey && !e.shiftKey && usesMultipage) {
978+
if (e.altKey || e.ctrlKey || e.metaKey || e.shiftKey) {
979+
return;
980+
}
981+
if (e.key === 'm' && usesMultipage) {
977982
let pathParts = location.pathname.split('/');
978983
let hash = location.hash;
979984
if (pathParts[pathParts.length - 2] === 'multipage') {
@@ -990,6 +995,13 @@ function doShortcut(e) {
990995
} else {
991996
location = 'multipage/' + hash;
992997
}
998+
} else if (e.key === 'u') {
999+
if (stylesheetWorkaroundForCanCallUserCodeAnnotation.innerText === '') {
1000+
stylesheetWorkaroundForCanCallUserCodeAnnotation.textContent =
1001+
'a.e-uc::before { display: block !important; }';
1002+
} else {
1003+
stylesheetWorkaroundForCanCallUserCodeAnnotation.textContent = '';
1004+
}
9931005
}
9941006
}
9951007

@@ -1017,4 +1029,6 @@ document.addEventListener('keypress', doShortcut);
10171029
document.addEventListener('DOMContentLoaded', () => {
10181030
Toolbox.init();
10191031
referencePane.init();
1032+
stylesheetWorkaroundForCanCallUserCodeAnnotation = document.createElement('style');
1033+
document.head.appendChild(stylesheetWorkaroundForCanCallUserCodeAnnotation);
10201034
});

‎src/Clause.ts

+28-2
Original file line numberDiff line numberDiff line change
@@ -20,9 +20,11 @@ export default class Clause extends Builder {
2020
subclauses: Clause[];
2121
number: string;
2222
aoid: string | null;
23+
type: string | null;
2324
notes: Note[];
2425
editorNotes: Note[];
2526
examples: Example[];
27+
effects: string[];
2628

2729
constructor(spec: Spec, node: HTMLElement, parent: Clause, number: string) {
2830
super(spec, node);
@@ -33,6 +35,7 @@ export default class Clause extends Builder {
3335
this.notes = [];
3436
this.editorNotes = [];
3537
this.examples = [];
38+
this.effects = [];
3639

3740
// namespace is either the entire spec or the parent clause's namespace.
3841
let parentNamespace = spec.namespace;
@@ -53,6 +56,11 @@ export default class Clause extends Builder {
5356
this.aoid = node.id;
5457
}
5558

59+
this.type = node.getAttribute('type');
60+
if (this.type === '') {
61+
this.type = null;
62+
}
63+
5664
let header = this.node.firstElementChild;
5765
while (header != null && header.tagName === 'SPAN' && header.children.length === 0) {
5866
// skip oldids
@@ -87,7 +95,7 @@ export default class Clause extends Builder {
8795
}
8896
// if we find such a DL, treat this as a structured header
8997

90-
const type = this.node.getAttribute('type');
98+
const type = this.type;
9199

92100
const { name, formattedHeader, formattedParams, formattedReturnType } = parseStructuredHeaderH1(
93101
this.spec,
@@ -120,7 +128,7 @@ export default class Clause extends Builder {
120128
header.innerHTML = formattedHeader;
121129
}
122130

123-
const { description, for: _for } = parseStructuredHeaderDl(this.spec, type, dl);
131+
const { description, for: _for, effects } = parseStructuredHeaderDl(this.spec, type, dl);
124132

125133
const paras = formatPreamble(
126134
this.spec,
@@ -158,6 +166,14 @@ export default class Clause extends Builder {
158166
this.node.setAttribute('aoid', name);
159167
this.aoid = name;
160168
}
169+
170+
this.effects = effects;
171+
for (const effect of effects) {
172+
if (!this.spec._effectWorklist.has(effect)) {
173+
this.spec._effectWorklist.set(effect, []);
174+
}
175+
this.spec._effectWorklist.get(effect)!.push(this);
176+
}
161177
}
162178

163179
buildNotes() {
@@ -184,6 +200,16 @@ export default class Clause extends Builder {
184200
}
185201
}
186202

203+
canHaveEffect(effectName: string) {
204+
// The following effects are runtime only:
205+
//
206+
// user-code: Only runtime can call user code.
207+
if (this.title !== null && this.title.startsWith('Static Semantics:')) {
208+
if (effectName === 'user-code') return false;
209+
}
210+
return true;
211+
}
212+
187213
static async enter({ spec, node, clauseStack, clauseNumberer }: Context) {
188214
if (!node.id) {
189215
spec.warn({

‎src/Spec.ts

+61
Original file line numberDiff line numberDiff line change
@@ -256,6 +256,17 @@ function isEmuImportElement(node: Node): node is EmuImportElement {
256256
return node.nodeType === 1 && node.nodeName === 'EMU-IMPORT';
257257
}
258258

259+
function maybeAddClauseToEffectWorklist(effectName: string, clause: Clause, worklist: Clause[]) {
260+
if (
261+
!worklist.includes(clause) &&
262+
clause.canHaveEffect(effectName) &&
263+
!clause.effects.includes(effectName)
264+
) {
265+
clause.effects.push(effectName);
266+
worklist.push(clause);
267+
}
268+
}
269+
259270
/*@internal*/
260271
export default class Spec {
261272
spec: this;
@@ -290,6 +301,8 @@ export default class Spec {
290301
}[];
291302
_prodRefs: ProdRef[];
292303
_textNodes: { [s: string]: [TextNodeContext] };
304+
_effectWorklist: Map<string, Clause[]>;
305+
_effectfulAOs: Map<string, Clause>;
293306
refsByClause: { [refId: string]: [string] };
294307

295308
private _fetch: (file: string, token: CancellationToken) => PromiseLike<string>;
@@ -332,6 +345,8 @@ export default class Spec {
332345
this._ntStringRefs = [];
333346
this._prodRefs = [];
334347
this._textNodes = {};
348+
this._effectWorklist = new Map();
349+
this._effectfulAOs = new Map();
335350
this.refsByClause = Object.create(null);
336351

337352
this.processMetadata();
@@ -455,6 +470,8 @@ export default class Spec {
455470

456471
this.autolink();
457472

473+
this.log('Propagating can-call-user-code annotations...');
474+
this.propagateEffects();
458475
this.log('Linking xrefs...');
459476
this._xrefs.forEach(xref => xref.build());
460477
this.log('Linking non-terminal references...');
@@ -652,6 +669,50 @@ export default class Spec {
652669
return true;
653670
}
654671

672+
private propagateEffects() {
673+
for (const [effectName, worklist] of this._effectWorklist) {
674+
this.propagateEffect(effectName, worklist);
675+
}
676+
}
677+
678+
private propagateEffect(effectName: string, worklist: Clause[]) {
679+
const usersOfAoid: Map<string, Set<Clause>> = new Map();
680+
for (const xref of this._xrefs) {
681+
if (xref.clause == null || xref.aoid == null) continue;
682+
if (!xref.canHaveEffect(effectName)) continue;
683+
684+
if (xref.hasAddedEffect(effectName)) {
685+
maybeAddClauseToEffectWorklist(effectName, xref.clause, worklist);
686+
}
687+
688+
const usedAoid = xref.aoid;
689+
if (!usersOfAoid.has(usedAoid)) {
690+
usersOfAoid.set(usedAoid, new Set());
691+
}
692+
usersOfAoid.get(usedAoid)!.add(xref.clause);
693+
}
694+
695+
while (worklist.length !== 0) {
696+
const clause = worklist.shift() as Clause;
697+
const aoid = clause.aoid;
698+
if (aoid == null || !usersOfAoid.has(aoid)) {
699+
continue;
700+
}
701+
702+
this._effectfulAOs.set(aoid, clause);
703+
for (const userClause of usersOfAoid.get(aoid)!) {
704+
maybeAddClauseToEffectWorklist(effectName, userClause, worklist);
705+
}
706+
}
707+
}
708+
709+
public getEffectsByAoid(aoid: string): string[] | null {
710+
if (this._effectfulAOs.has(aoid)) {
711+
return this._effectfulAOs.get(aoid)!.effects;
712+
}
713+
return null;
714+
}
715+
655716
private async buildMultipage(wrapper: Element, tocEles: Element[], jsSha: string) {
656717
let stillIntro = true;
657718
const introEles = [];

‎src/Xref.ts

+97-6
Original file line numberDiff line numberDiff line change
@@ -4,15 +4,19 @@ import type * as Biblio from './Biblio';
44
import type Clause from './Clause';
55

66
import Builder from './Builder';
7+
import { validateEffects } from './utils';
78

89
/*@internal*/
910
export default class Xref extends Builder {
1011
namespace: string;
1112
href: string;
1213
aoid: string;
14+
isInvocation: boolean;
1315
clause: Clause | null;
1416
id: string;
1517
entry: Biblio.BiblioEntry | undefined;
18+
addEffects: string[] | null;
19+
suppressEffects: string[] | null;
1620

1721
static elements = ['EMU-XREF'];
1822

@@ -30,6 +34,61 @@ export default class Xref extends Builder {
3034
this.aoid = aoid;
3135
this.clause = clause;
3236
this.id = node.getAttribute('id')!;
37+
this.isInvocation = node.hasAttribute('is-invocation');
38+
node.removeAttribute('is-invocation');
39+
40+
// Check if there's metadata adding or suppressing effects
41+
this.addEffects = null;
42+
this.suppressEffects = null;
43+
if (node.parentElement && node.parentElement.tagName === 'EMU-META') {
44+
if (node.parentElement.hasAttribute('effects')) {
45+
const addEffects = node.parentElement.getAttribute('effects')!.split(',');
46+
if (addEffects.length !== 0) {
47+
this.addEffects = validateEffects(spec, addEffects, node.parentElement);
48+
}
49+
}
50+
if (node.parentElement.hasAttribute('suppress-effects')) {
51+
const suppressEffects = node.parentElement.getAttribute('suppress-effects')!.split(',');
52+
if (suppressEffects.length !== 0) {
53+
this.suppressEffects = validateEffects(spec, suppressEffects, node.parentElement);
54+
}
55+
}
56+
if (this.addEffects !== null && this.suppressEffects !== null) {
57+
for (const e of this.addEffects) {
58+
if (this.suppressEffects.includes(e)) {
59+
throw new Error('effect suppression is contradictory');
60+
}
61+
}
62+
for (const e of this.suppressEffects) {
63+
if (this.addEffects.includes(e)) {
64+
throw new Error('effect suppression is contradictory');
65+
}
66+
}
67+
}
68+
69+
// Strip an outer <emu-meta> if present
70+
const children = node.parentElement.childNodes;
71+
node.parentElement.replaceWith(...children);
72+
}
73+
}
74+
75+
canHaveEffect(effectName: string) {
76+
if (!this.isInvocation) return false;
77+
if (this.clause && !this.clause.canHaveEffect(effectName)) {
78+
return false;
79+
}
80+
if (this.suppressEffects !== null) {
81+
return !this.suppressEffects.includes(effectName);
82+
}
83+
return true;
84+
}
85+
86+
hasAddedEffect(effectName: string) {
87+
if (!this.isInvocation) return false;
88+
if (this.addEffects !== null) {
89+
return this.addEffects.includes(effectName);
90+
}
91+
return false;
3392
}
3493

3594
static async enter({ node, spec, clauseStack }: Context) {
@@ -141,7 +200,34 @@ export default class Xref extends Builder {
141200
this.entry = spec.biblio.byAoid(aoid, namespace);
142201

143202
if (this.entry) {
144-
buildAOLink(node, this.entry);
203+
let effects = null;
204+
let classNames = null;
205+
206+
if (this.isInvocation) {
207+
effects = spec.getEffectsByAoid(aoid);
208+
}
209+
if (this.addEffects !== null) {
210+
if (effects !== null) {
211+
effects.push(...this.addEffects);
212+
} else {
213+
effects = this.addEffects;
214+
}
215+
}
216+
217+
if (effects) {
218+
if (this.suppressEffects !== null) {
219+
effects = effects.filter(e => !this.suppressEffects!.includes(e));
220+
}
221+
if (effects.length !== 0) {
222+
const parentClause = this.clause;
223+
effects = parentClause ? effects.filter(e => parentClause.canHaveEffect(e)) : effects;
224+
if (effects.length !== 0) {
225+
classNames = effects.map(e => `e-${e}`).join(' ');
226+
}
227+
}
228+
}
229+
230+
buildAOLink(node, this.entry, classNames);
145231
return;
146232
}
147233

@@ -179,11 +265,11 @@ function buildProductionLink(xref: Element, entry: Biblio.ProductionBiblioEntry)
179265
}
180266
}
181267

182-
function buildAOLink(xref: Element, entry: Biblio.BiblioEntry) {
268+
function buildAOLink(xref: Element, entry: Biblio.BiblioEntry, classNames: string | null) {
183269
if (xref.textContent!.trim() === '') {
184-
xref.innerHTML = buildXrefLink(entry, xref.getAttribute('aoid'));
270+
xref.innerHTML = buildXrefLink(entry, xref.getAttribute('aoid'), classNames);
185271
} else {
186-
xref.innerHTML = buildXrefLink(entry, xref.innerHTML);
272+
xref.innerHTML = buildXrefLink(entry, xref.innerHTML, classNames);
187273
}
188274
}
189275

@@ -276,6 +362,11 @@ function buildStepLink(spec: Spec, xref: Element, entry: Biblio.StepBiblioEntry)
276362
xref.innerHTML = buildXrefLink(entry, text);
277363
}
278364

279-
function buildXrefLink(entry: Biblio.BiblioEntry, contents: string | number | undefined | null) {
280-
return '<a href="' + entry.location + '#' + (entry.id || entry.refId) + '">' + contents + '</a>';
365+
function buildXrefLink(
366+
entry: Biblio.BiblioEntry,
367+
contents: string | number | undefined | null,
368+
classNames: string | null = null
369+
) {
370+
const classSnippet = classNames == null ? '' : ' class="' + classNames + '"';
371+
return `<a href="${entry.location}#${entry.id || entry.refId}"${classSnippet}>${contents}</a>`;
281372
}

‎src/autolinker.ts

+14-3
Original file line numberDiff line numberDiff line change
@@ -42,7 +42,7 @@ export function autolink(
4242
const spec = clause.spec;
4343
const template = spec.doc.createElement('template');
4444
const content = escape(node.textContent!);
45-
const autolinked = content.replace(replacer, match => {
45+
const autolinked = content.replace(replacer, (match, offset) => {
4646
const entry = autolinkmap[narrowSpace(match)];
4747
if (!entry) {
4848
return match;
@@ -56,9 +56,20 @@ export function autolink(
5656
}
5757

5858
if (entry.aoid) {
59-
return '<emu-xref aoid="' + entry.aoid + '">' + match + '</emu-xref>';
59+
let isInvocationAttribute = '';
60+
// Matches function-style invocation with parentheses and SDO-style 'of'
61+
// invocation.
62+
if (
63+
content[offset + match.length] === '(' ||
64+
(content[offset + match.length] === ' ' &&
65+
content[offset + match.length + 1] === 'o' &&
66+
content[offset + match.length + 2] === 'f')
67+
) {
68+
isInvocationAttribute = ' is-invocation';
69+
}
70+
return `<emu-xref aoid="${entry.aoid}"${isInvocationAttribute}>${match}</emu-xref>`;
6071
} else {
61-
return '<emu-xref href="#' + (entry.id || entry.refId) + '">' + match + '</emu-xref>';
72+
return `<emu-xref href="#${entry.id || entry.refId}">${match}</emu-xref>`;
6273
}
6374
});
6475

‎src/header-parser.ts

+15-3
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
import type Spec from './Spec';
2-
import { offsetToLineAndColumn } from './utils';
2+
import { offsetToLineAndColumn, validateEffects } from './utils';
33

44
export function parseStructuredHeaderH1(
55
spec: Spec,
@@ -224,9 +224,10 @@ export function parseStructuredHeaderDl(
224224
spec: Spec,
225225
type: string | null,
226226
dl: Element
227-
): { description: Element | null; for: Element | null } {
227+
): { description: Element | null; for: Element | null; effects: string[] } {
228228
let description = null;
229229
let _for = null;
230+
let effects: string[] = [];
230231
for (let i = 0; i < dl.children.length; ++i) {
231232
const dt = dl.children[i];
232233
if (dt.tagName !== 'DT') {
@@ -285,6 +286,17 @@ export function parseStructuredHeaderDl(
285286
}
286287
break;
287288
}
289+
case 'effects': {
290+
// The dd contains a comma-separated list of effects.
291+
if (dd.textContent !== null) {
292+
effects = validateEffects(
293+
spec,
294+
dd.textContent.split(',').map(c => c.trim()),
295+
dd
296+
);
297+
}
298+
break;
299+
}
288300
case '': {
289301
spec.warn({
290302
type: 'node',
@@ -305,7 +317,7 @@ export function parseStructuredHeaderDl(
305317
}
306318
}
307319
}
308-
return { description, for: _for };
320+
return { description, for: _for, effects };
309321
}
310322

311323
export function formatPreamble(

‎src/utils.ts

+25
Original file line numberDiff line numberDiff line change
@@ -219,3 +219,28 @@ export function attrValueLocation(
219219
return { line: attrLoc.line, column: attrLoc.col + (tagText.match(matcher)?.[0].length ?? 0) };
220220
}
221221
}
222+
223+
const KNOWN_EFFECTS = ['user-code'];
224+
export function validateEffects(spec: Spec, effectsRaw: string[], node: Element) {
225+
const effects = [];
226+
const unknownEffects = [];
227+
228+
for (const e of effectsRaw) {
229+
if (KNOWN_EFFECTS.indexOf(e) !== -1) {
230+
effects.push(e);
231+
} else {
232+
unknownEffects.push(e);
233+
}
234+
}
235+
236+
if (unknownEffects.length !== 0) {
237+
spec.warn({
238+
type: 'node',
239+
ruleId: 'unknown-effects',
240+
message: `unknown effects: ${unknownEffects}`,
241+
node,
242+
});
243+
}
244+
245+
return effects;
246+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,75 @@
1+
<!doctype html>
2+
<head><meta charset="utf-8"></head><body><div id="spec-container">
3+
4+
<emu-clause id="sec-user-code" type="abstract operation" aoid="UserCode">
5+
<h1><span class="secnum">1</span> UserCode ( )</h1>
6+
<p>The abstract operation UserCode takes no arguments. It performs the following steps when called:</p>
7+
<emu-alg><ol><li>Call user code.</li></ol></emu-alg>
8+
</emu-clause>
9+
10+
<emu-clause id="sec-nop" type="abstract operation" aoid="Nop">
11+
<h1><span class="secnum">2</span> Nop ( )</h1>
12+
<p>The abstract operation Nop takes no arguments. It performs the following steps when called:</p>
13+
<emu-alg><ol><li>Do nothing.</li></ol></emu-alg>
14+
</emu-clause>
15+
16+
<emu-clause id="sec-direct-call" type="abstract operation" aoid="DirectCall">
17+
<h1><span class="secnum">3</span> DirectCall()</h1>
18+
<p>The abstract operation DirectCall takes no arguments. Calling AOs that can call user code should insert <code>e-user-code</code> as a class into the AO link. It performs the following steps when called:</p>
19+
<emu-alg><ol><li><emu-xref aoid="UserCode" id="_ref_0"><a href="#sec-user-code" class="e-user-code">UserCode</a></emu-xref>().</li></ol></emu-alg>
20+
</emu-clause>
21+
22+
<emu-clause id="sec-transitive-call" type="abstract operation" aoid="TransitiveCall">
23+
<h1><span class="secnum">4</span> TransitiveCall()</h1>
24+
<p>The abstract operation TransitiveCall takes no arguments. Calling AOs that can transitively call user code should insert <code>e-user-code</code> as a class into the AO link. It performs the following steps when called:</p>
25+
<emu-alg><ol><li><emu-xref aoid="DirectCall" id="_ref_1"><a href="#sec-direct-call" class="e-user-code">DirectCall</a></emu-xref>().</li></ol></emu-alg>
26+
</emu-clause>
27+
28+
<emu-clause id="sec-suppressed-direct-call" type="abstract operation" aoid="SuppressedDirectCall">
29+
<h1><span class="secnum">5</span> SuppressedDirectCall()</h1>
30+
<p>The abstract operation SuppressedDirectCall takes no arguments. Can-call-user-code callsites that are suppressed do not get <code>e-user-code</code> as a class in the AO link. It performs the following steps when called:</p>
31+
<emu-alg><ol><li><emu-xref aoid="TransitiveCall" id="_ref_2"><a href="#sec-transitive-call">TransitiveCall</a></emu-xref>().</li></ol></emu-alg>
32+
</emu-clause>
33+
34+
<emu-clause id="sec-suppressed-transitive-call" type="abstract operation" aoid="SuppressedTransitiveCall">
35+
<h1><span class="secnum">6</span> SuppressedTransitiveCall()</h1>
36+
<p>The abstract operation SuppressedTransitiveCall takes no arguments. Can-call-user-code callsites that are suppressed do not propagate the effect It performs the following steps when called:</p>
37+
<emu-alg><ol><li><emu-xref aoid="SuppressedDirectCall" id="_ref_3"><a href="#sec-suppressed-direct-call">SuppressedDirectCall</a></emu-xref>().</li></ol></emu-alg>
38+
</emu-clause>
39+
40+
<emu-clause id="sec-added-direct-call" type="abstract operation" aoid="AddedDirectCall">
41+
<h1><span class="secnum">7</span> AddedDirectCall()</h1>
42+
<p>The abstract operation AddedDirectCall takes no arguments. AOs can have manually added user-code effect at a callsite that propagates. It performs the following steps when called:</p>
43+
<emu-alg><ol><li><emu-xref aoid="Nop" id="_ref_4"><a href="#sec-nop" class="e-user-code">Nop</a></emu-xref>().</li></ol></emu-alg>
44+
</emu-clause>
45+
46+
<emu-clause id="sec-added-transitive-call" type="abstract operation" aoid="AddedTransitiveCall">
47+
<h1><span class="secnum">8</span> AddedTransitiveCall()</h1>
48+
<p>The abstract operation AddedTransitiveCall takes no arguments. AOs can have manually added user-code effect at a callsite that propagates. It performs the following steps when called:</p>
49+
<emu-alg><ol><li><emu-xref aoid="AddedDirectCall" id="_ref_5"><a href="#sec-added-direct-call" class="e-user-code">AddedDirectCall</a></emu-xref>().</li></ol></emu-alg>
50+
</emu-clause>
51+
52+
<emu-clause id="sec-sdo-invocation" type="abstract operation" aoid="SDOInvocations">
53+
<h1><span class="secnum">9</span> SDOInvocations()</h1>
54+
<p>The abstract operation SDOInvocations takes no arguments. SDO-style invocations of AOs that can call user code also have the <code>e-user-code</code> class in the link. It performs the following steps when called:</p>
55+
<emu-alg><ol><li><emu-xref aoid="UserCode" id="_ref_6"><a href="#sec-user-code" class="e-user-code">UserCode</a></emu-xref> of Bar.</li></ol></emu-alg>
56+
</emu-clause>
57+
58+
<emu-clause id="sec-non-invocations" type="abstract operation" aoid="NonInvocations">
59+
<h1><span class="secnum">10</span> NonInvocations()</h1>
60+
<p>The abstract operation NonInvocations takes no arguments. Non-invocations (i.e. not followed by () or " of") do not have <code>e-user-code</code> as a class in the AO link. It performs the following steps when called:</p>
61+
<emu-alg><ol><li><emu-xref aoid="UserCode" id="_ref_7"><a href="#sec-user-code">UserCode</a></emu-xref> is an abstract operation.</li></ol></emu-alg>
62+
</emu-clause>
63+
64+
<emu-clause id="sec-static-direct-call" type="sdo" aoid="StaticDirectCall">
65+
<h1><span class="secnum">11</span> Static Semantics: StaticDirectCall</h1>
66+
<p>The <emu-xref href="#sec-algorithm-conventions-syntax-directed-operations"><a href="https://tc39.es/ecma262/#sec-algorithm-conventions-syntax-directed-operations">syntax-directed operation</a></emu-xref> StaticDirectCall takes no arguments. <emu-xref href="#sec-static-semantic-rules"><a href="https://tc39.es/ecma262/#sec-static-semantic-rules">Static semantics</a></emu-xref> suppress user-code.</p>
67+
<emu-alg><ol><li><emu-xref aoid="UserCode" id="_ref_8"><a href="#sec-user-code">UserCode</a></emu-xref>().</li></ol></emu-alg>
68+
</emu-clause>
69+
70+
<emu-clause id="sec-static-transitive-call" type="sdo" aoid="StaticTransitiveCall">
71+
<h1><span class="secnum">12</span> Static Semantics: StaticTransitiveCall</h1>
72+
<p>The <emu-xref href="#sec-algorithm-conventions-syntax-directed-operations"><a href="https://tc39.es/ecma262/#sec-algorithm-conventions-syntax-directed-operations">syntax-directed operation</a></emu-xref> StaticTransitiveCall takes no arguments. <emu-xref href="#sec-static-semantic-rules"><a href="https://tc39.es/ecma262/#sec-static-semantic-rules">Static semantics</a></emu-xref> suppress user-code.</p>
73+
<emu-alg><ol><li><emu-xref aoid="StaticDirectCall" id="_ref_9"><a href="#sec-static-direct-call">StaticDirectCall</a></emu-xref> of Baz.</li></ol></emu-alg>
74+
</emu-clause>
75+
</div></body>

‎test/baselines/generated-reference/multipage.html/multipage/index.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<!doctype html>
22
<head><meta charset="utf-8">
33
<link rel="icon" href="../img/favicon.ico">
4-
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=nov2YnIu" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-intro"></head>
4+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=ZToCK906" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-intro"></head>
55
<body><div id="menu"><div id="menu-search"><input type="text" id="menu-search-box" placeholder="Search..."><div id="menu-search-results" class="inactive"></div></div><div id="menu-pins"><div class="menu-pane-header">Pins</div><ul id="menu-pins-list"></ul></div><div class="menu-pane-header">Table of Contents</div><div id="menu-toc"><ol class="toc"><li><span class="item-toggle-none"></span><a href="./#sec-intro" title="Intro">Intro</a></li><li><span class="item-toggle-none"></span><a href="second.html#sec-second" title="Second Clause"><span class="secnum">1</span> Second Clause</a></li><li><span class="item-toggle">◢</span><a href="third.html#sec-third" title="Third Clause"><span class="secnum">2</span> Third Clause</a><ol class="toc"><li><span class="item-toggle-none"></span><a href="third.html#sec-alg" title="Algorithm"><span class="secnum">2.1</span> Algorithm</a></li></ol></li></ol></div></div>
66
<div id="menu-spacer"></div>
77
<div id="menu-toggle"><svg xmlns="http://www.w3.org/2000/svg" style="width:100%; height:100%; stroke:currentColor" viewBox="0 0 120 120">

‎test/baselines/generated-reference/multipage.html/multipage/second.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<!doctype html>
22
<head><meta charset="utf-8">
33
<link rel="icon" href="../img/favicon.ico">
4-
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=nov2YnIu" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-second"></head>
4+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=ZToCK906" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-second"></head>
55
<body><div id="menu"><div id="menu-search"><input type="text" id="menu-search-box" placeholder="Search..."><div id="menu-search-results" class="inactive"></div></div><div id="menu-pins"><div class="menu-pane-header">Pins</div><ul id="menu-pins-list"></ul></div><div class="menu-pane-header">Table of Contents</div><div id="menu-toc"><ol class="toc"><li><span class="item-toggle-none"></span><a href="./#sec-intro" title="Intro">Intro</a></li><li><span class="item-toggle-none"></span><a href="second.html#sec-second" title="Second Clause"><span class="secnum">1</span> Second Clause</a></li><li><span class="item-toggle">◢</span><a href="third.html#sec-third" title="Third Clause"><span class="secnum">2</span> Third Clause</a><ol class="toc"><li><span class="item-toggle-none"></span><a href="third.html#sec-alg" title="Algorithm"><span class="secnum">2.1</span> Algorithm</a></li></ol></li></ol></div></div>
66
<div id="menu-spacer"></div>
77
<div id="menu-toggle"><svg xmlns="http://www.w3.org/2000/svg" style="width:100%; height:100%; stroke:currentColor" viewBox="0 0 120 120">

‎test/baselines/generated-reference/multipage.html/multipage/third.html

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
<!doctype html>
22
<head><meta charset="utf-8">
33
<link rel="icon" href="../img/favicon.ico">
4-
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=nov2YnIu" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-third"></head>
4+
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/highlight.js/11.0.1/styles/base16/solarized-light.min.css"><link rel="stylesheet" href="../ecmarkup.css"><script src="../ecmarkup.js?cache=ZToCK906" defer=""></script><script src="multipage.js?cache=fWLoMF3T" defer=""></script><link rel="canonical" href="../#sec-third"></head>
55
<body><div id="menu"><div id="menu-search"><input type="text" id="menu-search-box" placeholder="Search..."><div id="menu-search-results" class="inactive"></div></div><div id="menu-pins"><div class="menu-pane-header">Pins</div><ul id="menu-pins-list"></ul></div><div class="menu-pane-header">Table of Contents</div><div id="menu-toc"><ol class="toc"><li><span class="item-toggle-none"></span><a href="./#sec-intro" title="Intro">Intro</a></li><li><span class="item-toggle-none"></span><a href="second.html#sec-second" title="Second Clause"><span class="secnum">1</span> Second Clause</a></li><li><span class="item-toggle">◢</span><a href="third.html#sec-third" title="Third Clause"><span class="secnum">2</span> Third Clause</a><ol class="toc"><li><span class="item-toggle-none"></span><a href="third.html#sec-alg" title="Algorithm"><span class="secnum">2.1</span> Algorithm</a></li></ol></li></ol></div></div>
66
<div id="menu-spacer"></div>
77
<div id="menu-toggle"><svg xmlns="http://www.w3.org/2000/svg" style="width:100%; height:100%; stroke:currentColor" viewBox="0 0 120 120">
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,135 @@
1+
<pre class=metadata>
2+
toc: false
3+
copyright: false
4+
assets: none
5+
</pre>
6+
7+
<emu-clause id="sec-user-code" type="abstract operation">
8+
<h1>UserCode ( )</h1>
9+
<dl class="header">
10+
<dt>effects</dt>
11+
<dd>user-code</dd>
12+
</dl>
13+
<emu-alg>
14+
1. Call user code.
15+
</emu-alg>
16+
</emu-clause>
17+
18+
<emu-clause id="sec-nop" type="abstract operation">
19+
<h1>Nop ( )</h1>
20+
<dl class="header">
21+
</dl>
22+
<emu-alg>
23+
1. Do nothing.
24+
</emu-alg>
25+
</emu-clause>
26+
27+
<emu-clause id="sec-direct-call" type="abstract operation">
28+
<h1>DirectCall()</h1>
29+
<dl class="header">
30+
<dt>description</dt>
31+
<dd>Calling AOs that can call user code should insert <code>e-user-code</code> as a class into the AO link.</dd>
32+
</dl>
33+
<emu-alg>
34+
1. UserCode().
35+
</emu-alg>
36+
</emu-clause>
37+
38+
<emu-clause id="sec-transitive-call" type="abstract operation">
39+
<h1>TransitiveCall()</h1>
40+
<dl class="header">
41+
<dt>description</dt>
42+
<dd>Calling AOs that can transitively call user code should insert <code>e-user-code</code> as a class into the AO link.</dd>
43+
</dl>
44+
<emu-alg>
45+
1. DirectCall().
46+
</emu-alg>
47+
</emu-clause>
48+
49+
<emu-clause id="sec-suppressed-direct-call" type="abstract operation">
50+
<h1>SuppressedDirectCall()</h1>
51+
<dl class="header">
52+
<dt>description</dt>
53+
<dd>Can-call-user-code callsites that are suppressed do not get <code>e-user-code</code> as a class in the AO link.</dd>
54+
</dl>
55+
<emu-alg>
56+
1. <emu-meta suppress-effects="user-code">TransitiveCall()</emu-meta>.
57+
</emu-alg>
58+
</emu-clause>
59+
60+
<emu-clause id="sec-suppressed-transitive-call" type="abstract operation">
61+
<h1>SuppressedTransitiveCall()</h1>
62+
<dl class="header">
63+
<dt>description</dt>
64+
<dd>Can-call-user-code callsites that are suppressed do not propagate the effect</dd>
65+
</dl>
66+
<emu-alg>
67+
1. SuppressedDirectCall().
68+
</emu-alg>
69+
</emu-clause>
70+
71+
<emu-clause id="sec-added-direct-call" type="abstract operation">
72+
<h1>AddedDirectCall()</h1>
73+
<dl class="header">
74+
<dt>description</dt>
75+
<dd>AOs can have manually added user-code effect at a callsite that propagates.</dd>
76+
</dl>
77+
<emu-alg>
78+
1. <emu-meta effects="user-code">Nop()</emu-meta>.
79+
</emu-alg>
80+
</emu-clause>
81+
82+
<emu-clause id="sec-added-transitive-call" type="abstract operation">
83+
<h1>AddedTransitiveCall()</h1>
84+
<dl class="header">
85+
<dt>description</dt>
86+
<dd>AOs can have manually added user-code effect at a callsite that propagates.</dd>
87+
</dl>
88+
<emu-alg>
89+
1. AddedDirectCall().
90+
</emu-alg>
91+
</emu-clause>
92+
93+
<emu-clause id="sec-sdo-invocation" type="abstract operation">
94+
<h1>SDOInvocations()</h1>
95+
<dl class="header">
96+
<dt>description</dt>
97+
<dd>SDO-style invocations of AOs that can call user code also have the <code>e-user-code</code> class in the link.</dd>
98+
</dl>
99+
<emu-alg>
100+
1. UserCode of Bar.
101+
</emu-alg>
102+
</emu-clause>
103+
104+
<emu-clause id="sec-non-invocations" type="abstract operation">
105+
<h1>NonInvocations()</h1>
106+
<dl class="header">
107+
<dt>description</dt>
108+
<dd>Non-invocations (i.e. not followed by () or " of") do not have <code>e-user-code</code> as a class in the AO link.</dd>
109+
</dl>
110+
<emu-alg>
111+
1. UserCode is an abstract operation.
112+
</emu-alg>
113+
</emu-clause>
114+
115+
<emu-clause id="sec-static-direct-call" type="sdo">
116+
<h1>Static Semantics: StaticDirectCall</h1>
117+
<dl class="header">
118+
<dt>description</dt>
119+
<dd>Static semantics suppress user-code.</dd>
120+
</dl>
121+
<emu-alg>
122+
1. UserCode().
123+
</emu-alg>
124+
</emu-clause>
125+
126+
<emu-clause id="sec-static-transitive-call" type="sdo">
127+
<h1>Static Semantics: StaticTransitiveCall</h1>
128+
<dl class="header">
129+
<dt>description</dt>
130+
<dd>Static semantics suppress user-code.</dd>
131+
</dl>
132+
<emu-alg>
133+
1. StaticDirectCall of Baz.
134+
</emu-alg>
135+
</emu-clause>

0 commit comments

Comments
 (0)
Please sign in to comment.