



UC Berkeley  
Teaching Professor  
Dan Garcia

# CS61C

## Great Ideas in Computer Architecture (a.k.a. Machine Structures)



UC Berkeley  
Lecturer  
Justin Yokota

## Intro to RISC-V

# The RISC-V Instruction Set Architecture

- The RISC-V Instruction Set Architecture
- Elements of Architecture:  
Registers
- Add/Sub Instructions
- Immediates

# Great Idea #1: Abstraction (Levels of Representation/Interpretation)



**High Level Language  
Program (e.g., C)**

```
temp = v[k];
v[k] = v[k+1];
v[k+1] = temp;
```

*| Compiler*  
**Assembly Language  
Program (e.g., RISC-V)**

```
lw x3, 0(x10)
lw x4, 4(x10)
sw x4, 0(x10)
sw x3, 4(x10)
```

Anything can be represented  
as a number,  
i.e., data or instructions

*| Assembler*  
**Machine Language  
Program (RISC-V)**

```
1000 1101 1110 0010 0000 0000 0000 0000
1000 1110 0001 0000 0000 0000 0000 0100
1010 1110 0001 0010 0000 0000 0000 0000
1010 1101 1110 0010 0000 0000 0000 0100
```

**Hardware Architecture Description  
(e.g., block diagrams)**

*| Architecture Implementation*

**Logic Circuit Description  
(Circuit Schematic Diagrams)**



# Assembly Language

- Basic job of a CPU: execute lots of instructions
- Instructions are the primitive operations that the CPU may execute
  - Like a sentence: *operations* (verbs) applied to *operands* (objects), processed in sequence ...
- Different CPUs implement different sets of instructions. The set of instructions a particular CPU implements is an Instruction Set Architecture (ISA)
  - Examples: ARM (cell phones), Intel x86 (i9, i7, i5, i3), IBM/Motorola PowerPC (old Macs), MIPS, RISC-V, ...



# Book: Programming From the Ground Up

“A new book was just released which is based on a new concept – teaching computer science through assembly language (Linux x86 assembly language, to be exact). This book teaches how the machine itself operates, rather than just the language. I've found that the key difference between mediocre and excellent programmers is whether or not they know assembly language. Those that do tend to understand computers themselves at a much deeper level. Although [almost!] unheard of today, this concept isn't really all that new -- there used to not be much choice in years past. Apple computers came with only BASIC and assembly language, and there were books available on assembly language for kids. This is why the old-timers are often viewed as 'wizards': they had to know assembly language programming.”

-- *slashdot.org comment, 2004-02-05*



# Instruction Set Architectures

- Early trend was to add more and more instructions to new CPUs to do elaborate operations
  - VAX architecture had an instruction to multiply polynomials!
- Reduced Instruction Set Computer (RISC) philosophy:
  - Cocke IBM, Patterson, Hennessy, 1980s.
  - Keep the instruction set small and simple, makes it easier to build *fast* hardware
  - Let software do complicated operations by composing simpler ones
  - This went against the conventional wisdom of the time  
(he who laughs last, laughs best)

# Patterson and Hennessy win Turing!



<https://engineering.berkeley.edu/news/2018/06/patterson-wins-turing-award/> Garcia, Yokota



# RISC-V Architecture

IBM 360 Green Card

- New open-source, license-free ISA spec
  - Supported by growing shared software ecosystem
  - Appropriate for all levels of computing system, from microcontrollers to supercomputers
  - 32-bit, 64-bit, and 128-bit variants (class/textbook uses 32-bit)
- Why RISC-V instead of Intel x86-64?
  - RISC-V is simple and elegant. We don't need to get bogged down in gritty details
  - RISC-V has exponential adoption, from microcontrollers to CPUs to warehouse-scale computers

<https://cs61c.org/> → Resources

07-Intro to RISC-V (8)

