

# VLSI DESIGN INTERNSHIP

# RISC-V ISA & RV32I

## RTL CODING

A.NARMATHA  
22BEC1280



## CERTIFICATE OF COMPLETION

This is to certify that Mr./Ms. Narmatha A (22BEC1280) has successfully completed VLSI Design Internship Program from 05-June-2024 to 13-July-2024.

During the internship program with us, he/she worked on RISC-V ISA & RV32I RTL Design project.

Date: 02/08/2024

Place: Bengaluru

MSUID: MSV-DI/2024-25/487



SIVAKUMAR P R  
FOUNDER & CEO, MAVEN SILICON



Centre of Excellence in VLSI

## MAVEN SILICON GRADE REPORT

### Grading Index

>90%    >70%    >50%    >30%    >10%    <10%  
● "O"    ● "A+"    ● "A"    ● "B+"    ● "B"    ● "C"

Project Test Score (in %)

70

Project Submission Report Score (in %)

60

Grade

A

Name

Narmatha A

College Registration Number

22BEC1280

Course Name

VLSI Design Internship Program

Project Name

RISC-V ISA & RV32I RTL Design

Date

July 2024

# ABOUT MAVEN SILICON

01

Maven Silicon is a distinguished company in the domain of VLSI (Very Large Scale Integration) design and semiconductor technology. It has established itself as a leading institution that offers a wide array of services and training programs related to VLSI design and verification.

02

- One of the standout features of Maven Silicon is its comprehensive VLSI training programs. These programs cover a wide spectrum of topics, from the fundamentals of digital design to advanced concepts such as ASIC and FPGA design. This ensures that students are not only taught theoretical concepts but also gain insights into the latest industry trends and best practices



# OBJECTIVE OF THE TRAINING

The VLSI Design Methodologies Course by MAVEN SILICON provided an overview of VLSI and explains various VLSI technology. Following project assignment, each student received project-specific training. In this case, detailed concepts of

- Instruction Set Architecture (ISA): Understand the fundamental principles of how instructions are encoded, decoded, and executed.
- Pipeline Design: Learn how to design efficient pipelines to improve processor performance.
- Memory Hierarchy: Explore the concepts of cache memory, virtual memory, and memory management units.
- Control Unit Design: Grasp the design of control units to generate control signals for the data path.
- Data Path Design: Understand the components of the data path, such as registers, ALU, and memory interface.

# RISC-V ISA INSTRUCTION FORMAT



# RISC V R-TYPE I TYPE

| <b>name</b> | <b>31:25</b> | <b>24:20</b> | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                                           |
|-------------|--------------|--------------|--------------|--------------|-------------|---------------|--------------------------------------------------------------|
| <b>add</b>  | 0000000      | rs2          | rs1          | 000          | rd          | 0110011       | $(rd) = (rs1) + (rs2)$                                       |
| <b>sub</b>  | 0100000      | rs2          | rs1          | 000          | rd          | 0110011       | $(rd) = (rs1) - (rs2)$                                       |
| <b>slt</b>  | 0000000      | rs2          | rs1          | 010          | rd          | 0110011       | $(rd) = \text{signed}(rs1) < \text{signed}(rs2) ? 1 : 0$     |
| <b>sltu</b> | 0000000      | rs2          | rs1          | 011          | rd          | 0110011       | $(rd) = \text{unsigned}(rs1) < \text{unsigned}(rs2) ? 1 : 0$ |
| <b>xor</b>  | 0000000      | rs2          | rs1          | 100          | rd          | 0110011       | $(rd) = (rs1) ^ (rs2)$                                       |
| <b>or</b>   | 0000000      | rs2          | rs1          | 110          | rd          | 0110011       | $(rd) = (rs1)   (rs2)$                                       |
| <b>and</b>  | 0000000      | rs2          | rs1          | 111          | rd          | 0110011       | $(rd) = (rs1) \& (rs2)$                                      |
| <b>sll</b>  | 0000000      | rs2          | rs1          | 001          | rd          | 0110011       | $(rd) = (rs1) \ll \text{unsigned}(rs2[4:0])$                 |
| <b>srl</b>  | 0000000      | rs2          | rs1          | 101          | rd          | 0110011       | $(rd) = (rs1) \gg \text{unsigned}(rs2[4:0])$                 |
| <b>sra</b>  | 0100000      | rs2          | rs1          | 101          | rd          | 0110011       | $(rd) = \text{signed}(rs1) \ggg \text{unsigned}(rs2[4:0])$   |

