

# **Topics**

- Exception vs Interruption
- Common Exception
- Exception Handler
  - Register in coprocessor 0
  - Trap, eret (MIPS32)

### **Exception vs Interruption**

- An exception is an event that disrupts the normal flow of the execution of your code.
  - ➤ When an exception occurs, the CPU will figure out what is wrong by checking its status, see if it can be corrected and then continue the execution of the normal code like nothing happened, or it stops the code execution.
  - > E.g. Accessing to the 0x0 address in user mode will trigger an exception.

- An interruption is an event caused by a device which is external to the CPU.
  - ➤ E.g. 'syscall' is an interruption.

# **Common Exception**

The following exceptions are the most common in the main processor:

- > Arithmatic type
  - Overflow exceptions
    - ➤ Which occur when **arithmetic operations** compute **signed** values and the destination lacks the precision to store the result.
  - > Divide-by-zero exceptions
    - > Which occur when a divisor is zero.
- > Address error exceptions
  - Which occur when the machine references a data item that is NOT on its proper memory alignment
  - Which occur when an address is invalid for the executing process.

# **Bad Address Exception**



| Registers Coproc 1 Co | oproc 0 |            |
|-----------------------|---------|------------|
| Name                  | Number  | Value      |
| \$8 (vaddr)           | 8       | 0x00000000 |
| \$12 (status)         | 12      | 0x0000ff13 |
| \$13 (cause)          | 13      | 0x00000010 |
| \$14 (epc)            | 14      | 0x0040000c |

| Bkpt | Address    | Code       | Basic                       |     | Source                        |   |
|------|------------|------------|-----------------------------|-----|-------------------------------|---|
|      | 0x00400000 | 0x23bdfffc | addi \$29, \$29, 0xfffffffc | 5:  | addi \$sp, \$sp, -4           | - |
|      | 0x00400004 | 0xafa20000 | sw \$2,0x00000000(\$29)     | 6:  | sw \$v0, (\$sp)               |   |
|      | 0x00400008 | 0x24020004 | addiu \$2, \$0, 0x00000004  | 9:  | li \$v0,4                     |   |
|      | 0x0040000c | 0x0000000c | syscall                     | 10: | syscall                       | П |
|      | 0x00400010 | 0x8fa20000 | lw \$2,0x00000000(\$29)     | 12: | lw \$v0, (\$sp)               |   |
|      | 0x00400014 | 0x23bd0004 | addi \$29, \$29, 0x00000004 | 13: | addi \$sp, \$sp, 4            |   |
|      | 0x00400018 | 0x201fffff | addi \$31, \$0, 0xffffffff  | 15: | addi \$ra, \$zero, 0xffffffff |   |
|      | 0x0040001c | 0x03e00008 | jr \$31                     | 16: | jr \$ra                       |   |

Runtime exception at 0x0040000c: address out of range 0x00000000

\$a0's default value is 0x00000000, which is not allowed to access in user mode

# The Register in Coprocessor 0

| Registers Coproc 1 C | oproc 0 |            |
|----------------------|---------|------------|
| Name                 | Number  | Value      |
| \$8 (vaddr)          | 8       | 0x00000000 |
| \$12 (status)        | 12      | 0x0000ff13 |
| \$13 (cause)         | 13      | 0x00000030 |
| \$14 (epc)           | 14      | 0x00400010 |

| Register name | Register number | Usage                                                          |
|---------------|-----------------|----------------------------------------------------------------|
| VAddr         | 8               | memory address at which an offending memory reference occurred |
| Status        | 12              | interrupt mask and enable bits                                 |
| Cause         | 13              | exception type(bit6~bit2) and pending interrupt bits           |
| EPC           | 14              | address of instruction that caused exception                   |

# **Exception Control Registers**

BD = Branch Delay, UM = User Mode, EL = Exception Level, IE =Interrupt Enable

#### **EXCEPTION CODES**

| Number | Name | Cause of Exception                                  | Number | Name | Cause of Exception                |
|--------|------|-----------------------------------------------------|--------|------|-----------------------------------|
| 0      | Int  | Interrupt (hardware)                                | 9      | Bp   | Breakpoint Exception              |
| 4      | AdEL | Address Error Exception (load or instruction fetch) | 10     | RI   | Reserved Instruction<br>Exception |
| 5      | AdES | Address Error Exception (store)                     | 11     | CpU  | Coprocessor<br>Unimplemented      |
| 6      | IBE  | Bus Error on<br>Instruction Fetch                   | 12     | Ov   | Arithmetic Overflow<br>Exception  |
| 7      | DBE  | Bus Error on<br>Load or Store                       | 13     | Tr   | Trap                              |
| 8      | Sys  | Syscall Exception                                   | 15     | FPE  | Floating Point Exception          |

# **Bad Address Exception** continued