| RISC-V System/360 Reference Data                       |                                                                  |      |  |  |  |
|--------------------------------------------------------|------------------------------------------------------------------|------|--|--|--|
| RV32 BASE INTEGRAL INSTRUCTIONS, in alphabetical order |                                                                  |      |  |  |  |
| MNEMONIC                                               | DESCRIPTION (in Verilog)                                         | NOTE |  |  |  |
| addi,addiu                                             | R ADDI (Word) R[d0] = R[r1] + imm[11:0]                          | 1)   |  |  |  |
| and                                                    | R AND R[d0] = R[r1] & R[r2]                                      | 1)   |  |  |  |
| andi                                                   | I AND Immediate R[d0] = R[r1] & imm[11:0]                        | 1)   |  |  |  |
| auipc                                                  | S Branch EQEqual I[R[R[r1]] - R[r2]]                             |      |  |  |  |
| beq                                                    | S Branch EQEqual I[R[R[r1]] - R[r2]]                             |      |  |  |  |
| bge                                                    | SB Branch Greater than or Equal I[R[R[r1]] >= R[r2]]             |      |  |  |  |
| bgeu                                                   | SB Branch > Unsigned I[R[R[r1]] > R[r2]]                         |      |  |  |  |
| blt                                                    | SB Branch Less Than I[R[R[r1]] < R[r2]]                          |      |  |  |  |
| bltu                                                   | SB Branch Less Than Unsigned I[R[R[r1]] < R[r2]] PC-PC+imm[11:0] | 2)   |  |  |  |
| beqz                                                   | S Branch Not Equal I[R[R[r1]] & R[r2]] PC-PC+imm[11:0]           | 2)   |  |  |  |
| ecrcrd                                                 | I Cont.Stat.RegRead&Clear R[d0] = CSR.CSR & ~CSR & ~imm          |      |  |  |  |
| ecrrs                                                  | I Cont.Stat.RegRead&Set R[d0] = CSR.CSR & R[r1]                  |      |  |  |  |
| ecrrsel                                                | I Cont.Stat.RegRead&Set R[d0] = CSR.CSR   imm                    |      |  |  |  |
| ecrrw                                                  | I Cont.Stat.RegRead&Write R[d0] = CSR.CSR   R[r1]                |      |  |  |  |
| ecrrwr                                                 | I Cont.Stat.RegRead&Write R[d0] = CSR.CSR   imm                  |      |  |  |  |
| break                                                  | I Environment BREAK Transfer control to debugger                 |      |  |  |  |
| call                                                   | I Environment CALL Transfer control to operating system          |      |  |  |  |
| fence                                                  | I Synch thread Synchronizes threads                              |      |  |  |  |
| fence.i                                                | I Synch Inst & Data Synchronizes writes to instruction           |      |  |  |  |
| jal                                                    | U Jump & Link R[d0] = PC[4:1] + imm[11:0]                        |      |  |  |  |
| jalr                                                   | I Jump & Link Register R[d0] = R[r1] + imm[11:0]                 |      |  |  |  |
| lb                                                     | I Load Byte R[d0] = M[R[r1]+imm[31:0]]                           |      |  |  |  |
| ld                                                     | I Load Halfword Unsigned R[d0] = (15 M[R[r1]+imm[31:0]])         |      |  |  |  |
| ld                                                     | I Load Doubleword R[d0] = M[R[r1]+imm[63:0]]                     |      |  |  |  |
| lh                                                     | I Load Halfword R[d0] = (48 M[R[r1]+imm[15:0]])                  |      |  |  |  |
| lbu                                                    | I Load Halfword Unsigned R[d0] = (48 O[M[R[r1]+imm[15:0]]])      |      |  |  |  |
| lu                                                     | I Load Upper Immediate R[d0] = (32 imm[31:16])                   |      |  |  |  |
| lw                                                     | I Load Word Unsigned R[d0] = (32 M[R[r1]+imm[31:0]])             |      |  |  |  |
| or                                                     | R OR Immediate R[d0] = R[r1]   R[r2]                             |      |  |  |  |
| ori                                                    | R OR Immediate R[d0] = R[r1]   imm                               |      |  |  |  |
| sb                                                     | S Store Byte M[R[r1]+imm[7:0]] = R[r2][7:0]                      |      |  |  |  |
| sd                                                     | S Store Doubleword M[R[r1]+imm[63:0]] = R[r2][63:0]              |      |  |  |  |
| sh                                                     | S Store Halfword M[R[r1]+imm[15:0]] = R[r2][15:0]                |      |  |  |  |
| sll                                                    | R Shift Left (Word) R[d0] = R[r1] << R[r2]                       |      |  |  |  |
| slli                                                   | R Shift Left Immediate (Word) R[d0] = R[r1] << imm[11:0]         |      |  |  |  |
| srl                                                    | R Shift Right (Word) R[d0] = R[r1] >> R[r2]                      | 1)   |  |  |  |
| srlt                                                   | R Set Less Than Immediate R[d0] = (R[r1] < imm) ? 1 : 0          | 1)   |  |  |  |
| srltu                                                  | R Set Less Than Unsigned R[d0] = (R[r1] < imm) ? 1 : 0           | 2)   |  |  |  |
| srltiu                                                 | R Set Less Than Unsigned R[d0] = (R[r1] < imm) ? 1 : 0           | 2)   |  |  |  |
| sra                                                    | R Shift Right Arithmetic (Word) R[d0] = R[r1] >> imm             | 1,5) |  |  |  |
| sra                                                    | R Shift Right Arithmetic (Word) R[d0] = R[r1] >> imm             | 1,5) |  |  |  |
| srl                                                    | R Shift Right (Word) R[d0] = R[r1] >> R[r2]                      | 1)   |  |  |  |
| srlt                                                   | R Shift Right Immediate (Word) R[d0] = R[r1] >> imm              | 1)   |  |  |  |
| sub,subw                                               | R SUBtract (Word) R[d0] = R[r1] - R[r2]                          | 1)   |  |  |  |
| sw                                                     | S Store Word M[R[r1]+imm[31:0]] = R[r2][31:0]                    |      |  |  |  |
| xor                                                    | R XOR R[d0] = R[r1] ^ R[r2]                                      |      |  |  |  |
| not                                                    | R NOT R[d0] = R[r1] ^ imm                                        |      |  |  |  |

