-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsuppressedPoints.js
97 lines (84 loc) · 2.55 KB
/
suppressedPoints.js
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
/* -*- Mode: Javascript; tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */
"use strict";
var functionBodies;
function suppressAllPoints(id)
{
var body = null;
for (var xbody of functionBodies) {
if (sameBlockId(xbody.BlockId, id)) {
assert(!body);
body = xbody;
}
}
assert(body);
if (!("PEdge" in body))
return;
for (var edge of body.PEdge) {
body.suppressed[edge.Index[0]] = true;
if (edge.Kind == "Loop")
suppressAllPoints(edge.BlockId);
}
}
function isMatchingDestructor(constructor, edge)
{
if (edge.Kind != "Call")
return false;
var callee = edge.Exp[0];
if (callee.Kind != "Var")
return false;
var variable = callee.Variable;
assert(variable.Kind == "Func");
if (!/::~/.test(variable.Name[0]))
return false;
var constructExp = constructor.PEdgeCallInstance.Exp;
assert(constructExp.Kind == "Var");
var destructExp = edge.PEdgeCallInstance.Exp;
if (destructExp.Kind != "Var")
return false;
return sameVariable(constructExp.Variable, destructExp.Variable);
}
// Compute the points within a function body where GC is suppressed.
function computeSuppressedPoints(body)
{
var successors = [];
if (!("PEdge" in body))
return;
for (var edge of body.PEdge) {
var source = edge.Index[0];
if (!(source in successors))
successors[source] = [];
successors[source].push(edge);
}
for (var edge of body.PEdge) {
if (edge.Kind != "Call")
continue;
var callee = edge.Exp[0];
if (callee.Kind != "Var")
continue;
var variable = callee.Variable;
assert(variable.Kind == "Func");
if (!isSuppressConstructor(variable.Name[0]))
continue;
if (edge.PEdgeCallInstance.Exp.Kind != "Var")
continue;
var seen = [];
var worklist = [edge.Index[1]];
while (worklist.length) {
var point = worklist.pop();
if (point in seen)
continue;
seen[point] = true;
body.suppressed[point] = true;
if (!(point in successors))
continue;
for (var nedge of successors[point]) {
if (isMatchingDestructor(edge, nedge))
continue;
if (nedge.Kind == "Loop")
suppressAllPoints(nedge.BlockId);
worklist.push(nedge.Index[1]);
}
}
}
return [];
}