### Topic 2b Basic Back-End Optimization

### Instruction Selection Instruction scheduling Register allocation

### **ABET Outcome**

- Ability to apply knowledge of basic code generation techniques, e.g. Instruction scheduling, register allocation, to solve code generation problems.
- # An ability to identify, formulate and solve loops scheduling problems using software pipelining techniques
- H Ability to analyze the basic algorithms on the above techniques and conduct experiments to show their effectiveness.
- H Ability to use a modern compiler development platform and tools for the practice of above.
- **#** A Knowledge on contemporary issues on this topic.



## (1) K. D. Cooper & L. Torczon, Engineering a Compiler, Chapter 12 (2) Dragon Book, Chapter 10.1 ~ 10.4



### A Short Tour on Data Dependence

### **Basic Concept and Motivation**

B Data dependence between 2 accesses

△The same memory location

△Exist an execution path between them

At least one of them is a write

**#**Three types of data dependencies

**H**Dependence graphs

**#**Things are not simple when dealing with loops

### **Data Dependencies**

∺There is a data dependence between statements S<sub>i</sub> and S<sub>i</sub> if and only if

- Both statements access the same memory location and at least one of the statements writes into it, and
- C There is a feasible run-time execution path from  $S_i$  to  $S_j$

**Types of Data Dependencies**  
**# Flow (true) Dependencies** - write/read (
$$\delta$$
)  
**x** := 4;  
**y** := **x** + 1;  
**# Output Dependencies** - write/write ( $\delta^{\circ}$ )  
**x** := 4;  
**x** := **y** + 1;  
**# Anti-dependencies** - read/write ( $\delta^{-1}$ )  
**y** := **x** + 1;  
**x** := 4;  
**o**  $\delta^{-1}$ 

6

### **An Example of Data Dependencies**





#### Forms a data dependence graph between statements onodes = statements edges = dependence relation (type label)

### **Reordering Transformations using DDG**

Given a correct data dependence graph, any order-based optimization that does not change the dependences of a program is guaranteed not to change the results of the program.

### **Reordering Transformations**

 A reordering transformation is any program transformation that merely changes the order of execution of the code, without adding or deleting any executions of any statements.
 A reordering transformation preserves a dependence if it preserves the relative execution order of the source and sink of that dependence.

### **Reordering Transformations (Con't)**

- **#** Instruction Scheduling
- **#** Loop restructuring
- **Exploiting** Parallelism
  - Analyze array references to determine whether two iterations access the same memory location. Iterations I1 and I2 can be safely executed in parallel if there is no data dependency between them.

### **Data Dependence Graph**



 $S_x \delta S_y \Rightarrow$  flow dependence

### **Data Dependence Graph**



# Should we consider input dependence?



Is the reading of the same X important?

Well, it may be! (if we intend to group the 2 reads together for cache optimization!)

### **Applications of Data Dependence Graph**

- register allocation
- instruction scheduling
- loop scheduling
- vectorization
- parallelization
- memory hierarchy optimization

### **Data Dependence in Loops**

Problem: How to extend the concept to loops?

(s1) do 
$$i = 1,5$$
  
(s2)  $x := a + 1;$   
(s3)  $a := x - 2;$   
(s4) end do

s2 δ<sup>-1</sup> s3, s2 δ s3

s3  $\delta$  s2 (next iteration)



#### **Motivation**

Modern processors can overlap the execution of multiple *independent* instructions through pipelining and multiple functional units. Instruction scheduling can improve the performance of a program by placing independent target instructions in parallel or adjacent positions.



- Assume all instructions are essential, i.e., we have finished optimizing the IR.
- Instruction scheduling attempts to reorder the codes for maximum instruction-level parallelism (ILP).
- It is one of the instruction-level optimizations
- Instruction scheduling (IS) in general is NPcomplete, so heuristics must be used.

### Instruction scheduling: A Simple Example

time



Since all three instructions are independent, we can execute them in parallel, assuming adequate hardware processing resources.

### **Hardware Parallelism**

Three forms of parallelism are found in modern hardware: •pipelining •superscalar processing •VLIW •multiprocessing Of these, the first three forms are commonly exploited by today's compilers' instruction scheduling phase.

### Pipelining & Superscalar Processing

#### Pipelining