| ARITHMETIC CORE INSTRUCTION SET           |                                  |                                                            |      |  |  |
|-------------------------------------------|----------------------------------|------------------------------------------------------------|------|--|--|
| Reference Data                            |                                  |                                                            |      |  |  |
| MNEMONIC                                  | FMT NAME                         | DESCRIPTION (in Verilog)                                   | NOTE |  |  |
| mul,mulw                                  | R MULy (Word)                    | R[d0] = R[r1]*R[r2]                                        | 1)   |  |  |
|                                           | R MULy upper Half                | R[d0] = (R[r1]*R[r2])[17:0]                                |      |  |  |
|                                           | R MULy lower Half Sign UnSigned  | R[d0] = R[r1]*R[r2]                                        | 6)   |  |  |
|                                           | R MULy lower Half Unsigned       | R[d0] = (R[r1]*R[r2])[17:0]                                | 2)   |  |  |
|                                           | div,divw                         | R DIVide (Word)                                            |      |  |  |
|                                           | R DIVide Unsigned                | R[d0] = R[r1]/R[r2]                                        | 1)   |  |  |
|                                           | R REMander (Word)                | R[d0] = (R[r1]-R[r2])&1                                    | 2)   |  |  |
|                                           | R REMander Unsigned (Word)       | R[d0] = (R[r1]-R[r2])&1                                    | 1)   |  |  |
|                                           | load,loadw                       | R Load (Word)                                              |      |  |  |
|                                           | S Load (Word)                    | S[d0] = M[R[r1]]                                           |      |  |  |
|                                           | store,storew                     | R Store (Word)                                             |      |  |  |
|                                           | S Store (Word)                   | M[R[r1]] = S[d0]                                           |      |  |  |
| RV44F and RV64D Floating Point Extensions |                                  |                                                            |      |  |  |
|                                           | Load (Word)                      | F[d0] = F[r1]                                              |      |  |  |
|                                           | Store (Word)                     | M[R[r1]] = F[d0]                                           |      |  |  |
|                                           | Add                              | R ADD F[d0] = F[r1]+F[r2]                                  | 1)   |  |  |
|                                           | Subtract                         | R SUBtract F[d0] = F[r1]-F[r2]                             | 7)   |  |  |
|                                           | Multiply                         | R MULy F[d0] = F[r1]*F[r2]                                 | 7)   |  |  |
|                                           | Divide                           | R DIVide F[d0] = F[r1]/F[r2]                               | 7)   |  |  |
|                                           | Negative Multiply-SUBtract       | R NEGATE MULy F[d0] = -(F[r1]*F[r2])                       | 7)   |  |  |
|                                           | Negative Multiply-ADD            | R NEGATE MULy ADD F[d0] = -(F[r1]*F[r2])+F[r3]             | 7)   |  |  |
|                                           | SQRT                             | R SQRT F[d0] = SQRT(F[r1])                                 | 7)   |  |  |
|                                           | NEGATE ADD                       | R NEGATE ADD F[d0] = -(F[r1]+F[r2])                        | 7)   |  |  |
|                                           | MULy ADD                         | R MULy ADD F[d0] = F[r1]*F[r2]+F[r3]                       | 7)   |  |  |
|                                           | MULy SUBtract                    | R MULy SUBtract F[d0] = F[r1]*F[r2]-F[r3]                  | 7)   |  |  |
|                                           | Negative MULy ADD                | R NEGATE MULy ADD F[d0] = -(F[r1]*F[r2])+F[r3]             | 7)   |  |  |
|                                           | Negative SQRT                    | R NEGATE SQRT F[d0] = SQRT(-F[r1])                         | 7)   |  |  |
|                                           | NEGATION                         | R NEGATION F[d0] = ~F[r1]                                  | 7)   |  |  |
|                                           | NEGATION SOURCE                  | R NEGATION SOURCE F[d0] = ~F[r1]&F[r2]                     | 7)   |  |  |
|                                           | MINIMUM                          | R MINIMUM F[d0] = F[r1]&F[r2]                              | 7)   |  |  |
|                                           | MAXIMUM                          | R MAXIMUM F[d0] = F[r1] F[r2]                              | 7)   |  |  |
|                                           | COMPARE FLOAT EQUAL              | R COMPARE FLOAT EQUAL F[d0] = F[r1]==F[r2]                 | 7)   |  |  |
|                                           | COMPARE FLOAT LESS THAN          | R COMPARE FLOAT LESS THAN F[d0] = F[r1]<F[r2]              | 7)   |  |  |
|                                           | COMPARE FLOAT LESS THAN OR EQUAL | R COMPARE FLOAT LESS THAN OR EQUAL F[d0] = F[r1]<=F[r2]    | 7)   |  |  |
|                                           | CLASSIFY TYPE                    | R CLASSIFY TYPE F[d0] = CLASSIFY(F[r1])                    | 7,8) |  |  |
|                                           | MOVE FROM IMAGE                  | R MOVE FROM IMAGE F[d0] = MOVE(F[r1],F[r2],F[r3])          | 7)   |  |  |
|                                           | MOVE TO IMAGE                    | R MOVE TO IMAGE F[d0] = MOVE(F[r1],F[r2],F[r3])            | 7)   |  |  |
|                                           | CONVERT FROM FP TO SP            | R CONVERT FROM FP TO SP F[d0] = CONV(FP2SP)[31:0]          | 7)   |  |  |
|                                           | CONVERT FROM SP TO FP            | R CONVERT FROM SP TO FP F[d0] = CONV(SP2FP)[31:0]          | 7)   |  |  |
|                                           | CONVERT 32B INTEGER              | R CONVERT 32B INTEGER F[d0] = CONV(INT32)[31:0]            | 7)   |  |  |
|                                           | CONVERT 64B INTEGER              | R CONVERT 64B INTEGER F[d0] = CONV(INT64)[63:0]            | 7)   |  |  |
|                                           | CONVERT 32B FLOAT                | R CONVERT 32B FLOAT F[d0] = CONV(FP32)[31:0]               | 7)   |  |  |
|                                           | CONVERT 64B FLOAT                | R CONVERT 64B FLOAT F[d0] = CONV(FP64)[63:0]               | 7)   |  |  |
|                                           | CONVERT 32B DOUBLE               | R CONVERT 32B DOUBLE F[d0] = CONV(DOUBLE)[31:0]            | 2,7) |  |  |
|                                           | CONVERT 64B DOUBLE               | R CONVERT 64B DOUBLE F[d0] = CONV(DOUBLE)[63:0]            | 2,7) |  |  |
|                                           | CONVERT 32B INTEGER              | R CONVERT 32B INTEGER F[d0] = CONV(INT32)[31:0]            | 2,7) |  |  |
|                                           | CONVERT 64B INTEGER              | R CONVERT 64B INTEGER F[d0] = CONV(INT64)[63:0]            | 2,7) |  |  |
|                                           | CONVERT 32B FLOAT                | R CONVERT 32B FLOAT F[d0] = CONV(FP32)[31:0]               | 2,7) |  |  |
|                                           | CONVERT 64B FLOAT                | R CONVERT 64B FLOAT F[d0] = CONV(FP64)[63:0]               | 2,7) |  |  |
|                                           | CONVERT 32B DOUBLE               | R CONVERT 32B DOUBLE F[d0] = CONV(DOUBLE)[31:0]            | 2,7) |  |  |
|                                           | CONVERT 64B DOUBLE               | R CONVERT 64B DOUBLE F[d0] = CONV(DOUBLE)[63:0]            | 2,7) |  |  |
|                                           | MINIMUM                          | R MINIMUM F[d0] = MIN(F[r1],F[r2])                         | 9)   |  |  |
|                                           | MAXIMUM                          | R MAXIMUM F[d0] = MAX(F[r1],F[r2])                         | 9)   |  |  |
|                                           | MAXIMUM UNSIGNED                 | R MAXIMUM UNSIGNED F[d0] = MAX(F[r1],F[r2])                | 2,9) |  |  |
|                                           | MINIMUM UNSIGNED                 | R MINIMUM UNSIGNED F[d0] = MIN(F[r1],F[r2])                | 9)   |  |  |
|                                           | SWAP                             | R SWAP F[d0] = SWAP(F[r1],F[r2])                           | 9)   |  |  |
|                                           | XOR                              | R XOR F[d0] = XOR(F[r1],F[r2])                             | 9)   |  |  |
|                                           | LOAD RESERVED                    | R LOAD RESERVED F[d0] = LOAD_RESERVED(F[r1],F[r2])         | 9)   |  |  |
|                                           | STORE CONDITIONAL                | R STORE CONDITIONAL F[d0] = STORE_CONDITIONAL(F[r1],F[r2]) | 9)   |  |  |