| <b>name</b>  | <b>31:20</b> | <b>24:20</b> | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                                           |
|--------------|--------------|--------------|--------------|--------------|-------------|---------------|--------------------------------------------------------------|
| <b>addi</b>  | imm[11:0]    |              | rs1          | 000          | rd          | 0010011       | $(rd) = (rs1) + \text{signed(imm)}$                          |
| <b>slti</b>  | imm[11:0]    |              | rs1          | 010          | rd          | 0010011       | $(rd) = \text{signed(rs1)} < \text{signed(imm)} ? 1 : 0$     |
| <b>stliu</b> | imm[11:0]    |              | rs1          | 011          | rd          | 0010011       | $(rd) = \text{unsigned(rs1)} < \text{unsigned(imm)} ? 1 : 0$ |
| <b>xori</b>  | imm[11:0]    |              | rs1          | 100          | rd          | 0010011       | $(rd) = (rs1) \wedge \text{signed(imm)}$                     |
| <b>ori</b>   | imm[11:0]    |              | rs1          | 110          | rd          | 0010011       | $(rd) = (rs1)   \text{signed(imm)}$                          |
| <b>andi</b>  | imm[11:0]    |              | rs1          | 111          | rd          | 0010011       | $(rd) = (rs1) \& \text{signed(imm)}$                         |
| <b>slli</b>  | 0000000      | imm[4:0]     | rs1          | 001          | rd          | 0010011       | $(rd) = (rs1) \ll \text{unsigned(imm)}$                      |
| <b>srlti</b> | 0000000      | imm[4:0]     | rs1          | 101          | rd          | 0010011       | $(rd) = (rs1) \gg \text{unsigned(imm)}$                      |
| <b>srai</b>  | 0100000      | imm[4:0]     | rs1          | 101          | rd          | 0010011       | $(rd) = \text{signed(rs1)} \ggg \text{unsigned(imm)}$        |

# RISC V L-TYPE S-TYPE

## Load Instructions (I-Type)

| <b>name</b> | <b>31:20</b> | <b>24:20</b> | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                             |
|-------------|--------------|--------------|--------------|--------------|-------------|---------------|------------------------------------------------|
| <b>lb</b>   | imm[11:0]    |              | rs1          | 000          | rd          | 0000011       | (rd) = signed(mem_byte[(rs1) + signed(imm)])   |
| <b>lh</b>   | imm[11:0]    |              | rs1          | 001          | rd          | 0000011       | (rd) = signed(mem_hw[(rs1) + signed(imm)])     |
| <b>lw</b>   | imm[11:0]    |              | rs1          | 010          | rd          | 0000011       | (rd) = mem[(rs1) + signed(imm)]                |
| <b>lbu</b>  | imm[11:0]    |              | rs1          | 100          | rd          | 0000011       | (rd) = unsigned(mem_byte[(rs1) + signed(imm)]) |
| <b>lhu</b>  | imm[11:0]    |              | rs1          | 101          | rd          | 0000011       | (rd) = signed(mem_hw[(rs1) + signed(imm)])     |

## Store Instructions (S-Type)

| <b>name</b> | <b>31:20</b> | <b>24:20</b> | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                         |
|-------------|--------------|--------------|--------------|--------------|-------------|---------------|--------------------------------------------|
| <b>sb</b>   | imm[11:5]    | rs2          | rs1          | 000          | imm[4:0]    | 0100011       | mem_byte[(rs1) + signed(imm)] = (rs2)[7:0] |
| <b>sh</b>   | imm[11:5]    | rs2          | rs1          | 001          | imm[4:0]    | 0100011       | mem_hw[(rs1) + signed(imm)] = (rs2)[15:0]  |
| <b>sw</b>   | imm[11:5]    | rs2          | rs1          | 010          | imm[4:0]    | 0100011       | mem[(rs1) + signed(imm)] = (rs2)           |

# RISC V B-TYPE J-TYPE

## Branch Instructions (B-Type)