Decompose an instruction's execution into a sequence of stages, so that multiple instruction executions can be overlapped. It has the same principle as the assembly line.

#### **Superscalar Processing**

Multiple instructions proceed simultaneously assisted by hardware dynamic scheduling mechanism. This is accomplished by adding more hardware, for parallel execution of stages and for dispatching instructions to them.

### **A Classic Five-Stage Pipeline**

IF RF EX ME WB

time

- instruction fetch
- decode and register fetch
- execute on ALU
- memory access
- write back to register file



### **Parallelism in a pipeline**

#### Example:

| il: | add | rl, | rl,   | r2 | Assume:                           |
|-----|-----|-----|-------|----|-----------------------------------|
| i2: | add | r3  | r3,   | rl | Register instruction 1 cycle      |
| i3: | lw  | r4, | 0(r1) |    | Memory instruction <b>3</b> cycle |
| i4: | add | r5  | r3,   | r4 | Wembry instruction 5 cycle        |

Consider two possible instruction schedules (permutations):



### **Superscalar Illustration**



Multiple instructions in the same pipeline stage at the same time

### **Parallelism Constraints**

#### **Data-dependence constraints**

If instruction A computes a value that is read by instruction B, then B can't execute before A is completed.

#### **Resource hazards**

Finiteness of hardware function units means limited parallelism.

### **Scheduling Complications**

#### Hardware Resources

- finite set of FUs with instruction type, and width, and latency constraints
- cache hierarchy also has many constraints
- Data Dependences
- can't consume a result before it is produced
- ambiguous dependences create many challenges
- Control Dependences
  - impractical to schedule for all possible paths
  - choosing an "expected" path may be difficult
  - recovery costs can be non-trivial if you are wrong

### Legality Constraint for Instruction Scheduling

Question: when must we preserve the order of two instructions, *i* and *j* ?

Answer: when there is a *dependence* from *i* to *j*.

### **Construct DDG with Weights**

Construct a DDG by assigning weights to nodes and edges in the DDG to model the pipeline as follows:

- Each DDG node is labeled a resource-reservation table whose value is the resource-reservation table associated with the operation type of this node.
- Each edge *e* from node *j* to node *k* is labeled with a weight (latency or delay) d<sub>e</sub> indicting that the destination node *j* must be issued no earlier than d<sub>e</sub> cycles after the source node *k* is issued.

### **Example of a Weighted Data Dependence Graph**

Assume:

Register instruction 1 cycle Memory instruction 3 cycle



### **Legal Schedules for Pipeline**

Consider a basic block with m instructions,  $i_1, ..., i_m$ .

A legal sequence, **S**, for the basic block on a pipeline consists of:

A permutation **f** on **1**...**m** such that **f**(**j**) identifies the new position of instruction **j** in the basic block. For each DDG edge form **j** to **k**, the schedule must satisfy **f**(**j**) <= **f**(**k**)

### Legal Schedules for Pipeline (Con't)

#### **Instruction start-time**

An instruction start-time satisfies the following conditions:

- Start-time (j) >= 0 for each instruction j
- No two instructions have the same start-time value
- For each DDG edge from *j* to *k*,
   start-time(k) >= completion-time (j)

where

completion-time (j) = start-time (j)

+ (weight between *j* and *k*)

### Legal Schedules for Pipeline (Con't)

We also define: make\_span(S) = completion time of schedule S = MAX ({ completion-time (j)})  $1 \le j \le m$ 

### Example of Legal Schedules



#### Instruction Scheduling (Simplified)

#### **Problem Statement:**

- Given an acyclic weighted data dependence graph G with:
  - Directed edges: precedence
  - Undirected edges: resource constraints



Determine a schedule S such that the length of the schedule is minimized!

#### Simplify Resource Constraints

- Assume a machine M with **n** functional units or a "clean" pipeline with **k** stages.
- What is the complexity of a optimal scheduling algorithm under such constraints ?
- Scheduling of M is still hard!
  - n = 2 : exists a polynomial time algorithm [CoffmanGraham]
  - **n** = 3 : remain open, conjecture: NP- hard

## **General Approaches of Instruction Scheduling**

% List Scheduling % Trace scheduling % Software pipelining % .....

## **Trace Scheduling**

A technique for scheduling instructions across basic blocks.

