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
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
|
// Version of bug1494537.js for plain objects.
let offsets = [213, 559, 255, 515, 30, 507, 252, 329, 487, 7];
function update_index(i, j) {
var offset = offsets[j % offsets.length];
return i + offset;
}
function compute_index(initial, count) {
for (var i = 0; i < count; i++) {
initial = update_index(initial, i);
}
return initial;
}
// This is written so that the IC added in the bug activates.
function mutate_object(obj, count, epsilon = 0) {
var index = 0;
for (var i = 0; i < count; i++) {
index = update_index(index, i);
obj[index] = i + epsilon;
}
return obj[offsets[0]+offsets[1]] === (1 + epsilon) &&
obj[10] === undefined;
}
// Monomorphizing mutate_object to ensure we get the IC chains we want
function create_variant(variant) {
var source = mutate_object.toString().replace("mutate_object", "mutate_object_"+variant);
return source;
}
function test_basic() {
eval(create_variant("basic"));
var x = {};
var count = 100;
assertEq(mutate_object_basic(x, count), true);
var end = compute_index(0, count);
assertEq(x[end], count - 1);
assertEq(x[end - 1], undefined);
}
// Ensure the IC respects frozen.
function test_frozen() {
eval(create_variant("frozen"));
var x = {};
Object.freeze(x);
var count = 100;
assertEq(mutate_object_frozen(x, count), false);
var end = compute_index(0, count);
var y = {};
assertEq(mutate_object_frozen(y, count), true);
assertEq(y[end], count - 1);
Object.freeze(y);
// After a mutated object is frozen, can't subsequently modify elements
assertEq(mutate_object_frozen(x, count, 10), false);
assertEq(y[end], count - 1);
}
// Let's make sure updates to the object happen as expected.
function test_update() {
eval(create_variant("update"));
var x = {};
var count = 100;
assertEq(mutate_object_update(x, count), true);
var end = compute_index(0, count);
assertEq(x[end], count - 1);
assertEq(x[end - 1], undefined);
var epsilon = 2;
mutate_object_update(x, 200, epsilon);
assertEq(x[end], count -1 + epsilon)
}
// Elements may be non-writable, let us not write them.
function test_nonwritable() {
eval(create_variant("nonwritable"));
var x = {};
var count = 100;
var index = compute_index(0, 10);
Object.defineProperty(x, index, {value: -10, writable: false});
mutate_object_nonwritable(x, count);
assertEq(x[index], -10);
}
// Random indices can get setters, let's make sure we honour those.
function test_setter() {
eval(create_variant("setter"));
var x = {};
var count = 100;
var index = compute_index(0, 80);
var sigil = 0;
Object.defineProperty(x, index, {set(newVal) {sigil++; }});
mutate_object_setter(x, count);
assertEq(sigil, 1);
assertEq(x[index], undefined);
}
// Ensure indexes on the prototype don't break things.
function test_proto_indices() {
eval(create_variant("proto_indices"));
var x = {};
var count = 100;
var index = compute_index(0, 80);
x.__proto__[index] = "hello";
mutate_object_proto_indices(x, count);
assertEq(x.__proto__[index], "hello");
assertEq(x[index], 79);
}
test_basic();
test_frozen();
test_update();
test_nonwritable();
test_setter();
test_proto_indices();
|