# **Register Integration: A Simple and Efficient Implementation of Squash Reuse**

Amir Roth and Gurindar S. Sohi Computer Sciences Department, University of Wisconsin - Madison {amir, sohi}@cs.wisc.edu

### Abstract

Register integration (or simply integration) is a mechanism for incorporating speculative results directly into a sequential execution using data-dependence relationships. In this paper, we use integration to implement squash reuse, the salvaging of instruction results that were needlessly discarded during the course of sequential recovery from a control- or data- misspeculation.

To implement integration, we first allow the results of squashed instructions to remain in the physical register file past mis-speculation recovery. As the processor re-traces portions of the squashed path, integration logic examines each instruction as it is being renamed. Using an auxiliary table, this circuit searches the physical register file for the physical register belonging to the corresponding squashed instance of the instruction. If this register is found, integration succeeds and the squashed result is re-validated by a simple update of the rename table. Once integrated, an instruction is complete and may bypass the out-of-order core of the machine entirely. Integration reduces contention for queuing and execution resources, collapses dependent chains of instructions and accelerates the resolution of branches. It achieves this using only rename-table manipulations; no additional values are read from or written to the physical registers.

Our preliminary evaluation shows that a minimal integration configuration can provide performance improvements of up to 8% when applied to current-generation micro-architectures and up to 11.5% when applied to more aggressive microarchitectures. Integration also reduces the amount of wasteful speculation in the machine, cutting the number of instructions executed by up to 15% and the number of instructions fetched along mis-speculated paths by as much as 6%.

## **1** Introduction

Modern microprocessors rely heavily on *speculative execution* to achieve performance. Sequential processors (ones that execute sequential programs) speculate on both control and data, executing instructions before all of their input dependences are known with certainty. Successful speculation improves performance by sparing the speculated instructions the wait of having their execution context verified. On the other hand, unsuccessful speculation, or *mis-speculation*, hurts performance by forcing the processor to *recover* to some prior non-speculative state and start over. This paper presents *register integration*, a mechanism for overcoming an inherent inefficiency in conventional sequential mis-speculation recovery.

The inefficiency we speak of is born of a basic antagonistic combination found in sequential programs. While a sequential program is composed of many *locally independent computa-tions*, the *state* of the program is only defined sequentially at dynamic instruction boundaries. Since mis-speculation recovery is defined in terms of this sequential state, a mis-speculation in one computation inadvertently but necessarily causes

valid work from sequentially younger computations to be aborted, or *squashed*, and re-executed. Register integration can be used to perform *squash reuse* [2, 18], to salvage the results of squashed computations that are in fact control- and data- independent of the particular mis-speculation event that precipitated the recovery action.

Many processors implement speculation using a level of indirection that maps the architectural register name space to a larger physical register storage space. The larger physical space allows multiple versions of each architectural location (all but one of which is speculative) to simultaneously coexist. Successful speculation involves the promotion of newer mappings to non-speculative status; mis-speculation recovery restores prior mappings and recycles the speculative storage. Integration is motivated by the observation that only restoration of previous mappings is required for correct recovery. If the speculative values are left intact past a recovery event, then should the processor re-trace part of the squashed path and discover that some of the instructions were useful after all, only the corresponding mappings will need to be restored; the values themselves will already exist and will not need to be recomputed.

The matching of squashed results with re-traced instructions is accomplished using a second mapping into the physical register file, the Integration Table (IT). The IT differs from the sequential mapping (map table) in a fundamental way. The map table describes the contents of the physical registers in a transient, sequentially dependent way from the point of view of the architectural registers. In contrast, the IT describes the contents of the physical registers in a persistent, order-independent way that reflects the operations and dataflow relationships used to create the values they contain. While an instruction is being register-renamed, the IT is used to search the physical register file for a physical register that holds the result of a previous squashed instance of the same instruction. If a register is found such that its creating instruction instance had the same physical register inputs as the currently renamed instance, then the currently-renamed instruction is "recognized" as having been previously executed and squashed. The instruction is *integrated* by setting the sequential mapping for its output to point to the physical register allocated during the initial (squashed) execution. The integrated instruction is complete for all intents and purposes; it can commit as soon as the retirement algorithm allows.

Integration has many advantages. Obviously, it reduces consumption of and contention for execution resources. It also collapses data-dependent chains of instructions: a data-dependent chain of dependent instructions cannot be executed in a single cycle, but a completed chain of instructions may be integrated in a single cycle. Integrated branch instructions are resolved immediately, and should these be mis-predicted branches the mis-prediction penalty and subsequent demand on the fetch engine are also reduced. From an engineering standpoint, integration is simple to implement. It is unambiguously correct, involves no explicit verification and does not require additional data paths to either read or write any values into the physical registers. In general, integration involves modifications only to the register renaming stage in the processor; the rest of the pipeline is oblivious to its existence.

Our initial experiments show that a minimal integration configuration can achieve speedups of up to 8% on a representative current-generation microarchitecture. We estimate that the speedup increases to up to 11.5% for more aggressive microarchitectures. Integration also reduces the level of wasteful speculation in a processor, cutting the number of instructions fetched along mis-speculated paths by as much as 6% and the number of instructions executed by 15%.

The rest of the paper is organized as follows. The next section presents the basic integration algorithm and argues for its correctness properties. Section 3 addresses some issues involved in the implementation of integration. In section 4 we evaluate integration using cycle-level simulation. Section 5 discusses related work. Section 6 presents our conclusions.

# 2 Integration

In this work, we use integration to implement squash reuse, the salvaging of results that were unnecessarily discarded during the process of sequential mis-speculation recovery. In this section, we discuss the basic integration algorithm and describe the principles that allow it to accomplish its goal in a straightforward way. We specifically address the integration of load instructions, which requires additional attention.

## 2.1 Basic Algorithm

During the course of processing, the program's dataflow graph, in the form of the results of its individual instructions, is stored in the physical register file. At any point in the program, the "active" vertices (results) of this graph are available through a set of mappings that maps architectural register names to physical register locations and their values. New portions of the dataflow graph can only be attached to these "active" vertices. As each instruction is added to the graph, a physical register to hold its value is allocated and mapped to the architectural output. Each instruction is annotated with both the physical register holding its value and the prior physical register mapping of the same architectural location. Recovery entails backtracking over a portion of the program, restoring the previous mapping of each instruction's output while recycling the storage for the squashed result.

Integration exploits the observation that mis-speculation recovery is obligated only to restore some prior sequential mapping into the physical register file. That the results associated with the discarded mappings are also recycled during recovery is an implementation convenience; leaving them intact past the mis-speculation does not impact correctness (of course, they must be recycled eventually lest the processor "leak" away all physical registers). Assuming the results are kept, let us consider the point immediately after the completion of a recovery sequence. Just at this point, all squashed instructions are, in principle, still "attached" to the current state (dependence graph) of the program as defined by the register mapping. The inputs of the oldest squashed instructions are found in this mapping. The fact that the inputs are valid validates the outputs, which are themselves inputs of younger squashed instructions, and so on. Integration is the process of transitively recognizing this validity, instruction by instruction. For every instruction sequenced by the processor, the integration logic looks for the result of a squashed instruction that had the same input mappings. If one is found, the corresponding physical register is "un-squashed" or "pulled back into the sequential flow" simply by setting the sequential mapping to point to it. This action re-validates the physical register mapping, and makes the input mappings of squashed instructions that depend on it valid, allowing them to be subsequently integrated. Notice that this same mechanism naturally avoids the re-use of instructions whose data inputs have been invalidated. As the processor sequences instructions from paths different than the squashed one, the results of these instructions create mappings to new physical registers not found in the squashed dataflow graph. These new mappings effectively "detach" those portions of the squashed dataflow graph that depend on the corresponding architectural name, and prevent them from being integrated.

