

# UVM Basic TestBench Environment

## For 16x32 Memory

### Contents:

- 1- UVM Environment Basic Architecture/ With Connections
- 2- UVM Class Tree and How to extend from UVM Classes
- 3- Phasing
- 4- Factory
- 5- Building Environment Components
- 6- UVM Resources and Databases
- 7- How to pass virtual interface handle
- 8- TLM Connections
  - a. Monitor to Scoreboard and Subscriber
  - b. Driver to Sequencer
- 9- Sequence-to-Sequencer Dynamic Connection
- 10- How to write sequences
- 11- How to drive to/ sample from DUT
- 12- Scoreboard Code to check memory
- 13- Coverage and Subscriber

# UVM Basic Architecture



# UVM Basic Architecture with Connections



- 1 - Monitor Analysis Port (for broadcasting)
- 2 - Agent Analysis Port (for passing monitor port outwards)
- 3- Scoreboard Analysis Export (for passing monitor port inwards to FIFO)
- 4- FIFO built-in analysis\_export connector (for writing data in FIFO)
- 5- FIFO built-in get\_peek\_export connector (for getting data from FIFO)
- 6- Subscriber built-in analysis\_export connector (for getting data)
- 7- Driver built-in seq\_item\_port connector (for getting data from sequence through sequencer and sending back a response)
- 8- Sequencer built-in seq\_item\_export connector (for getting requests and responses from driver)
- 9- Dynamic Seq-to-Seqr connection

# DUT 16x32 Single-Port Sync-Reset Memory

```
1  module memory_16x32 (
2      input logic clk,
3      input logic re,
4      input logic en,
5      input logic rst,
6      input logic [3:0] addr,
7      input logic [31:0] data_in,
8      output logic [31:0] data_out,
9      output logic valid_out
10 );
11
12 // Declare a 2D array of 32-bit words with 16 rows
13 logic [31:0] mem [15:0];
14
15 // Write data to memory when enabled and address is valid
16 always @ (posedge clk) begin
17     if (!rst) begin
18         mem <= '{default:'h0}; // Reset memory to zero
19         data_out <= 'h0; // Reset output data to zero
20         valid_out <= 'h0; // Reset output validity to zero
21     end else if (en) begin // Check enable and address range
22         mem[addr] <= data_in; // Assign memory location from input data
23         valid_out <= 'h0;
24     end else if (re) begin
25         data_out <= mem[addr]; // Assign output data from memory location
26         valid_out <= 'h1; // Set output validity to one
27     end
28 end
29
30 endmodule
```

## General Notes

- 1- Each entity will be in its own file , each file will be a package except for interface , tb-top and DUT file.
- 2- in each file you'll need to add `include "uvm-macros.svh"` and `import uvm_pkg::*;` inside the package , outside class
- 3- in each file you'll need to import the packages of the other files you'll use .. like `"driver-pkg"` inside `"agent-pkg"` and so on for till you reach tb-top where you'll import `"test-pkg"`.
- 4- run.do file : to automate working with Questasim
- 5- Source UVM Files are highlighted in this colour , lines of code are highlighted with this colour .
- 6- each step has a provided link that goes to a GmUnit in the github repo.

# UVM Notes and Steps

## 1 UNM Introduction :

- \* It stands for universal Verification Methodology, and it's a standardized way for verifying digital designs.
  - \* It's based on system verification and provides a set of classes that can be extended to create test bench components such as drivers, monitors, agents, etc.
  - \* It also standardized transaction-level communication between tb components.

## 2 UVM Class Tree :

Both have a Parent/child relation

A - Inheritance Tree is "Partial" → But differ in meaning

- Called : "is-a" relation . root : `UVM-void`
  - Defines : DNA of an object
  - Purpose : Code sharing and allows Poly morphism
  - Established via "extends" keyword



- \* UVM-Void is an empty class, acts as the base class for all UVM classes
  - \* UVM-object extends "inherits" UVM-Void, acts as a virtual base class for all Components (like driver, monitor) and transactions. It defines a set of methods like Copy, Paste, etc.
  - \* UVM-Component is the base class for all the UVM Components like env, agent, etc.
    - it supports hierarchical structure, each Comp. can have child Comp.
    - Components can have phases that organize simulation, Comp performs diff. tasks during each phase.
    - it also supports factory interface, the factory keeps track of created Comp
  - \* The classes UVM-agent, UVM-env, etc will be used to extend from to make the user defined classes for ex. my-agent or my-env.

Note: `WYM-object` is said to be a child of `wym-root`: because it's extended from it.

### 3 Virtual vs non-virtual classes and functions

A- Virtual class means : no instance can be created from this class . but it can be inherited . like `num-component` . it's a virtual class

### **B - Function vs Virtual function**

