



# Design & Verification of an ALU using **UVM & SVA**

*Abdelrahman Mohamed Yassien*

*dir:*  
*/home/svgpdv25abohamad/alu\_uvm\_project*

# Project Timeline



# ALU DESIGN SPECIFICATIONS

| <b>A_op</b> | <b>C = ?</b>       | <b>Description</b>               |
|-------------|--------------------|----------------------------------|
| 0           | $A + B$            | Sum of inputs                    |
| 1           | $A - B$            | B subtracted from A              |
| 2           | $A \wedge B$       | A bitwise XOR B                  |
| 3           | $A \& B$           | A bitwise & B                    |
| 4           | $A \& B$           | A bitwise & B                    |
| 5           | $A   B$            | A bitwise OR B                   |
| 6           | $\sim(A \wedge B)$ | A XNOR B                         |
| 7           | NULL               | Not permitted to take this value |



| <b>(A_EN &amp; ~B_EN)</b> | <b>B_op</b> | <b>C= ?</b>  | <b>Description</b>  |
|---------------------------|-------------|--------------|---------------------|
|                           | 0           | $A + B$      | Sum of inputs       |
|                           | 1           | $A - B$      | B subtracted from A |
|                           | 2           | $A \wedge B$ | A bitwise XOR B     |
|                           | 3           | $A \& B$     | A bitwise & B       |

| <b>(~A_EN &amp; B_EN)</b> | <b>B_op</b> | <b>C = ?</b>   | <b>Description</b>               |
|---------------------------|-------------|----------------|----------------------------------|
|                           | 0           | $\sim(A \& B)$ | A NAND B                         |
|                           | 1           | $A + B$        | Sum of inputs                    |
|                           | 2           | $A + B$        | Sum of inputs                    |
|                           | 3           | NULL           | Not permitted to take this value |

# TOP ALU TESTBENCH



# RST AGENT TLM DIAGRAM