Integration of a result requires locating a squashed instance of the corresponding instruction with input physical registers identical to those of the current instance being renamed. To facilitate this search, integration relies on the *Integration Table* (*IT*), an auxiliary structure that indexes and tags squashed results using instruction identity and input mapping information. Each entry in the IT corresponds to a squashed instruction instance and contains that instruction's PC and the physical registers used for that instance's inputs and output. The IT also contains three fields whose purpose will be made clear later: *Jump-Target* which is meaningful only for control instructions, and *Memory-Address* and *Memory-Value* fields which are meaningful only for loads and stores.

We illustrate the basic algorithm using an example. Figure 1 shows a short program fragment with four variables X,Y,Z and W each allocated to a different logical register. For each dynamic instruction, we show the instruction preceded by its PC, the state of the Map and Integration Tables immediately after the renaming of the instruction and descriptions of the actions taking place during sequential processing and in the IT. The shaded boxes and circled markers highlight the handling of instruction A5. The program undergoes three processing phases. In the first, instructions A1 through A8 are renamed and executed; a new physical register is allocated to each newly created result (marker 2). The second phase begins after all the instructions have completed execution when a branch mis-prediction is detected at instruction A3. Instructions A8, A7, A6, A5 and A4 are recovered in reverse order and the original mappings for their output registers are restored (marker 3). However, instead of recycling the physical registers, each result is entered into the IT and tagged with the instruction PC and physical register inputs used to create it (marker 4). Integration comes into play in the final phase. Having recovered from the mis-prediction, the sequential processor resumes fetching at the re-convergent point beginning at A5. We follow the renaming and potential integration of each instruction carefully.

Intuitively, the re-traced instance of A5 *should* be integrated since removing A4 did not change the value of Y. Indeed,

when A5 is renamed for a second time Y is mapped to 51 (marker 5), the same mapping it had during A5's original, squashed execution (marker 1). Properly, the IT contains an entry for an instance of A5 with input physical register 51 (marker 6). By comparing PC/input-register tuples from the dynamic instruction and map table with the corresponding IT tuples (marker 7 with 8, marker 5 with 6), we determine that integration can take place. The act itself consists of setting the output mapping of A5 to the physical register originally allocated for it, 53 (marker 9). The IT entry is removed so that the register will not be integrated by another instruction.

When A6 is renamed for the second time, it finds its input X mapped to register 50. Changing the path has removed A4 and changed the value of X with respect to A6, invalidating it. This invalidation is naturally reflected in the IT, as no entry for A6 with an input of 50 is found. The A6 IT entry has 52 as its input; 52 was created by A4, which was squashed and *not retraced*. Without a match, the instruction is left in the IT until it is evicted. A new physical register, 57, is allocated to the current instance of A6.

Recall, when we integrated A5, we entered its output (53) into the map table. That action set the stage for A7, an instruction that depends on A5, to be integrated now. The squashed version of A7 was executed with input register 53, the output of the squashed A5. When A7 is re-traced, its input is again 53 thanks to the integration of A5. A7 is integrated in exactly the same manner that A5 was.

The final instruction in the group, A8, should not be integrated since it depends on A6, which was itself not integrated. Such indeed is the case. When A6 was *not integrated*, a new mapping (57) was created for X. This new mapping prevents A8 from being integrated, much like the removal of A4 changed the mapping that prevented A6 from being integrated.

In a four wide super-scalar machine, the integration decision on these four instructions can be made in parallel. How this is done is the subject of a future section. However, the example demonstrated the four possible cases for super-scalar integration: basic integration of an instruction (A5), basic non-integration of an instruction (A6), the integration of an instruction that depends on an integrated instruction (A7), and the nonintegration of an instruction that depends on a non-integrated instruction (A8).

## 2.2 Integrating Loads

An integrated instruction can be thought of as having two executions: a physical execution where the instruction is actually executed and then squashed, and an architectural execution in which the integrated instruction is supposed to execute but doesn't actually do so. For most types of instructions, the algorithm we have shown so far is perfectly safe. The combination of operation and valid input values, denoted by PC and physical registers respectively, guarantees that the results of the physical execution are identical to those that would be produced in the architectural execution, allowing the former to be substituted for the latter. Loads are the exception. The integration of a particular load is not guaranteed to be safe because a conflicting store may have executed between the load's physical and architectural executions. A load that is either blindly integrated despite such a store conflict or that experiences a post-integration conflict is termed *mis-integrated*. Mis-integrations jeopardize correctness.

Loads present a problem because physical register names are not sufficient to detect load/store collisions. There are two ways to ensure that mis-integrated loads are not allowed to retire. The first is to re-execute all integrated loads and treat a change in the output value as a mis-speculation. The second is to store data addresses (and potentially values) with loads in the IT and use stores to invalidate matching loads. The first

| Insn Action             | Dynamic Insn      | Map Table |      |    | In | tegrati         | ion Ta | IT Action |    |                |
|-------------------------|-------------------|-----------|------|----|----|-----------------|--------|-----------|----|----------------|
|                         |                   | Х         | Y    | Ζ  | W  | PC              | I1     | I2        | 0  |                |
| Rename/Alloc            | A1: $X = 0;$      | 50        | 47   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A2: $Y = 1;$      | 50        | 51   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A3: if $(Z == 0)$ | 50        | 51   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A4: $X = 1;$      | 52        | 51   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A5: Y++;          | 52        | 53   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A6: X++;          | 54        | 53   | 48 | 49 |                 |        |           |    | No Match       |
| Rename/Alloc            | A7: $W = Y * Y;$  | 54        | 53   | 48 | 55 |                 |        |           |    | No Match       |
| Rename/Alloc            | A8: $Z = X * Y;$  | 54        | 53   | 56 | 55 |                 |        |           |    | No Match       |
| Recover                 | A8: $Z = X * Y;$  | 54        | 53   | 48 | 55 | A8              | 54     | 53        | 56 | Enter          |
| Recover                 | A7: $W = Y * Y;$  | 54        | 53   | 48 | 49 | A7              | 53     | 53        | 55 | Enter          |
| Recover                 | A6: X++;          | 52        | 53   | 48 | 49 | A6              | 52     |           | 54 | Enter          |
| Recover                 | A5: Y++;          | 52        | 51   | 48 | 49 | 4 A5            | 51     |           | 53 | Enter          |
| Recover                 | A4: $X = 1;$      | 50        | 51   | 48 | 49 | A4              | 50     |           | 52 | Enter          |
| <b>Rename/Integrate</b> | • A5: Y++;        | 50        | 9 53 | 48 | 49 | <sup>8</sup> A5 | 51     |           | 53 | Match/Remove   |
| Rename/Alloc            | A6: X++;          | 57        | 53   | 48 | 49 | A6              | 52     |           | 54 | No Match/Leave |
| Rename/Integrate        | A7: $W = Y * Y;$  | 57        | 53   | 48 | 55 | A7              | 53     | 53        | 55 | Match/Remove   |
| Rename/Alloc            | A8: $Z = X * Y;$  | 57        | 53   | 58 | 55 | A8              | 54     | 53        | 56 | No Match/Leave |