- \* A function is defined as virtual in a parent class to allow child class to override it with its own functionality.
  - \* If a function "best" is virtual in a parent class, then in all child classes of this parent, the function "best" is automatically virtual.   
 like phase functions (build-phase, connect-phase, etc) inside WM-Component.   
 will be explained in [§](#)
  - \* This matters if the handle-type "like a pointer" is of parent type but it actually points to a child. ∵ when calling the function using the handle, simulator goes to the function definition in the parent and only if it's virtual does it go look at the object-type that handle is pointing at (the "child") and executes the child function instead of the parent.

```

Ex: class parent;
    virtual function void iam();
        $display("i am in parent");
    endfunction
end class

class child extends parent;
    function void iam();
        $display("i am in child");
    endfunction
end class

```

Parent P1 ; // handle type is parent  
child c1 = new();  
P1 = c1; // parent handle points to child object  
P1.iam(); // output is "i am in child"  
because of virtual.

<- Pure Virtual Functions : used in virtual class to force all extended child classes to implement this function.

#### 4 User-defined classes, Step 1 : links to git-repository commits

\* The needed classes for the environment are :

- Components : these are created once at runtime and stay throughout sim. Hence, they're called quasi-static.
  - test, env, agent, driver, monitor, sequencer, scoreboard and subscriber.
  - Extended from UVM-test, UVM-env, etc ⇒ these source codes can be found in GitHub UVM 1.2 @ SRC/Comps/...
  - Belong to a structural tree, for ex: driver is a child of agent which is a child to env. and so on.
- objects : created and destroyed as needed during runtime. Hence, they're dynamic and don't belong to the structural tree.
  - sequence & Packet or transaction class
  - sequence is extended from UVM-sequence and Packet is extended from UVM-sequence-item, found in SRC/seq/...

\* When extending from a class, note if :

- it's parameterized ⇒ class UVM-subscriber #(type T=int) extends UVM-Component;
    - Parameterization is either of type like the example  $\lceil$  or of value  $\#(P1=5)$ .
    - $\#(type T=int)$  : parameterizes the data type "int" and gives it a name "T", the logic behind this is simple; it's to make this class generic and can work for any data type indep. of the one it was written in.  
 $\Rightarrow$  why "int" ? because it can be overridden with any data type when extending from this class.
    - How will it be used? when we define our scoreboard class say "my-scoreboard" we'll override this parameter with the data type of our choice which is another user-defined class "my-seq-item" for ex
    - if the parent class is param, you'll have to override these params as needed
- ↳ extended from UVM-sequence-item
- my-sub extends UVM-subscriber #(my-seq-item); → this class file must be included in the scoreboard file.
- Notes: Don't forget that a class is a data type for ex: class X; | X c1; // c1 is a class instance of type "X".

- it has pure virtual functions ⇒ these must be implemented or code won't run.

for ex: "write" function in UVM-subscriber.

\* Default Constructor "new" :

- for objects, it just creates an instance of the class with the provided "name" in the parameters

```
function new (string name = "sequence");
```
  - for Components, it creates an instance with "name" and the hierarchical parent handle, as explained before.. env is the hierar. parent of agent and so on. Therefore, 2 steps are needed for this definition:
    - declare that the instance of this Component has a parent and assign it with null for now.
    - in the Component's hierar. parent class, for this example it's "my-env" class, create an instance of the agent and assign a value of its parent which is my-env or use the keyword "this" it'll point to the class you're in.
- ↳ The exact syntax will be explained in [6]
- inside build-phase function of env ⇒ `agent = my-agent::type_id::Create("agent", this);`
- ↳ inheritance parent

\* Calling the parent's default constructor using super.new inside the child's constructor.

- super points to the parent class, this child extended from for ex. inside your "my-env" that's extended from UVM-env, "super.new" calls constructor of UVM-env because it won't be automatically called.

## 5 Phasing :

- \* Phasing concept is important in SystemVerilog testbenches because it introduces classes that can be created in the middle of the simulation  $\Rightarrow$  this needs organization.
- \* Phasing does that through a set of defined methods for "UVM-Component" and they can be grouped into 3 categories :
  1. Build time phases : Functions executed before start of simulation.  $\rightarrow$  at 0 time.
    - most used  $\Rightarrow$  1.1 : "build\_phase" function : instances of all UVM-Component and extended class are created along with their sub-components  
for ex: my-env  $\rightarrow$  uvm-env  $\rightarrow$  uvm-component and all that's inside my-env like my-agent and so on.
    - 2.2 : "Connect-phase" function : after all instances have been created, their connections "TLM" are established.  
 $\hookrightarrow$  Transaction Level Modeling
  2. Run time phases : The actual simulation processing time happens in these phases.
    - 2.2 : "run-phase" task : is the most common but not the only one, runs in parallel with other run-time phases.  
and it contains the behaviour of the component
    3. Clean-up phases : after end of simulation, like report-phase or final-phase  $\hookrightarrow$  for last minute operations before exiting the sim.
- : All UVM-Component and extended class of it are aware of the phase concept. So, all components go through a pre-defined set of phases and can't proceed to the next phase until all components finish the current phase  $\Rightarrow$  UVM phases act as a synchronising mechanism, so no comp. is called before it's created for ex.

## 6 Factory :

- \* In SystemVerilog environment, we use the "new" function to build an instance of a class. It's hard-coded. For example, if you build a driver inside an agent using "new" method, you'd have to specify the type of this driver inside the agent. This is fine, but what if you have 2 drivers, 1 for normal functionality and another for a special one... then you'd have to change the code where this driver is used each time you use a different one.  
What the factory in UVM does is that it changes the process of creating instances of classes, adding new "overriding" features to the process... in the prev. ex, all you have to do is tell the factory to override the driver class by the special driver class in the "test" without going back to the agent class... imagine doing this on a large scale, where you can change the elements of your tb environment from the test code only without the headache of changing every code.

- \* This is done by a set of classes defined in UVM, all you have to do is 3 steps :  $\rightarrow$  a macro is a substitution of text, for example, a code, so instead of repeating lines of code, it's replaced by class name, my-test, my-env, etc.
  1. Register all objects and components in the factory using these UVM-defined-macros :  
`UVM_Component_Utils(class-handle); // for components, and UVM_Object_Utils(class-handle); // for objects`  
 $\Rightarrow$  This step means UVM-Factory (which has a single instance in the whole sim) adds this item to its table, to be known by factory.  
 $\Rightarrow$  What does it do?
    - a. Create an object inside your class to represent it, it's a parameterized instance of UVM-Component-registry class, Param. with your class type  $\Rightarrow$  `UVM_Component_Registry #(class-handle, "name")`.  
This registry is your class's door to UVM-Factory, it's like the middle man UVM-Factory orders to call "new" for your item.
    - b. Define a new data type using typedef for that registry named type-id, so it's a standard name for all Comp. registry without having to know the specific class name. `typedef UVM_Component_Registry #(class-handle, "name") type-id;`
  2. Keep the definition of the "new" constructor as it is  $\Rightarrow$  objects take string "name" and components take "name" and a handle to "parent". why? because the factory will call the "new" function for you, with the right type of parent or overridden parent. and if you change its arguments, the factory won't be able to adapt.
  3. Create the item with the defined "create" method, ex: driver my-driver inside the agent  
`drv = my-driver::type-id::create("my-driv", this)`, instead of `drv = my-driver.new("my-driv", this)`; the 2nd code will be called either way, it's just that calling "new" by yourself means you're asking for a certain type of driver, "my-driver". but through the create method, it checks with the factory 1st if there's an override request for "my-driver". if there is, for ex: an order to override any "my-driver" with "special\_driver", "create" method will handle this and build your driver with the wanted type.  $\text{C issued in the test code}$

Note that "Create" is a registry function that calls the factory's Create function called "Create\_Component\_by-type".

- what about "this" keyword? it's a special pointer to the current instance of the class, it's used for path accuracy.
- in the "Create" line of Gde, it points to the wanted parent for your item, that was assigned "null" in "new".
- what if instead of using "this" you use a fixed parent handle for ex "my-agent"?
- this works fine until you decide to override "my-agent" with another agent, "special-agent" for ex.
- now "Create" orders to builds an instance of my driver inside the agent, but which agent?
- using "this", it will point to the correct agent → advanced-agent
- using fixed-parent method will point to my-agent always ⇒ driver will attached to wrong parent.

Note: when creating any Component, "this" keyword will point to its parent, except my-test. its parent will remain as "null" and this tells UVM that this instance will be the top. Explained in [?](#)

<https://github.com/yasmine-eldesoukie/uvm-based verification environment for memory/commit/e40e55790401488daaaa5fe602910ae28bfd6af>  
<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/4f572c058bba8ec21956a49d1d31fa6fb2c04ba08>

## 7 Building the environment Components, Step 2: links to commits ↗

\* Building is done Top-down ⇒ Test → Env. → Agent → ...

\* let's divide this process to 3 steps :

1- Building from Test down to driver and monitor : → or-object- for objects.

- Register the classes in the factory. `uvm_Component_Utils::Component_name();`
- Implement the build\_phase function. `function void build_phase(uvm_phase phase);`
- Call build\_phase of the parent. `super.build_phase(phase);`
- Create instances of sub-Components and objects, for ex: env and sequence inside test, using "Create" method.

2- Building and running "my-test" :

- Call the global task "run-test" in the tb-top module inside the initial block ⇒ `run-test("my-test");` this's the name used when registering in factory
- This task can be found in `src/base/uvm-globals.svh` and it Calls "`uvm_top.run-test`" that :
  - Calls factory to build an instance of the "my-test" class but name it "uvm-test-top" and assign it to parent=null so that "uvm-top" becomes its parent ⇒ this's defined by UVM.
  - Starts the phasing mechanism, build → Connect → run → clean-up. This's like pushing the last piece of domino
  - Calls \$finish; to end simulation. That's why any code after `run-test(1);` doesn't execute.

3- Overriding the building blocks -if needed. This step is done in the "build-phase" of "my-test" before the "create" lines. and it can be by name or by type, it's to tell the factory that whenever something asks for "my-driver" for ex. give them an "advanced-driver" ex: `set-type-override-by-type(my-driver::get-type(), advanced-driver::get-type());`

### Remember:

an instance of `uvm_root` called `uvm_top` is automatically instantiated and is the root for the hierarchical tree.

## 8 UVM Resources and Databases :

### Note:

A) **Associative Array:** `int x[J];` is a normal array which has an index starting from 0 / `int x[string];` is still an array but its index is a key of type "string". ∴ instead of `x[0]=32;` it's `x[bus-width]=32;`

B) **Static Class:** a class that contains static methods and variables, it's not meant for instantiation or inheritance. It acts as a tool box, you don't need to "own" the toolbox you can just grab the tool and use it.  
↳ no new();  
 ↳ its contents are accessed by the class name and the scope operator ":"

