summaryrefslogtreecommitdiffstats
path: root/js/src/jit-test/tests/gc/weak-marking-varying.js
blob: 76dc39dfb2517353346069b314455360e16cd338 (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
// |jit-test| allow-unhandlable-oom

// Crash test. Try lots of different combinations of mark colors and
// sequencing, and rely on the in-code asserts to detect problems.

// This test requires precise control over GC timing. The SM(cgc) job will fail
// without this, because startgc() asserts if an incremental GC is already in
// progress.
gczeal(0);

function global(where) {
  if (where == 'same-realm')
    return globalThis;
  if (where == 'same-compartment')
    return newGlobal();
  if (where == 'same-zone')
    newGlobal({sameZoneAs: {}});
  return newGlobal({newCompartment: true});
}

function varying(mapColor, keyColor, delegateColor, order, where) {
  const vals = {};
  const g = global(where);

  vals.m = new WeakMap();
  vals.key = g.eval("Object.create(null)");
  vals.val = Object.create(null);
  vals.m.set(vals.key, vals.val);

  addMarkObservers([vals.m, vals.key]);
  g.delegate = vals.key;
  g.eval('addMarkObservers([delegate])');
  g.delegate = null;
  addMarkObservers([vals.val]);

  for (const action of order) {
    if (action == 'key' && ['black', 'gray'].includes(keyColor)) {
      enqueueMark(`set-color-${keyColor}`);
      enqueueMark(vals.key);
      enqueueMark("unset-color");
      enqueueMark("yield");
    } else if (action == 'map' && ['black', 'gray'].includes(mapColor)) {
      enqueueMark(`set-color-${mapColor}`);
      enqueueMark(vals.m);
      enqueueMark("drain");
      enqueueMark("unset-color");
      enqueueMark("yield");
    } else if (action == 'delegate' && ['black', 'gray'].includes(delegateColor)) {
      g.delegate = vals.key;
      g.eval(`enqueueMark("set-color-${delegateColor}")`);
      g.eval('enqueueMark(delegate)');
      g.eval('enqueueMark("unset-color")');
      g.eval('enqueueMark("yield")');
      g.delegate = null;
    }
  }

  vals.m = vals.key = vals.val = null;

  if (delegateColor == 'uncollected')
    schedulezone({});
  startgc(100000);
  print('  ' + getMarks().join("/"));
  gcslice(100000);
  print('  ' + getMarks().join("/"));
  finishgc();
  print('  ' + getMarks().join("/"));

  clearMarkQueue();
  clearMarkObservers();
}

if (this.enqueueMark) {
  for (const mapColor of ['gray', 'black']) {
    for (const keyColor of ['gray', 'black', 'unmarked']) {
      for (const delegateColor of ['gray', 'black', 'unmarked', 'uncollected']) {
        for (const order of [['map', 'key'],
                             ['key', 'map'],
                             ['map', 'delegate'],
                             ['delegate', 'map']])
        {
          for (const where of ['same-realm', 'same-compartment', 'same-zone', 'other-zone']) {
            print(`\nRunning variant map/key/delegate=${mapColor}/${keyColor}/${delegateColor}, key is ${where}, order: ${order.join(" ")}`);
            varying(mapColor, keyColor, delegateColor, order, where);
          }
        }
      }
    }
  }
}