```
.data
     str: .asciiz "hello"
.text
print_string:
     addi $sp,$sp,-4
     sw $v0,($sp)
     la $a0,str
     li $v0,4
     syscall
     lw $v0,($sp)
     addi $sp,$sp,4
     addi $ra,$zero,0xffffffff
     jr $ra
```

| \$ra | 31 | 0xffffffff |
|------|----|------------|
| рс   |    | 0xffffffff |

| Registers    | Coproc 1 | Coproc 0 |            |
|--------------|----------|----------|------------|
| 1            | lame     | Number   | Value      |
| \$8 (vaddr)  |          | 8        | 0xffffffff |
| \$12 (status | )        | 12       | 0x0000ff13 |
| \$13 (cause) |          | 13       | 0x00000010 |
| \$14 (epc)   |          | 14       | 0xffffffff |

Error in : invalid program counter value: 0xffffffff

### **Bad Address Exception** continued

#### Which one will trigger the exception?

| Registers Cop | proc 1 Coproc 0 |            |
|---------------|-----------------|------------|
| Name          | Number          | Value      |
| \$8 (vaddr)   | 8               | 0x10010009 |
| \$12 (status) | 12              | 0x0000ff13 |
| \$13 (cause)  | 13              | 0x00000010 |
| \$14 (epc)    | 14              | 0x0040001c |

While use 'sw' intead of 'lw' in the demos, which one will trigger the exception? What's the value of \$13 in Coproc 0 on this situation?

```
.include "../macro print str.asm"
.data
     str: .asciiz "data is:"
          .bvte 1:10
     ws: .word 2:10
.text
     print string("data is:")
     add $t0,$zero,$zero
     addi $t1,$zero,10
loop out:
     lw $a0,bs
     li $v0,1
     syscall
     add $t0,$t0,1
     bne $t0,$t1,loop out
     end
```

```
.include "../macro print str.asm"
.data
     str: .asciiz "data is:"
          .bvte 1:10
     bs:
     ws: .word 2:10
.text
     print_string("data is:")
     add $t0,$zero,$zero
     addi $t1,$zero,10
loop out:
     lw $a0,ws
     li $v0,1
     syscall
     add $t0,$t0,1
     bne $t0,$t1,loop out
     end
```

Runtime exception at 0x0040001c: fetch address not aligned on word boundary 0x10010009

# **Arithmetic Exception**

Will the 'addu' trigger an exception? How about 'sub', 'div'? How about 'addiu \$a0, \$t0, -1'?

| .data                     |
|---------------------------|
| addend1: .word 0x7fffffff |
| addend2: .word 0x7fffffff |
| .text                     |
| print_string:             |
| lw \$t0,addend1           |
| lw \$t1,addend2           |
| add \$a0,\$t0,\$t1        |
|                           |
| li \$v0,1                 |
| syscall                   |
|                           |
| li \$v0,10                |
| syscall                   |

| Registers     | Coproc 1 | Coproc 0 |            |
|---------------|----------|----------|------------|
| N             | ame      | Number   | Value      |
| \$8 (vaddr)   |          | 8        | 0x00000000 |
| \$12 (status) |          | 12       | 0x0000ff13 |
| \$13 (cause)  |          | 13       | 0x00000030 |
| \$14 (epc)    |          | 14       | 0x00400010 |

| Bkpt  | Address    | Code       | Basic                      |     | Source               |
|-------|------------|------------|----------------------------|-----|----------------------|
|       | 0x00400000 | 0x3c011001 | lui \$1,0x00001001         | 6:  | lw \$t0, addend1     |
|       | 0x00400004 | 0x8c280000 | lw \$8,0x00000000(\$1)     | 12  |                      |
|       | 0x00400008 | 0x3c011001 | lui \$1,0x00001001         | 7:  | lw \$t1, addend2     |
| 1100  | 0x0040000c | 0x8c290004 | lw \$9,0x00000004(\$1)     | 10  |                      |
|       | 0x00400010 | 0x01092020 | add \$4, \$8, \$9          | 8:  | add \$a0, \$t0, \$t1 |
| 1000  | 0x00400014 | 0x24020001 | addiu \$2, \$0, 0x00000001 | 10: | li \$v0,1            |
|       | 0x00400018 | 0x0000000c | syscall                    | 11: | syscall              |
| 11-11 | 0x0040001c | 0x2402000a | addiu \$2, \$0, 0x0000000a | 13: | li \$v0,10           |
|       | 0x00400020 | 0x0000000c | syscall                    | 14: | syscall              |

Runtime exception at 0x00400010: arithmetic overflow

# **How MIPS Acts When Taking An Exception?**

- > Step1. It sets up the EPC to point to the restart location
- > Step2. CPU changes into kernel mode and disables the interrupts (MIPS does this by setting EXL bit of Status register)
- > Step3. Set up the Cause register to indicate which is wrong so that software can tell the reason for the exception. If it is for address exception, for example, TLB miss and so on, the BadVaddr register is set.
- > Step4. CPU starts fetching instructions from the exception entry point and then goes to the exception handler.