#### **%The Basic Idea of trace scheduling**

□ Uses information about actual program behaviors to select regions for scheduling.

## **Software Pipelining**

A technique for scheduling instructions across loop iterations.

#### **K** The Basic Idea of software pipelining

Rewrite the loop as a repeating pattern that overlaps instructions from different iterations.

## **List Scheduling**

A most common technique for scheduling instructions within a basic block.

#### **%The Basic Idea of list scheduling**

□ All instructions are sorted according to some priority function.

Also Maintain a list of instructions that are ready to execute

(i.e. data dependence constraints are satisfied)

□ Moving cycle-by-cycle through the schedule template:

 choose instructions from the list & schedule them (provided that machine resources are available)

• update the list for the next cycle

41

### List Scheduling (Con't)

- □ Uses a greedy heuristic approach
- Has forward and backward forms
- Is the basis for most algorithms that perform scheduling over regions larger than a single block.

#### Heuristic Solution: Greedy List Scheduling Algorithm

- 1. Build a priority list **L** of the instructions in nondecreasing order of some rank function.
- 2. For each instruction j, initialize
  pred-count[j] := #predecessors of j in DDG
- 3. Ready-instructions := {j | pred-count[j] = 0 }
- 4. While (ready-instructions is non-empty) do j := first ready instruction according to the order in priority list, L; Output j as the next instruction in the schedule;

Consider resource constraints beyond a single clean pipeline

### Heuristic Solution: Greedy List Scheduling Algorithm

Ready-instructions := ready-instructions- { j };
for each successor k of j in the DDG do
 pred-count[k] := pred-count[k] - 1;
 if (pred-count[k] = 0 ) then
 ready-instructions := ready-instruction
 + { k };
 end if

#### end for end while

#### Special Performance Bounds

For a single two stage pipeline,

*m* = 1 and *k* = 2 ==>

(here m is the number of pipelines, and k is the number of pipeline stages per pipeline)

makespan (greedy)/makespan(OPT) <= 1.5

#### **Properties of List Scheduling**

- Complexity: O(n^2) --- where n is the number of nodes in the DDG
- In practice, it is dominated by DDG building which itself is also O(n<sup>2</sup>)
- The result is within a factor of two from the optimal for pipelined machines (Lawler87)

## A Heuristic Rank Function Based on Critical paths

1. Compute EST (Earliest Starting Times) for each node in the augmented DDG as follows:

EST[START] = 0

EST{y] =
MAX ({EST[x] + node-weight (x) + edge-weight (x,y) |
 there exists an edge from x to y })

- 2. Set CPL := EST[END], the critical path length of the augmented DDG.
- 3. Similarly, compute LST (Latest Starting Time of All nodes);
- 4. Set rank (*i*) = LST [*i*] EST [*i*], for each instruction *i*

(all instructions on a critical path will have zero rank)

**NOTE: there are other heurestics** 

## **Example of Rank Computation**





**Instruction Scheduling for a Basic Block** 

- 1. Build the data dependence graph (DDG) for the basic block
  - Node = target instruction
  - Edge = data dependence (flow/anti/output)
- 2. Assign weights to nodes and edges in the DDG so as to model target processor e.g., for a two-stage pipeline
  - Node weight = 1, for all nodes
  - Edge weight = 1 for edges with load/store instruction as source node; edge weight = 0 for all other edges



- 3. A legal schedule for a weighted DDG must obey all ordering and timing constraints of the weighted DDG
- 4. **Goal**: find a legal schedule with minimum completion time

Con'd

#### **Other Heurestics for Ranking**

% Number of successors ?
% Number of total decendents ?
% Latency ?
% Last use of a variable ?
% Others ?

Note: these heuristics help break ties, but none dominates the others.

### Hazards Preventing Pipelining

- Structural hazards
- Data dependent hazard
- Control hazard

# Local vs. Global Scheduling

- 1. Straight-line code (basic block) Local scheduling
- 2. Acyclic control flow Global scheduling
  - Trace scheduling
  - Hyperblock/superblock scheduling
  - IGLS (integrated Global and Local Scheduling)
- 3. *Loops* a solution for this case is loop unrolling+scheduling, another solution is

*software pipelining or modulo scheduling* i.e., to rewrite the loop as a repeating pattern that overlaps instructions from different iterations.





