

### Computer Systems and Networks

ECPE 170 – University of the Pacific

# **MIPS** Assembly

# Lab Schedule

#### Activities

#### **This Week**

- MIPS discussion
- Practice problems (whiteboard)
- Using the QtSPIM simulator

#### **Assignments Due**

- **Tuesday Nov 5**<sup>th</sup>
  - **7** Lab 8 due by 11:59pm
- Tuesday Nov 12<sup>th</sup>
  - **7** Lab 9 due by 11:59pm

# Person of the Day – John Cocke



- Computer architecture pioneer 7
  - "Father of RISC Architecture" 7
  - Developed IBM 801 processor, 7 1975-1980
- Winner, ACM Turing Award, 1987

**RISC = Reduced Instruction Set Computing** 

Achieve higher performance with simple instructions that execute faster

# Person of the Day – John Hennessy



- Computer architecture pioneer
- Popularized RISC architecture in early 1980's
- Founder of MIPS Computer Systems in 1984
- Currently president of an obscure school: Stanford University

# Class to Date



# **Class Now**







7

### MIPS Overview

- Family of computer processors first introduced in 1981
- Microprocessor without Interlocked Pipeline Stages
  - Original acronym
  - ↗ Now MIPS stands for nothing at all...

# MIPS Products

#### Embedded devices

- Cisco/Linksys routers
- Cable boxes
- MIPS processor is buried inside System-on-a-Chip (SOC)
- - Nintendo 64
  - Playstation 2, PSP
- Computers?
  - **7** Not so much anymore...
  - **↗** SGI / DEC / NEC workstations back in 1990's

# **MIPS** Design

#### RISC – What does this mean?

- **R**educed Instruction Set Computing
- Simplified design for instructions
- Use more instructions to accomplish same task
- 32 bits (originally) What does this mean?
  - ↗ 1 "word" = 32 bits
  - **↗** Size of data processed by an integer add instruction
  - New(er) MIPS64 design is 64 bits, but we won't focus on that

# MIPS Assembly Programming

### Quotes – Donald Knuth



"People who are more than casually interested in computers should have at least **some idea of what the underlying hardware is like**. Otherwise the programs they write will be pretty weird." – Donald Knuth

This is your motivation in Labs 10 and 11!

# Why Learn Assembly Programming?

#### **Computer Science** track

- Understand capabilities (and limitations) of physical machine
- Ability to optimize program performance (or functionality) at the assembly level *if necessary*

#### Computer Engineer track

- Future courses (e.g. ECPE 173) will focus on processor design
- Start at the assembly programming level and move into hardware
  - How does the processor implement the add instruction?
  - How does the processor know what data to process?

### Instruction Set Architecture

- Instruction Set Architecture (ISA) is the interface between hardware and software
  - **7** Specifies the format of processor instructions
  - Specifies the format of memory addresses (and addressing modes)
  - Specifies the primitive operations the processor can perform

### Instruction Set Architecture

- ISA is the "contract" between the hardware designer and the assembly-level programmer
- Documented in a manual that can be hundreds or thousands of pages long
  - Example: Intel 64 and IA-32 Architectures Software Developers Manual
  - http://www.intel.com/content/www/us/en/processors/ar chitectures-software-developer-manuals.html
  - No joke the manual PDF from August 2012 is <u>3020 pages long</u>!
    - Affordable Care Act ("Obamacare") is only 906 pages!

### Instruction Set Architecture

- Processor families share the same ISA
- **Example ISAs:** 
  - Intel x86
  - ↗ Intel / AMD x86-64
  - Intel Itanium
  - **ARM**

  - MIPS

All completely different, in the way that C++, Java, Perl, and PHP are all different...

> ... and yet learning one language makes learning the next one much easier

# Why MIPS?

- → Why choose MIPS?
  - The MIPS ISA manual (volume 1, at least) is a svelte 108 pages!
  - Extremely common ISA in textbooks
  - **7** Freely available simulator
  - Common embedded processor
  - Good building-block for other RISC-style processors
  - Aligns with ECPE 173 course

# Arithmetic Instructions

#### Addition

add <result>, <input1>, <input2>

#### Subtraction



### Task : Write Code

#### **Write MIPS assembly for**

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

### **Congratulations!**

You're now an assembly programming expert!

