

# RISC-V Instruction Set Summary

---



This work is licensed under a [Creative Commons Attribution-NonCommercial-ShareAlike 4.0 International License](#).

Copyright © Codecubix.eu, fbourg

---

|                                                           |   |
|-----------------------------------------------------------|---|
| TABLE 1 : RISC-V REGISTERS NAMES AND NUMBERS              | 2 |
| TABLE 2 : RISC-V (32-BITS) INSTRUCTION FORMATS            | 3 |
| TABLE 3 : RV32I RISC-V INTEGER INSTRUCTIONS               | 4 |
| TABLE 4 : RISC-V COMPRESSED (16-BITS) INSTRUCTION FORMATS | 4 |



Copyright © Codecubix.eu, fbourg

---

**Table 1 : RISC-V Registers names and numbers**

| #     | RISC-V Base                                                           | Ext           | ABI Name | Register | Description                       |
|-------|-----------------------------------------------------------------------|---------------|----------|----------|-----------------------------------|
| 0     | Integer<br>Base<br>-<br><b>RV32I</b><br><b>RV64I</b><br><b>RV128I</b> | <b>RV32-E</b> | zero     | x0       | Hard-wired zero                   |
| 1     |                                                                       |               | ra       | x1       | Return address                    |
| 2     |                                                                       |               | sp       | x2       | Stack pointer                     |
| 3     |                                                                       |               | gp       | x3       | Global pointer                    |
| 4     |                                                                       |               | tp       | x4       | Thread pointer                    |
| 5     |                                                                       |               | t0       | x5       | Temporary/alternate link register |
| 6-7   |                                                                       |               | t1-2     | x6-7     | Temporaries                       |
| 8     |                                                                       |               | s0 / fp  | x8       | Saved register/frame pointer      |
| 9     |                                                                       |               | s1       | x9       | Saved register                    |
| 10-11 |                                                                       |               | a0-a1    | x10-11   | Function arguments/return values  |
| 12-15 |                                                                       |               | a2-a5    | x12-15   | Function arguments                |
| 16-17 |                                                                       |               | a6-a7    | x16-17   | Function arguments                |
| 18-27 |                                                                       |               | s2-s11   | x18-27   | Saved registers                   |
| 28-31 |                                                                       |               | t3-t6    | x28-31   | Temporaries                       |
| 32-39 | Floating<br>Point<br>-<br><b>F</b>                                    | <b>F</b>      | ft0-7    | f0-7     | FP temporaries                    |
| 40-41 |                                                                       |               | fs0-1    | f8-9     | FP saved registers                |
| 42-43 |                                                                       |               | fa0-1    | f10-11   | FP arguments/return values        |
| 44-49 |                                                                       |               | fa2-7    | f12-17   | FP arguments                      |
| 50-59 |                                                                       |               | fs2-11   | f18-27   | FP saved registers                |
| 60-63 |                                                                       |               | ft8-11   | f28-31   | FP temporaries                    |



**Table 2 : RISC-V (32-bits) instruction formats**

| 31:25                           | 24:20        | 19:15 | 14:12        | 11:7                  | 6:0       |                |
|---------------------------------|--------------|-------|--------------|-----------------------|-----------|----------------|
| <b>func7</b>                    | rs2          | rs1   | <b>func3</b> | rd                    | <b>op</b> | <b>R-TYPE</b>  |
| imm <sub>11:0</sub>             |              | rs1   | <b>func3</b> | rd                    | <b>op</b> | <b>I-TYPE</b>  |
| imm <sub>11:5</sub>             | rs2          | rs1   | <b>func3</b> | imm <sub>4:0</sub>    | <b>op</b> | <b>S-TYPE</b>  |
| imm <sub>12,10:5</sub>          | rs2          | rs1   | <b>func3</b> | imm <sub>4:1,11</sub> | <b>op</b> | <b>SB-TYPE</b> |
| imm <sub>31:12</sub>            |              |       |              | rd                    | <b>op</b> | <b>U-TYPE</b>  |
| imm <sub>20,10:1,11,19:12</sub> |              |       |              | rd                    | <b>op</b> | <b>UJ-TYPE</b> |
| fs3                             | <b>func2</b> | fs2   | fs1          | <b>func3</b>          | fd        | <b>op</b>      |
| 5bits                           | 2bits        | 5bits | 5bits        | 3bits                 | 5bits     | 7bits          |
|                                 |              |       |              |                       |           | 32bits         |

## Instruction formats details

In the base ISA, there are four core instruction formats (R/I/S/U) :

- **R-TYPE** Register-register ALU instructions : add, xor, mul
- **I-TYPE** Immediate ALU instructions, load instructions : addi, lw, jalr, slli
- **S-TYPE** Store instructions : sw, sb
- **SB-TYPE** Comparison and branch instructions: beq, bge
- **U-TYPE** Instructions with upper immediates
- **UJ-TYPES** Jump instructions: jal
  
- **func2** type of operation on 2bits
- **func3** type of operation on 3bits
- **func4** type of operation on 4bits
- **func6** type of operation on 6bits
- **func7** type of operation on 7bits

## Glossary of instruction descriptions

- **rs1, rs2** Register descriptors\* : Source operands 1 and 2
- **rd** Register descriptor\* : Destination operand
- **op** Operation code

\*Register descriptors (rs1,rs2, rd) always need 5 bits to address all possible 32 ( $2^5$ ) working registers (from x0 to x31). That is partly why it is not possible to use CSR directly into regular instructions. In “Priviledged / CSR instructions” that are all I-TYPE, the csr operand is coded by imm<sub>11:0</sub> (on 12 bits), and that is what theoretically allows to address up to 4096 CSR ( $2^{12}$ ). In addition, in the “Priviledged / CSR instructions”, the 5-bit unsigned immediate (uimm) is coded in the rs1 field and not in the imm<sub>11:0</sub> field as it should be because of its previous use.

- **imm** signed immediate in imm<sub>11:0</sub>
- **uimm** 5-bit unsigned immediate in imm<sub>4:0</sub>
- **upimm** 20 upper bits of a 32-bit immediate, in imm<sub>31:12</sub>
- **Address** memory address : rs1 + SignExt(immm<sub>11:0</sub>)
- **[Address]** data at memory location Address
- **BTA** branch target address : PC + SignExt({imm<sub>12:1</sub>, 1'b0})
- **JTA** jump target address : PC + SignExt({imm<sub>20:1</sub>, 1'b0})
- **Label** text indicating instruction address
- **SignExt** value sign-extended to 32 bits
- **ZeroExt** value zero-extended to 32 bits
- **csr** constrol and status register



**Table 3 : RV32I RISC-V Integer instructions**

| op           | Func3 | Func7   | Type | Mnemonic                    | Description                         | Operation                               |
|--------------|-------|---------|------|-----------------------------|-------------------------------------|-----------------------------------------|
| 0000011(3)   | 000   |         | I    | <b>lb</b> rd, imm(rs1)      | Load byte                           | rd = SignExt([Address]7:0)              |
| 0000011(3)   | 001   |         | I    | <b>lh</b> rd, imm(rs1)      | Load half                           | rd = SignExt([Address]15:0)             |
| 0000011(3)   | 010   |         | I    | <b>lw</b> rd, imm(rs1)      | Load word                           | rd = ([Address]31:0)                    |
| 0000011(3)   | 100   |         | I    | <b>lbu</b> rd, imm(rs1)     | Load byte unsigned                  | rd = ZeroExt([Address]7:0)              |
| 0000011(3)   | 101   |         | I    | <b>lhu</b> rd, imm(rs1)     | Load half unsigned                  | rd = ZeroExt([Address]15:0)             |
| 0010011(19)  | 000   |         | I    | <b>addi</b> rd, rs1, imm    | ADD immediate                       | rd = rs1 + SignExt(immm)                |
| 0010011(19)  | 001   |         | I    | <b>slli</b> rd, rs1, uimm   | Shift left logical immediate        | rd = rs1 << uimm                        |
| 0010011(19)  | 010   |         | I    | <b>slti</b> rd, rs1, imm    | Set less than immediate             | rd = rs1 < SignExt(immm)                |
| 0010011(19)  | 011   | 0000000 | I    | <b>sltiu</b> rd, rs1, imm   | Set less than imm. unsigned         | rd = rs1 < SignExt(immm)                |
| 0010011(19)  | 100   |         | I    | <b>xori</b> rd, rs1, imm    | XOR immediate                       | rd = rs1 ^ SignExt(immm)                |
| 0010011(19)  | 101   | 0000000 | I    | <b>srli</b> rd, rs1, uimm   | Shift right logical immediate       | rd = rs1 >> uimm                        |
| 0010011(19)  | 101   | 0100000 | I    | <b>srai</b> rd, rs1, uimm   | Shift right arithmetic immediate    | rd = rs1 >> uimm                        |
| 0010011(19)  | 110   |         | I    | <b>ori</b> rd, rs1, uimm    | OR immediate                        | rd = rs1   SignExt(immm)                |
| 0010011(19)  | 111   |         | I    | <b>andi</b> rd, rs1, uimm   | AND immediate                       | rd = rs1 & SignExt(immm)                |
| 0010111(23)  |       |         | U    | <b>auipc</b> rd, rs1, uimm  | ADD upper immediate to PC           | rd = (upimm, 12'b0) + PC                |
| 0100011(35)  |       |         | S    | <b>sb</b> rs2, imm(rs1)     | Store byte                          | [Address]7:0 = rs27:0                   |
| 0100011(35)  |       |         | S    | <b>sh</b> rs2, imm(rs1)     | Store half                          | [Address]15:0 = rs215:0                 |
| 0100011(35)  |       |         | S    | <b>sw</b> rs2, imm(rs1)     | Store word                          | [Address]31:0 = rs2                     |
| 0110011(51)  | 000   | 0000000 | R    | <b>add</b> rd, rs1, rs2     | ADD                                 | rd = rs1 + rs2                          |
| 0110011(51)  | 000   | 0100000 | R    | <b>sub</b> rd, rs1, rs2     | SUB                                 | rd = rs1 - rs2                          |
| 0110011(51)  | 001   | 0000000 | R    | <b>sll</b> rd, rs1, rs2     | Shift left logical                  | rd = rs1 << rs24:0                      |
| 0110011(51)  | 010   | 0000000 | R    | <b>slt</b> rd, rs1, rs2     | Set less than                       | rd = rs1 < rs2                          |
| 0110011(51)  | 011   | 0000000 | R    | <b>sltu</b> rd, rs1, rs2    | Set less than unsigned              | rd = rs1 < rs2                          |
| 0110011(51)  | 100   | 0000000 | R    | <b>xor</b> rd, rs1, rs2     | XOR                                 | rd = rs1 ^ rs2                          |
| 0110011(51)  | 101   | 0000000 | R    | <b>srl</b> rd, rs1, rs2     | Shift right logical                 | rd = rs1 >> rs24:0                      |
| 0110011(51)  | 101   | 0100000 | R    | <b>sra</b> rd, rs1, rs2     | Shift right arithmetic              | rd = rs1 >>> rs24:0                     |
| 0110011(51)  | 110   | 0000000 | R    | <b>or</b> rd, rs1, rs2      | OR                                  | rd = rs1   rs2                          |
| 0110011(51)  | 111   | 0000000 | R    | <b>and</b> rd, rs1, rs2     | AND                                 | rd = rs1 & rs2                          |
| 0110111(55)  | -     |         | U    | <b>lui</b> rd, upimm        | Load upper immediate                | rd = {upimm, 12'b0}                     |
| 1100011(99)  | 000   |         | B    | <b>beq</b> rs1, rs2, label  | Branch if equal =                   | if (rs1 == rs2) PC = BTA                |
| 1100011(99)  | 001   |         | B    | <b>bne</b> rs1, rs2, label  | Branch if not equal ≠               | if (rs1 != rs2) PC = BTA                |
| 1100011(99)  | 010   |         | B    | <b>blt</b> rs1, rs2, label  | Branch if lower than <              | if (rs1 < rs2) PC = BTA                 |
| 1100011(99)  | 011   |         | B    | <b>bge</b> rs1, rs2, label  | Branch if greater / equal ≥         | if (rs1 ≥ rs2) PC = BTA                 |
| 1100011(99)  | 100   |         | B    | <b>bltu</b> rs1, rs2, label | Branch if lower than unsigned <     | if (rs1 < rs2) PC = BTA                 |
| 1100011(99)  | 101   |         | B    | <b>bgeu</b> rs1, rs2, label | Branch if greater / equal unsign. ≥ | if (rs1 ≥ rs2) PC = BTA                 |
| 1100111(103) | 000   |         | I    | <b>jalr</b> rd, rs1, label  | Jump and link register              | PC = rs1 + SignExt(immm)<br>rd = PC + 4 |
| 1101111(111) | -     |         | J    | <b>jal</b> rd, label        | Jump and link                       | PC = JTA<br>rd = PC + 4                 |

**Table 4 : RISC-V compressed (16-bits) instruction formats**

| 15:13        | 12     | 11          | 10       | 9   | 8            | 7         | 6:5       | 4:2       | 1:0    |  |  |
|--------------|--------|-------------|----------|-----|--------------|-----------|-----------|-----------|--------|--|--|
| <b>func4</b> | rd/rs1 |             |          |     |              |           | rs2       | <b>op</b> |        |  |  |
| <b>func3</b> | imm    | rd/rs1      |          | imm |              | <b>op</b> |           |           |        |  |  |
| <b>func3</b> | imm    | rs1'        |          | imm |              | rs2'      | <b>op</b> |           |        |  |  |
| <b>func6</b> |        |             | rd/rs1   |     | <b>func2</b> | rs2'      | <b>op</b> |           |        |  |  |
| <b>func3</b> | imm    | rs1'        |          | imm |              | <b>op</b> |           |           |        |  |  |
| <b>func3</b> | imm    | <b>func</b> | rd'/rs1' |     | imm          |           | <b>op</b> |           |        |  |  |
| <b>func3</b> | imm    |             |          |     |              |           |           | <b>op</b> |        |  |  |
| <b>func3</b> | imm    | rs2         |          |     |              | <b>op</b> |           |           |        |  |  |
| <b>func3</b> | imm    |             |          |     |              |           | rd'       | <b>op</b> |        |  |  |
| <b>func3</b> | imm    | rs1'        |          | imm | rd'          |           | <b>op</b> |           |        |  |  |
| 3bits        | 1bit   | -           | -        | -   | -            | 2bits     | 3bits     | 2bits     | 16bits |  |  |

**CR-TYPE**  
**CI-TYPE**  
**CS-TYPE**  
**CS' -TYPE**  
**CB-TYPE**  
**CB' -TYPE**  
**CJ-TYPE**  
**CSS-TYPE**  
**CIW-TYPE**  
**CL-TYPE**