| <b>name</b> | <b>31:25</b> | <b>24:20</b> | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                                                      |
|-------------|--------------|--------------|--------------|--------------|-------------|---------------|-------------------------------------------------------------------------|
| <b>beq</b>  | imm[12 10:5] | rs2          | rs1          | 000          | imm[4:1 11] | 1100011       | $pc = pc + ((rs1) == (rs2) ? \text{signed}(imm) : 4)$                   |
| <b>bne</b>  | imm[12 10:5] | rs2          | rs1          | 001          | imm[4:1 11] | 1100011       | $pc = pc + ((rs1) != (rs2) ? \text{signed}(imm) : 4)$                   |
| <b>blt</b>  | imm[12 10:5] | rs2          | rs1          | 100          | imm[4:1 11] | 1100011       | $pc = pc + ((rs1) < (rs2) ? \text{signed}(imm) : 4)$                    |
| <b>bge</b>  | imm[12 10:5] | rs2          | rs1          | 101          | imm[4:1 11] | 1100011       | $pc = pc + ((rs1) >= (rs2) ? \text{signed}(imm) : 4)$                   |
| <b>bltu</b> | imm[12 10:5] | rs2          | rs1          | 110          | imm[4:1 11] | 1100011       | $pc = pc + (\text{unsigned}((rs1) < (rs2)) ? \text{signed}(imm) : 4)$   |
| <b>bgeu</b> | imm[12 10:5] | rs2          | rs1          | 111          | imm[4:1 11] | 1100011       | $pc = pc + (\text{unsigned}((rs1) >= (rs2))) ? \text{signed}(imm) : 4)$ |

## Jump and Link Instructions (J-Type, I-Type)

| <b>name</b> | <b>31:25</b>          | <b>19:15</b> | <b>14:12</b> | <b>11:7</b> | <b>opcode</b> | <b>Description</b>                                               |
|-------------|-----------------------|--------------|--------------|-------------|---------------|------------------------------------------------------------------|
| <b>jal</b>  | imm[20 10:1 11 19:12] |              |              | rd          | 1101111       | $(rd) = pc + 4, pc = pc + \text{signed}(imm)$                    |
| <b>jalr</b> | imm[11:0]             | rs1          | 000          | rd          | 1100111       | $(rd) = pc + 4, pc = ((rs1) + \text{signed}(imm)) \& \sim 32'h1$ |

# INSTRUCTION DECODER



# RISC V PC MUX



# PC MUX

## Internal Signals

- **i\_addr** (reg, 32 bits): The internal register for the instruction address.
- **next\_pc** (wire, 32 bits): The calculated next PC value based on branching.

## Misaligned Instruction Detection:

- `omisaligned_instr_out` is set if `next_pc` is misaligned (i.e., if the least significant bit is 1) and `branch_taken_in` is high.

## PC Increment:

- `opc_plus_4_out` is calculated as the current `pc_in` plus 4, which is the address of the next sequential instruction.

## Next PC Calculation:

- `onext_pc` is calculated as:
- `{iaddr_in, 1'b0}` if `branch_taken_in` is high (indicating a branch),
- `pc_plus_4_out` otherwise.

# INSTRUCTION DECODER

## PC Multiplexer:

- The pc\_mux\_out value is selected based on pc\_src\_in:
- 2'b00: pc\_mux\_out is set to BOOT\_ADDRESS.
- 2'b01: pc\_mux\_out is set to epc\_in.
- 2'b10: pc\_mux\_out is set to trap\_address\_in.
- 2'b11: pc\_mux\_out is set to next\_pc.

## Instruction Address Update:

- The i\_addr register is updated:
- To BOOT\_ADDRESS if rst\_in is high (indicating a reset).
- To pc\_mux\_out if ahb\_ready\_in is high (indicating that the AHB is ready).

# RISC V INTEGER FILE



# INTEGER FILE

## Register File:

- The module uses an array of 32 registers (reg\_file) to store 32-bit integer values. Each register is identified by its address (0 to 31).

## Read Operations:

- The rs\_1\_addr\_in and rs\_2\_addr\_in signals specify the addresses of the registers to be read for the first and second source operands (RS1 and RS2), respectively.
- The module uses these addresses to access the corresponding registers in reg\_file and assign the read data to the rs\_1\_out and rs\_2\_out outputs.

## Write Operations:

- The wr\_en\_in signal controls write operations to the register file.
- If wr\_en\_in is high (active) and rd\_addr\_in is valid (not zero), the data provided by rd\_in is written to the register specified by the address rd\_addr\_in.

# LOAD UNIT



# LOAD UNIT

## Load Size:

- The load\_size\_in signal specifies the size of the data to be loaded.
- Based on this value, the module selects the appropriate portion of the data\_in for the output.

## Byte Address Offset:

- The iadder\_1\_to\_0\_in signal indicates the byte offset within the loaded word (data\_in) based on the base address used for the load instruction. This helps select the specific byte or half-word to be extracted.

## Sign Extension:

