summaryrefslogtreecommitdiffstats
path: root/js/src/jit-test/tests/cacheir/bug1494537-plain.js
blob: e17cd3bd1da8ca0fff3962ef0078f3d788720143 (plain)
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();