**FIGURE 1. A Working Example of Integration.** Shows the three-phase processing of a series of instructions. The three phases are: (i) initial execution (ii) recovery and (iii) squashed-path re-execution. The shaded quantities and circled markers highlight the actions surrounding instruction A5.

method uses a simple IT but reduces the positive impact of successful integration, forcing integrated loads to consume execution bandwidth. The second increases the potential impact of successful integration, but complicates the IT somewhat. Our framework models store invalidations.

## **3** Implementation Aspects

In this section we discuss several implementation aspects of integration including all modifications that must be made to the base microarchitecture, the integration circuit itself, and the mechanism that ensures the safe integration of loads.

### 3.1 Base Micro-architecture Requirements

Integration is not a technique that can be applied to all speculative microarchitectures. Its implementation requires that the base microarchitecture allow speculative results to remain intact past a mis-speculation recovery action and support the out-of-order allocation and freeing of speculative storage.

These requirements disqualify many current microarchitectures. In-order speculative microarchitectures like Sun's UltraSparc-III that use working (future) register files indexed by architectural register number both disallow arbitrary assignments of physical results to architectural names and overwrite the mis-speculated instructions results during recovery. Intel's P6 [10] core processors and HAL's SPARC64 V [7] keep speculative results in the re-order buffer, preventing their preservation past a mis-speculation recovery. IBM's Power [19] processors and (we believe) AMD's K7 [5] have physical register files separate from the re-order buffer, but also have an architectural register file and require that physical registers be allocated and freed in-order. Microarchitectures with physical register models that can support integration are the out-of-order Alpha processors starting with the 21264 [11], those of MIPS beginning with the R10000 [21], and (we believe) Intel's Pentium 4 NetBurst microarchitecture [9].

## 3.2 A Micro-architecture with Integration

We now examine a microarchitecture that includes integration and comment on changes in the flow of instructions through the modified pipeline. A pipeline with integration is shown in Figure 2(a); the structural modifications and new register tag and data paths are in bold. We work from the back of the pipeline to the front, explaining how instructions become candidates for integration before dealing with the flow of integrated instructions. A later subsection is dedicated to explaining the integration circuit itself in detail.

Since integration deals with salvaging the results of squashed instructions, the most natural time to insert instructions into the IT is during mis-speculation recovery. Implementation of IT insertion is straightforward for micro-architectures that implement recovery using serial rollback. Most microarchitectures, however, including the Alpha 21264 [11] and MIPS R10000 [21], implement recovery as a monolithic copy from a checkpoint. IT insertion is slightly more involved in this case, but its particulars do not affect integration performance. For clarity, we explain the process as serial.

One important qualification to the IT entry procedure is the exclusion of all instructions that have not completed execu-

tion. The reasoning behind this decision is that it is the integration of *completed* instructions that contributes most to performance. Integration provides two main performance benefits: it allows instructions to bypass the issue engine and it collapses dependent chains of instructions. Neither of these benefits applies to instructions that have not issued and only the first applies to instructions that have not issued but not completed. However, the number of instructions likely to be integrated while in this post-issue/pre-completion state is small, and in return for forfeiting them, we simplify the handling of integrated instructions by assuming that all integrated instructions are complete. Faulting instructions are also excluded from the IT, since faults may have side effects that would need to be reproduced on integration.

One of the principles of integration is that it allows speculative physical registers to "survive" recovery. This means that during recovery output registers of instructions that are entered into the IT are not reclaimed and added to the free list as usual. However, we must be explicit about who is responsible for eventually freeing the registers of instructions that are in the IT, so that these registers are not "leaked". The policy is actually quite straightforward. The IT assumes responsibility for the physical registers of its entries. If an entry is evicted without having been integrated, it physical register is added to the free list. Conversely, if an entry is integrated, responsibility for the register returns to the re-order buffer, which handles it in the usual way. One caveat is that the IT entry of an integrated instruction must be cleared so that no other sequential instruction will attempt to get ownership of the corresponding register (the output of two simultaneously active instructions may not be allocated to the same physical register). Notice, the change of ownership mechanism also allows the same instruction to be repeatedly squashed and integrated.

The next subsection describes the integration related modifications to the register renaming logic. Here, we describe what happens to an instruction after it has been integrated which, having decided that only completed instructions can be integrated, is not much. An integrated instruction is entered into the re-order buffer marked as completed and the integrated physical register is set as its "current mapping". Integrated loads (and stores) are allocated load (or store) queue entries that are filled using the IT Memory-Address and Memory-Value fields and marked as completed. These entries, too, are ordinary. Finally, if the integrated instruction is a branch, the resolution and potential recovery sequences are started immediately using the Jump-Target IT field as a recovery address. The integrated instruction can bypass the out-of-order execution core; it does not need to be allocated to a reservation station, scheduled, executed, or written back.

#### 3.3 Integration Circuit

The most delicate piece of the integration mechanism is the integration circuit itself. The integration circuit examines each dynamic instruction and decides whether or not that instruction may be integrated. Of course, it must do so for multiple, potentially dependent instructions in parallel. In this section, we describe one possible implementation of this logic and its complexity. We begin with a scalar description of the circuit, before proceeding to the super-scalar case.

Scalar register renaming occurs in two logical steps. First, an instruction's logical inputs are renamed to physical outputs

using lookups in the map table. Second, its logical output is allocated a new physical register and this new logical-to-physical mapping is entered into the sequential map table, allowing future instructions that need the value to obtain their inputs from the correct location. We call the two stages input routing and output allocation, respectively. Integration adds a piece called *output selection* in which the output mapping must be chosen between a newly allocated physical register and a physical register obtained from an IT entry. The output selection circuit occurs logically after the input routing circuit since the integration test must compare the input physical registers of the sequential instance with those in the IT entry. However, the scalar implementation of integration can be thought of as occurring in one of two ways. In the first, output selection is implemented serially after input routing with the integration table indexed by instruction PC and input physical registers. In the second, output selection is split into IT lookup, which happens in parallel with input routing, and an integration test, which occurs logically after it. In this organization, shown in Figure 2(b), the IT is indexed by PC only and the physical register numbers are used to match tags. Both schemes likely require pipelining register renaming into at least two stages.

The merits of each implementation are open to debate in the scalar realm, but in a super-scalar environment only the second is viable. While the first scheme interleaves and serializes the input routing and output selection decisions that must be made for each instruction, the PC-only indexed scheme permits a parallel prefix implementation similar to the one used to super-scalarize conventional register renaming. Let us review conventional super-scalar renaming because its input routing decisions must reflect intra-group dependences. To do so, dependency-check logic acts in parallel with output allocation. This logic compares the logical input of each instruction in the group with the logical output of each previous in-group instruction; a match overrides the input to the appropriate

newly allocated physical register. For example, in a group of four two-input, one-output instructions each of the second instruction's inputs has to be compared with the first instruction's output, each of the third instruction's inputs has to be compared with the outputs of the first two instructions and each of the fourth instruction's inputs has to be compared with the outputs of the first two is be compared with the outputs of the first two instructions and each of the fourth instruction's inputs has to be compared with the outputs of the first three instructions. The total number of comparisons for this case is 12 and in general I \* N(N-1)/2, with *I* the number of inputs per instruction and *N*, the superscalar width or the number of parallel renaming operations. In general, the depth of the circuit is linear with *N* and the number of comparisons grows as  $N^2$ .

