

# Hierarchal Testbench Configuration Using uvm\_config\_db

June 2014



#### **Authors**

**Hannes Nurminen** Professional Services, Synopsys

Satya Durga Ravi Professional Services, Synopsys

## **Abstract**

SoC designs have become extremely complex as more and more IP blocks are integrated into them. This increases the verification challenge manifold in terms of configuration and data handling, as well as architecting and maintaining a large verification environment. Hence it has become very important to create a robust and reusable testbench using a proven methodology that does not just facilitate but also improves the efficiency in verifying different configurations of the device under test (DUT).

Accellera Systems Initiative's Universal Verification Methodoloveefic.6(i)-12Vinreleto create a scalable, robust and reusab base classes that help facilitate the creation of these testbenches, including an easy way to pass objects and variables across testbench hierarchy.

For engineers who are new to verification methodologies or are in the process of adopting UVM, this paper focuses on the UVM configuration mechanism "uvm \_ confg \_ db", which helps in passing different class properties across hierarchy testbench components. Through the use of examples, the usage, techniques, and limitations of uvm \_ confg \_ db are explained.

## Introduction

To address the needs of today's verification architecture, a hierarchical setup of components is necessary to easily move or share configurations and other parameters across different testbench components. To enable this, UVM provides the infrastructure to maintain a database of objects and variables that can be populated and accessed using strings. This is achieved using the UVM syntax uvm \_ confg \_ db.

Using uvm \_ confg \_ db, objects can share the handle to their data members with other objects. Other testbench components can get access to the object without knowing where it exists in the hierarchy. It's almost like making some class variables global or public. Any testbench component can place handles and function defines the object type, the name and hierarchical path to the object

searched for.

# **How to Use It – Different Syntax and Operation**

Explicit set() and get() call functions are how you interact with the uvm \_ confg \_ db. The uvm \_ confg \_ db class functions are static, so they must be called using the ": " operator.

```
2
                    #(<type>)::
                                                         cntxt,
                                                         inst _ name,
3
4
                                                         field _ name,
                                     . . .
5
                                    <type>
                                                         value)
6
7
                   #(<type>)::
                                                         cntxt,
8
                                                         inst _ name,
9
                                                         field _ name,
10
                                                         value)
```

Figure 1: set() and get() function syntax

"cntxt" and "inst\_ name" are used to specify the storage location or address of the object handle. When used properly these parameters define the hierarchical path to the object data.

- "¿HOG B "Os In Prisme for the object. It does not have to match the object's actual name in the source code. Objects using **set()** and **get()** must use exactly the same name, otherwise the receiving party (**get()**) will fail to find the object from **uvm \_ confg \_ db**.
- " Y D O' K the actual object handle shared through the uvm \_ confg \_ db. Multiple recipients accessing an object via get(), will access the same object.
- "<type> " is used as a parameter for the uvm \_ confg \_ db class to identify the object from the uvm \_ confg \_ db. "<type>" which may be either an integral or string, is the class name of the " Y D O.XThe exception is with enumerated type variables which must use int otherwise the set() won't work as expected.

```
13 ..... {sin
```

Figure 2: config\_db for enum type

The set() specifies the "address" (cntxt & inst \_ name ) where the object handle is stored to control the recipient(s) of the object. The get() has the same flexibility, and can freely select from where the information is to be fetched. In practice get() can be used to fetch an object destined to any component in the hierarchy. Typically for set() and get(), this is used in the "cntxt" field to specify the current instance/scope. set() uses "inst \_ name" to address the object to the appropriate sub-block in the hierarchy. get() often uses empty ("") inst \_ name , since it typically is getting the objects destined for itself.

#### Figure 3: set() and get() typical use

uvm \_ confg \_ db has two additional functions exists() and wait \_ modifed(). exists() verifies that the
defined variable is found in the uvm \_ confg \_ db. The wait \_ modifed() function blocks execution until the
defined variable is accessed with the set() call.

Figure 4: exists() and wait\_modified() typical use

## **Automatic Configuration**

UVM also offers build-time configuration of uvm \_ component (and extended) classes utilizing uvm \_ confg \_
db. In automatic configuration, it is sufficient to call set() from an upper layer in the hierarchy and the get() will automatically execute at build time without requiring an explicit call. Automatic configuration utilizes the uvm \_ confg \_ db feature "under the hood" to pass the configuration values from higher level testbench components in the hierarchy to its lower level components.

For automatic configuration to work there are two important requirements:

