Download presentation
Presentation is loading. Please wait.
Published byTamara Critchley Modified over 9 years ago
1
Code Generation Steve Johnson
2
May 23, 2005Copyright (c) Stephen C. Johnson 2005 2 The Problem Given an expression tree and a machine architecture, generate a set of instructions that evaluate the tree –Initially, consider only trees (no common subexpressions) –Interested in the quality of the program –Interested in the running time of the algorithm
3
May 23, 2005Copyright (c) Stephen C. Johnson 2005 3 The Solution Over a large class of machine architectures, we can generate optimal programs in linear time –A very practical algorithm –But different from the way most compilers work today –And the technique, dynamic programming, is powerful and interesting Work done with Al Aho, published in JACM
4
May 23, 2005Copyright (c) Stephen C. Johnson 2005 4 What is an Expression Tree? Nodes represent –Operators (including assignment) –Operands (memory, registers, constants) No flow of control operations A = + BC
5
May 23, 2005Copyright (c) Stephen C. Johnson 2005 5 Representing Operands In fact, we want the tree to represent where the operands are found MEM (A) = + MEM (B) MEM (C)
6
May 23, 2005Copyright (c) Stephen C. Johnson 2005 6 Possible Programs load B,r1; load C,r2; add r1,r2,r1 store r1,A or load B,r1 add C,r1 store r1,A or add B,C,A
7
May 23, 2005Copyright (c) Stephen C. Johnson 2005 7 (Assembler Notation) Data always moves left to right load B,r1 r1 = MEM(B) add r1,r2,r3 r3 = r1 + r2 store r1,A MEM(A) = r1
8
May 23, 2005Copyright (c) Stephen C. Johnson 2005 8 Which is Better? Not all sequences legal on all machines Longer sequences may be faster Situation gets more complex when –Complicated expressions run out of registers –Some operations (e.g., call) take a lot of registers –Instructions have complicated addressing modes
9
May 23, 2005Copyright (c) Stephen C. Johnson 2005 9 Example Code A = 5*B + asin(C/2 + sin(D)) might generate (machine with 2 registers) load B,r1 OR load D,r1 mul r1,#5,r1 call sin store r1,T1 load C,r2 load C,r1 div r2,#2,r2 div r1,#2,r1 add r2,r1,r1 store r1,T2 call asin load D,r1 load B,r2 call sin mul r2,#5,r2 load T2,r2 add r1,r2,r1 add r2,r1 store r1,A call asin load T1,r2 add r2,r1,r1 store r1,A
10
May 23, 2005Copyright (c) Stephen C. Johnson 2005 10 What is an Instruction An instruction is a tree transformation MEM (A) REG (r1) load A,r1 REG (r1) MEM (A) store r1,A * REG (r2) load (r1),r2 REG (r1)
11
May 23, 2005Copyright (c) Stephen C. Johnson 2005 11 These can be Quite Complicated * + REG (r1) REG (r2) INT 2 << REG (r3) load r1(r2),r3
12
May 23, 2005Copyright (c) Stephen C. Johnson 2005 12 Types and Resources Expression Trees (and instructions) typically have types associated with them –We’ll ignore this –Doesn’t introduce any real problems Instructions often need resources to work –For example, a temporary register or a temporary storage location –Will be discussed later
13
May 23, 2005Copyright (c) Stephen C. Johnson 2005 13 Programs A program is a sequence of instructions A program computes an expression tree if it transforms the tree according to the desired goal –Compute the tree into a register –Compute the tree into memory –Compute the tree for its side-effects Condition codes Assignments
14
May 23, 2005Copyright (c) Stephen C. Johnson 2005 14 Example Goal: compute for side effects MEM (A) = + MEM (B) MEM (C) load B,r1 load C,r2 add r1,r2,r1 store A,r1
15
May 23, 2005Copyright (c) Stephen C. Johnson 2005 15 Example (cont.) MEM (A) = + REG (r1) MEM (C) load C,r2 MEM (A) = + REG (r1) REG (r2)
16
May 23, 2005Copyright (c) Stephen C. Johnson 2005 16 Example (cont.) MEM (A) = + REG (r1) REG (r2) add r1,r2,r1 MEM (A) = REG (r1)
17
May 23, 2005Copyright (c) Stephen C. Johnson 2005 17 Example (concl.) store r1,A MEM (A) = REG (r1) (Side effect done)
18
May 23, 2005Copyright (c) Stephen C. Johnson 2005 18 Typical Code Generation Some variables are assigned to registers, leaving a certain number of scratch registers An expression tree is walked, producing instructions (greedy algorithm...). An infinite number of temporary registers is assumed
19
May 23, 2005Copyright (c) Stephen C. Johnson 2005 19 Typical Code Generation (cont.) A register allocation phase is run –Assign temporary registers to scratch register Often by doing graph coloring... –If you run out of scratch registers, spill Select a register Store it into a temporary When it is needed again, reload it
20
May 23, 2005Copyright (c) Stephen C. Johnson 2005 20 Practical Observation Many (most?) code generation bugs happen in this spill code –Choose a register that is really needed –Very hard to test... Create test cases that just barely fit or just barely don’t fit to test edge cases... Can be quite inefficient –“thrashing” of scratch registers Code may not be optimal
21
May 23, 2005Copyright (c) Stephen C. Johnson 2005 21 Complexity Results Simple machine with 2-address instructions: – r1 op r2 => r1 Cost = number of instructions Allow common subexpressions only of the form A op B, where A and B are leaf nodes Generating optimal code is N-P complete Even if there are an infinite number of registers! Implies exponential time for a tree with n nodes
22
May 23, 2005Copyright (c) Stephen C. Johnson 2005 22 Complexity Results (cont.) Simple 3-address machine –r1 op r2 => r3 Cost = number of instructions Allow arbitrary common subexpressions Infinite number of registers Can get optimal code in linear time –Topological sort –Each node in a different register
23
May 23, 2005Copyright (c) Stephen C. Johnson 2005 23 Complexity Results (cont.) In the 3-address model, finding optimal code that uses the minimal number of registers is N-P complete But that’s not what we are faced with in practice –We have a certain number of registers –We need to use them intelligently
24
May 23, 2005Copyright (c) Stephen C. Johnson 2005 24 Complexity Results (concl.) For many practical machine architectures (including 2-address machines), we can generate optimal code in linear time when there are no common subexpressions (tree) Can be extended to an algorithm exponential in the amount of sharing The optimal instruction sequence is not generated by a simple tree walk
25
May 23, 2005Copyright (c) Stephen C. Johnson 2005 25 Machine Model Restrictions Resources (temporary registers) must be interchangeable. We will assume that we have N of them Every instruction has a (positive) cost The cost of a program is the sum of the costs of the instructions No other constraints on the instruction shape or format (!)
26
May 23, 2005Copyright (c) Stephen C. Johnson 2005 26 Study Optimal Programs Suppose we have an expression tree T that we wish to compute into a register –For the moment, we assume T can be computed with no stores –We assume we have N scratch registers Suppose the root node of T is + Then, in an optimal program, the last instruction must have a + at the root of the tree that it transforms –We make a list of these instructions –Each has some preconditions for it to be legal
27
May 23, 2005Copyright (c) Stephen C. Johnson 2005 27 Preconditions: Example Suppose the last instruction was add r1,r2,r1 Suppose the tree T looks like Then our optimal program must compute T1 into r1 and T2 into r2 + T1T2
28
May 23, 2005Copyright (c) Stephen C. Johnson 2005 28 Precondition Resources If our optimal program ends in this add instruction, then we can assume that it contains two subprograms that compute T1 and T2 into r1 and r2, respectively
29
May 23, 2005Copyright (c) Stephen C. Johnson 2005 29 Precondition Resources (cont.) Look at the first instruction –If it computes part of T1, then (since no stores) at least one register is always in use computing T1. –So T2 must be computed using at most N-1 registers –Alternatively, if the first instruction computes part of T2, T1 must be computed using at most N-1 registers.
30
May 23, 2005Copyright (c) Stephen C. Johnson 2005 30 Reordering Lemma Let P be an optimal program without stores that computes T. Suppose it ends in an instruction X that has k preconditions. Then we can reorder the instructions in P so it looks like P1 P2 P3... Pk X where the Pi compute the preconditions of X in some order. Moreover, P2 uses at most N-1 registers, P3 uses at most N-2 registers, etc., and each Pi computes its precondition optimally using that number of registers
31
May 23, 2005Copyright (c) Stephen C. Johnson 2005 31 Cost Computation Define C(T,n) to be the cost of the optimal program computing T using at most n registers. Suppose X is an instruction matching the root of T with k preconditions, corresponding to subtrees T1 through Tk. Then C(T,n) <= C(X)+C(T1,p1)+...+C(Tk,pk) where C(X) is the cost of instruction X, and p1,...,pk are a permutation of the numbers n, n-1,... n-k+1 In fact, C(T,n) equals the minimum, over all instructions X and permutations, of this sum
32
May 23, 2005Copyright (c) Stephen C. Johnson 2005 32 Sketch of Proof By the reordering lemma, we can write any optimal program as a sequence of subprograms computing the preconditions in order, with decreasing numbers of scratch registers, followed by some instruction X. If any subprograms is not optimal, we can make the program shorter, contradicting optimality of the original program. Thus the optimal cost equals one of the sums (for some X and permutation)
33
May 23, 2005Copyright (c) Stephen C. Johnson 2005 33 How About Stores (spills)? We will now let C(T,n) represent the cost of computing T with n registers if stores (spills) are allowed. More notation: if T is a tree and S a subtree, T/S will represent T with S removed and replaced by a MEM node.
34
May 23, 2005Copyright (c) Stephen C. Johnson 2005 34 Another Rearrangement Lemma Suppose P is an optimal program computing a tree T, and suppose a subtree S is stored into a temporary location in this optimal program. Then P can be rewritten in the form P1 P2 where P1 computes S into memory and P2 computes T/S.
35
May 23, 2005Copyright (c) Stephen C. Johnson 2005 35 Consequences P1 can use all N registers. After P1 runs, all registers are free again. Let C(S,0) be the cost of computing S into a temporary ( MEM ) location. Then C(T,n) <= C(S,0) + C(T/S,n) One way to compute S into memory is to compute it into a register and store it (there may be other, cheaper, ways). Thus C(S,0) <= C(S,N) + Cstore
36
May 23, 2005Copyright (c) Stephen C. Johnson 2005 36 Optimal Algorithm 1.Recursively compute C(S,n) and C(S,0) for all subtrees of T, starting bottom up, and all n <= N. 2.Enumerate all instructions matching the root of T. Those that leave a result in memory contribute to C(T,0). Those leaving a result in a register contribute to C(T,n). Apply the cost formula for each permutation of the preconditions of the instruction, and remember the minimal costs. 3.Update C(T,0) using C(T,0) = min(C(T,0),C(T,N)+Cstore) 4.The result gives the minimal cost to compute the tree using n registers, or to compute it into memory
37
May 23, 2005Copyright (c) Stephen C. Johnson 2005 37 Dynamic Programming This bottom-up technique is called dynamic programming It has a fixed cost per tree node because: –There are a finite (usually small) number of instructions that match the root of each tree –The number of permutations for each instruction is fixed (and typically small) –The number of scratch registers N is fixed So the optimal cost can be determined in time linear in the size of the tree
38
May 23, 2005Copyright (c) Stephen C. Johnson 2005 38 Unravelling Going from the minimal cost back to the instructions can be done several ways: –Can remember the instruction and permutation that gives the minimal value for each node –At each node, recompute the desired minimal value until you find an instruction and permutation that attain it
39
May 23, 2005Copyright (c) Stephen C. Johnson 2005 39 Top-Down Memo Algorithm Instead of computing bottom up, you can compute top down (in a lazy manner) and remember the results. This might be considerably faster for some architectures
40
May 23, 2005Copyright (c) Stephen C. Johnson 2005 40 No Spills! Note that we do not have to have spill code in this algorithm. The subtrees that are computed and stored “fall out” of the algorithm. They are computed ahead of the main computation, when all registers are available. The resulting instruction stream is not typically a tree walk of the input.
41
May 23, 2005Copyright (c) Stephen C. Johnson 2005 41 Reality Check Major assumptions –Cost is the sum of costs of instructions Assumes single ALU, no overlapping Many machines now have multiple ALUs, overlapping operations – All registers identical True of most RISC machines Not true of X86 architectures But memory operations getting more expensive –Optimality for spills is important
42
May 23, 2005Copyright (c) Stephen C. Johnson 2005 42 Other Issues Register allocation across multiple statements, flow control, etc. –Can make a big difference in performance –Can use this algorithms to evaluate possible allocations Cost of losing a scratch register to hold a variable
43
May 23, 2005Copyright (c) Stephen C. Johnson 2005 43 Common Subexpressions A subtree S of T is used more than once (T is now not a tree, but a DAG) Say there are 2 uses of S. Then there are 4 strategies –Compute S and store it –Compute one use and save the result until the second use (2 ways, depending on which use is first –Ignore the sharing, and recompute S
44
May 23, 2005Copyright (c) Stephen C. Johnson 2005 44 Cost Computations Ignoring the sharing is easy Computing and storing is easy Ordering the two uses implies an ordering of preconditions in some higher-level instruction selection –And the number of free registers is affected, too Do the problem twice, once for each order
45
May 23, 2005Copyright (c) Stephen C. Johnson 2005 45 Summary Register spills are evil –Complicated, error-prone, hard to test If something is to be spilled, compute it ahead of time with all registers free The optimal spill points fall out of the dynamic programming algorithm
Similar presentations
© 2025 SlidePlayer.com. Inc.
All rights reserved.