In addition to the conventional dependence-check circuit that compares logical registers, integration requires that we implement output selection and any corrections it might imply for input routing for subsequent instructions. Recall, for the scalar integration test we compared each IT entry input with the corresponding register retrieved from the map table. In the super-scalar case, we must also compare it to the physical register outputs for all integration candidates of all prior instructions in the group. Note, we do not have to compare the candidate inputs with the newly allocated physical registers corresponding to each prior instruction: the situation in which an instruction is dependent on a prior instruction in the group and is integrated while the prior instruction is not is obviously impossible. Nevertheless, although the priority encoding depth of the circuit is still N, the superscalar width, the number of physical register comparisons now grows with both N and the number of possible IT matches, M. The precise formula is I \* (((N(N-1)/2)M + N) \* M); the growth of the function is  $IN^2M^2$ . The complexity of the circuit is very close to that of register renaming for a direct-mapped IT, but diverges for higher-associativity implementations. For instance, a fourwide machine with a direct-mapped IT requires 20 physical register comparisons to implement integration. The same machine with a 2-way IT needs 64 comparisons. Just for scale, an 8-wide machine with a 4-way IT requires 960 com-



**FIGURE 2.** Implementation Aspects. (a) A micro-architecture with integration. Integration-specific modifications in bold. In addition to the actual integration table (IT) and modified rename logic, there are additional paths from the instruction ordering buffer (ROB) to the IT that are used during recovery, a path from the IT to the free list, and paths between the IT and the load and store queues. (b) Scalar, PC-indexed integration circuit. A scalar integration circuit in which the IT and map table are accessed in parallel. An extension of this circuit implements super-scalar integration. The diagram traces the IT, map table and free list, as well as the instruction itself through the two steps of integration-enabled register renaming. At the top of the figure, the instruction shown is raw and the structures are as they appear before the instruction is renamed. At the bottom, the instruction is renamed and the structures reflect that fact.

parisons! Certainly, a highly associative integration circuit is challenging to build. In the evaluation section, we quantify the performance impact of higher associativity.

We should mention here that some of the complexity of the integration circuit may be moved off-line into the IT itself. For instance, the IT could internally perform the intra-group dependence checks and store groups of dependent instructions in a kind of "trace" that can be integrated using  $I^*N^*M$  comparisons. However, IT management becomes much more complex in this case, and there is the added problem of choosing the grouping of instructions into traces. An investigation of such optimizations is outside the scope of this work.

### 3.4 Safe Load Integration via Store Invalidation

When first presenting integration, we remarked that special support must be provided to ensure that loads that have been invalidated by intervening stores are removed from integration consideration. At the very least, the mis-integration should be detected so that alternative corrective action can be taken. Mis-integration detection and avoidance are implemented using a simple extension to the processor's basic load speculation (advancing loads past incomplete stores) detect store/load ordering violations as follows. The load and store queues contain address and value fields. Completed stores check their address and value against address/value pairs of *younger* previously completed loads. An address overlap coupled with a value mismatch signals a memory ordering violation which is handled by replaying the load in some way.

The solution handles two cases. The mis-integration detection case covers conflicts with stores that complete after the load has been integrated. Mis-integration detection is implemented naturally by the native load speculation mechanism. Recall, loads are entered into the IT along with their address and value fields from the load queue. When they are integrated, these fields are restored to the load queue. To a completing store, therefore, an integrated load looks just like any other completed load and conflicts are handled in the usual way. Misintegration avoidance targets conflicts with stores that complete before a load is integrated. To implement avoidance, we simply extend the store-invalidation procedure to include IT loads. The IT essentially "snoops" completed stores, matching their address/value pairs with the Memory-Address/Memory-Value pairs of IT loads. An address match/value mismatch causes the invalidation of the corresponding load, preventing it from being integrated. Detection and avoidance can also be implemented using purely address-based criteria.

Our results show that most mis-integrations are avoided. Those that aren't, while not impacting correctness, can degrade performance as they are equivalent to normal load or value mis-speculations. Our performance evaluation section will measure the prevalence of mis-integration.

## 3.5 Handling Data Mis-Speculations

The discussion of load integration brings up an important note regarding integration and the way it must deal with instructions squashed due to data mis-speculations like speculative memory-ordering violations [14, 22] and value mis-speculations [12]. Specifically, for micro-architectures like the Alpha 21264 [11], in which data mis-speculations are handled by

squashing, integration must be careful not to confuse a value mis-speculated instruction and its dependent instructions with correctly executed squashed instructions. IT entries that correspond to data mis-speculated results must not be integrated. One broad solution to this problem would be to not enter squashed instructions into the IT during recovery from these kinds of mis-speculations. However, this solution is too harsh since it prevents the correctly executed instructions that were lost during recovery from being salvaged. An effective trick is to enter all completed instructions *except* for the value mis-speculated instruction *itself* into the IT. This omission effectively "detaches" all dependent instructions intact.

There is an interesting interaction between integration and another technique for salvaging work lost to a data mis-speculation, *selective squashing* [8, 12, 15, 16]. In selective squashing, instructions are kept in reservation stations until retirement allowing them to simply re-issue as data mis-speculations are resolved. If selective squashing *is* implemented, integration is not "activated" during data mis-speculations since the instructions are not squashed and re-fetched. Integration, on the other hand, still handles control mis-speculation squashes which, quite conveniently, cannot be handled by selective squashing. Integration and selective squashing complement each other nicely. However, we do not explore their interaction experimentally; our simulations model full squashing for all mis-speculations.

#### 3.6 Setting the Size of the Physical Register File

A final implementation note concerns the size of the IT and its relationship to the total size of the physical register file. To avoid resource stalls, the number of physical registers should be equal to the maximum number of values (both architectural and speculative) that can be "in play" at any time. For a speculative machine this is equal to the number of architected registers plus the maximum number of renamed in-flight instructions (the size of the re-order buffer). Now, the IT is simply a mechanism for keeping physical registers "in circulation" for longer periods of time; values in the IT are still considered "in play". Consequently, to avoid resource stalls in a micro-architecture with integration, the size of the physical register file should be equal to the number of architected registers plus the size of the re-order buffer plus the size of the IT. In our simulated configurations, we use this formula to ensure that the machine never stalls for lack of a free physical register.

## **4** Performance Evaluation

We evaluate the potential performance impact of integration using cycle-level simulation. We present a full set of results for one specific design meant to represent a potential currentgeneration (or very near future) microprocessor. We then briefly look at two dimensions in the IT design space, size and associativity. To be fair, we quantify the adverse performance effects of any additional pipeline stages required by integration. Finally, we try to project integration's impact on more aggressive future-generation microarchitectures.

## 4.1 Experimental Framework

We evaluate integration using the SPEC2000 integer benchmark suite. The programs are compiled for the Alpha EV6 architecture by the Digital UNIX V4 cc compiler with optimizations -O3 -fast. We use the test datasets for reporting performance for all benchmarks except *perlbmk*. There we are forced to use the training set because the test set contains fork and exec calls that our simulation environment does not support. Where multiple test data sets are given we use the longer running one, specifically *place* for *vpr* and *kajiya* for *eon*. We simulate all programs in their entirety.