\* The testbench environment has many components that will eventually need data from one another, we can declare these data as global variables but it's not safe; anyone can change it! and it's a nightmare to debug.

\* How does UVM resolve this issue? it created an organized mechanism for storing, searching and retrieving these data through a set of classes:

- 1. uvm\_resource #(data-type):** uvm stores each item created in the sim as a "resource" using "uvm-resource" class. and each resource has a name, a type, a value and context "scope" for visibility or where and who can access it.
- 2. uvm\_resource\_base :** this's the parent or base class of "uvm-resource", it's not tied to a certain data type and will be used for the queue that will store these resources.
- 3. uvm\_queue #(data-type) :** this's a queue of "data-type", this "data-type" will be param. by "uvm-resource-base" so that it's not limited to "int" items for ex, or even `uvm_resource #(int)` ... it can store any uvm resource because we used the generic `uvm-resource-base`.

4. **uvm\_pool**: this's basically an **associative array** with added features. ∵ it can carry an associative array of these uvm-queues of uvm-resources.

5. **uvm\_resource\_pool**: this class has 2 groups of "uvm\_pool" instances:

- a) Name **uvm\_pool**: the key of the associative array is a **string name**, and the entries of this array are queues of diff. resources sharing the **same name**.
- b) Type **uvm\_pool**: same as name **uvm\_pool** but the key is a **data type** and the stored resources are of the **same type**.

