

# UVM Project

**Objective:** The project is divided into three main parts, with each part focusing on building and extending UVM environments for the individual components and the complete system. The final integration will demonstrate reuse of UVM environments.

## Important Notes:

- **Options:** There are two options either to do only part 1 of this project as individual project or do the full 3 parts as a team project.
- **Design Specifications:** explained in details in the link [here](#).
- **Golden Model implementation:** Write your golden model as a Verilog design module to make it easier for you and practice writing the design as well (Reach out to your assistant in case you are not sure how the scoreboard will check the DUT output against yours. Also, reach out to your assistant in case you implement the golden model as a task)
- **Team Size:** 3 and you must insert your team members [here](#)
- **Here is the snippet of expected waveform:**



The project will be executed in three distinct phases:

- RAM Verification
- SPI Slave Verification
- SPI Wrapper Verification re-using the RAM and SPI Slave environments

## Part 1: UVM Environment for SPI-Slave

In this part, you will create a full UVM environment to verify the **SPI slave design**.



## Sequences Requirements:

Split the sequences into the following sequences. You can switch on and off the constraints or use inline constraints to control the sequence item generated in the sequence

- reset\_sequence
- main\_sequence

## Constraint Requirements:

- 1- The reset signal (rst\_n) shall be deasserted most of the time.
- 2- The SS\_n signal to be high for one cycle every 13 cycles for all cases except read data to be high for one cycle every 23 cycles
- 3- Declare a randomized array of 11 bits to drive bit by bit the MOSI and the first 3 bits sent serially to the MOSI when the SS\_n falls are valid combinations only (000, 001, 110, 111)
- 4- The tx\_valid signal to be high in case of read data

(Hint: You can use post\_randomize for points 2 and 3)

## Functional Coverage Requirements:

- 1- Add coverpoints on rx\_data[9:8] to take all possible values and all possible transitions
- 2- Add coverpoints on SS\_n to capture:
  - Check full transaction duration:  $1 \rightarrow 0 [ *13 ] \rightarrow 1$  for normal operations.
  - Check extended transaction:  $1 \rightarrow 0 [ *23 ] \rightarrow 1$  for READ\_DATA.
- 3- Add coverpoints on MOSI to validate correct transitions:
  - 000 (Write Address)
  - 001 (Write Data)
  - 110 (Read Address)
  - 111 (Read Data)
- 4- Cross coverage between SS\_n and MOSI bins (Exclude irrelevant bins to focus on legal operation scenarios)

## Assertions Requirements:

- 1- An assertion ensures that whenever reset is asserted, the outputs (MISO, rx\_valid, and rx\_data) are all low.

2- An assertion checks that after any valid command sequence (write\_add\_seq(000), write\_data\_seq(001), read\_add\_seq(110), or read\_data\_seq(111)), the rx\_valid signal must assert exactly after 10 cycles and the SS\_n should eventually after the 10 cycles to close communication.

3- Add the following assertions in the design to check correct FSM transitions and guard the assertions using conditional compilation with the `ifdef directive with macro named "SIM", and then include the macro in the vlog command +define+SIM option in the vlog command of your do file. Refer to [this](#) link to learn more about conditional compilation.

- IDLE → CHK\_CMD
- CHK\_CMD → WRITE or READ\_ADD or READ\_DATA
- WRITE → IDLE
- READ\_ADD → IDLE
- READ\_DATA → IDLE

**Note:** You are free to add assertions, covergroups or constraints to enrich your verification.

## Part 2: UVM Environment for Single-Port RAM

In this part, you will create a full UVM environment to verify the **RAM design** with default parameters (MEM\_DEPTH: 256, ADDR\_SIZE: 8)

### Interface:

| Name     | Type   | Size    | Description                                                                                                                                 |
|----------|--------|---------|---------------------------------------------------------------------------------------------------------------------------------------------|
| din      |        | 10 bits | Data Input                                                                                                                                  |
| clk      |        | 1 bit   | Clock                                                                                                                                       |
| rst      | Input  | 1 bit   | Active low synchronous reset                                                                                                                |
| rx_valid |        | 1 bit   | If HIGH: accept din[7:0] to save the write/read address internally or write a memory word depending on the most significant 2 bits din[9:8] |
| dout     |        | 8 bits  | Data Output                                                                                                                                 |
| tx_valid | Output | 1 bit   | Whenever the command is memory read the tx_valid should be HIGH                                                                             |

### Sequences Requirements:

Split the RAM sequences into the following sequences. You can switch on and off the constraints or use inline constraints to control the sequence item generated in the sequence

- reset\_sequence
- write\_only\_sequence
- read\_only\_sequence
- write\_read\_sequence

## Constraint Requirements:

- 1- The reset signal (rst\_n) shall be deasserted most of the time.
- 2- The rx\_valid signal shall be asserted most of time.
- 3- For a *write-only sequence*, every Write Address operation shall always be followed by either Write Address or Write Data operation.
- 4- For a *read-only sequence*, every Read Address operation shall always be followed by either Read Address or Read Data operation.
- 5- For a *randomized read/write sequence*, the following ordering rules shall be enforced:
  - Every Write Address operation shall always be followed by either Write Address or Write Data operation.
  - After a Write Data, the next operation shall be chosen with the following probability distribution: 60% → Read Address & 40% → Write Address
  - Every Read Address operation shall always be followed by either Read Address or Read Data operation. After a Read Data, the next operation shall be chosen with the following probability distribution: 60% → Write Address & 40% → Read Address

(Hint: You can use post\_randomize and inline constraints)

## Functional Coverage Requirements:

- 1- Write Coverpoint to check transaction ordering for din[9:8]:
  - Check din[9:8] takes 4 possible values
  - Check write data after write address
  - Check read data after read address
  - Check write address => write data => read address => read data

### 3- Cross coverage:

- Between all bins of din[9:8] and rx\_valid signal when it is high
- Between din[9:8] when it equals read data and tx\_valid when it is high

## Assertions Requirements: (Add the assertions file and bind it in the top module)

- 1- An assertion ensures that whenever reset is asserted, the output signals (tx\_valid and dout) are low
- 2- An assertion checks that during address or data input phases (write\_add\_seq, write\_data\_seq, read\_add\_seq), the tx\_valid signal must remain deasserted.
- 3- An assertion checks that after a read\_data\_seq occurs, the tx\_valid signal must rise to indicate valid output and after it rises by one clock cycle, it should eventually fall.

4- An assertion checks that every **Write Address** operation must be eventually followed by a **Write Data** operation.

5- An assertion checks that every **Read Address** operation must be eventually followed by a **Read Data** operation.

**Note:** You are free to add assertions, covergroups or constraints to enrich your verification.

## Part 3: UVM Environment for SPI Wrapper

In this part, you will build a complete UVM environment to verify the end to end functionality of the **SPI wrapper** design. The environment should integrate both the **Single-Port RAM** and the **SPI Slave** environments developed previously to reuse them but with **passive** agents (check the last extra assignment to learn how to do it).

### Constraint Requirements:

- 1- The reset signal (rst\_n) shall be deasserted most of the time.
- 2- The SS\_n signal to be high for one cycle every 13 cycles for all cases except read data to be high for one cycle every 23 cycles
- 3- Declare a randomized array of 11 bits to drive bit by bit the MOSI and the first 3 bits sent serially to the MOSI when the SS\_n falls are valid combinations only (000, 001, 110, 111)
- 4- For a *write-only sequence*, every Write Address operation shall always be followed by either Write Address or Write Data operation.
- 5- For a *read-only sequence*, every Read Address operation shall always be followed by either Read Address or Read Data operation.
- 6- For a *randomized read/write sequence*, the following ordering rules shall be enforced:
  - Every Write Address operation shall always be followed by either Write Address or Write Data operation.
  - After a Write Data, the next operation shall be chosen with the following probability distribution: 60% → Read Address & 40% → Write Address
  - Every Read Address operation shall always be followed by either Read Address or Read Data operation. After a Read Data, the next operation shall be chosen with the following probability distribution: 60% → Write Address & 40% → Read Address

*(Hint: You can use post\_randomize)*

### Assertions Requirements:

- 1- An assertion ensures that whenever reset is asserted, the outputs (MISO, rx\_valid, and rx\_data) are all inactive.

- 2- An assertion to make sure that the MISO remains with a stable value as long as it is not a read data operation

>> Bind the assertion modules for the RAM and SPI Slave and SPI wrapper in the top module

### Sequences Requirements:

Split the sequences into the following sequences. You can switch on and off the constraints or use inline constraints to control the sequence item generated in the sequence

- reset\_sequence
- write\_only\_sequence
- read\_only\_sequence
- write\_read\_sequence

**Note:** You are free to add assertions, covergroups or constraints to enrich your verification.

### Requirements:

- Verification plan
- Draw your UVM testbench showing the UVM structure using PPT, draw.io or MS Visio
  - Write a section where you will describe in detail how your UVM testbench work.
- Code snippets
- Code Coverage report
  - 100% (You can exclude the 2-D RAM declaration from the toggle coverage in the RAM design)
- Functional Coverage report, and Sequential Domain Coverage report
- Bug report
- Sections with QuestaSim snippets to for each UVM sequence and how the interface is driven in each sequence using the waveform snippets
- Also, provide a table showing the assertions used as follows:

| Feature                                   | Assertion                                        |
|-------------------------------------------|--------------------------------------------------|
| Whenever the rst is asserted, MISO is low | <code>@(posedge clk) (!rst_n  &gt; ~MISO)</code> |

### Submission File:

.rar file (<your\_name>\_Project2 for example Kareem\_Waseem\_Project2) having:

- 1- 3 Folders (Folder for each part containing the design files, UVM Codes, Do File, and the coverage report)
- 2- PDF file has the requirements above with your names on the first page. Please use the same certificate names [here](#).