The simulation environment is built on top of the SimpleScalar 3.0 [1] Alpha toolkit. The cycle-level simulator models an outof-order machine similar in organization to an unclustered Alpha 21264 [11] with nominal stages fetch, register rename and dispatch, schedule, execute, writeback and commit. The out-of-order scheduling logic speculates loads aggressively, issuing them even in the presence of prior stores with unavailable addresses. A mis-speculation causes the load and all downstream instructions to be squashed and re-fetched. Our model does not include a dependence-speculation mechanism that may reduce the incidence of memory-ordering violations [3, 14, 22]. However, we don't believe that the inclusion of such a mechanism would take away a significant portion of the impact of integration, since most integration candidates are produced by control mis-speculation. The recovery mechanism itself is modeled as serial with bandwidth equal to commit. Recovery stalls renaming, but execution and retirement from the head of the machine may continue. We model a memory system with non-blocking caches, finite write-buffers and miss-status holding registers (MSHR), and cycle accurate bus utilization. Table 1 shows the simulation parameters in detail. IT configuration is specified inline with the respective presentation of results. The Alpha has 64 architectural registers; the number of physical registers for a given configuration is therefore always set to be 64 + ROB size + IT size.

## 4.2 Base Configuration Results

Table 2, which is split into two for readability, shows the performance impact of integration using a 256-entry directmapped IT on the configuration described above. Data is presented in four main parts. The first two characterize the performance of the base and modified system in terms of instructions fetched and executed, branch mis-predictions and branch misprediction resolution latency, and total memory-ordering violations. These numbers give a feel for the degree of mis-speculation in each program and its causes. Comparing these groups of numbers pair-wise gives an idea of the overall effect of integration on speculative (mis-speculative) processor activity. The next two parts measure the activity and effectiveness of integration using more direct metrics. We report absolute counts of instructions integrated, loads integrated, and mispredicted branches integrated (and ostensibly, immediately resolved).

The shaded at the bottom computes the characteristic and performance metrics of integration and its impact on performance. The *contribution rate* is the number of instructions integrated as a percentage of the total number of instructions committed; it is the amount of work integration contributes to the architectural execution of the program. The *salvage rate* is number of instructions integrated as a percentage of squashed (and completed) instructions and measures the rate at which integration candidates are harvested. The contribution and salvage rates measure both a program's inherent suitability for integration *and* our mechanism's ability to capture integration candidates. The final three metrics measure the percentage of instructions fetched, instructions executed and total execution time saved by integration.

The performance figures show that integration is equally effective on all benchmarks. On some, like gzip, vpr, crafty and twolf, it cuts execution time by upwards of 5%. On others, it achieves speedups of less than 1%. To explain this behavior we appeal to the structure of the programs and to the contribution and salvage rates, which help correlate this structure with suitability for integration. There are some programs that for structural reasons simply cannot take advantage of integration. One possibility is that the programs have few squash-causing branch mis-predictions and memory-ordering violations. Another is that branch mis-predictions are present but that the code within the conditional arms is so long that the processor does not have time to fetch and execute the re-convergent region before the branch is resolved. Finally, if the re-convergent region is reachable along the mis-speculated path, it is possible that it contains no data-independent instructions, the ones that can later be integrated.

How do the benchmarks break down according to these criteria? *Bzip2*, for instance, encounters branch mis-predictions

| Front-End                  | Symmetric 16K-entry combined 10-bit history gshare and 2-bit predictors. 2K entry, 4-way associative BTB, 32 entry return-address-stack. 3-cycle fetch. 32-entry instruction buffer. Up to 8 instructions from two cache blocks fetched per cycle. A maximum of one taken branch per cycle. 8-wide single-cycle decode. Direct, unconditional jump mis-predictions recovered at decode.                                                                                                                                                                                                                                                                                                                        |
|----------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|
| Issue<br>Mechanism         | 8-way superscalar out-of-order speculative issue with a maximum of 128 instructions or 64 loads or 32 stores in flight. 2-cycle schedule/register read. Loads speculatively issue in the presence of earlier stores with unknown addresses. The load and subsequent instructions are squashed and re-fetched on a memory ordering violation. Recovery from all forms of mis-speculation is serial with a bandwidth of 8 instructions per cycle. Recovery stalls register renaming, but execution of unrecovered instructions may proceed in parallel. Store to load bypass takes 2 cycles. Memory and control instructions have the highest scheduling priority. Priority within a group is determined by age. |
| Memory<br>System           | 32KB, 32B lines, 2-way associative, 1-cycle access L1 instruction cache. 64KB, 32B lines, 2-way associa-<br>tive, 2-cycle access, L1 data cache. A maximum of 16 outstanding load misses. 16-entry store buffer. 16-<br>entry ITLB, 32-entry DTLB with 30-cycle hardware miss handling. Shared 1MB, 64B line, 4-way associa-<br>tive, 12 cycle access L2 cache. 70-cycle memory latency. 32B bus to L2 cache clocked at processor fre-<br>quency. 16B bus to memory clocked at 1/3 processor frequency. Cycle level bus utilization modeled.                                                                                                                                                                   |
| Functional Units (latency) | 8 INT ALU (1), 2 INT mult/div (3/20), 3 FP add (2), 1 FP mult/div (4/24), 4 load/store (2). The FP adders and all multipliers are fully pipelined.                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                                             |

TABLE 1. Simulated machine configuration.

infrequently (fewer than once every 400 instructions). It falls under the first category. *Bzip2*'s salvage rate is close to 40%, but it *executes* so few instructions along mis-speculated paths as compared to other programs that the overall pool of integration candidates is small. The second two categories are somewhat more difficult to distinguish from one another, but five of the other benchmarks: *gcc*, *mcf*, *parser*, *perlbmk* and *gap* fall into them. These programs incur branch mis-predictions or memory ordering violations every 100 instructions or so (or more frequently), execute (and squash) somewhat more instructions than they commit, yet permit the successful integration of only around 20% of squashed instructions. *Vortex* is a strange case. It executes many instructions along squashed paths but, since many squashes are due to load mis-speculation, integrates only a relatively low percentage of them. Performance gain is achieved because many of the integrated instructions are mispredicted branches. The four benchmarks we mentioned at the top execute a lot of work along mis-speculated paths and integrate that work at a high rate. These programs benefit the most from integration. Other factors that contribute to the observed impact of integration but are difficult to quantify directly are the parallelism in the high-integration regions and the extent to which the integrated instructions help collapse dependence chains.