Each **uvm-resource** will be stored inside both of these **uvm\_pools**, this will make things more organized and faster to get when called for.

6. **uvm\_resource\_db**: the **uvm\_resource\_pool** is like an actual library storing resources and **uvm\_resource\_db** is like the librarian, it's an API  
an interface to **uvm\_resource\_pool**. it's a **static class**. This is what we use in the tb, the prev. classes are the behind-the-scenes workers

→ when used **UVM\_RESOURCE\_DB #(int)::set ("global", "bus\_width", 32);** it creates a **uvm\_resource #(int)** to be stored in

**uvm\_resource\_pool**: → by name under "bus\_width" in name **uvm\_pool** ↗ stored in both. "global" is the scope, that needs to be  
by type under "int" in type **uvm\_pool** ↗ matched why you use **::get (" ", " ", )**;

→ It's used for items that don't belong to a physical path but rather created, run and destroyed in sim. like sequences.

7. **UVM\_Config\_db**: this's a restricted database, when used it has the feature to include the physical path as the scope, and the getter has to  
be in that scope to get! unlike **uvm\_resource\_db** scope, it has no idea that the tb is a tree of Components and it has no idea where that  
resource is, "global" to it is just a string to match at the getter.

→ ∵ **UVM\_Config\_db** is used in tb for sharing Components and restricting the other Components access to it. it's mostly used for passing  
the virtual interface handle from tb-top to driver and monitor. Explained in [?]

→ Syntax: **UVM\_CONFIG\_DB #(data-type)::set (Context, Instance-path, Field-name, Value);** to store

\* Context: start of the path. "null" for modules and "this" for classes.

\* Instance-path: string representing "name\_of\_Component\_inside\_path".

\* Field-name: key "Name" to be stored under in **uvm\_resource\_pool**.

\* Value: actual Data to store.

**UVM\_CONFIG\_DB #(data-type)::get (Context, Instance-path, Field-name, Value);** to get

by default it's get by name from  
the **uvm\_resource\_pool** to search  
by name in its "name uvm pool"  
and match the exact type in that  
queue, rather than search by type  
because then, uvm jumps straight  
to the "type pool" and gets the  
most recent resource.



## 9 Passing Virtual Interface Handle, Step 3 :

[link to Commit](#)

Note: "include "interface.sv" must be outside tb-top module to be global

- \* Interface instance is created at tb-top and needs to be passed down to driver and monitor. This will be done using `UVM_Config_Db::set/get`.
- \* It could be made global to the whole env. but this's useless since not all components need the interface and it will just take too long to find when needed since the scope is wide  $\Rightarrow$  Performance b.
- \* What's better is to pass it down layer by layer: test  $\rightarrow$  env  $\rightarrow$  agent  $\rightarrow$  driver/monitor

\* Steps :

1. in tb-top module, create an instance of the interface  $\Rightarrow$  `intf my_intf;`
2. in tb-top, Pass it down to test as virtual : `UVM_Config_Db #(virtual intf)::set("null", "uvm-test-top", "my-vif", my-intf);`  
"null" because this's a module, "uvm-test-top" is my-test instance where you want to deliver this handle.
3. in test class, also create a virt. handle to store the passed handle from tb-top to be able to pass it down again to env.

$\Rightarrow$  `virtual intf test-vif;` || a diff name just to since the difference btwn field-name that can't change and "value" that's like the specific container for the passed handle in each class.

$\Rightarrow$  in build-phase

$\left\{ \begin{array}{l} \text{if } (!(\text{UVM_Config_Db} \#(\text{virtual intf}))::\text{get}(\text{this}, "", "my-vif", \text{test-vif})) \text{ // this and "" mean to get here because in tb-top we specified this exact location } \Rightarrow \text{"uvm-test-top"} \\ \text{UVM_Fatal("my-test", "Failed to get my-intf"); // Fatal because not getting the interface handle is severe.} \\ \text{UVM_Config_Db} \#(\text{virtual intf})::\text{set}(\text{this}, \text{env}, "my-vif", \text{test-vif}); // set it to "my-env" which is the inst. of my-env in my-test. } \end{array} \right.$

4. Repeat in my-env, my-agent and my-driver/my-monitor classes.

Just remember to set twice in agent, 1 for driver and another for monitor  
also note to only get in driver and monitor, no more set!

Note: If interface is Parameterized, for ex:

`interface intf #(Parameter width = 8)();`

then it has to be declared as `(virtual intf #())`

in the `UVM_Config_Db` line

## 10 Transaction Level Modeling (TLM)

### Transaction Level Modeling (TLM) :



- \* TLM has multiple advantages over using mailboxes in terms of flexibility, direction and added features that make TLM smarter.
- \* The mechanism is like a faucet and a drain, each is defined in its prospective Component without knowing or caring about the other Comp. unlike mailboxes that have to be defined in both Components (both share handles to the same mailbox), if one is to be swapped, the other Comp. code has to change.
- \* UVM Provides many versions of TLMs and they can be Categorized based on :

#### 1. Role : Who starts the call and where the logic lives.

- Port : initiates transaction by calling a method on the other end.
- Imp : "implementation", terminates transaction, this's where that method is implemented.
- Export : Middleman, Passes a Connection through a Component boundary. Explained in [II](#)

#### 2. Interface : action and direction, what is being done and in which direction does data flow.

direction  $\Rightarrow$  A) unidirectional . (Put, get, Peek) family  $\Rightarrow$  Methods

Actions  $\Rightarrow$  Put : initiator "puts" data to target.  $\rightarrow$  get : initiator asks to "get" data from target.

$\rightarrow$  Peek : initiator "looks" or "peeks" on data at target but it's not removed from target.

B) bidirectional : (Master, Pull) family , used when a request-response handshake is needed

$\rightarrow$  Pull : This one is actually built-in for driver and sequencer, driver pulls data using "get-next-item" method and then pushes a completion status "item-done".

$\rightarrow$  other examples like master/slave and transparent.

C) analysis : (Broadcast) family 1-to-many

$\rightarrow$  write : this's the only method, initiator "writes" in each target, and this's the method used between monitor and both scoreboard and subscriber or even better ... if there are many of them not just one. and this is another advantage of TLM over mailboxes, monitor having an analysis-port just writes and it doesn't care if it's 1 Comp. or a 100.

#### 3. Execution :

A) Blocking : implemented methods are task, so they take time, it forces initiator to wait.  
related methods are like "put" and "get".

B) Non-blocking : implemented methods are functions, so they execute in no time.  
 $\rightarrow$  Just checking needs no wait  