RISC-V Green Card



# RISC-V Origins

- Started in Summer 2010 to support open research and teaching at UC Berkeley
  - Lineage can be traced to RISC-I/II projects (1980s)
- As the project matured, it migrated to RISC-V foundation ([www.riscv.org](http://www.riscv.org))
- Many commercial and research projects based on RISC-V
  - Open-source and proprietary
  - Widely used in education
- Read more:
  - <https://riscv.org/risc-v-history/>
  - <https://riscv.org/risc-v-genealogy/>



RISC-I



RISC-II



Garcia, Yokota



# Elements of Architecture: Registers

- The RISC-V Instruction Set Architecture
- Elements of Architecture: Registers
- Add/Sub Instructions
- Immediates

# Instruction Set

Preliminary discussion of the logical design of an electronic computing instrument<sup>1</sup>

Arthur W. Burks / Herman H. Goldstine /  
John von Neumann

“instruction sets”

3.1. It is easy to see by formal-logical methods that there exist codes that are *in abstracto* adequate to control and cause the execution of any sequence of operations which are individually available in the machine and which are, in their entirety, conceivable by the problem planner. The really decisive considerations from the present point of view, in selecting a code, are more of a practical nature: simplicity of the equipment demanded by the code, and the clarity of its application to the actually important problems together with the speed of its handling of those problems. It would take us much too far afield to discuss these questions at all generally or from first principles. We will therefore restrict ourselves to analyzing only the type of code which we now envisage for our machine.

- Instruction set for a particular architecture (e.g. RISC-V) is represented by the assembly language
- Each line of assembly code represents one instruction for the computer

add                     $x_1, x_2, x_3$   
                        \underbrace{\hspace{1cm}} \quad \underbrace{\hspace{1cm}} \quad \underbrace{\hspace{1cm}}  
                        operation name                    registers

# Higher-Level Language vs. Assembly Language (1/2)

## C, Java

- In C (and most high-level languages), variables declared first and given a type
  - int fahr, celsius;  
char a, b, c, d, e;
- Each variable can ONLY represent a value of the type it was declared as (cannot mix/match int/char variables)
- In high-level languages, variable types determine operation
  - int \*p = ...;  
p = p + 2;  
int x = 42;  
x = 3 \* x;

## RISC-V

- Assembly operands are registers
  - Registers are limited number of special locations, built directly into the hardware
  - RISC-V: Operations can only be performed on register operands
- In assembly language, the registers have no type
  - Register contents are just bits
- Operation determines “type,” i.e., how register contents are treated
  - E.g., as value, memory address, etc.

# Registers are Inside the Processor



**Benefit:**  
Since registers are directly in hardware, they're very fast (faster than 0.25ns)

- Speed of light:  
 $3 \times 10^8 \text{ m/s} = 0.3 \text{ m/ns} = 30 \text{ cm/ns} = 10 \text{ cm}/0.3 \text{ ns}!!!$
- 0.3ns is the clock period of a 3.33GHz computer

# Great Idea #3: Principle of Locality / Memory Hierarchy

Processor  
chip



Extremely fast  
Extremely expensive  
Tiny capacity



Garcia, Yokota



# 32 Registers in RISC-V

- **Drawback: Each ISA a predetermined number of registers**
  - Why? Registers are built into hardware
  - Solution: RISC-V code must be very carefully put together to efficiently use registers
    - Nowadays: compiler maps C variables to RISC-V registers across declarations, function calls, etc.
- **32 registers in RISC-V**
  - Why 32? Smaller is faster, but too small is bad
  - Goldilocks principle (“This porridge is too hot; This porridge is too cold; this porridge is just right”)
- **Each RISC-V register is 32 bits wide (in RV32 variant)**
  - Groups of 32 bits called a word in RV32

# Register Names and Numbers

- Registers are numbered from 0 to 31
  - Referred to by number  $x_0 - x_{31}$
- $x_0$  is special, always holds value zero
  - So only 31 registers able to hold variable values
- Each register can be referred to by number or name
  - We'll explain names next week
- Ok, enough already...gimme my RV32!!!

# Add/Sub Instructions

- The RISC-V Instruction Set Architecture
- Elements of Architecture:  
Registers
- Add/Sub Instructions
- Immediates

# Higher-Level Language vs. Assembly Language (2/2)

C, Java

- **Each line can contain multiple operations**

```
a = b * 2 - (arr[2] + *p);
```

- **Common operations are:**

- `=, +, -, *, /`
- Here, `*` includes both multiply and dereference

RISC-V

- **Each statement is an instruction**
  - An instruction executes exactly one short list of simple commands
- **Each line of assembly code contains at most 1 instruction**

|     |            |
|-----|------------|
| add | x1, x2, x3 |
| sub | x1, x1, x4 |

The compiler translates from higher-level language down to assembly...  
...so RISC-V instructions are often closely related to common C/Java operations.

# RISC-V Arithmetic Instruction Syntax



- **Syntax is rigid:**
  - 1 operator, 3 operands
  - Why? Keep hardware simple via regularity ([more in a few weeks](#))

# RISC-V Addition and Subtraction

| opname                | rd,                                        | rs1,                                      | rs2                                       |
|-----------------------|--------------------------------------------|-------------------------------------------|-------------------------------------------|
| operation,<br>by name | "destination"<br>operand<br>getting result | "source1"<br>1st operand<br>for operation | "source2"<br>2nd operand<br>for operation |

## ▪ Addition

RISC-V    add x1, x2, x3  
C      a = b + c;



## ▪ Subtraction

RISC-V    sub x4, x5, x6  
C      d = e - f;  
(remember!  
order of operands  
matter in sub)

# RISC-V Arithmetic, Example 1

- How could we translate this C statement?

```
a = b + c + d - e;
```

x10 x1 x2 x3 x4

- Assume the above mapping for C variables  $\leftrightarrow$  RISC-V registers.

- Solution: Break into multiple instructions!

```
add x10, x1, x2
add x10, x10, x3
sub x10, x10, x4
```

```
# a_temp = b + c
# a_temp = a_temp + d
# a = a_temp - e
```

A single line of C may break up into several lines of RISC-V.

In-line comments prefixed by #, work like C99's // .  
Unlike C, no multi-line comment /\* \*/ support.

# RISC-V Arithmetic, Example 2

- How do we do this?

$$f = (g + h) - (i + j);$$

x19    x20    x21    x22    x23    x5    x6

- Assume the above mapping for C variables  $\leftrightarrow$  RISC-V registers.

- Solution: Use intermediate temporary registers!

```
add x5, x20, x21          # a_temp = g + h
add x6, x22, x23          # b_temp = i + j
sub x19, x5, x6           # f = (g + h) - (i + j)
```

- Note: A good compiler may actually do the following. Why?

$$f = g + h - i - j; \quad \text{Can use zero temp registers!}$$

# Aside: Apollo Guidance Computer

- Margaret Hamilton  
Director of Software Engineering  
Division of MIT Instrumentation Laboratory
  - Built Apollo Guidance Computer Command Module (navigation, lunar landing)
    - 72KB of memory on board *Eagle* (Apollo 11, 1969)

```

179      TC  BANKCALL    # TEMPORARY, I HOPE HOPE HOPE
180      CADR STOPRATE   # TEMPORARY, I HOPE HOPE HOPE
181      TC  DOWNFLAG   # PERMIT X-AXIS OVERRIDE
  
```

```

245      CAF  CODE500    # ASTRONAUT: PLEASE CRANK THE
246      TC   BANKCALL   #
247      CADR GOPERF1   SILLY THING AROUND
248      TCF  GOTOP00H   # TERMINATE
249      TCF  P63SPOT3   # PROCEED SEE IF HE'S LYING
250
251 P63SPOT4  TC  BANKCALL  # ENTER      INITIALIZE LANDING RADAR
252      CADR SETPOS1
253
254      TC   POSTJUMP   # OFF TO SEE THE WIZARD ...
255      CADR BURNBABY
  
```

39 ## It traces back to 1965 and the Los Angeles riots, and was inspired  
 40 ## by disc jockey extraordinaire and radio station owner Magnificent Montague.  
 41 ## Magnificent Montague used the phrase "Burn, baby! BURN!" when spinning the  
 42 ## hottest new records. Magnificent Montague was the charismatic voice of  
 43 ## soul music in Chicago, New York, and Los Angeles from the mid-1950s to  
 44 ## the mid-1960s.



Margaret Hamilton and  
Apollo code  
Wikimedia Commons



Presidential  
Medal of  
Freedom  
(2016)

# Immediates

- The RISC-V Instruction Set Architecture
- Elements of Architecture: Registers
- Add/Sub Instructions
- Immediates

# Immediates

- **Immediates are numerical constants.**
  - They appear often in code; hence, they have separate instructions.
- **Add Immediate instruction:**

RISC-V      addi x3, x4, 10

C      f = g + 10;

Syntax similar to add instruction. But the *last operand must be a number, not a register.*

# Immediates

- Immediates are numerical constants.
  - They appear often in code; hence, they have separate instructions.
- Add Immediate instruction:

RISC-V      addi x3, x4, 10  
C      f = g + 10;

Syntax similar to add instruction. But the *last operand* must be *a number*, not a register.

- There is no Subtract Immediate instruction in RISC-V. Why?

- While there are add and sub instructions, to subtract an immediate, just use addi:

RISC-V      addi x3, x4, -10  
C      f = g - 10;

**RISC Philosophy: Reduce** the possible types of operations to an absolute minimum. If an operation can be decomposed into a simpler operation, don't include it in the ISA.

# Register Zero

- One particular immediate, the number zero (**0**), appears very often in code.
- RISC-V hardwires the register zero (**x0**) to value **0**:

RISC-V    add    x3, x4, **x0**  
C      f = g;



RISC-V    addi x3, **x0**, 0xff  
C      f = 0xff;



- Defined in hardware, so an instruction add x0, x3, x4 will not do anything!

# Concept Check: True or False?

1. Types are associated with declaration in C (normally), but are associated with instructions (operators) in RISC-V.
2. Since there are only 32 registers, we can't write RISC-V for C expressions that contain > 32 variables.
3. If p (stored in x9) were a pointer to an array of ints, then p++; would be

addi x9 x9 1

123

- A. FFF
- B. FFT
- C. FTF
- D. FTT
- E. TFF
- F. TFT
- G. TTF
- H. TTT

Start the presentation to see live content. For screen share software, share the entire screen. Get help at [pollev.com/app](https://pollev.com/app)

# Concept Check: True or False?

1. Types are associated with declaration in C (normally), but are associated with instructions (operators) in RISC-V.
2. Since there are only 32 registers, we can't write RISC-V for C expressions that contain > 32 variables.
3. If p (stored in x9) were a pointer to an array of ints, then p++; would be

addi x9 x9 1

1. True, we saw that on an earlier slide.
2. False. we saw how to break up longer equations to smaller ones already.
3. False. Don't forget that ints are (usually) 4 bytes wide, so instruction would be addi x9, x9, 4.

123

- A. FFF  
B. FFT  
C. FTF  
D. FTT  
E. TFF  
F. TFT  
G. TTF  
H. TTT

Garcia, Yokota