|                                      |                                   | gzip    | vpr     | gcc     | mcf    | crafty  | parser  |
|--------------------------------------|-----------------------------------|---------|---------|---------|--------|---------|---------|
| Committed instructions (M)           |                                   | 3367.27 | 1566.70 | 2015.64 | 259.63 | 4264.78 | 4203.56 |
| Base                                 | Fetched instructions (M)          | 5555.67 | 3667.92 | 3816.01 | 527.87 | 8080.35 | 7515.99 |
|                                      | Executed instructions (M)         | 4114.58 | 2069.79 | 2327.15 | 292.49 | 5158.60 | 4854.72 |
|                                      | Mispredicted branches (M)         | 16.61   | 20.48   | 22.93   | 2.54   | 38.80   | 38.08   |
|                                      | Misprediction resolution lat. (c) | 29.72   | 18.41   | 16.85   | 33.37  | 21.48   | 20.78   |
|                                      | Mis-speculated loads (M)          | 2.50    | 0.00    | 0.20    | 0.01   | 1.35    | 0.14    |
| Base                                 | Fetched instructions (M)          | 5376.16 | 3424.83 | 3709.65 | 509.96 | 7659.44 | 7374.33 |
| + IT                                 | Executed instructions (M)         | 3481.16 | 1774.06 | 2133.07 | 271.98 | 4649.16 | 4582.10 |
|                                      | Mispredicted branches (M)         | 15.91   | 20.90   | 22.97   | 2.54   | 38.84   | 38.05   |
|                                      | Misprediction resolution lat. (c) | 27.56   | 15.66   | 15.86   | 31.96  | 19.27   | 20.15   |
|                                      | Mis-speculated loads (M)          | 3.29    | 0.59    | 0.36    | 0.02   | 1.41    | 0.20    |
| Integra                              | ated instructions (M)             | 640.70  | 249.35  | 167.73  | 15.85  | 450.31  | 274.49  |
| Integrated loads (M)                 |                                   | 177.12  | 90.69   | 55.60   | 3.28   | 200.29  | 78.19   |
| Integrated mispredicted branches (M) |                                   | 0.78    | 0.59    | 0.17    | 0.01   | 0.53    | 0.54    |
| Integrated/committed (%) (contrib.)  |                                   | 19.0    | 15.9    | 8.3     | 6.1    | 10.6    | 6.5     |
| Integrated/squashed (%) (salvage)    |                                   | 61.9    | 46.7    | 29.1    | 24.0   | 45.3    | 28.3    |
| Fetched insns saved (%)              |                                   | 3.2     | 6.6     | 2.8     | 3.7    | 5.2     | 1.9     |
| Executed insns saved (%)             |                                   | 15.4    | 15.3    | 8.3     | 7.0    | 9.9     | 5.6     |
| Execution time saved (%)             |                                   | 4.8     | 8.1     | 2.0     | 1.1    | 5.2     | 1.1     |

|                                      |                                   | eon    | perlbmk  | gap     | vortex   | bzip2    | twolf  |
|--------------------------------------|-----------------------------------|--------|----------|---------|----------|----------|--------|
| Committed instructions (M)           |                                   | 458.29 | 27684.23 | 1169.58 | 9808.12  | 8822.14  | 258.73 |
| Base                                 | Fetched instructions (M)          | 987.32 | 51890.55 | 1738.94 | 17977.94 | 10694.62 | 530.94 |
|                                      | Executed instructions (M)         | 554.43 | 30300.91 | 1227.20 | 11673.81 | 9067.05  | 295.94 |
|                                      | Mispredicted branches (M)         | 4.34   | 261.86   | 9.80    | 34.98    | 24.40    | 2.89   |
|                                      | Misprediction resolution lat. (c) | 14.32  | 60.65    | 24.82   | 12.41    | 19.56    | 16.56  |
|                                      | Mis-speculated loads (M)          | 3.92   | 13.66    | 0.15    | 43.15    | 0.16     | 0.32   |
| Base<br>+ IT                         | Fetched instructions (M)          | 957.22 | 51341.83 | 1722.18 | 17111.10 | 10638.29 | 505.40 |
|                                      | Executed instructions (M)         | 501.30 | 28964.36 | 1186.67 | 9919.20  | 8917.34  | 268.77 |
|                                      | Mispredicted branches (M)         | 4.31   | 262.07   | 9.87    | 33.85    | 24.49    | 2.89   |
|                                      | Misprediction resolution lat. (c) | 13.56  | 59.88    | 24.35   | 10.36    | 19.10    | 14.98  |
|                                      | Mis-speculated loads (M)          | 3.74   | 13.56    | 0.18    | 40.14    | 0.52     | 0.32   |
| Integrated instructions (M)          |                                   | 41.35  | 1308.39  | 3.80    | 157.36   | 132.05   | 22.35  |
| Integrated loads (M)                 |                                   | 12.37  | 435.56   | 1.04    | 34.93    | 44.27    | 8.38   |
| Integrated mispredicted branches (M) |                                   | 0.30   | 7.67     | 0.02    | 11.73    | 0.27     | 0.27   |
| Integrated/committed (%) (contrib.)  |                                   | 9.0    | 4.7      | 0.3     | 1.6      | 1.5      | 8.6    |
| Integrated/squashed (%) (salvage)    |                                   | 44.8   | 22.4     | 22.4    | 7.3      | 33.4     | 41.4   |
| Fetched instructions saved (%)       |                                   | 3.1    | 1.1      | 1.0     | 4.8      | 0.5      | 4.8    |
| Executed instructions saved (%)      |                                   | 9.6    | 4.4      | 3.3     | 15.1     | 1.7      | 9.2    |
| Execution time saved (%)             |                                   | 3.0    | 0.9      | 0.4     | 3.1      | 0.4      | 5.6    |

**TABLE 2.** Detailed Performance Impact of Adding a Direct-Mapped, 256-entry IT to a Current Generation Microarchitecture. Raw quantities are listed in millions of events (M) or cycles (c).

To a first order, integration is primarily a technique for reducing the number of instructions executed in a program. To that end it is fairly successful, reducing the consumption of execution bandwidth by 1% to 15%. However, a rather striking trend is the incredibly strong correlation between the performance of integration and its second order effect, reducing the number of instructions fetched, which it does at rates that vary from close to nil to near 7%. Integration is a technique that operates at decode/rename time. It is is therefore unable to eliminate the latency and bandwidth of fetch from the cost of an integrated instruction. Integration frees up execution bandwidth for new instructions, but does not directly free up more fetch bandwidth to fetch those new instructions (it actually can, but only indirectly via the accelerated resolution of mispredicted branches). As a result, the reduced consumption of execution bandwidth generally leaves bubbles and open slots in the execution pipelines. Actual performance gain is more closely related to the number of instructions eliminated from processing completely.

One opportunity for integration to do harm is by precipitating squashes through mis-integrations. However, our figures show that although memory-ordering squashes are sometimes increased with integration, the number of introduced squashes is small in comparison with the number of loads integrated. On the whole, integration reduces the amount of mis-speculation activity in the processor, cutting down the number of instructions fetched and (to a lesser degree) executed. This fact suggests two interesting applications for integration. The first is as a dynamic power and energy reduction technique [13]. This use, of course, requires that the power characteristics of integration itself be acceptable, something that has not yet been investigated. The second application is in a simultaneous multithreading (SMT) processor [6, 20], where several narrow front-ends share a large out-of-order execution engine. This could be an ideal environment for integration, which would reduce contention in the back end, and would require only (replicated) narrow, low-complexity integration circuits.

### 4.3 Impact of Table Size and Associativity

Two important parameters in the design of the IT are its size and associativity. Since the IT always contains the *most recently squashed* instructions, its size determines the degree to which it can salvage work from *older* squashed regions. If the IT is too small, older squashed instructions would be evicted before they could be integrated. However, an overly large IT is also undesirable since it implies an overly large (and overly slow) physical register file.

The effect of IT size on the performance impact of integration is shown in Figure 3(a). The trends certainly support our program-structure explanation for the bimodal nature of integration, as each group of benchmarks responds differently to changes in IT size. Those benchmarks that fail to benefit from integration for structural reasons do so consistently, regardless of IT size. More integration resources do not change the fact that the product of program and machine does not produce many valid integration candidates. On the other hand, programs whose structure does allow them to support integration, can draw additional benefit from additional integration resources. In general, however, a very large IT is not necessary. A significant fraction of the benefit can be achieved with a small IT that can buffer the squashed results from the last mis-speculated region. For this set of programs and our machine configuration, 256 entries (enough space to buffer instructions from between 4 to 8 mis-speculated regions) appears to be sufficient. The corresponding number of physical registers is 448.