related methods always start with "try\_" or "Can\_"  $\Rightarrow$  "try-put" : returns 1 if target is ready to accept transaction

C) Combined : for ports that can do both you can use the plain "UVM-put-port" instead of either "UVM-blocking-put-port" or "UVM-nonblocking-put-port"

Note: analysis-ports are by default non-blocking, no receiver target should stall the process. That's why "write" is a function not a task.



Ex: uvm-analysis-port



Ex: uvm-nonblocking-port-port  
uvm-blocking-get-imp  
uvm-peek-export



Ex: uvm-nonblocking-pull-imp  
uvm-blocking-master-port  
uvm-slave-export



#### \* General Method :

- 1- Create a `uvm-*-port` at `initiator`, for ex: `uvm-blocking-pkt-port #(packet) CompA-p`; at `Comp A`. and call its "new()" constructor in `Comp.A` "build-phase": `CompA-p = new("CompA-p",this);`
  - 2- Call the defined method for the chosen port at `initiator` in its "run-phase". for this ex, it's "put": `CompA-p.put(pkt);` remember in [5], run-phase task contains behaviour of
  - 3- Create a `uvm-*-imp` at `target`, for ex: `uvm-blocking-pkt-imp #(packet) CompB-imp`; at `Comp. B`. and call its "new()" constructor.
  - 4- Implement the defined method "put" at `target`. This method defines what the target does with the received data.
  - 5- Connect them in their hierarchical parent "Connect-phase", for this ex: it's `Comp.C`.
- `CompA.CompA-p.Connect(CompB.CompB-p);`

#### 11 TLM Connections in TB , Step 4 :

link to Commit ↗

Note: building is top-down  
and connecting is bottom-up

<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/d7ceba589c675ebf5a6d0c4b7eef61e5f7794e31>

\* Before we start, we have to add Connect-phase function and run-phase task to all Comp. Just like build-phase in [7]

#### 1- Monitor to Scoreboard / subscriber :

- \* one-to-many → analysis ports will be used. Monitor is the initiator ⇒ `uvm-analysis-port` → naming is for historic reasons.
- \* subscribers extended from `uvm-subscriber` have built-in "`uvm-analysis-imp`" connection called "`analysis-export`". That's why it was mandatory to implement the "write" function in "`my-subscriber`" class.
- \* for the scoreboard, it's one of two scenarios : and implement the write function.
  - one monitor ⇒ we will create a "`uvm-analysis-imp`" in scoreboard and connect it to the monitor's port in "env".
  - multiple monitors working with diff. speeds, so to avoid data loss... each monitor will store its data in a FIFO inside scoreboard.

\* Step-by-step following the general method in [10] :

##### 1- Monitor (The initiator) :

- a) Create a `uvm-analysis-port #(my-seq-item) mon-ap`; and call "new()" in build phase `mon-ap = new("mon-ap",this);`
- b) Call "write" function in its run-phase `mon-ap.write(seq-item);` notice how monitor just "writes" not caring which Comp receives! ∵ if sub. was to be swapped by another, monitor code won't change, that wouldn't be the case if we're using mailboxes.

##### 2- Subscriber (Target) :

- a) its "imp" is already built-in and named "analysis-export".
- b) implement "write" function and what you want sub. to do with the received "seq-item".

##### 3- Scoreboard (Target) :

- a) Create a `uvm-tlm-analysis-fifo #(my-seq-item) scr-fifo`; / scr-fifo = new ("scr-fifo",this);
- This FIFO also has a "uvm-analysis-imp" called "analysis-export"

For the available methods of each Connection, check  
uvm\_src → tlm1 → "uvm-tlm-fifos.svh"

- b) create a `UVM-analysis-export` # (my\_seq\_item) `scr_ax`; / `scr_ax = new ("scr_ax", this);`  
 remember in 10 when "export" was described as a middleman to pass connections through component boundaries.