### Data Sources

- Previous example was (just a little bit) fake...
  - We made up some variables: temp0, temp1, f, g, h, i, and j
  - This is what you do when programming in C++ (or any high level language)

### Problem: You can't make up variables in assembly! (as least, not in this fashion)

### Data Sources

#### Where can we explicitly place data in assembly programming?



#### 1. Registers

- On the CPU itself
- Very close to ALU
- **7** Tiny
- Access time: 1 cycle
- 2. Memory
  - Off-chip
  - オ Large
  - Access time: 100+ cycles

# Aside – Cache

- Review: Does the programmer explicitly manage the cache?
- Answer: No!
  - The assembly programmer just reads/writes memory addresses
  - Cache is managed automatically in hardware
  - **Result:** Memory *appears* to be faster than it really is

# ECPE 71

From your knowledge of ECPE 71 (Digital Design), how would you construct a register?

Flip Flops! (D Flip Flop shown)



| D | Q(t+1) |
|---|--------|
| 0 | 0      |
| 1 | 1      |
|   |        |



# ECPE 71 – Group of Registers



# Registers

# MIPS design: 32 integer registers, each holding 32 bits "Word size" = 32 bits

| Name      | Use                  |  |
|-----------|----------------------|--|
| \$zero    | Constant value: ZERO |  |
| \$s0-\$s7 | Local variables      |  |
| \$t0-\$t9 | Temporary results    |  |

#### This is only 19 – where are the rest of the 32?

- Reserved by convention for other uses
- ➤ We'll learn a few more later...

### Task : Write Code

#### **Write MIPS assembly <u>using registers</u> for:**

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

| Map:                 |     | Code: |       |      |  |  |
|----------------------|-----|-------|-------|------|--|--|
| \$s0 = g             | add | \$t0, | \$s0, | \$s1 |  |  |
| \$s1 = h<br>\$s2 = i | add | \$t1, | \$s2, | \$s3 |  |  |
| \$s3 = j             | sub | \$s4, | \$t0, | \$t1 |  |  |
| \$s4 = f             |     |       |       |      |  |  |

### More Arithmetic Instructions

#### Add Immediate



### Task : Write Code

#### **Write MIPS assembly <u>using registers</u> for:**

$$f = g + 20$$

### Memory

#### Challenge: Limited supply of registers

- Physical limitation: We can't put more on the processor chip, and maintain their current speed
- Many elements compete for space in the CPU...
- Solution: Store data in memory
- MIPS provides instructions that transfer data between memory and registers

# Memory Fundamentals

# MIPS <u>cannot</u> directly manipulate data in memory!

Data must be moved to a register first! (And results must be saved to a register when finished)

This is a common design in *RISC-style* machine: a *load-store architecture*.

# Memory Fundamentals

Yes, it's a **pain** to keep moving data between registers and memory.

But consider it your *motivation* to reduce the number of memory accesses. That will **improve program performance**!

# Memory Fundamentals

- Four questions to ask when accessing memory:
  - What direction do I want to copy data? (i.e. to memory, or from memory?)
  - 2. What is the specific **memory address**?
  - 3. What is the specific **register name**? (or number)
  - 4. How **much data** do I want to move?

# Memory – Fundamental Operations

### <u>Load</u>

Copy data from memory to register



#### <u>Store</u>

Copy data from register to memory



# Memory – Determining Address

- There are many ways to calculate the desired memory address
  - **7** These are called *addressing modes*
  - We'll just learn one mode now:
    base + offset
- The base address could be HUGE! (32 bits)
  - **オ** We'll place it in a register
- The offset is typically small
  - We'll directly include it in the instruction as an "immediate"



**MIPS notation: offset(base)** 

# Memory – Register Name

- What is the name of the register to use as either the data destination (for a *load*) or a data source (for a *store*)?
- Use the same register names previously learned
# Memory - Data Transfer Size

- How much data do I want to load or store?
  - ↗ A full word? (32 bits)
  - A "half word"? (16 bits)
  - A byte? (8 bits)
- We'll have a different instruction for each quantity of data
- ↗ No option to load an entire array!
  - Will need a loop that loads 1 element at a time...

# Memory – Data Transfer Instructions

#### Load (copy from memory to register)



**Store** (copy from register to memory)



# Example