Up to MIPS III, eret instruction is used to return to the original location before falling into the exception. Note that eret behavior is: clear the SR[EXL] bit and return control to the adress stored in EPC.

### **Exception Related Instructions**

### > Conditional trap

```
\triangleright teq $s0, $s1 ##trap(jump to the ktext), if s0==s1
```

```
> tne $s0, $s1 ##trap(jump to the ktext), if s0!=s1
```

```
\triangleright teqi $s0, 1 ##trap(jump to the ktext), if s0==1
```

#### >mfc0,mtc0

```
>mfc0 $k0,$14 ##Move from coproc0 reg#14(epc) to $k0
```

```
\rightarrow mtc0 $k0,$14 ##Move from $k0 to coproc0 reg#14(epc)
```

#### > eret

- > Returns from an interrupt, exception or error trap.
- Similar to a branch or jump instruction, eret executes the next instruction before taking effect. Use this on R4000 processor machines in place of rfe.

#### Demo

```
.data
                  .asciiz "\ndata over"
         dmsg:
.text
         li $v0,5
main:
         syscall
         teqi $v0,0
         la $a0,dmsg
         li $v0,4
         syscall
         li $v0,10
         syscall
```

```
.ktext
          0x80000180
     move $k0,$v0
     move $k1,$a0
     la $a0,msg
     li $v0,4
     syscall
     move $v0,$k0
     move $a0,$k1
     mfc0 $k0,$14
     addi $k0,$k0,4
     mtc0 $k0,$14
     eret
.kdata
msg: .asciiz "\nTrap generated"
```

Q1. How to tigger the trap?

Q2. When will the string
"\ndata over" be printed out?

Q3. Use "break" in text
segment and ktext segment
seperately,
what happens?

Q4. Remove the instruction "addi \$k0,\$k0,4", while the trap is trigged, what will happen?

### Practice(1)

- 1. Implementing a procedure to read a list of number from input, store them into an array and print every item of the array out:
  - 1. The size of the array and its space are determined by user's input
  - 2. The space of array item is determined by user's input
  - 3. While storing the array items into array, if the item exceed the bounday of array, an exception will be triggered and exit the program.
    - 1. For example: the size of array is 10 and its space is 10Bytes, the space of each array item is 4Bytes. Using loop to read the value of array items and write them into the array's space. While processing the 3th array item and write it into array's space, this value will pollute other areas. Your procedure should trigger an exception on this situation, print the warning infomation and exit the program.
  - 4. While all items are stored into the array correctly, print every item out.

The exception handler do the following things:

- 1. stop the program running
- 2. output prompt information, including "runtime exception at  $0x_***"$  (the address of the instruction which triggered the exception), the cause of the exception ("Arrayltem pollutes other areas", index is:\*\*\*) and the index of array item which triggered the exception.
- 3. exit the program.

# Practice(2)

- 2. How many types of exception are list in this slides? List their exception code in the cause register(\$13 in coproc 0)
- 3. Answer the question on page 9,10 and 13.
- 4. Create a macro named "dive" with 3 parameters(the 1st stores the quotient, the 2nd stores the dividend and the last one stores the divisor) to do the division. If the value of divisor is 0, trigger a trap to output prompt information, including "runtime exception at 0x\_\*\*\* "(the address of the instruction which triggered the exception), the cause of the exception ( "divisor is 0" ) and exit the program.

### **Tips**

.align Align the next datum on a 2 <sup>n</sup> byte boundary.

For example, .align 2 aligns the next value on a word boundary.

.align 0 turns off automatic alignment of .half, .word, .float, and .double directives until the next .data or .kdata directive.

.kdata subsequent items are stored in the kernel data segment, If the optional argument addr is present, subsequent items are stored starting at address addr.

**.ktext** subsequent items are stored in the kernel text segment, In SPIM, these items may only be instructions or words. If the optional argument *addr* is present, subsequent items are stored starting at address *addr*.

# Tips: usage of '.align'





#### .data

str: .asciiz "data is:"

bs: .byte 1:10

ws: .word 2:10

#### .data

str: .asciiz "data is:"

.align 2

bs: .byte 1:10

ws: .word 2:10

|            |            |            |            | Value (+c) | Value (+10) | Value (+14) | Value (+18) | Value (+1c) |
|------------|------------|------------|------------|------------|-------------|-------------|-------------|-------------|
| 0x10010000 | 0x61746164 | 0x3a736920 | 0000000000 | 0x01010101 | 0x01010101  | 0x00000101  | 0x00000002  | 0x0000000   |
| 0x10010020 | 0x00000002 | 0x00000002 | 0x00000002 | 0x00000002 | 0x00000002  | 0x00000002  | 0x00000002  | 0x0000000   |
| Address    | Value (+0) | Value (+4) | Value (+8) | Value (+c) | Value (+10) | Value (+14) | Value (+18) | Value (+1c) |