The associativity of the IT has two different uses that impact performance in two ways. From the standard viewpoint, associativity is a mechanism for more efficient management of collisions in the IT. Specific to the integration circuit, however, associativity can also determine the number of squashed instances of the same static instruction that are simultaneously considered for integration. Although the first use does not



**FIGURE 3.** Effect of IT Size and Associativity on Performance Impact of Integration. Percentage of execution time saved using (a) a direct-mapped IT of four sizes: 64, 128, 256 and 512. (the corresponding physical register file sizes are 256, 320, 448 and 704) and (b) a 256-entry IT with associativities 1, 2 and 4.

necessarily imply the second, we use associativity to quantify both IT eviction policy and integration circuit complexity in order to simplify the discussion. The impact of IT associativity on integration performance is shown in Figure 3(b). The trends are similar to those observed when changing the size of the IT; the bimodal effect is still present for the same programstructural reasons. The trends are much less pronounced, however. Except for in the cases of *gzip* and *vortex*, there is little benefit to having anything more complex than a direct-mapped IT that supplies a single integration candidate per instruction. That higher associativities that would overly complicate the integration circuit are unnecessary is good news indeed.

## 4.4 Impact of Increased Pipelining

Earlier we mentioned that an implementation of integration *may* require register renaming to be pipelined into two stages. Such an increase in pipeline depth will erode some of the performance gained by integration, and potentially induce absolute slow-downs for programs that did not originally benefit from integration and would now be forced to pay for its implementation. The increased number of physical registers may also require adding additional register read/schedule cycles.

The impact of increased pipelining for both register renaming and register read is shown in Figure 4(a). Integration-induced increased pipelining does mitigate the performance impact of integration, even producing slow-downs for those benchmarks which integration does not help. The dominant effect is an increase in the branch resolution latency which cuts integration's fetch savings. There is an interesting interplay between increased pipelining and integration. On one hand, it lengthens the branch resolution latency, increasing the number of instructions that can be executed along mis-speculated paths. On the other, it slows down the execution of *all* instructions, reducing the completion rate of squashed instructions. The overall effect on the number of integration candidates and integrations is small. Although the effects of pipeline depth increases take away some of integration's performance, such increases are by no means mandatory. The access times of large physical register files can be controlled using techniques like replication [11, 19] or banking [4] and while integration probably requires two-stage register renaming, it should not add stages to already pipelined renaming implementations.

## 4.5 Impact of Base Microarchitecture

One final piece of data we would like to provide is an estimate of the impact of integration for more aggressive microarchitectures. To model a microarchitecture that hopefully represents a next-generation microprocessor, we begin with the organization of our basic 8-way machine. We double the re-ordering capability by doubling the sizes of the instruction and memory ordering buffers; the number of physical registers is increased accordingly. In the memory system, we double the size of the L2 cache to 2 MB and increase the number of simultaneously outstanding misses to 16. To simulate a faster clock, we deepen the pipeline to 5-cycle fetch, 3-cycle decode/rename and 4-cycle register read, lengthen cache array access time to 3 cycles, and slow raw memory access time and the memory bus by 50%. In Figure 4(b), we compare the speedups achieved by our baseline integration configuration (a direct-mapped 256entry IT) when applied to both the current-generation and next-generation microarchitectures.

One trend that is noticeable by its novelty is that, unlike increasing IT size or associativity, a more aggressive microarchitecture *does* increase the impact of integration on programs that do not benefit from it in a more conservative implementation. The reason for this is that a more speculative machine *changes the structural behavior* of the program. Larger re-order buffers that provide more room for speculation and a deeper pipeline that increases the time it takes to discover and resolve branch mis-predictions combine to raise the total number of instructions executed along mis-speculated



**FIGURE 4.** Effect of Increased Pipelining and a More Aggressive Base Microarchitecture on Performance Impact of Integration. Execution time saved using a direct-mapped 256-entry IT for (a) our base microarchitecutre with integration-deepend pipeline and (b) a more aggressive base microarchitecture.

paths. That increases the number of potential integration candidates and, in turn, successful integrations. For example, a larger machine can mis-speculate longer along a conditional arm and is more likely to reach (and squash) the re-convergent region along the mis-speculated path. Our results indicate that between 5% and 50% *more* instructions are integrated in the more aggressive, more-speculative configuration.

The relative increase in the effectiveness of integration is probably larger than a simple increase in integrated instructions can account for. As the graph shows, integration is 50% to 120% more effective in reducing execution time in the aggressive configuration than in the base configuration. Absolute performance improvements for the next-generation micro-architecture are close to or over 10% for several benchmarks. The reason for this boost is that in the more aggressive, more deeply pipelined implementation, the benefit of each integrated instruction is also relatively higher. Specifically, the longer register-read times make integration's ability to collapse dependent chains of instructions more important. The absolute importance of instant branch mis-prediction resolution is also increased by longer register-read times. However, the relative impact of this effect is somewhat mitigated because the depth of the front end increases as well.

# **5** Related Work

The term squash reuse was introduced to describe one of the tasks performed by Instruction Reuse (IR) [18]. IR is a tablebased technique for avoiding the execution of an instruction that has been previously executed with the same inputs. In addition to squash reuse, in which the reused value comes from the same instance of the instruction that has merely been squashed, IR implements general reuse, in which the reused value comes from a different (not necessarily squashed) previous instance that just happens to have the same input operands. Integration implements only squash reuse because it requires that the value already exist in the register file and that the physical register inputs of the squashed instruction match exactly with the inputs of the instruction it will "replace". IR lifts these constraints by storing the squashed value inside the lookup table (which is called a reuse buffer or RB) and writing it into the register file when reuse is detected and by basing the reuse criterion itself is on instance-independent architectural quantities like values or logical register names, rather than instance-dependent micro-architectural ones like physical register numbers. IR is very applicable, it can exploit general reuse and be implemented on any microarchitecture, but has a somewhat complex implementation. A value-based reuse test implies the need to read registers, which not only complicates the register file, but also moves IR further back in the pipeline, reducing its impact. An architectural-name-based reuse test removes the need to read registers but requires an explicit dependence-tracking scheme within the RB so as not to become too conservative. Both IR forms require additional write data-paths into the register file. In integration, the reused values are already stored in physical registers so no additional register data-paths to read or write any values are required. At the same time, the physical-register-based nature of the reuse test implements dependence-tracking naturally.

The *Dynamic Control Independence (DCI)* [2] buffer is another result salvage mechanism that operates in a centralized window environment. The DCI buffer is a shadow re-order buffer whose contents persist past mis-speculation events that invalidate the architectural buffer (this is a familiar theme). Shadow buffer tags and results can be re-used if the instruction proves to be control- and data- independent. Control independent instructions are found by associatively searching the squashed region of the shadow buffer; their data-independent nature is checked using an architectural-name-based invalidation scheme. The DCI buffer is essentially an architectural-name-based implementation of squash re-use similar to IR that uses a shadow re-order buffer rather than an RB.

We have already alluded to the interplay between integration and selective squashing [8, 12, 15, 16], which allows instruction instances to execute multiple times "in-place" before retirement. Selective squashing is an effective way of dealing with data mis-speculations, in which the correct instructions are already in the machine. Selective squashing allows the penalty of squash and re-fetch to be avoided at the cost of keeping instructions in the reservation-station longer and increasing reservation-station contention. Selective squashing, however, cannot salvage work lost to control mis-speculation. Integration and selective squashing are duals. Both techniques salvage instructions by keeping around information for longer than is conventionally required, physical registers for integration and reservation stations for selective squashing. However, while selective squashing actively picks out instructions dependent on the mis-speculation, integration waits for all squashed instructions to be re-processed then picks out the ones that were actually mis-speculation independent.