### What will this instruction do?

- Load word copies from memory to register:

  - Offset: 20 bytes
  - Destination register: \$s1
  - Amount of data transferred: 1 word (32 bits)

#### Write MIPS assembly for:

$$g = h + array[16]$$

(Array of words. Can leave g and h in registers)

Code: # Assume \$s3 is already set lw \$t0, 16(\$s3) add \$s1, \$s2, \$t0

# Memory Address

- ✓ Slight flaw in previous solution
  - The programmer intended to load the 16<sup>th</sup> array element
  - Each element is 4 bytes (1 word)
  - The offset is in <u>bytes</u>
  - **7** 16 \* 4 = 64

#### Write MIPS assembly for:

### array[12] = h + array[8]

(Array of words. Assume h is in register)

#### Write MIPS assembly for:

### g = h + array[i]

(Array of words. Assume g, h, and i are in registers)

| Map:                                                                   | Code:                                                                                                                                                                                                                                  |  |  |
|------------------------------------------------------------------------|----------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|--|--|
| \$s1 = g<br>\$s2 = h<br>\$s3 = base<br>address of<br>array<br>\$s4 = i | <pre># "Multiply" i by 4<br/>add \$t1, \$s4, \$s4 # x2<br/>add \$t1, \$t1, \$t1 # x2 again<br/># Get addr of array[i]<br/>add \$t1, \$t1, \$s3<br/># Load array[i]<br/>lw \$t0, 0(\$t1)<br/># Compute add<br/>add \$a1 \$a2 \$t0</pre> |  |  |
|                                                                        | add \$s1, \$s2, \$t0                                                                                                                                                                                                                   |  |  |

# Aside – Compiler

- When programming in C / C++, are your variables (int, float, char, ...) stored in memory or in registers?
- **↗** Answer: It depends
- **Compiler will choose** where to place variables
  - Registers: Loop counters, frequently accessed scalar values, variables local to a procedure
  - Memory: Arrays, infrequently accessed data values

# MIPS Branches / Loops

45

# Branches, Tests, Jump

Branch on Equal (if \$1 == \$2, goto dest)

beq <reg1>, <reg2>, <destination>

Set on Less Than (if \$2 < \$3, set \$1 = 1, otherwise 0)</p>

slt <reg1>, <reg2>, <reg3>

Jump (goto dest)

j <destination>

### Write MIPS assembly for:

if (A == B)
{
 <equal-code>
}
else
{
 <not-equal-code>
}
<after-if-code>



### **Write MIPS assembly:**

**Map:** \$s0 = A \$s1 = B

| Code:  |                                   |  |  |
|--------|-----------------------------------|--|--|
|        | beq \$s0,\$s1,equal               |  |  |
|        | <not-equal-code></not-equal-code> |  |  |
|        | j done                            |  |  |
| equal: | <equal-code></equal-code>         |  |  |
|        | j done                            |  |  |
| done:  | <after-if-code></after-if-code>   |  |  |
|        |                                   |  |  |

### Write MIPS assembly for:

while (A != B)

{

}

```
<loop-body>
```

<post-loop-code>



### **Write MIPS assembly:**

| Map:                 |        | Code:                               |
|----------------------|--------|-------------------------------------|
| \$s0 = A<br>\$s1 = B | start: | beq \$s0,\$s1,done                  |
|                      |        | <loop-body><br/>j start</loop-body> |
|                      | done:  | <post-loop-code></post-loop-code>   |
|                      |        |                                     |

50

There are many, **many**, variations of branch or test instructions intended to simplify programming



- . <u>Show</u>: Appendix A Reference
- 2. <u>Discuss</u>: Instruction versus *Pseudo-Instruction*

### Resources

- Resources on Website view "Resources" page
   MIPS Instruction Set (partial guide)
- **Resources available in Sakai site (under ECPE 170)** 
  - - Appendix A from famous Hennessy & Patterson Computer Organization textbook
    - Assemblers, Linkers, and the SPIM simulator
    - Starting on page 51 is an overview of the MIPS assembly commands!
  - MIPS\_Green\_Sheet.pdf
    - "Cheat sheet" for <u>expert programmers</u>
    - MIPS commands, registers, memory conventions, ...

# MIPS Simulator Walkthrough



**Computer Systems and Networks**