This export connection just connects the "mon" with the "scr-fifo", why?  
 to understand the reasoning behind this we have to go a couple of steps forward.

Don't touch  
a code that  
isn't broken!

When connecting the monitor with the FIFO of scoreboard inside env connect-phase, if we use:

`agt_mon.mon_ap.Connect (scr.scr_fifo.analysis_export);` This means env. has to be aware that

1) scoreboard has a FIFO that has an analysis\_export. if any of that changes, env. breaks or you have to change its code.

2) agent has a monitor that has mon\_ap  $\Rightarrow$  same thing and we will fix it in  $\square$  with the same approach.

$\therefore$  to avoid that it'll be `agt_agt_ap.Connect (scr.scr_ax);`  $\Rightarrow$  env. doesn't know about its children contents.

c) Connect `scr_ax` with `scr_fifo`'s `analysis-export` in the `connect-phase`: `scr_ax.Connect (scr_fifo.analysis-export);`

d) Call "get" or "peek" from FIFO in "run-phase", it's established that the FIFO has an analysis\_IMP called `analysis-export`

and it's the front-door of the FIFO for writing data. Well, there's another "imp" acting as the "back-door" for collecting data

from FIFO, it's a "uvm-get-peek-imp" named "get-peek-export" providing both "peek" methods for retrieving without

removing from FIFO and "get" methods for retrieving and removing from FIFO. including both blocking/non-blocking versions

$\therefore$  in run-phase call `scr_fifo.get-peek-export.get (seq_item);` for ex. like get, try-get, can-get, etc.

#### 4- Agent (Parent of Monitor):

a) Create a door for passing mon\_ap Connection to env. Just like `scr_ax`, the only difference is that the connection in scoreboard is going top-down from a parent "scr" to a child "FIFO"  $\Rightarrow$  export is used. But in the agent's case it's bottom-up from child "monitor" to parent "agent"  $\Rightarrow$  port is used.

`UVM-analysis-port #(my_seq_item) agnt_ap; / agnt_ap = new ("agt_ap", this); // in build-phase`

b) Connect it to mon\_ap in `connect-phase`: `mon.mon_ap.Connect (agt_ap);`

Connect Rules:

Port to Imp/Export  $\Rightarrow$  Port.Connect (Imp/Export) Port to Port  $\Rightarrow$  Child\_Port.Connect (parent\_Port) ex: "agt\_ap"

Export to Imp  $\Rightarrow$  Parent\_export.Connect (child\_imp) ex: "scr\_ax"

#### 5- Environment (Parent of init. & targets):

a) Connect `agt_ap` to both sub. and SCR in `connect-phase`:

`agt.agt_ap.Connect (scr.scr_ax); / agt.agt_ap.Connect (sub.analysis_export);`

### 2 - Driver-Sequencer Connection:

\* in SystemVerilog Verification environment, we had a stimulus generator, a driver and a shared mailbox. in UVM two things changed:  
 the use of TLM Connections over a mailbox and dividing the stim\_gen into a sequencer and sequence ... while stim\_gen works fine in simple designs,  
 it can get very messy and hard to implement in more complex designs, that's why it's divided into a "script"  $\rightsquigarrow$  sequencer and a "director"  $\rightsquigarrow$  seqr.

\* in the sequence, you write the scenarios for ex: reset\_seq, write\_seq, read\_seq, etc and in seqr, you can manage how they run, their priorities and so on.

\* before jumping to the TLM Connection of driver\_seqr, let's talk about two important tasks in the sequence:

1- `start-item (seq_item)`: This's a request to the seqr, it says "I have a "seq-item", tell me when the driver is ready to receive it".

This request blocks seq code until seqr is not busy and driver is ready.

2- `finish-item (seq_item)`: Once the sequence is granted the permission, it randomizes or assigns data in general to its packet's data "seq-item" upon the req. of driver of course. once it's done with generating data, it tells seqr that it's done and that "seq-item" can be sent to driver and then this line also blocks seq code until driver announces that it finished with this data.

\* How to Establish this Connection:

#### 1- Driver:

$\rightarrow \text{src} \rightarrow \text{tm1} \rightarrow \text{UVM-Sqr-Connections.svh}$

- Note: This is a bidirectional conn.  
 because driver "gets" data from seqr, through seqr, and then sends some kind of response dep. on the protocol.
- it has a special built-in port of type `UVM-seq-item-pwlr-port` named "seq-item-port" and is ready to use.
  - it has both blocking and non-blocking methods to accommodate for your driver's protocol.  $\text{src} \rightarrow \text{tm1} \rightarrow \text{UVM-Sqr-Ifc.svh}$
  - Call the desired methods, the standard protocol is a simple blocking "get-next-item" and nonblocking "item\_done" as a response.
- \* `get-next-item`: tells seqr that driver is ready for new data  $\rightarrow$  this's the signal seqr is waiting for to generate data.
  - \* `item_done`: tells seqr that it finished with this data  $\rightarrow$  this's the signal seqr is waiting for at `finish-item`.

$\Rightarrow$  There're other methods for diff. driver functionality.

## 2- Sequencer :

- \* it also has a built-in port of type `uvm-seq-item-pull-ifm` named "seq-item-export", and it has all the methods implemented you don't need to do anything here.

## 3- Agent (Parent) :

- \* in its Connect-phase, you call `drv.seq_item_port.Connect(seqr.seq_item_export)`;

<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/bce6e6bd01ec88db718427392fb7019faea308b>

<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/317085782aa2725dba33440cb1601ec1b8ae141a>