- The reset awareness was inspired from The published paper [Handling Asynchronous Reset\(s\) Testing by building reset-awareness into UVM testbench components by Wei Wei Cheong, Katherine Garden, Ana Sanz Carretero which was previewed @DVCON 2021](#)
- As shown in the figure, the rst agent monitor class is always waiting for reset assertion, and as soon as it is asserted, the monitor class writes to the uvm\_tlm\_analysis\_port so it reaches every reset\_aware component and hence allows them to act accordingly.



# ALU AGENT TLM DIAGRAM

- The reset awareness was inspired from The published paper [Handling Asynchronous Reset\(s\) Testing by building reset-awareness into UVM testbench components by Wei Wei Cheong, Katherine Garden, Ana Sanz Carretero which was previewed @DVCON 2021](#)
- As shown in the figure, the alu agent is fully reset aware, every component is connected with uvm\_tlm\_analysis\_imp s to allow it to react to asynchronous reset assertion.
- Note: the sequencer in this agent the **stop\_sequences()** function is called within the write function body defined @it, therefore stopping all sequences and calling item\_done to all the current running items by the driver.



# SCOREBOARD TLM DIAGRAM

The hierarchy of this scoreboard is inspired from [Clifford Cumming's OVM-UVM Scoreboards-Fundamental Architectures](#)

The reset awareness was inspired from The published paper [Handling Asynchronous Reset\(s\) Testing by building reset-awareness into UVM testbench components](#) by Wei Wei Cheong, Katherine Garden, Ana Sanz Carretero which was previewed @DVCON 2021



# SUBSCRIBER TLM DIAGRAM

The reset awareness was inspired from The published paper The  
reset awareness was inspired from The published paper [Handling  
Asynchronous Reset\(s\) Testing by building reset-awareness into UVM  
testbench components](#) by Wei Wei Cheong, Katherine Garden, Ana  
Sanz Carretero which was previewed @DVCON 2021



## OBJECT ORIENTED PROGRAMMING



# Classes? What do they do?

# Configurations

# Configurations





# Sequence Items

# ALU Sequence Item



# RST Sequence Item



Defines RAND RST stimulus of the DUT, and also RST\_DURATION, RST\_DELAY...and other static variables to control coverage and rsts assertion during the test

Registers all the stimulus within the factory to be compared, copied, cloned, printed, packed or unpacked (overriding each function do\_X respectively)

Defines the constraints for the rst stimulus and control variables/signals

# Environment & It's Components



# ENVIRONMENT



# AGENT



# ALU ACTIVE AGENT



# ALU DRIVER



# ALU INPUT MONITOR



# ALU OUTPUT MONITOR



# ALU SEQUENCER



# RST ACTIVE AGENT



# RST DRIVER



RST  
DRIVER



# RST MONITOR



# RST SEQUENCER



# SCOREBOARD



# SCOREBOARD



Instantiates the COMPARATOR & PREDICTOR and creates them using uvm\_factory accordingly

Connects the EXPORTS coming from the environment to the EXPORTS inside the COMPARATOR & the PREDICTOR  
Connects the EXPORT Inside the Comparator to the PORT inside the predictor to get the PREDICTED ITEM and compare it to the ACTUAL SAMPLED ITEM

Halts the test using TEST TERMINATION TECHNIQUE used in [Clifford Cummings' OVM & UVM TECHNIQUES FOR TERMINATING TESTS](#). Raising the objection using the Phase\_ready\_to\_end\_phase

Keeps track of Incorrect & Correct comparisons during simulation and displays them in a report

# PREDICTOR



Gets the input stimulus sent by the input Monitor using the `.get()` blocking task from the `tlm_analysis_fifo` declared within it.

Predicts the Expected Result, assigns it to a `seq_item` and writes it to the `tlm_analysis_port` declared within it to the **COMPARATOR**

Is reset aware, disables all the operations as soon as a reset is detected through the reset TLM

# COMPARATOR



Gets the output that was sampled by the output monitor through the uvm\_tlm\_analysis\_fifo using .get() blocking task

Compares the sampled outputs to the expected seq\_item sent by the predictor

Is reset aware, disables all the operations as soon as a reset is detected through the reset TLM



# SUBSCRIBER

# SUBSCRIBER



Receives input stimulus & outputs monitored by the monitor using `fork_joined .get()` task to each analysis FIFOs which are declared for INPUT STIMULUS OUTPUTS MONITORED RESPECTIVELY and samples each time the write function is called by the `uvm_tlm_analysis_port` inside the monitors.

When the parametrized coverage group of each test reaches 100% coverage, it asserts `cov_target` accordingly and therefore the test ends if all the resets are finished as well.

Is reset aware therefore it is disabled when a reset assertion happens.

# SEQUENCES

# SEQUENCES

# ALU random sequence



Randomizes the input stimulus of the DUT according to the specified constraints using the alu\_seq\_item

keeps looping in a forever loop until coverage for the perspective test that runs the sequence hits 100.

keeps looping in a forever loop until the number of resets that is to be asserted which is defined by the test\_config is finished as well.

# ALU Error sequence



Randomizes the input stimulus of the DUT according to the specified constraints using the `error_seq_item`

keeps looping in a forever loop until coverage for the perspective test that runs the sequence hits 100.

keeps looping in a forever loop until the number of resets that is to be asserted which is defined by the `test_config` is finished as well.

# ALU repetition sequence



Randomizes the input stimulus of the DUT according to the specified constraints using the `alu_seq_item`

keeps looping in a forever loop until coverage for the perspective test that runs the sequence hits 100.

keeps looping in a forever loop until the number of resets that is to be asserted which is defined by the `test_config` is finished as well.

# Reset Sequence



Randomizes the `rst_seq_item` properties  
(`rst_duration`, `rst_delay`, `rst_on..etc`)

keeps looping in a forever loop until the  
number of resets that is to be asserted  
which is defined by the `test_config` is  
finished as well.

# Virtual Sequencer



# Virtual Sequencer



Instantiates the alu\_sequencer & the  
rst\_sequencer so they can be used later  
to point to the sequencers of the same  
type inside the virtual sequence and  
start sequences respectively on them

# ALU Interface



ALU Interface has all of the input signals of the DUT except the rst

# RST Interface



RST Interface has `rst_n`

Also has the event which is triggered to start the `rst` assertion from the `top_testbench`

Receives the `rst_delay` & `rst_duration` from the verification environment and asserts the `rst` according to those values

# TESTS & VIRTUAL SEQUENCES



# Virtual Sequences



# Test & Test Base



**Reset  
Awareness &  
Asynchronous  
reset**

# Asynchronous RST

- To assert a true asynchronous reset one would need to do it through the top\_testbench @random times, which is exactly what the RST Agent does, it asserts the reset @Randomized times Asynchronously by triggering an event in the RST Interface which in turn starts a reset assertion in the top\_testbench.
- The **resets are spaced using @Reset\_DELAY** which is also randomized and constrained.
- The **reset\_DURATION** is also randomized and constrained.
- They are **both passed to the interface task and then to the top\_testbench.**

```
//-----
// Connect phase
//-----
function void connect_phase(uvm_phase phase);
super.connect_phase(phase);

// Connect alu agent to env TLM
alu_agent_h.inputMon2agent.connect(alu_in_env_ap);
alu_agent_h.outputMon2agent.connect(alu_out_env_ap);

// Connect env to scoreboard
alu_in_env_ap.connect(scb_h.analysis_export_inputs);
alu_out_env_ap.connect(scb_h.analysis_export_actual_outputs);

// Connect env to subscriber
alu_in_env_ap.connect(sub_h.analysis_export_inputs);
alu_out_env_ap.connect(sub_h.analysis_export_outputs);

//****************************************************************************

// Connect reset agent to env
rst_agent_h.reset_collected_port_n.connect(reset_collected_port_n); // reset_agent_to_env
rst_agent_h.reset_collected_port_p.connect(reset_collected_port_p); // reset_agent_to_env

// Connect env to alu agent
reset_collected_port_n.connect(alu_agent_h.reset_collected_export_n);
reset_collected_port_p.connect(alu_agent_h.reset_collected_export_p);

// Connect reset to alu agent
reset_collected_port_n.connect(scb_h.reset_collected_export); // scoreboard we may send posedge rst later
reset_collected_port_n.connect(sub_h.reset_collected_export); // subscriber we may send posedge rst later

//****************************************************************************

// sequencers connection
v_seqr.seqr_ALU = alu_agent_h.sequencer_h;
v_seqr.seqr_RST = rst_agent_h.sequencer_h;
```

# Reset Awareness

- So, the **driver calls the interface task**, but what does the monitor do?
- **The monitor permanently observes the state of the reset through the virtual interface and as soon as it is asserted either high or low.**

it notifies all the other components within the Environment through UVM\_TLM\_ANALYSIS\_PORTS passing through the RST Agent, Env to UVM\_TLM\_ANALYSIS\_EXPORTS in the the ALU Agent and its components, Subscriber & Scoreboard.

Some of these components have direct UVM\_TLM\_ANALYSIS\_IMPS or a mix of UVM\_TLM\_ANALYSIS\_EXPORTS & UVM\_TLM\_ANALYSIS\_FIFOS



```

//-
// Connect phase
//-----
function void connect_phase(uvm_phase phase);
  super.connect_phase(phase);

  // Connect alu agent to env TLM
  alu_agent_h.inputMon2agent.connect(alu_in_env_ap);
  alu_agent_h.outputMon2agent.connect(alu_out_env_ap);

  // Connect env to scoreboard
  alu_in_env_ap.connect(scb_h.analysis_export_inputs);
  alu_out_env_ap.connect(scb_h.analysis_export_actual_outputs);

  // Connect env to subscriber
  alu_in_env_ap.connect(sub_h.analysis_export_inputs);
  alu_out_env_ap.connect(sub_h.analysis_export_outputs);

  //-----
  // Connect reset agent to env
  rst_agent_h.reset_collected_port_n.connect(reset_collected_port_n); // reset_agent_to_env
  rst_agent_h.reset_collected_port_p.connect(reset_collected_port_p); // reset_agent_to_env

  // Connect env to alu agent
  reset_collected_port_n.connect(alu_agent_h.reset_collected_export_n);
  reset_collected_port_p.connect(alu_agent_h.reset_collected_export_p);

  // Connect reset to alu agent
  reset_collected_port_n.connect(scb_h.reset_collected_export); // scoreboard we may send posedge rst later
  reset_collected_port_n.connect(sub_h.reset_collected_export); // subscriber we may send posedge rst later

  //-----
  // sequencers connection
  v_seqr.seqr_ALU = alu_agent_h.sequencer_h;
  v_seqr.seqr_RST = rst_agent_h.sequencer_h;
  
```



# TLM debugging & Tracing

The get provided to & get connected to functions help debugging the tlm of your uvm hierarchy if there is ever an issue with it.

It gets passed a property of type uvm\_port\_list which I named list in this case.

```
//-----  
// End of elaboration phase - for reporting connection details  
//-----  
function void end_of_elaboration_phase(uvm_phase phase);  
    super.end_of_elaboration_phase(phase);  
  
    // Set verbosity level for the scoreboard  
    scoreboard_h.set_report_verbosity_level_hier(UVM_HIGH);  
  
    // Log connections and provided ports/exports  
    alu_agnt.driver.reset_collected_imp.get_provided_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_agnt.driver.reset_collected_imp.get_connected_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_reset_agnt.monitor.reset_collected_port.get_provided_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_reset_agnt.monitor.reset_collected_port.get_connected_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_reset_agnt.reset_collected_port.get_provided_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_reset_agnt.reset_collected_port.get_connected_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
  
    alu_agnt.reset_collected_export.get_provided_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
    alu_agnt.reset_collected_export.get_connected_to(list);  
    `uvm_info(get_name(), $sformatf("%p", list), UVM_LOW);  
endfunction
```

# Virtual Sequence Structure

- If a reset is asserted, all the reset aware components get disabled and restart their processes again, therefore a forever loop is used in the sequences, and a do-while loop is used in the virtual sequences(in the repititon\_v\_seq).
- **Both sequences (RST & repititon sequence) need to be running together not to ruin the driver/sequencer handshake, therefore causing errors.**
- If the reset sequence finishes, it doesn't affect the other alu\_sequence since they have **stop\_sequences()** called within their sequencers.
- On the other hand, if the roles were reversed, **the reset\_sequencer would generate errors due to wait for grant task being terminated without an ack**

```
fork
begin
    rst_seq.start(p_sequencer.seqr_RST);
    `uvm_info(get_name, $sformatf("seqr_RST Arbitration mode = %s", seqr_RST.get_arbit
end
begin
    fork
        op_a_seq_h.start(p_sequencer.seqr_ALU , .this_priority(100));
        op_b01_seq_h.start(p_sequencer.seqr_ALU , .this_priority(200));
        op_b11_seq_h.start(p_sequencer.seqr_ALU , .this_priority(300));
    join
end
begin
    #test_timeout;
    `uvm_info(get_name, ("TIME OUT!!"), UVM_LOW)
    alu_seq_item::cov_target=1;
    rst_seq_item::resets_done=1;
end
join_any
    disable fork;
end while ((!alu_seq_item::cov_target) || (!rst_seq_item::resets_done));
task
```

# Virtual Sequence Structure

- A timeout is added to ensure the test doesn't hang if the coverage target isn't hit.
- The timeout is set by the top testbench and is transferred to the environment using the **test\_config** through the base\_test.
- Sequence Arbitration is added for practice, in this mode, the sequence **with the highest priority inside the fork\_join gets started first**
- The base of the Virtual Sequence (base\_v\_seq) is used **to instantiate the sequencers, declare the p\_sequencer**
- Each v\_seq is responsible for **starting the sequences using the p\_sequencer**

```
fork
begin
    rst_seq.start(p_sequencer.seqr_RST);
    `uvm_info(get_name, $sformatf("seqr_RST Arbitration mode = %s", seqr_RST.get_arbit
end
begin
    fork
        op_a_seq_h.start(p_sequencer.seqr_ALU , .this_priority(100));
        op_b01_seq_h.start(p_sequencer.seqr_ALU , .this_priority(200));
        op_b11_seq_h.start(p_sequencer.seqr_ALU , .this_priority(300));
    join
end
begin
    #test_timeout;
    `uvm_info(get_name, ("TIME OUT!!"), UVM_LOW)
    alu_seq_item::cov_target=1;
    rst_seq_item::resets_done=1;
end
join_any
    disable fork;
end while ((!alu_seq_item::cov_target) || (!rst_seq_item::resets_done));
task
```

# Test Termination

This Test termination technique is mentioned in [Clifford cummings' OVM & UVM Techniques for Terminating Tests](#)

- The **phase ready to end** checks if the run phase is still running, then checks for the test's termination flag.
- **(alu\_seq\_item::cov\_target) which is asserted only when respective coverage for the respective test reaches 100%.**
- **(rst\_seq\_item::resets\_done) which is only asserted when the specified number of resets which were defined through the test\_config file are finished.**
- Used the **Final Phase** for the scoreboard to determine when the scoreboard was shutting down to make sure no items are slipping through the test without being compared properly.

```
// -----
// phase_ready_to_end phase
// -----
function void phase_ready_to_end(uvm_phase phase);
  if (phase.get_name() != "run") return;
  if (~alu_seq_item::cov_target || ~rst_seq_item::resets_done) begin
    phase.raise_objection(.obj(this));
    fork
      begin
        delay_phase(phase);
      end
      join_none
    end
  endfunction

//-----
// delay phase
//-----
task delay_phase(uvm_phase phase);
  wait(alu_seq_item::cov_target && rst_seq_item::resets_done);
  phase.drop_objection(.obj(this));
endtask

//-----
// final phase
//-----
function void final_phase(uvm_phase phase);
  super.final_phase(phase);
  `uvm_info("SCOREBOARD", "Scoreboard is stopping.", UVM_LOW)
endfunction
```

# Ensuring Coverage target is hit

- Both reset sequence & random the other sequence which is running alongside it is in a forever loop waiting to hit 100% coverage or to hit test timeout
- Breaking from the body task only when the number of specified resets is done, and the cov\_target is done.

```
//-----  
virtual task body();  
  
    forever begin  
        req = alu_seq_item::type_id::create("req");  
        start_item(req);  
  
        if(!req.randomize())  
            `uvm_error(get_type_name(), "Randomization failed")  
  
        finish_item(req);  
  
        if(alu_seq_item::cov_target && rst_seq_item::resets_done) break;  
  
    //-----  
virtual task body();  
    req = rst_seq_item::type_id::create("req");  
    forever begin  
        start_item(req);  
        ok = req.randomize();  
        if(!ok)  
            `uvm_error(get_type_name(), "Randomization @rst_seq_item failed")  
        finish_item(req);  
        if(rst_seq_item::resets_done)begin  
            $display("atyab? ma fe atyab mn hek");  
            break;  
        end  
    end
```

# Type Overrides

```
virtual function void build_phase(uvm_phase phase);
  super.build_phase(phase);
  // override virtual sequence
  base_v_seq::type_id::set_type_override(random_v_seq::type_id::get());
  v_seq = base_v_seq::type_id::create("v_seq", this);

  cmd = uvm_cmdline_processor::get_inst();
  cmd.get_plusargs(plusargs_queue);
  `uvm_info (get_type_name(), $sformatf("PLUSARGS = %p", plusargs_queue), UVM_LOW)
endfunction : build_phase
```

**Utilizing report phase to report after the conclusion of the test**

```
//-----
// Report phase
//-----
function void report_phase(uvm_phase phase);
  `uvm_info(get_type_name(),
    $sformatf("\nReset Driver Report:\n\tTotal Resets: %0d",rst_seq_item::reset_count), UVM_LOW)

  `uvm_info(get_type_name(), "Reset Driver Report Phase Complete", UVM_LOW)
endfunction : report_phase
```

```
  virtual function void build_phase(uvm_phase phase);
    super.build_phase(phase);
    // override virtual sequence
    base_v_seq::type_id::set_type_override(error_v_seq::type_id::get());
    alu_seq_item::type_id::set_type_override(error_seq_item::type_id::get());
```

**Changing verbosity level of a certain part in the hierarchy**

```
// Set verbosity level for the scoreboard
scoreboard_h.set_report_verbosity_level_hier(UVM_HIGH);
```

# Random Test coverage target hit

```
Total Incorrect Items: 0
JVM_INFO env/env_components/scoreboard.sv(105) @ 22925: uvm_test_top.env_h.scb_h [scoreboard] Scoreboard Report Phase Complete
JVM_INFO env/env_components/subscriber.sv(524) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Received transactions: 1112
JVM_INFO env/env_components/subscriber.sv(526) @ 22925: uvm_test_top.env_h.sub_h [subscriber]
Coverage Report:
JVM_INFO env/env_components/subscriber.sv(527) @ 22925: uvm_test_top.env_h.sub_h [subscriber] TEST_NAME = random_test
JVM_INFO env/env_components/subscriber.sv(532) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Control Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(533) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Input Values Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(534) @ 22925: uvm_test_top.env_h.sub_h [subscriber] A Operations Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(535) @ 22925: uvm_test_top.env_h.sub_h [subscriber] B Operations01 Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(536) @ 22925: uvm_test_top.env_h.sub_h [subscriber] B Operations11 Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(537) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Output Values Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(538) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Data Ranges Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(539) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Stability Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(540) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Special Cases Coverage: 100.00%
JVM_INFO env/env_components/subscriber.sv(553) @ 22925: uvm_test_top.env_h.sub_h [subscriber] Total Coverage: 100.00%
JVM_INFO env/env.sv(168) @ 22925: uvm_test_top.env_h [env] Environment Report Complete
JVM_INFO tests/base_test.sv(103) @ 22925: uvm_test_top [random_test] TEST PASSED
JVM_INFO env/env_components/scoreboard.sv(94) @ 22925: uvm_test_top.env_h.scb_h [SCOREBOARD] Scoreboard is stopping.
JVM_INFO /tools/Synopsys/install/vcs/V-2023.12-1/etc/uvm-1.2/base/uvm_report_catcher.svh(705) @ 22925: reporter [UVM/REPORT/CATCHER]
```



# Repetition Test Coverage target hit

```
Reset Driver Report:  
  Total Resets: 4  
VM_INFO env/env_components/rst_agnt_components/rst_driver.sv(72) @ 10985: uvm_test_top.env_h.rst_agent_h.driver_h [rst_driver] Reset Driver Report Phase Complete  
VM_INFO env/env_components/scoreboard.sv(101) @ 10985: uvm_test_top.env_h.scb_h [scoreboard]  
coreboard Report:  
  Total Transactions: 515  
  Total Correct Items: 515  
  Total Incorrect Items: 0  
VM_INFO env/env_components/scoreboard.sv(105) @ 10985: uvm_test_top.env_h.scb_h [scoreboard] Scoreboard Report Phase Complete  
VM_INFO env/env_components/subscriber.sv(524) @ 10985: uvm_test_top.env_h.sub_h [subscriber] Received transactions: 515  
VM_INFO env/env_components/subscriber.sv(526) @ 10985: uvm_test_top.env_h.sub_h [subscriber]  
coverage Report:  
VM_INFO env/env_components/subscriber.sv(527) @ 10985: uvm_test_top.env_h.sub_h [subscriber] TEST_NAME = repitition_test  
VM_INFO env/env_components/subscriber.sv(544) @ 10985: uvm_test_top.env_h.sub_h [subscriber] repitition cg for op_A Coverage: 100.00%  
VM_INFO env/env_components/subscriber.sv(545) @ 10985: uvm_test_top.env_h.sub_h [subscriber] repitition cg for op_B1 Coverage: 100.00%  
VM_INFO env/env_components/subscriber.sv(546) @ 10985: uvm_test_top.env_h.sub_h [subscriber] repitition cg for op_B2 Coverage: 100.00%  
VM_INFO env/env_components/subscriber.sv(553) @ 10985: uvm_test_top.env_h.sub_h [subscriber] Total Coverage: 100.00%  
VM_INFO env/env.sv(168) @ 10985: uvm_test_top.env_h [env] Environment Report Complete  
VM_INFO tests/base_test.sv(103) @ 10985: uvm_test_top [repitition_test] TEST PASSED  
VM_INFO env/env_components/scoreboard.sv(94) @ 10985: uvm_test_top.env_h.scb_h [SCOREBOARD] Scoreboard is stopping.  
VM_INFO env/env.sv(168) @ 10985: uvm_test_top [repitition_test] TEST PASSED
```



# Assertions



# Assertions

- Asserting over every possible scenario, even the error ones, providing CEXs to see them being triggered during the error\_test run
- Asserting over each property with an \$error if the property fails
- Covering each property

```
property p_output_update;
  disable iff (~rst_n)
  @alu_intf.driver_cb
  $changed({alu_intf.ALU_EN_STATE_o, alu_intf.OP_MODE_o, alu_intf.OP_A_o, alu_intf.OP_B01_o,
            alu_intf.OP_B11_o, alu_intf.A, alu_intf.B}) |=>
    // First check for NULL operation - this should cause assertion failure
    !is_null_operation($past(alu_intf.ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o),
                        $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o)) &&
    // Then check for forbidden results (-32, or MODE_B2 special forbidden values)
    !is_forbidden_result($past(alu_intf.ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o),
                          $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o),
                          $past(alu_intf.A), $past(alu_intf.B)) &&
    // Then check for expected output value based on the mode
    alu_intf.C == compute_result(rst_intf.rst_n, $past(alu_intf.ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o),
                                $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o),
                                $past(alu_intf.A), $past(alu_intf.B));
endproperty
```

```
OP_B11_t op_B11; // Operation
int signed low_range, int signed high_range,           // Valid range [low:high]
int signed invalid_min, int signed invalid_max        // Invalid values or specific exclusions
);
@(alu_intf.monitor_cb) disable iff (!rst_n)
  (alu_intf.ALU_EN_STATE_o == ALU_ON &&
   alu_intf.OP_MODE_o == op_MODE &&
   (op_MODE == MODE_A ? (alu_intf.OP_A_o == op_A) : 1'b1) &&
   (op_MODE == MODE_B01 ? (alu_intf.OP_B01_o == op_B01) : 1'b1) &&
   (op_MODE == MODE_B11 ? (alu_intf.OP_B11_o == op_B11) : 1'b1) &&
   (! (op_MODE == MODE_IDLE)))
  |=> (alu_intf.C inside {[low_range:high_range]} && !(alu_intf.C inside {[invalid_min:invalid_max]}));
endproperty

// Properties for specific operations using parameterized templates
// Mode A operations
property p_mode_a_add;
  p_op_result_range(MODE_A, A_ADD, 'x, 'x, -30, 30, -32, -31);
endproperty

property p_mode_a_sub;
  p_op_result_range(MODE_A, A_SUB, 'x, 'x, -30, 30, -32, -31);
endproperty
```

# Error Test CEX assertions firing

```
Offending '((alu_intf.C inside ([-13]:17)) && !(alu_intf.C == (-14))))'
error: "interfaces/sva_interface.sva_interface.sv", 196: alu_top_tb.sva_intf.a_mode_b2_b_add_2: at time 8015 ns
code B2 B+2 operation resulted in invalid value
interfaces/sva_interface.sva_interface.sv", 45: alu_top_tb.sva_intf.a_output_update: started at 8020ns failed at 8020ns
    Offending '(((!sva_helper_pkg::is_null_operation($past(alu_intf.ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o), $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o))) && (!sva_helper_pkg::alu_intf.ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o), $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o), $past(alu_intf.A), $past(alu_intf.B))) && (alu_intf.C == sva_helper_pkg::ALU_EN_STATE_o), $past(alu_intf.OP_MODE_o), $past(alu_intf.OP_A_o), $past(alu_intf.OP_B01_o), $past(alu_intf.OP_B11_o), $past(alu_intf.A), $past(alu_intf.B)))'
error: "interfaces/sva_interface.sva_interface.sv", 45: alu_top_tb.sva_intf.a_output_update: at time 8020 ns
Invalid ALU computation result detected en_a[1, en_b[1, a_op[1, b_op[3, A[0, B[16, actual_c -14, expected_c -14
VM_INFO /tools/Synopsys/install/vcs/V-2023.12-1/etc/uvm-1.2/base/uvm_objection.svh(1276) @ 8025: reporter [TEST_DONE] 'run' phase is ready to proceed to the 'extract' phase
VM_INFO env/env_components/alu_agnt_components/alu_driver.sv(100) @ 8025: uvm_test_top.env_h.alu_agent_h.driver_h [alu_driver]
ALU driver Report:
    Total driven_pkts: 11
VM_INFO env/env_components/alu_agnt_components/alu_driver.sv(103) @ 8025: uvm_test_top.env_h.alu_agent_h.driver_h [alu_driver] ALU Driver Report Phase Complete
VM_INFO env/env_components/alu_agnt_components/alu_monitor_in.sv(117) @ 8025: uvm_test_top.env_h.alu_agent_h.monitor_in_h [alu_monitor_in]
Monitor_In Report:
    Total Inputs_monitored_pkts: 367
VM_INFO env/env_components/alu_agnt_components/alu_monitor_in.sv(120) @ 8025: uvm_test_top.env_h.alu_agent_h.monitor_in_h [alu_monitor_in] Monitor_In Report Phase Complete
VM_INFO env/env_components/alu_agnt_components/alu_monitor_out.sv(102) @ 8025: uvm_test_top.env_h.alu_agent_h.monitor_out_h [alu_monitor_out]
Monitor_out Report:
    Total Outputs_monitored_pkts: 367
VM_INFO env/env_components/alu_agnt_components/alu_monitor_out.sv(105) @ 8025: uvm_test_top.env_h.alu_agent_h.monitor_out_h [alu_monitor_out] Monitor_out Report Phase Complete
VM_INFO env/env_components/rst_agnt_components/rst_driver.sv(69) @ 8025: uvm_test_top.env_h.rst_agent_h.driver_h [rst_driver]
Reset Driver Report:
    Total Resets: 4
VM_INFO env/env_components/rst_agnt_components/rst_driver.sv(72) @ 8025: uvm_test_top.env_h.rst_agent_h.driver_h [rst_driver] Reset Driver Report Phase Complete
VM_INFO env/env_components/scoreboard.sv(101) @ 8025: uvm_test_top.env_h.scb_h [scoreboard]
Scoreboard Report:
    Total Transactions: 367
    Total Correct Items: 367
    Total Incorrect Items: 0
VM_INFO env/env_components/scoreboard.sv(105) @ 8025: uvm_test_top.env_h.scb_h [scoreboard] Scoreboard Report Phase Complete
VM_INFO env/env_components/subscriber.sv(524) @ 8025: uvm_test_top.env_h.sub_h [subscriber] Received transactions: 367
VM_INFO env/env_components/subscriber.sv(526) @ 8025: uvm_test_top.env_h.sub_h [subscriber]
Coverage Report:
VM_INFO env/env_components/subscriber.sv(527) @ 8025: uvm_test_top.env_h.sub_h [subscriber] TEST_NAME = error_test ←
VM_INFO env/env_components/subscriber.sv(550) @ 8025: uvm_test_top.env_h.sub_h [subscriber] Error Cases Coverage: 100.00% ←
VM_INFO env/env_components/subscriber.sv(553) @ 8025: uvm_test_top.env_h.sub_h [subscriber] Total Coverage: 100.00% ←
```

---

# Waveforms

---



# Error Test Waveform



# Random Test Waveform



# Repetition Test Waveform



# Verbosity

# Different Verbosity for Different Debugging Needs

UVM\_LOW for Reporting & Checking that the hierarchy is starting correctly & Scoreboard Mismatches

```
function void report_phase(uvm_phase phase);
    `uvm_info(get_type_name(), $sformatf("Received transactions: %0d", count_trans), UVM_LOW)
    `uvm_info(get_type_name(), "\nCoverage Report:", UVM_LOW)
    `uvm_info(get_type_name(), $sformatf("TEST_NAME = %s", test_name), UVM_LOW)

    case(test_name)
        "random_test": begin
            `uvm_info(get_type_name(), $sformatf("Control Coverage: %.2f%%", control_cg.get_coverage()), UVM_LOW)
```

UVM\_HIGH for deep-level & case Debugging (usually during the development phase)

```
virtual task run_phase(uvm_phase phase);
    //-----
    //Wait for reset de-assertion
    //-----
    @(posedge reset_deassertion); // first reset obologatry
    `uvm_info(get_full_name(), $sformatf("first reset detect :\n"), UVM_HIGH)
    + 1 - 4) + 1;
```

UVM\_MEDIUM for Correct Cases tracing & Expected behaviour

```
// Compare expected vs actual
if (expected_item.C === actual_item.C) begin
    correct_counter++;
    `uvm_info(get_type_name(), $sformatf("CORRECT %0d: Expected = %0d, Actual = %0d",
    correct_counter, expected_item.C, actual_item.C), UVM_MEDIUM)

    `uvm_info(get_type_name(), $sformatf("CORRECT %0d: Expected_TXN_counter = %0d, Actual_TXN_counter = %0d",
    correct_counter, alu_seq_item::txn_counter_expected, alu_seq_item::txn_counter_actual), UVM_MEDIUM)
end
```

---

**COVERAGE?**  
**What? For**  
**What?**

**100%**

# Coverage Examples

Fixed/Cross – based coverage method

```
covergroup inputs_cg();
  A_df:coverpoint input_cov_copied.A iff (input_cov_copied.ALU_en & input_cov_copied.rst_n){
    option.weight = ((input_cov_copied.A == -16)?0:1);
    bins A[] = {[16:-16]};
    `ifndef RANDOM_ERROR_INJECTION_TEST
      ignore_bins A_ignored[] = {-16};
    `else
      illegal_bins A_illegal[] = {-16};
    `endif
  }

  B_df:coverpoint input_cov_copied.B iff (input_cov_copied.ALU_en & input_cov_copied.rst_n){
    ...
  }

  A_B_Aen_Ben_Aop_Bop: cross A_df, B_df, A_en_df, B_en_df, A_op_df, B_op01_df, B_op11_df{
    bins A_op_cases = binsof(A_df) && binsof(B_df) && binsof(A_en_df) intersect {1} &&
      binsof(B_en_df) intersect {0} && binsof(A_op_df);
    bins B_op01_cases = binsof(A_df) && binsof(B_df) && binsof(A_en_df) intersect {0} &&
      binsof(B_en_df) intersect {1} && binsof(B_op01_df);
    bins B_op11_cases = binsof(A_df) && binsof(B_df) && binsof(A_en_df) intersect {1} &&
      binsof(B_en_df) intersect {1} && binsof(B_op11_df);
  }
}
```



```
inputs_cg = new();
```

```
outputs_cg = new();
```

```
inputs_cg.sample();
outputs_cg.sample();
```

Instance – based coverage method

```
covergroup A_cg_df(input int signed i, input transaction_base cov);
  option.weight = ((i == -16)? 0:1);
  option.name = $sformatf("df = %0d", i);
  option.per_instance = 1;
  df: coverpoint cov.A iff (cov.rst_n && cov.ALU_en) {
    bins A[] = {i};
    `ifndef RANDOM_ERROR_INJECTION_TEST
      ignore_bins A_ignored[] = {-16};
    `else
      illegal_bins A_illegal[] = {-16};
    `endif
  }
endgroup : A_cg_df
```

```
covergroup A_op_cg_dt(input int i, input int k , input transaction_base cov);
  option.weight = ((k == 7 || i == 7)? 0:1);
  option.name = $sformatf("dt %0d => %0d", i, k);
  option.per_instance = 1;
  dt: coverpoint cov.a_op iff (cov.rst_n && (cov.a_en & ~cov.b_en) && cov.ALU_en) {
    bins A_op[] = (i => k);
    `ifndef RANDOM_ERROR_INJECTION_TEST
      ignore_bins A_op_ignored1[] = (i => 7);
      ignore_bins A_op_ignored2[] = (7 => k);
    `else
      illegal_bins A_op_illegal1[] = (i => 7);
      illegal_bins A_op_illegal2[] = (7 => k);
    `endif
  }
endgroup : A_op_cg_dt
```

## Construction

## Sampling

```
foreach(A_op_cg_df_vals[i]) A_op_cg_df_vals[i] = new(i, input_cov_copied);

foreach(A_cg_df_vals[i]) A_cg_df_vals[i].sample();
foreach(B_cg_df_vals[i]) B_cg_df_vals[i].sample();

foreach(A_op_cg_dt_vals[i,j]) A_op_cg_dt_vals[i][j].sample();
foreach(B_op01_cg_dt_vals[i,j]) B_op01_cg_dt_vals[i][j].sample();
foreach(B_op11_cg_dt_vals[i,j]) B_op11_cg_dt_vals[i][j].sample();
```

---

# Coverage Closure



# Dashboard.txt for valid Test

```
Dashboard
Date: Mon Mar 24 10:16:12 2025
User: svgpdv25abohamad
Version: V-2023.12-1
Command line: urg -dir random_test_simulation_executable.vdb -elfile random_test.el -dir repetition_test_simulation_executable.vdb -elfile repetition_test.el -format text -report coverage_closure/valid_tests_cov_merged
Number of tests: 2

Total Coverage Summary
SCORE LINE COND TOGGLE BRANCH ASSERT GROUP
99.07 100.00 100.00 100.00 94.44 100.00

Hierarchical coverage data for top-level instances
SCORE LINE COND TOGGLE BRANCH ASSERT NAME
99.61 100.00 100.00 100.00 98.04 alu_top_tb(x)
33.33 -- -- -- 33.33 uvm_pkg

Total Module Definition Coverage Summary
SCORE LINE COND TOGGLE BRANCH ASSERT
80.81 75.59 78.12 92.41 63.49 94.44

Total Groups Coverage Summary
SCORE WEIGHT
100.00 1
```

```
Date: Mon Mar 24 10:16:12 2025
User: svgpdv25abohamad
Version: V-2023.12-1
Command line: urg -dir error_test_simulation_executable.vdb -format text -elfile error_test.el -report coverage_closure/error_injection_cov_merged
Number of tests: 1
```

```
Total Coverage Summary
SCORE LINE COND TOGGLE BRANCH ASSERT GROUP
86.73 100.00 100.00 100.00 100.00 20.37 100.00
Most assertions failing due to error injection which is a good sign...

Hierarchical coverage data for top-level instances
SCORE LINE COND TOGGLE BRANCH ASSERT NAME
83.92 100.00 100.00 100.00 100.00 19.61 alu_top_tb(x)
33.33 -- -- -- 33.33 uvm_pkg

Total Module Definition Coverage Summary
SCORE LINE COND TOGGLE BRANCH ASSERT
55.26 68.50 56.25 83.54 47.62 20.37

Total Groups Coverage Summary
SCORE WEIGHT
100.00 1
```

# Dashboard.txt for Error Injection Test

# Exclusions and .el files

Exclusions can be done in the merge command line using –exclude but the naming of the branches, points, ..etc is often hard to make out and understand to just do it from the line, therefore using dve to make the .el file is much easier and thereafter including them in the report command.



# Scripting & Makefiles

```

COVERAGE_DIR := coverage_closure
VERDI := verdi
DVE := dve

# Define the default test names (can be overridden via command line)
TEST_NAME ?= random_test # Default test name if not specified
TEST_NAME1 ?= random_test                                defaultname for tests incase of no inputs
TEST_NAME2 ?= repetition_test
TEST_NAME3 ?= error_test

# Default tools
WAVEFORM_VIEWER ?= dve # Default waveform viewer (can be 'verdi' or 'dve')
COVERAGE_MERGE_TOOL ?= urg # Default coverage merge tool (can be 'urg' or 'verdi')

# Derived variables based on TEST_NAME
SIM_EXEC := $(TEST_NAME)_simulation_executable
VDP_FILE := $(TEST_NAME).vdp                         each file parametrised by the test_name
COMPILE_LOG := $(TEST_NAME).compile.log
SIM_LOG := $(TEST_NAME).sim.log
COVERAGE_DB := $(SIM_EXEC).vdb

# Compilation and simulation flags
FILE_LIST_FLAGS := -f
COVERAGE_FLAGS := -cm branch+cond+tg1+line+assert
VCS_UVM_LIB_FLAG := -ntb_opts uvm-1.2
VCS_FLAGS := -full64 -svrilog $(VCS_UVM_LIB_FLAG) -timescale=$(TIME_SCALE) $(COVERAGE_FLAGS) -l $(OUTPUT_DIR)/$(COMPILE_LOG) -debug_access+r +R
VDP_FLAGS := -vpd $(OUTPUT_DIR)/$(VDP_FILE)
UVM_TESTNAME := +UVM_TESTNAME=
VERBOSE_LEVEL ?= UVM_LOW
UVM_VERBOSITY := +UVM_VERBOSITY

# Create necessary directories
$(OUTPUT_DIR) $(COVERAGE_DIR) $(VDP_OUTPUT_DIR):
    @mkdir -p $@
    @echo "Created directory: $@"

# Compile the design and testbench with the specified test name
compile: $(OUTPUT_DIR) $(VDP_OUTPUT_DIR) $(DESIGN_FILE) $(TESTBENCH_FILE)
    @echo "Compiling design and testbench for test: $(TEST_NAME)..."
    @$(VCS) $(VCS_FLAGS) $(FILE_LIST_FLAGS) $(DESIGN_FILE) $(FILE_LIST_FLAGS) $(TESTBENCH_FILE) -o $(SIM_EXEC) || { echo "Compilation failed"; exit 1; }
    @echo "Compilation complete for $(TEST_NAME).."

# Run the simulation and dump the VPD file with the test-specific name
run: compile
    @echo "Running simulation for test: $(TEST_NAME)..."
    @./$(SIM_EXEC) $(COVERAGE_FLAGS) $(UVM_TESTNAME)$(TEST_NAME) $(UVM_VERBOSITY)$(VERBOSE_LEVEL) +ntb_random_automatic -l $(OUTPUT_DIR)/$(SIM_LOG) $(VDP_FLAGS) || { echo "Simulation failed"; exit 1; }
    @echo "Simulation complete. VPD file dumped to $(OUTPUT_DIR)/$(VDP_FILE)."

# View waveform with selected viewer (verdi or dve)
view-waveform: $(OUTPUT_DIR)
    @if [ "$!WAVEFORM_VTFWR!" = "verdi" ]; then \
        @echo "Verdi viewer selected" \
        @$(VERDI) $(TEST_NAME).vdp \
    else \
        @echo "DVE viewer selected" \
        @$(DVE) $(TEST_NAME).vdp \
    fi

```

Choosing between Verdi &  
DVE for wave viewing  
Choosing between urg and  
Verdi for vdb merging  
USE make help to  
understand the tasks

```

# Generate coverage reports
generate-reports:
    @echo "Generating coverage reports....."
    urg -dir $(TEST_NAME1)_simulation_executable.vdb $(TEST_NAME2)_simulation_executable.vdb -format text -elfile RANDOM_TEST.el -report $(COVERAGE_DIR)/valid_tests_cov_merged.txt
    urg -dir $(TEST_NAME3)_simulation_executable.vdb -format text -elfile RANDOM_ERROR_INJECTION_TEST.el -report $(COVERAGE_DIR)/error_injection_cov_merged.txt

clean:
    @echo "Cleaning up..."
    @rm -rf "$(OUTPUT_DIR)/* $(COVERAGE_DIR)/* \
        $(TEST_NAME1)_simulation_executable" \
        $(TEST_NAME2)_simulation_executable" \
        $(TEST_NAME3)_simulation_executable" \
        $(TEST_NAME1)_simulation_executable.daidir" \
        $(TEST_NAME2)_simulation_executable.daidir" \
        $(TEST_NAME3)_simulation_executable.daidir" \
        $(TEST_NAME1)_simulation_executable.vdb" \
        $(TEST_NAME2)_simulation_executable.vdb" \
        $(TEST_NAME3)_simulation_executable.vdb"
    @echo "Clean up complete."

```

Makefile to compile, run, check the simulation waveform using the .vdp files & clean each time the clean task is called or when the run\_all.sh is executed

Saving compile and simulation.log files so they can be searched for any errors using a script if I have the time to make one

Generating the coverage reports of the valid tests \_merging them and of the error\_injection test

Cleaning the previous run's output files as mentioned in the bash\_script slide

# Scripting & Makefiles

```
# Create log directory
mkdir -p "$LOG_DIR"
log "Starting regression test suite"
log "Using makefile: $MAKEFILE"
log "Waveform viewer: $WAVEFORM_VIEWER"
log "Coverage merge tool: $COVERAGE_MERGE_TOOL"

# Interactive tool selection
if $INTERACTIVE; then
    echo "Would you like to select a different waveform viewer? [verdi/dve/no] (default: $WAVEFORM_VIEWER): "
    read viewer_choice
    if [[ "$viewer_choice" == "verdi" || "$viewer_choice" == "dve" ]]; then
        WAVEFORM_VIEWER="$viewer_choice"
        log "Changed waveform viewer to: $WAVEFORM_VIEWER"
    fi

    echo "Would you like to select a different coverage merge tool? [urg/verdi/no] (default: $COVERAGE_MERGE_TOOL): "
    read coverage_choice
    if [[ "$coverage_choice" == "urg" || "$coverage_choice" == "verdi" ]]; then
        COVERAGE_MERGE_TOOL="$coverage_choice"
        log "Changed coverage merge tool to: $COVERAGE_MERGE_TOOL"
    fi
fi

# Clean environment
log "Cleaning environment"
run_cmd "make -f $MAKEFILE clean"

# Run tests
log "Running random_test"
run_cmd "make -f $MAKEFILE TEST_NAME=random_test full-sim"

log "Running repiition_test"
run_cmd "make -f $MAKEFILE TEST_NAME=repiition_test full-sim"

log "Running error_test"
run_cmd "make -f $MAKEFILE TEST_NAME=error_test full-sim"

# Generate reports with specified tools
log "Generating coverage reports using $COVERAGE_MERGE_TOOL for merging"
run_cmd "make -f $MAKEFILE TEST_NAME1=random_test TEST_NAME2=repiition_test TEST_NAME3=error_test COVERAGE_MERGE_TOOL=$COVERAGE_MERGE_TOOL generate-reports"

# Optionally view a waveform if in interactive mode
if $INTERACTIVE; then
    echo "Would you like to view a waveform? [yes/no]: "
    read view_choice
    if [[ "$view_choice" == "yes" || "$view_choice" == "y" ]]; then
        echo "Which test would you like to view? [random_test/repiition_test/error_test]: "
        read test_choice
        run_cmd "make -f $MAKEFILE TEST_NAME=$test_choice WAVEFORM_VIEWER=$WAVEFORM_VIEWER view-waveform"
    fi
```

Bash script that runs regression using the Makefile tasks