# 6 Conclusions and Future Work

We present register integration (or just *integration*), a technique for salvaging valid results that have been unavoidably lost due to the sequential nature of speculation and mis-speculation recovery. Integration is a discipline that allows speculative results to remain in the physical register file past recovery events with the hope that they were independent of the misspeculation in question and can be used once the particulars of that mis-speculation have been resolved. Integration logic is implemented as a modification to conventional register renaming that recognizes the validity of squashed results using their data-dependences and spares the processor from having to reexecute the corresponding instructions.

Our initial evaluation shows that integration has the potential for noticeable performance improvements of up to 8% at configurations representative of current-generation processors and up to 11.5% for more aggressive, more speculative, moredeeply pipelined next-generation configurations. These speedups are achieved through a combination of reduction in the consumption of execution and fetch bandwidths, the collapsing of dependent instruction chains, and the acceleration of branch resolution. Our numbers indicate that programs typically are able to reuse between 20% and 60% of all squashed instructions that have completed execution prior to squashing, representing between 1% and 19% of committed instructions.

Perhaps more important than integration's performance characteristics, are its mis-speculation reduction characteristics. In addition to improving performance, integration reduces the overall level of wasted work performed by the processor. It reduces the number of instructions executed by re-using squashed computations and its acceleration of branch resolution reduces the number of instructions fetched along misspeculated paths. According to our results, the number of instruction fetches saved can reach 6% and the number of instruction executions saved, 15%. Both of these numbers grow relatively as the underlying micro-architecture becomes more aggressive. These characteristics make integration an interesting candidate for reducing dynamic-power and energy and also suggest its use in reducing resource contention in simultaneously multi-threaded (SMT) processors.

The implementation of integration is simple, requiring only an integration table (IT), a small cache-like structure with limited content-addressible capabilities and an integration circuit, which is added to the register renaming logic. No changes to either the fetch or execution engines themselves are necessary and integration does not require the reading or writing of any register values, only map table manipulations are used. The performance improvements we present are all achievable with the minimal complexity implementation of integration.

Future work in the area of integration includes a more thorough evalutation of the IT design space, experiments with more varied benchmarks, and a more detailed investigation into the interaction of different micro-architectural parameters with integration. A study of the high-level characteristics of programs that draw benefit from integration is also interesting. We have mentioned possibility for interesting synergy between integration and selective squashing; that possibility needs further investigation. The power aspects of integration and its potential use as a power-reduction technique are also subjects of open research.

The most interesting future direction for integration lies in its ability to support new speculation models. As we have presented it, integration is a mechanism that can re-impose lost sequential semantics on a set of instructions using only their data-dependences. The real power of integration, however, may be in its ability to impose such semantics on a set of instructions that were not executed sequentially in the first place. Integration enables a new form of speculation, datadriven speculation, in which speculative execution proceeds along statically annotated data-dependence arcs with no regards to sequencing. Integration is used subsequently to sequence the results into a control-driven sequential form required by the architectural interface. In fact, integration was invented during the course of our investigation into a new form of speculative multithreading called speculative data-driven multithreading (DDMT) [17].

#### Acknowledgements

This work was supported in part by National Science Foundation grants MIP-9505853 and CCR-9900584, donations from Intel Corp. and Sun Microsystems, the University of Wisconsin Graduate School and an Intel Ph.D Fellowship. The authors thank the anonymous referees for their reviews.

#### References

- D. Burger and T. Austin. The SimpleScalar Tool Set, Version 2.0. Technical Report CS-TR-97-1342, University of Wisconsin-Madison, Jun. 1997.
- [2] Y. Chou, J. Fung, and J. Shen. Reducing Branch Misprediction Penalties via Dynamic Control Independence Detection. In *Proc. 1999 International Conference on Supercomputing*, pages 109–118, Jun. 1999.

- [3] G. Chrysos and J. Emer. Memory Dependence Prediction using Store Sets. In *Proc. 25th International Sympo*sium on Computer Architecture, pages 142–153, Jun. 1998.
- [4] J.-L. Cruz, A. Gonzalez, M. Valero, and N. Topham. Multiple-Banked Register File Architectures. In Proc. 27th Annual International Symposium on Computer Architecture, pages 316–325, Jun. 2000.
- [5] K. Diefendorf. K7 Challenges Intel. *Microprocessor Report*, 12(14), Nov. 1998.
- [6] K. Diefendorf. Compaq Chooses SMT for Alpha. *Microprocessor Report*, 13(16), Dec. 1999.
- [7] K. Diefendorf. HAL Makes SPARCS Fly. *Microprocessor Report*, 13(5), Nov. 1999.
- [8] M. Franklin. *The Multiscalar Architecture*. PhD thesis, University of Wisconsin-Madison, Madison, WI 53706, Nov. 1993.
- [9] P. Glaskowsky. Pentium 4 (Partially) Previewed. *Micro-processor Report*, 14(8), Aug. 2000.
- [10] L. Gwenapp. Intel's P6 Uses Decoupled Superscalar Design. *Microprocessor Report*, 9(2), Feb. 1995.
- [11] R. Kessler. The Alpha 21264 Microprocessor. *IEEE Micro*, 19(2), Mar./Apr. 1999.
- [12] M. Lipasti. Value Locality and Speculative Execution. PhD thesis, Department of Electrical and Computer Engineering, Carnegie-Mellon University, May 1997.
- [13] S. Manne, A. Klauser, and D. Grunwald. Pipeline Gating: Speculation Control for Energy Reduction. In Proc. 25th Annual International Symposium on Computer Architecture, pages 132–141, Jun. 1998.
- [14] A. Moshovos and G. Sohi. Memory Dependence Speculation Tradeoffs in Centralized, Continuous-Window Superscalar Processors. In Proc. 6th Annual International Symposium on High-Performance Computer Architecture, pages 301–312, Feb. 2000.
- [15] E. Rotenberg, Q. Jacobson, Y. Sazeides, and J. Smith. Trace Processors. In *Proc. 30th International Symposium* on *Microarchitecture*, pages 138–148, Dec. 1997.
- [16] E. Rotenberg and J. Smith. Control Independence in Trace Processors. In *Proc. 32nd International Sympo*sium on Microarchitecture, pages 4–15, Nov. 1999.
- [17] A. Roth and G. Sohi. Speculative Data-Driven Multithreading. In *Proc. 7th International Symposium on High-Performance Computer Architecture (to appear)*, Jan. 2001.
- [18] A. Sodani and G. S. Sohi. Dynamic Instruction Reuse. In Proc. 24th International Symposium on Computer Architecture, pages 194–205, Jun 1997.
- [19] P. Song. IBM's Power3 to Replace P2SC. Microprocessor Report, 11(15), Nov. 1997.
- [20] D. M. Tullsen, S. J. Eggers, and H. M. Levy. Simultaneous Multithreading: Maximizing On-Chip Parallelism. In Proc. 22nd International Symposium on Computer Architecture, pages 392–403, Jun. 1995.
- [21] K. Yeager. The MIPS R10000 Superscalar Microprocessor. *IEEE Micro*, Apr. 1996.
- [22] A. Yoaz, M. Erez, R. Ronen, and S. Jourdan. Speculation Techniques for Improving Load-Related Instruction Scheduling. In Proc. 26th Annual International Symposium on Computer Architecture, pages 42–53, May 1999.