## 12 Sequence to Sequencer Connection , Step 5 : [ links to commits ]

- \* Unlike the TLM hard Connections, this Conn. is **dynamic** because the seq. is an object created in runtime.
- \* The Connection happens in "my-test" class "run-phase" as so : note that `seq` is an instance of `my_sequencer` and `seqr` is of `J`
- ```
seq.start(env.agnt.seqr);
```
- \* That line of Code on its own finishes right away, to make the test wait until the sequence is actually finished, we use the concept of objections, this's like a global Counter, to ↑ the Counter use "`raise_objection`", to ↓ it, use "`drop_objection`" so that Code looks like this : =>
- \* **Drain-time** : to insure "test" doesn't finish right after the last Packet is sent, and that it has time to be processed, set drain-time.

Before drain time



```
task run_phase (uvm_phase phase);
  phase.get_objection().set_drain_time(this, 50ns);
  phase.raise_objection(this);
  seq.start(env.agnt.seqr);
  phase.drop_objection(this);
end task
```

After drain time



up to this point, the tb environment is generic, next it will be specialized for the DUT in Page 4

## 13 Sequence Item Class , step 6 :

- \* This is the class carrying the "Packet", it's extended from "uvm-sequence-item". it carries the dut signals and it's where we add their Constraints.
- \* It's also registered in the factory (as an object) and it has a Constructor.

- \* 16x32 Memory Signals :   
clk, control signals (rst, en, we), data busses ([31:0] data\_in & data\_out) and a valid\_out flag.
- \* Signals that are to be randomized are declared as rand and for Cyclic\_Randomization => randc, to hit all values before repeating

## 14 Sequence Class(es) , step 7 : [ link to step 6,7 Commit ]

- \* For simplicity, you can write all your scenarios in one class, but the better approach is to write each unique sequence in its own class, this allows reusability and also gives you the freedom to run them in parallel from your "test" and also add priority.
- \* Assume having multiple sequences, this's how you run them in "test" sequentially vs in parallel with priority :

```
phase.raise_objection(this);
write_seq.start(env.agnt.seqr);
reset_seq.start(env.agnt.seqr);
phase.drop_objection(this);
```

```
phase.raise_objection(this);
```

fork

```
  write_seq.start(env.agnt.seqr, null, 100);
  reset_seq.start(env.agnt.seqr, null, 200);
```

join

```
phase.drop_objection(this);
```

null refers to parent seq.

seqr has a built-in arbitration algorithm and by default it's (First Come, First served) and if two come at the same time, it looks at the higher priority.

The 2nd Code runs as so:

=> both sequences are alive at the same time but the sequencer sees that reset has a higher priority and calls it first once driver asks for next item, then it goes to write\_seq.

- \* UNM-sequence class that extends your seq classes, has a set of virtual empty methods that the seqr knows of and executes them in order when you use "start()" method in test.
- These methods are: pre-start() function / Pre-body(), body() & post-body() tasks and post-start function.
- usually only "body" is needed and you need to keep its name that way or it will never run!
- \* Where can you "create" an instance of the Packet class my-seq-item? in the "body" task, inside the loop to create a new instance with every randomization to avoid overwriting in the same memory location that other components might still be using.

→ extended from uvm-sequence

### Overview of the sequence class for 16x32 Memory module :

- ① Inside the "body" task, create a reset packet where reset is ON, if you want to add delay to keep this pattern for 50ns for ex, add this delay # 50ns after finish\_item(reset-packet); and a new packet won't be sent to driver until this delay finishes.
- ② Then make 2 loops, each of m iterations, for ex: 16 as the memory addresses. → between start-item(); & finish-item();
- ③ for the 1st loop, adjust the input signal for a "write" scenario. And for the 2nd, adjust them for a "read" scenario.  
The addr signal could be incremental from 0 to 15 or randomized, i will use the latter.
- ④ Randomization : to insure all addresses values are hit before repeating we use randc for the addr signal in my-seq-item class, and it works just fine  
If we create one instance of my-seq-item and overwrite it with each iteration. But, if we create a new one with each iteration, then each one will have the full range and randc will be = just rand ... The solution is 1. declare a randc logic [3:0] local-addr ; signal in "my-sequence" class.  
2. randomize it in the loop using this.randomize(); 3. Assign local-addr to write-pkt.addr for ex write-pkt.addr = local-addr;
- ⑤ Remember using "assert" in SV environment to detect if randomization failed, in UVM, we use 'uvm-fatal' macro, it's a reporting macro.  

```
if (!this.randomize())
    `ID` "message"
    `uvm-fatal("Randomization Failed", "local-addr rand. failed")
```
- ⑥ You can add different patterns for input signal dominance and so on.

## 15 Interface, Step 8 :

- \* Later on when we get to the driver and monitor we will have to take care of driving and sampling timing and race conditions, the robust way to avoid race conditions is to use a clocking block for each driver and monitor.
- \* In a clocking block, there's an 1) output skew : defines when to drive the output after the clocking event (like posedge clk).  
2) input skew : defines when to sample the inputs before the clocking event, to insure data is stable.

### Clocking Blocks of Driver and Monitor in the Interface :

```
clocking driv_cb @ (posedge clk);
  default output #1ns,
  output rst, en, re, data_in, addr;
endclocking
```

```
clocking mon_cb @ (posedge clk);
  default input #1ns,
  input rst, en, re, data_in, addr, data_out, valid_out;