- The load\_unsigned\_in signal determines whether sign extension is performed during the load operation.
- If load\_unsigned\_in is high (active), the module performs zero extension for byte and half-word loads. This means the sign bit (bit 7 for bytes and bit 15 for half-words) is replicated to fill the higher-order bits of the output.  
If load\_unsigned\_in is low (inactive), the sign bit from the loaded data is replicated to extend the sign for signed byte or half-word loads

# STORE UINT



# STORE UNIT

## Address Calculation:

- The d\_addr\_out is assigned the base address from iadder\_in with the two least significant bits cleared to ensure alignment for byte or halfword stores.

## Data Formatting (always blocks):

- byte\_dout: This register stores the rs2\_in value shifted and padded with zeros depending on the byte address within the word
- halfword\_dout: This register stores the rs2\_in value shifted and padded with zeros depending on the halfword address within the word (

## Output Selection and Bus Signals (always blocks):

- If ahb\_ready\_in is high (bus ready), the block selects the appropriate data (byte\_dout, halfword\_dout, or rs2\_in) based on funct3\_in for writing. If ahb\_ready\_in is low (bus not ready), the block sets ahb\_htrans\_out to 0 (no transfer).

## Memory Write Request:

- The wr\_req\_out is simply assigned the same value as mem\_wr\_req\_in. This module doesn't modify the memory write request itself.

# WRITE BACK MUX



# WRITE BACK MUX

## ALU Second Source Mux (always block assignment):

- This assignment selects the second operand for the ALU based on the alu\_source\_reg\_in signal. It's done outside the main always block for efficiency.
- If alu\_source\_reg\_in is high, it selects the value from the second source register (rs2\_reg\_in).
- If alu\_source\_reg\_in is low, it selects the immediate value (imm\_reg\_in).

## Write-back Mux Selection (always block):

- The always@\* block selects the data source for writing back to the register file based on the wb\_mux\_sel\_reg\_in control signal.
- It uses a case statement to compare the control signal with predefined constants (WB\_ALU, WB\_LU, etc.). These constants represent different stages or results within the pipeline that can provide data for writing back.
- Depending on the matched case, the block assigns the corresponding data input (e.g., alu\_result\_in for ALU output, lu\_output\_in for Load Unit output, etc.) to the wb\_mux\_out register.

# WRITE ENABLE GENERATOR



# WRITE ENABLE GENERATOR

This module uses a simple assignment strategy to generate the write enable signals based on the flush\_in signal and the corresponding register file write enable signals from the pipeline (rf\_wr\_en\_reg\_in and csr\_wr\_en\_reg\_in).

## Flush Signal Priority:

- The flush\_in signal is an active high signal, likely used for pipeline flushes or other events where writes need to be disabled.
- The assignments prioritize the flush\_in signal. If flush\_in is high, both output write enables (wr\_en\_integer\_file\_out and wr\_en\_csr\_file\_out) are set to low (writes disabled), regardless of the original write enable signals from the pipeline.

## Normal Operation:

- If flush\_in is low (inactive), the module assigns the respective register file write enable signals (rf\_wr\_en\_reg\_in and csr\_wr\_en\_reg\_in) to the outputs (wr\_en\_integer\_file\_out and wr\_en\_csr\_file\_out). This allows writes to proceed based on the pipeline's write enable signals when there's no flush condition.

# DECODER



# DECODER

## Opcode Decoding:

- Decodes opcode\_in[6:2] to determine the type of instruction and sets corresponding flags (e.g., is\_branch, is\_jal, is\_jalr).

## Function Code Evaluation:

- Evaluates the funct3\_in field to determine specific operations for immediate ALU instructions and sets corresponding flags (is\_addi, is\_slti, is\_sltiu, is\_andi, is\_ori, is\_xori)..

## Memory Load/Store Handling:

- load\_size\_out and load\_unsigned\_out are set based on funct3\_in.
- mem\_wr\_req\_out is set for store instructions unless there's a misalignment or a trap.
- Immediate Address Handling:
- Sets iadder\_src\_out for load, store, and jump register instructions.
- Determines misaligned load/store conditions based on funct3\_in and iadder\_1\_to\_0\_in.

# INSTRUCTION DECODER

## Control Signal Generation:

- `csr_wr_en_out` and `csr_op_out` are set for CSR instructions.
- `rf_wr_en_out` is set for instructions that require writing back to the register file.
- `wb_mux_sel_out` and `imm_type_out` are set based on the type of instruction.
- `illegal_instr_out` is set if the instruction is not implemented or has an invalid opcode.

## Trap Handling:

- Misalignment checks (`misaligned_load_out`, `misaligned_store_out`) ensure proper handling of memory alignment issues

# IMMEDIATE GENERATOR



# IMMEDIATE GENERATOR

- **R-Type (imm\_type\_in = 000):** These instructions (e.g., add, sub) don't use an immediate value. The module sets imm\_out to zero by replicating the sign bit (bit 31) of the instruction 20 times and concatenating it with the instruction bits 31:20.
- **I-Type (imm\_type\_in = 001):** These instructions (e.g., addi, lw, sw) use a sign-extended immediate value in the range of  $-2^{12}$  to  $2^{12}-1$ .
- **S-Type (imm\_type\_in = 010):** These instructions (e.g., sw, addi) use a sign-extended immediate value in the range of  $-2^{12}$  to  $2^{12}-1$  for memory access..
- **B-Type (imm\_type\_in = 011):** These instructions (e.g., beq, jal) use a branch offset. The module creates a 32-bit immediate value by replicating the sign bit (bit 31) 19 times, concatenating it with bits 31, 7, 30:25, 11:8, and a 0 bit at the least significant position. This sign-extends the branch offset based on the branch target location.
- **U-Type (imm\_type\_in = 100):** These instructions (e.g., lui) use a 20-bit unsigned immediate value. The module takes instruction bits 31:12 and concatenates them with 12 zero bits, forming a 32-bit unsigned immediate value.
- **J-Type (imm\_type\_in = 101):** These instructions (e.g., jal) use a jump offset. The module replicates the sign bit (bit 31) 11 times, concatenates it with bits 31, 19:12, 20, 30:21, and a 0 bit, forming a 32-bit immediate value for the jump target address.

# ALU BLOCK



# ALU BLOCK

## Arithmetic:

- Addition:  $op\_1\_in + op\_2\_in$  (or  $op\_1\_in - op\_2\_in$  if the most significant bit of  $opcode\_in$  is 1)

## Logical:

- AND:  $op\_1\_in \& op\_2\_in$
- OR:  $op\_1\_in | op\_2\_in$
- XOR:  $op\_1\_in ^ op\_2\_in$

## Shift:

- Left Shift Logical:  $op\_1\_in << op\_2\_in[4:0]$
- Right Shift Logical:  $op\_1\_in >> op\_2\_in[4:0]$
- Right Shift Arithmetic:  $op\_1\_in >>> op\_2\_in[4:0]$  (sign-extended shift)

## Comparison:

- Set Less Than:  $op\_1\_in < op\_2\_in$  (sets the least significant bit of the result to 1 if  $op\_1\_in$  is less than  $op\_2\_in$ , otherwise 0)
- Set Less Than Unsigned:  $op\_1\_in < op\_2\_in$  (unsigned comparison)

# BRANCH UNIT



# BRANCH UNIT

## Opcode Decoding:

- Determines the instruction type (branch, jump) based on the opcode\_6\_to\_2\_in input.
- Sets corresponding flags is\_jal, is\_jalr, or is\_branch accordingly.

## Branch Condition Evaluation:

- For branch instructions, evaluates the branch condition based on funct3\_in and rs1\_in, rs2\_in values.
- Sets the take flag to indicate whether the branch should be taken.

## Jump Condition:

- For jump instructions, sets the pc\_mux\_sel flag to indicate that the PC should be updated with the jump target.

## Branch Taken Output:

- Combines the pc\_mux\_sel\_en and pc\_mux\_sel signals to determine the final branch\_taken\_out value.

# INSTRUCTION DECODER



# INSTRUCTION DECODER

- opcode\_out: Captures the opcode bits (bits 6:0) of the instruction.
- funct7\_out: Captures the funct7 bits (bits 31:25) of the instruction. (Might be unused depending on the implementation)
- funct3\_out: Captures the funct3 bits (bits 14:12) of the instruction.
- csr\_addr\_out: Captures the CSR address bits (bits 31:20) of the instruction. (Only used for CSR instructions)
- rs1\_addr\_out: Captures the register address bits for the first source operand (RS1) (bits 19:15).
- rs2\_addr\_out: Captures the register address bits for the second source operand (RS2) (bits 24:20).
- rd\_addr\_out: Captures the register address bits for the destination register (RD) (bits 11:7).
- instr\_31\_7\_out: Captures the remaining bits of the instruction from bit 31 to bit 7 (25 bits). This might be used for immediate value extraction or other instruction-specific purposes.

# REG BLOCK 1



# REG BLOCK 1

- The module uses a positive edge-triggered clock (posedge clk\_in) to synchronize its operation with the system clock.
- It includes an always block that executes on every positive clock edge.
- Inside the always block, there's an if-else statement to control the value written to the pc\_out register.
- If rst\_in is high (active reset), the pc\_out is reset to the BOOT\_ADDRESS. This ensures the program starts from the beginning when the system is reset.
- If rst\_in is low (reset inactive), the pc\_out is updated with the value provided by the pc\_mux\_in input. This allows the program counter to change based on the program execution flow.

# REG BLOCK 2



# REG BLOCK 2

The module uses a positive edge-triggered clock (posedge clk\_in) to synchronize its operation with the system clock.

- It includes an always block that executes on every positive clock edge.
- Inside the always block, there's an if-else statement to control the value written to the pc\_out register.
- If rst\_in is high (active reset), the pc\_out is reset to the BOOT\_ADDRESS. This ensures the program starts from the beginning when the system is reset.
- If rst\_in is low (reset inactive), the pc\_out is updated with the value provided by the pc\_mux\_in input. This allows the program counter to change based on the program execution flow.

# TOP MODULE



# TOP MODULE

- **Integration:** It integrates and connects all the lower-level modules of the design, establishing the overall system's architecture.
- **Interface Definition:** It defines the external interface of the design, including input and output ports, clock signals, and reset signals.
- **Configuration:** It includes parameters and settings that configure the entire design, such as clock frequencies, memory sizes, and other system-level parameters.
- **Control Logic:** It may contain control logic that orchestrates the interactions between different modules, ensuring proper sequencing and data flow.
- **Testbench Connection:** In simulation, it connects to the testbench, allowing the design to be verified and validated.

NarmathaA@mavenserver-RH2:~

Terminal Sessions View X server Tools Games Settings Macros Help



Quick connect...



```
# -----#
# B888888b. d8888 .d8888b. .d8888b.
# B88 Y88b d88888 d88P Y88b d88P Y88b
# B88 888 d88P888 Y88b. Y88b.
# B88 d88P d88P 888 *Y888b. *Y888b.
# B888888P* d88P 888 *Y88b. *Y88b.
# B88 d88P 888 *888 *888
# B88 d8888888888 Y88b d88P Y88b d88P
# B88 d88P 888 *Y8888P* *Y8888P*
# -----
# ----- NOP or NO_INSTRUCTION -----
# UVM_INFO verilog_src/uvm-1.1d/src/base/uvm_objection.svh(1267) @ 1800091500: reporter [TEST_DONE] 'run' phase is ready to proceed to
# 
# --- UVM Report Summary ---
# 
# ** Report counts by severity
# UVM_INFO :60008
# UVM_WARNING : 1
# UVM_ERROR : 0
# UVM_FATAL : 0
# ** Report counts by id
# [IN AGENT TOP] 1
# [Questa UVM] 2
# [RNTST] 1
# [TEST_DONE] 1
# [UVMTOP] 1
```

# CONCLUSION

The combination of advanced VLSI technology, the RISC-V ISA, and efficient pipeline designs makes it possible to develop high-performance, cost-effective processors suitable for a wide range of applications. The careful management of hazards and design considerations ensures that these processors meet the stringent demands of modern computing environments.

# LEARNING OUTCOME

- **Digital Design:** A deep understanding of digital design principles, including logic gates, Boolean algebra, and sequential logic.
- **Verilog HDL:** Proficiency in writing and simulating Verilog code, a crucial skill for hardware design.
- **Computer Architecture:** Knowledge of processor architecture, instruction set design, pipelining, and memory systems.
- **VLSI Design Flow:** Familiarity with the complete VLSI design flow, from RTL design to physical synthesis and layout.
- **Testing and Verification:** Experience in writing testbenches, simulating designs, and debugging issues.

# FUTURE SCOPE

- **AI and Machine Learning:** Custom RISC-V cores can be designed to accelerate specific AI and ML workloads, improving performance and efficiency.
- **Automotive and Aerospace:** RISC-V's open-source nature and potential for customization make it attractive for safety-critical applications in these industries.
- **Advanced Microarchitectures:** Continued research and development in microarchitecture techniques, such as out-of-order execution and speculative execution, will enhance the performance of RISC-V processors.
- The future of RISC-V RV32 processors looks bright. Its flexibility, open-source nature, and growing ecosystem make it a compelling choice for a wide range of applications. As the technology continues to mature, we can expect to see even more innovative and powerful RISC-V-based products in the years to come.