References:
- A Unified Approach to Global Program Optimization by Gary A. Kildall, POPL 1973. Gary's Ph.D. thesis at U.W. He is also known for the CP/M operating system that was an early competitor with MS-DOS (Microsoft).
Analyses and Optimizations:
- constant propagation and folding
- common subexpression elimination
- live expression analysis
A: a = 1
B: c = 0
for i = 1...10 {
C: b = 2
D: d = a + b
E: e = b + c
F: c = 4
}
Hasse Diagram, Finite Descending Chains
{(a,...), (b,...), (c,...), (d,...), (e, ...) } (5 pairs)
|
...
|
{(a,1),(b,2)}
|
{(a,1)}
{}
Peel one iteration:
A: a = 1
B: c = 0
i = 1
C: b = 2
D: d = 3
E: e = 2
F: c = 4
for i = 2...10 {
E: e = b + c
}
Peel two iterations:
A: a = 1
B: c = 0
i = 1
C: b = 2
D: d = 3
E: e = 2
F: c = 4
i = 2
E: e = 6
{} a = 3 {(a,3)} if (eq? (read) 0) { {(a,3)} b = 5 {(a,3),(b,5)} } else { {(a,3)} b = 10 {(a,3),(b,10)} } {(a,3),(b,5)} /\ {(a,3),(b,10)} = {(a,3)} c = b {(a,3)}
Consider every possible execution path, propagating constants along each path. Then the analysis information P_N for a node N is the intersection of the information from all paths to that node.
Let P^i_N be the information for the ith path from the entry node up to but not including node N.
P^3(C) = {(a,1),(c,0)} P^7(C) = {(a,1),(b,2),(c,4),(d,3),(e,2)}
A {}
A->B {(a,1)}
A->B->C {(a,1),(c,0)}
A->B->C->D *{(a,1),(b,2),(c,0)}
A->B->C->D->E {(a,1),(b,2),(c,0),(d,3)}
A->B->C->D->E->F {(a,1),(b,2),(c,0),(d,3),(e,2)}
A->B->C->D->E->F->C {(a,1),(b,2),(c,4),(d,3),(e,2)}
A->B->C->D->E->F->C->D *{(a,1),(b,2),(c,4),(d,3),(e,2)}
A->B->C->D->E->F->C->D->E {(a,1),(b,2),(c,4),(d,3),(e,2)}
A->B->C->D->E->F->C->D->E->F {(a,1),(b,2),(c,4),(d,3),(e,6)}
...
...
->D *{...}
Then P_N = Inter_i P^i_N.
P_D = {(a,1),(b,2)}
This approach is known as meet-over-all-paths (MOP).
This approach is not practical because the number of paths are often infinite.
Instead of waiting to the end to take the intersection, we could do it eagerly, intersecting along the way with prior results.
A {}
A->B {(a,1)}
A->B->C {(a,1),(c,0)}
A->B->C->D {(a,1),(b,2),(c,0)}
A->B->C->D->E {(a,1),(b,2),(c,0),(d,3)}
A->B->C->D->E->F {(a,1),(b,2),(c,0),(d,3),(e,2)}
A->B->C->D->E->F->C {(a,1)}
= {(a,1),(b,2),(c,4),(d,3),(e,2)} /\ {(a,1),(c,0)}
A->B->C->D->E->F->C->D {(a,1),(b,2)}
= {(a,1),(b,2)} /\ {(a,1),(b,2),(c,0)}
...C->D->E->F->C->D->E {(a,1),(b,2),(d,3))}
= {(a,1),(b,2),(d,3)} /\ {(a,1),(b,2),(c,0),(d,3)}
...D->E->F->C->D->E->F {(a,1),(b,2),(d,3))}
= {(a,1),(b,2),(c,4)(d,3))} /\ {(a,1),(b,2),(c,0),(d,3),(e,2)}
...E->F->C->D->E->F->C {(a,1)}
= {(a,1),(b,2),(c,4)(d,3))} /\ {(a,1)}
...E->F->C->D->E->F->C->D {(a,1),(b,2)}
= {(a,1),(b,2)} /\ {(a,1),(b,2)}
We got the same answer for P_C, so going around the loop again isn't going to change the results.
Informal Global Analysis Algorithm For Constant Propagation & Folding
- For entry node E, set P_E = {}
- Process node E, send info to the successor nodes.
- Intersect the incoming pools for each node with the prior pool for the node, if there is one. If not, take the incoming pool.
- If the result for a node changed, then send updated info to the successor nodes.
-
An "transfer" function f that maps a node and an input pool to an output pool. Recall the example program:
A: a = 1 B: c = 0 for i = 1...10 { C: b = 2 D: d = a + b E: e = b + c F: c = 4 }
The transfer function for constant propagation for this program is:
f(A,S) = {(a,1)} U (S - a) f(B,S) = {(c,0)} U (S - c) f(C,S) = {(b,2)} U (S - b) f(D,S) = if (a,n1) in S and (b,n2) in S then {(d,n1+n2)} U (S - d) else S - d
-
Initial value for the entry node.
-
The information must form a meet-semilattice.
-
The meet operation /\ is commutative and associative.
-
The semilattice has a "bottom" element (e.g. empty set). (It may also have a "top" element, in which case it is a lattice.)
-
The elements are partially ordered in a way that jives with the meet operation.
A <= B iff A /\ B = A
A subset or equal B iff A /\ B = A
-
-
The transfer functions must be monotonic.
X <= Y --> f(N,X) <= f(N,Y)
which implies that taking the meet "early" gives you correct but possibly less precise results.
f(N, X /\ Y) <= f(N, X) /\ f(N, Y)
If the transfer function is distributive
f(N, X /\ Y) = f(N, X) /\ (N, Y)
then applying the meet "early" doesn't lose precision.
-
For termination, the lattice needs to have finite descending chains.
Example:
T: r = a + b
U: s = r + x
V: t = (a + b) + x
W: return t
==>
r = a + b
s = r + x
return s
Need to keep track of which expressions are equivalent, i.e., we need to track equivalence classes of expressions.
P_T = { a | b | x }
P_U = { a | b | x | a+b, r }
P_V = { a | b | x | a+b, r | r+x, s }
P_W = { a | b | x | a+b, r | r+x, s, (a + b) + x, t }
The transfer function for common subexpression elimination:
For each subexpression e in the node N:
- If e is already in an equiv. class, then it is redundant.
- If not, create a new class in P with singleton e.
- If node N is an assignment d = e, remove from P all expressions containing d. For each expression e' in P that contains e, create e'' by replacing e with d, and place e'' in the class of e'.
The meet operation P1 /\ P2 forms a partition over the expressions common to both P1 and P2, with
(P1 /\ P2)(e) = P1(e) /\ P2(e)
Generalizes liveness analysis from variables to expressions. An expression is live if it may be used in the future.
This analysis works backwards, from the end of the program to the beginning. Flip edges to have it work forwards.
The transfer function is:
If N is x := e,
P <- (P - {e'| e' in P, x in e' }) U {e' | e' in e}
The meet operation is set union.
Initial pools are {}.
Worklist Algorithm:
Place entry node in the worklist.
While the worklist is not empty:
Pop a node from the worklist
Apply the transfer function for that node.
Place its successors who changed in the worklist.
To speed up the ordering:
-
Form the condensation graph by collapsing strongly-connected components.
-
Process the condensation graph in topological order. For each component, run the worklist algorithm to a fixed point.