endclocking
```



<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/e80532cc31505e470a7bb36f474c5602cc0d9b>

## 16 Driver and Monitor Classes, Step 9 :

- \* Driver issues the DUT signals at the "driv-cb" event like so @ (drv-vif.driv-cb) using non-blocking assignments "<=" in "run-phase"
- \* Monitor samples at its clocking block event as well @ (mon-vif.mon-cb) but with blocking assignments "=" in "run-phase".

<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/e4eb0fe84767943dd605c3b3901cd50a5bf60ef>

## 17 Scoreboard Class, Step 10 :

- \* The scoreboard has to mimic the DUT behaviour with a ref model, Since this's a memory module ... we will create a 2D array to store the data\_in with the write command and retrieve that data to compare in case of a read command. [31:0] shadow\_mem [int];  
\* The DUT takes 1 clk cycle to process its commands:  
1- Reset is sync., so at clk 1 for ex, DUT samples "rst=0" but memory, data\_out & valid\_out = 0 at clk 2.

→ associative array

2. Read Command is sampled at edge x but the expected data\_out is only available at edge x+1 whether you use a clocking block or not



Driving at clk edge creates a race condition and depends on simulator scheduling using clocking blocks avoids that and is safer



∴ When scoreboard gets a packet with :

1. (rst = 0) :

① raises a flag "check-reset" for ex, for next call. check\_reset = 1'b1;

② Deletes "shadow\_mem" to mimic DUT shadow\_mem.delete();

③ In the next call there must be a piece of code controlled by the flag and checks output signals

```
if (check_reset) begin
    if (seq_item.data_out != 1'b0 || seq_item.valid_out != 1'b0)
        `uvm_error("Data_Mismatch", $sformatf("data_out = %o", seq_item.data_out));
    check_reset = 1'b0;
end
```

`uvm\_error is a reporting macro ; it also counts # errors .

2. (re = 1) :

① Raises a "check-read-data" flag for next call .

② Stores this Packet addr in a local variable "read\_addr" read\_addr = seq\_item.addr;

③ In the next call , Compares the data\_out of new Packet with that stored in "shadow\_mem [ read\_addr ] " . and turn down the flag .

3. (en = 1) :

① Just store data\_in of that Packet in shadow\_mem [ seq\_item.addr ] seq\_item is the Packet

Complete Scoreboard Code:

```
// Process Prev. Command
if (check_reset)
    ...
else if (check_read_data)
    ...
// Process Current Command
if (!seq_item.rst)
    ...
else if (!seq_item.en)
    ...
else if (seq_item.re)
    ...

```

## 18 Subscriber Class, Step II : link to Commit 3

<https://github.com/yasmine-eldesoukie/uvm based verification environment for memory/commit/d0742e0613b31914ee954115257e8a955a258eee>

\* This is where we write the Functional Coverage Code to check if the input signals hit the desired cases or test , or to find holes in them to create dedicated tests to cover these scenarios .

\* Before the Constructor Code , add the Covergroups with their Coverpoints , Cross\_Coverage , bins and so on .

\* Then inside the Constructor , use cg = new (); where cg is the name of the Covergroup .

\* Inside "write" Function : Parameterize it with "my\_seq\_item" class

Sample the data of the received signals from monitor packet "seq\_item" . But make it under the condition of rst off .

```
virtual function void write ( my_seq_item t );
    seq_item = t; // seq_item is an instance of my_seq_item
    if (!t.rst)
        cg.sample();
endfunction
```

→ created in brief\_phase

\* Covergroup cg ; keyword

```
name_of_Coverpoint : Coverpoint seq_item.input_signal_1;
name_of_Coverpoint : Coverpoint seq_item.input_signal_2 [
    illegal_bins name_of_bins = { value1, value2 };
    bins name_of_bins [ ] = { [ value1 : value2 ] };
    ignore_bins name = { value };
];
```

```
Cross_Cover_1: cross name_of_cp1, name_of_cp2 ;
Cross_Cover_2: cross name_of_cp2, name_of_cp3 [
    bins name = cross_Cover_2 with ( Condition );
];
endgroup
```

- \* Each Covergroup has Coverpoints, each Coverpoint can have bins (for 1 or multiple values), illegal\_bins (must not happen) and ignore\_bins (to exclude from coverage). Simulator creates "bins" for each Covergroup, each bin can have 1 or more values dep. on the total number of possible values, naming them like bins all\_Zeros = ['h0']; is for readability.
- \* The difference between cp-a Coverpoint seq-item.a; & cp-a Coverpoint seq-item.a { bins all\_Zeros = ['h0']; } is that the 1st only hits 100% if all possible value of a are hit, The latter hits 100% if all its "bins" are hit, in this case it's "h0".
- \* Cross Coverage is between two Coverpoints, it checks all their combinations if cross\_Cover: cross cp-a, cp-b; and checks certain combinations of them that follow a certain condition cross\_Cover: cross cp-a, cp-b, { bins cross\_1 = cross\_Cover with (Condition); } Multiple Examples Can be found in the provided link above.

### Examples of Coverage for Memory DUT :

- 1 @ check data\_out hit special cases : all 1's, all 0's, toggle, etc.
- 2 @ check different combinations of en & re : z'b10, z'b01, z'b11 .
- 3 @ check addr hit all its possible values, all values transitioned to every other value, or that it went from all 1's to all 0's and vice versa.
- 4 @ check that each addr has been written in and read from with cross coverage between 2 & 3.

Note: in this step, you'll probably need to make adjustment to constraints in "seq-item" class and "sequence" class and maybe add directed tests.