```
The variable or object must have the appropriate FLAG in uvm _ feld _ * macros super() must be called in build _ phase()

3 class agents extends uvm _ agent ;
4 int i4;
5 ` uvm _ component _ utils _ begin (agent)
6 ` uvm _ feld _ int (i4, UVM _ ALL _ ON )
7 ` uvm _ component _ utils _ end
8
9
```

During the build phase of the simulation the agent object's "i4" variable would get value 1111. It is important to note that automatic configuration happens only at build phase.

## **Command Line**

Compilation and simulation time are the major contributors to verification overhead. The ability to change the configuration or parameters without being forced to recompile is critical. The UVM class uvm \_ cmdline \_
processor provides a mechanism to capture the command line argument and pass to verification components the testcase name, verbosity, configuration and other attributes.

Configuration overriding can only be done from the command line for integer and string using the following:

```
XYPBVHWBFRQ¿JBLQW FRPS! ¿OHG! YDOXH!
XYPBVHWBFRQ¿JBVWULQJ FRPS! ¿HOG! YDOXH!
```

There is no way to override the object from the command line, because X Y P B R EdwhhblFbWpassed to the simulation.

When using the command line argument to set the configuration, make sure that the "<type>" used in uvm \_ confg \_ db set() and get() functions is uvm \_ bitstream \_ t for integer and the "<type>" for string is as shown below:

The log message generated during simulation is:

UVM\_INFO@ 0: U H S R U \$W8H9UO B & O ' / , 1 (\$BS3\$50 \$2 L&FO&PU Q setting I U R tPre F R P P DO 02 Q H X Y P B V H W B F R Q ¿ J B L Q W X Y P Ba,W6H V W B W R S H Q Y B L

UVM\_INFO@ 0: U H S R U \$W8H9UO B & O  $^{\prime}$  / , 1 (\$BS3S50 2 L&FOQRU Q setting I U R tPre F R P P DO 0. Q H X Y P B V H W B F R Q  $\stackrel{.}{_{\sim}}$  J B BY WWW HL QV JW X BY PWF RR CS R WHH CG Y B L

## **Cross-Hierarchical Access**

The set() and get() parameters "cntxt", "inst\_name" and "¿HOGB" @nDkPil-possible to use a number of different paths to the same object. "cntxt" uses actual object hierarchy whereas "inst\_name" and "¿HOGB" @nDkPil-possible to use a number of different paths to the same object. "cntxt" uses actual object hierarchy whereas "inst\_name" and "¿HOGB" B" "QuDxSPhled hierarchy path with names given to the objects in create(ynew()) method. It is good practice to create the objects with the same name as the object name.

When referencing down in hierarchy, it should be enough to use this in "cntxt" and then provide the path and/

### Figure 9: set() and get() functions need "cntxt" parameter of type uvm\_component or null

One common usage of uvm \_ confg \_ db outside uvm \_ components, is delivering values from K G O B tw R S the testbench, including access to interfaces instantiated on K G O B. Who K Stended from any UVM class, uvm \_ confg \_ db can still be utilized and communication with the UVM part of the testbench is possible.

If set() or get() function is used with "cntxt" parameter not pointing to object of uvm \_ component extended classes, there will be a compile error as shown below.



Figure 10: Example error messages when trying to use "this" for non-uvm\_component in set/get

## Problems, Errors and Debug

Even though operation and use of **set()** and **get()** functions with **uvm \_ confg \_ db** are logical and quite simple, **uvm \_ confg \_ db** related debugging is often needed. Some errors may stop the compile or simulation making them easy to find, as opposed to a coding error that simulates without error even though the **get()** function was receiving incorrect objects. Some common types of errors are:

- `Compile time errors
  - Parameter type does not match provided T value
  - Trying to use this-pointer from class not extended from uvm \_ component
- Simulation time errors
- get() does not find what was set using set() due to misspelling of "inst \_ name " or " ¿ H O G Y D O X H
- null object access attributed to get() used before set()



Figure 11: Synopys' VCS DVE UVM debug dialog window

The UVM command line option + 8 9 0 B & 2 1 ), \* B '%mBkes all set() (and get() calls visible in the simulation log. However doing this makes the log file too verbose and difficult to interpret. For this reason tracing is typically turned on only when finding a specific uvm \_ confg \_ db problem. Below is an example of log messages printed out when set() and get() functions are executed.

## Don'ts:

- Avoid using the uvm \_ confg \_ db mechanism excessively as it may cause
  performance issues
- `Avoid using the automatic configuration or implicit get() method call

Apart from the above recommendations, it is recommended to use a UVM-aware GUI-based debugging tool such as Synopsys' VCS Discovery Visualization Environment (DVE). As part of Synopsys Professional Services we have

```
QHZBYDt@i\gdex(); #1;
                                                   UHJH[S QDPH B DJHQW B "
                          WR HYHU\ DJHQW
117
       $display ‡
118
       uvm _ confg _ db#(int )::set(this ,
119
                    "name _ agent _ ?",
                     ‡LBRIBHQY.
121
                    i4 _ env
122
        QHZBYDt@i%gdex(); #1;
124
                          WR HYHU\ DJHQW
                                                   UHJH[S QDPH B DJHQW
       $display ‡
125
       uvm _ confg _ db#(int )::set(this ,
126
                    "name _ agent _ *",
                     ‡LBRIBHQY.
128
                    i5 _ env
129
                    );
        QHZBYDt@i%gdex(); #1;
         GLVSOD\ ‡
                               WR HYHU\ DJHQW
                                                        UHJH[S
                                                                    DJHQW
132
       uvm _ confg _ db#(int )::set(uvm _ root::get(),
133
                    "*agent*",
           H p O
                                           uvm _ confg _ db#) • p(~VhH W Pp ð W
```