1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
|
// Test case to cover constant atom guards.
//
// GuardSpecificAtom for short (≤32 characters) constant atoms is optimised.
function* characters(...ranges) {
for (let [start, end] of ranges) {
for (let i = start; i <= end; ++i) {
yield i;
}
}
}
const ascii = [...characters(
[0x41, 0x5A], // A..Z
[0x61, 0x7A], // a..z
[0x30, 0x39], // 0..9
)];
const latin1 = [...characters(
[0xC0, 0xFF], // À..ÿ
)];
const twoByte = [...characters(
[0x100, 0x17E], // Ā..ž
)];
function toRope(s) {
try {
return newRope(s[0], s.substring(1));
} catch {}
// newRope can fail when |s| fits into an inline string. In that case simply
// return the input.
return s;
}
function atomize(s) {
return Object.keys({[s]: 0})[0];
}
for (let i = 1; i <= 32; ++i) {
let strings = [ascii, latin1, twoByte].flatMap(codePoints => [
// Same string as the input.
String.fromCodePoint(...codePoints.slice(0, i)),
// Same length as the input, but a different string.
String.fromCodePoint(...codePoints.slice(1, i + 1)),
// Shorter string than the input.
String.fromCodePoint(...codePoints.slice(0, i - 1)),
// Longer string than the input.
String.fromCodePoint(...codePoints.slice(0, i + 1)),
]).flatMap(x => [
x,
toRope(x),
newString(x, {twoByte: true}),
atomize(x),
]);
// Must be small enough to transition to megamorphic ICs.
const stringsPerLoop = 4;
for (let codePoints of [ascii, latin1, twoByte]) {
let str = String.fromCodePoint(...codePoints.slice(0, i));
for (let i = 0; i < strings.length; i += stringsPerLoop) {
let fn = Function("strings", `
var obj = {["${str}"]: 0};
for (let i = 0; i < 250; ++i) {
let idx = i % strings.length;
let str = strings[idx];
let actual = str in obj;
let expected = str === "${str}";
if (actual !== expected) throw new Error();
}
`);
fn(strings.slice(i, stringsPerLoop));
}
}
}
|