

## Lessons from the Trenches: Migrating Legacy Verification Environments to UVM™

Tutorial presented by members of the VIP TSC







### **Agenda**

- Anecdotes From Hundreds of UVM Adopters
  - John Aynsley, Doulos
- Migrating from OVM to UVM A Case Study
  - Hassan Shehab, Intel
- A Reusable Verification
   Testbench Architecture
   Supporting C/UVM Mixed Tests
  - Richard Tseng, Qualcomm
- UVM to Rescue Path to Robust Verification
  - Asad Khan, Texas Instruments

- OVM to UVM Migration
   There and Back Again, a Consultant's
   Tale
  - Mark Litterick, Verilab
- IBM Recommendations for OVM → UVM Migration
  - Wes Queen, IBM
- FPGA chip verification using UVM
  - Charles Zhang & Ravi Ram, Paradigm
- Please ask questions during/end of each talk
- Fill out survey questionnaire







# Anecdotes From Hundreds of UVM Adopters

#### John Aynsley









#### **General Comments**

Especially managers and self-teachers



- Underestimating the learning curve
- Directed tests versus constrained random verification
- Reuse and OOP expertise

UVM only gets you so far







#### **OVM to UVM**

- There exists plenty of guidance on migrating from OVM to UVM
- www.uvmworld.org

ovm2uvm\_migration.pdf

- verificationacademy.com/verification-methodology
- http://www.doulos.com/knowhow/sysverilog/uvm/ovm-to-uvm







#### **General Issues with UVM**

Studying the documentation is not enough!



- Certain UVM concepts are not straightforward
- SV/UVM terminology can be a barrier
- The sheer size of the UVM BCL
- Too much choice



Lack of recommended practice and naming conventions







## **Top UVM Time-Wasters**

- Field macros
- `uvm\_do macros
- Deprecated OVM sequence mechanisms
- Confusion over the config db (and the OVM legacy)







#### **Those Evil Field Macros?**

```
class basic transaction extends uvm sequence item;
  rand bit[7:0] addr, data;
  function new (string name = "");
    super.new(name);
  endfunction: new
  `uvm object utils begin(basic transaction)
    `uvm field int(addr, UVM DEFAULT)
    `uvm field int(data, UVM BIN | UVM NOCOPY)
  `uvm object utils end
endclass : basic transaction
```







### Field Macro Flags

Inclusion in operations

UVM DEFAULT

all on

UVM COPY

UVM COMPARE

UVM PRINT

UVM RECORD

UVM PACK

default

UVM NOCOPY

UVM NOCOMPARE

UVM NOPRINT

UVM NORECORD

UVM NOPACK

need to set explicitly

UVM READONLY



not configured

SYSTEMS INITIATIVE



#### Overriding do\_compare

```
class bus xact extends uvm sequence item;
                                                tx1.compare(tx2)
  function bit do compare(uvm object rhs, uvm comparer comparer);
   bus xact t;
   bit result = 1;
    $cast(t, rhs);
    result &= comparer.compare field("op", op, t.op, $bits(op));
    if (op != NOP)
      result &= comparer.compare field("addr", addr, t.addr,
                                                         $bits(addr));
                            Collects mismatches
    return result;
  endfunction
  `uvm object utils begin(bus xact)
    `uvm field int(op, UVM NOCOMPARE)
                                              Turn off default comparison
    `uvm field int(addr, UVM NOCOMPARE)
  `uvm object utils end
endclass
                                        Also uvm_packer, uvm_recorder, ...
```





SYSTEMS INITIATIVE

#### Field Macros and Overridden Methods

```
uvm_comparer comparer = new;
comparer.policy = UVM_SHALLOW;
comparer.show_max = 999;
tx1.compare(tx2, comparer);
```

#### Pseudo-code

```
begin

bit result = 1;

result &= tx1.field_automation(tx2);

result &= tx1.do_compare(tx2);

output_mismatch_report;

return result;
end
```

```
UVM_COMPARE

UVM_NOCOMPARE

UVM_REFERENCE
```







## **Stop Faffing Around!**

```
class basic transaction extends uvm sequence item;
  `uvm object utils(basic transaction)
  function bit do compare (uvm object
                                       rhs,
                          uvm comparer comparer);
   bit result = 1;
   basic transaction tx;
    $cast(tx, rhs);
    result &= (addr == tx.addr);
    result &= (data == tx.data);
    return result;
  endfunction
endclass : basic transaction
```







#### The Dreaded super.build\_phase

```
uvm_config_db#(int)::set(this, "m_env.m_driv", "count", 999);
```

```
int count;

'uvm_component_utils_begin(my_component)

'uvm_field_int(count, UVM_DEFAULT)

'uvm_component_utils_end

function void build_phase(uvm_phase phase);

super.build_phase(phase);

calls apply_config_settings
endfunction
```







## The Moderately Evil `uvm\_do

```
`uvm_do(req)
```

```
req = tx_type::type_id::create("req");
start_item(req);
if( !req.randomize() ) `uvm_error(...)
finish_item(req);
```

Equivalent?







#### **Expanded Invocation of `uvm\_do**

`uvm\_do(SEQ\_OR\_ITEM)

```
begin
 uvm sequence base seq;
 begin
 uvm object wrapper w ;
 w = SEQ OR ITEM.get type();
  $cast(SEQ OR ITEM , create item(w , m sequencer, `"SEQ OR ITEM`"));
 end
  if (!$cast( seq,SEQ OR ITEM)) start item(SEQ OR ITEM, -1);
  if (( seq == null || ! seq.do not randomize) && !SEQ OR ITEM.randomize()
with {} ) begin
    `uvm warning("RNDFLD", "Randomization failed in uvm do with action")
 end
  if (!$cast( seq,SEQ OR ITEM)) finish item(SEQ OR ITEM, -1);
 else seq.start(m sequencer, this, -1, 0);
 end
```







#### The OVM Sequencer Library

```
class my_sequencer extends ovm_sequencer #(basic_transaction);

`ovm_sequencer_utils(my_sequencer)

function new(string name, ovm_component parent);
   super.new(name,parent);
   `ovm_update_sequence_lib_and_item(basic_transaction)
   endfunction : new
```

Populates sequence lib with simple, random, & exhaustive sequences

endclass: my\_sequencer

Deprecated in UVM







#### The OVM Sequence

```
class my sequence extends ovm sequence #(instruction);
  function new(string name = "");
    super.new(name);
  endfunction: new
  task body;
 endtask
  `ovm sequence utils(my sequence, my sequencer)
```

endclass: my\_sequence

Deprecated in UVM







## Selecting a Sequence in OVM

```
set_config_string("*.m_seqr", "default_sequence", "my_sequence2");
```

```
set_config_string("*.m_seqr", "count", 10);
```

All firmly deprecated in UVM







## Starting a Sequence in UVM

sequence.start(sequencer, parent\_sequence, priority);







#### **Draft UVM Sequence Library**

```
class my_seq_lib extends uvm_sequence_library #(my_tx);
   `uvm_object_utils(my_seq_lib)
   `uvm_sequence_library_utils(my_seq_lib)

function new(string name = "");
   super.new(name);
   init_sequence_library();
   endfunction
endclass
```

```
my_seq_lib lib = my_seq_lib::type_id::create();
lib.add_sequence( seq1::get_type() );
lib.add_sequence( seq2::get_type() );
lib.selection_mode = UVM_SEQ_LIB_RAND;
if ( !lib.randomize() ) ...
lib.start(m_env.m_seqr);
```







#### Other Detailed UVM Issues

- Need for run-time phasing
- Confusion over when to raise/drop objections
- Register layer seems hard

- How to handle layered sequencers and agents
- Confusion over the semantics of lock/grab

Session 8: Hardcore UVM - I, Weds 10:30am - 12:00pm

The Finer Points of UVM: Tasting Tips for the Connoisseur (myself)

Beyond UVM: Creating Truly Reusable Protocol Layering (Janick)







## Things Missing from UVM

- Mixed language support (RTL)
- Mixed language support (TLM)
- Using UVM with analog/AMS







# Migrating from OVM to UVM A Case Study

Hassan Shehab
Technical Validation Lead
Intel Corporation









## **Agenda**

 We present a OVM compatibility layer on top of UVM that allows the use of OVM based IPs on UVM source code

 We look at the results of using the compatibility layer by migrating a SoC consisting of 25+ OVM VIPs







#### Introduction

- We present a case study of migrating a SoC environment fully developed on OVM to UVM
  - UVM recommends running a converter script on the source code to replace the ovm\_\* symbols with uvm\_\* symbols
  - This mandates either abandoning the OVM code base of the VIPs or maintaining two repositories
  - With heavy OVM in use, this is NOT practical as VIPs needs to go into SoCs with OVM base and UVM base running in parallel.
- Enhanced the OVM compatibility layer developed originally by Mark Glasser part of UVM EA
  - https://forum.verificationacademy.com/forum/uvmovm-kit-downloads-and-usercontributions-forum/kit-downloads-and-user-contributions/18304-uvm-ea-ovmcompatibility-kit
  - Enhanced the compatibility layer to work with UVM 1.1 release
  - This layer sits on top of UVM and allows the migration to UVM w/o having to modify the OVM IPs







## **OVM Compatibility Layer**

- The compatibility layer is done in a way that the existing OVM based environment can use it just as an OVM version change
  - The code below shows the ovm\_pkg content which is derived from the UVM code base (green), the compatibility layer code are split into the files (red) as shown below

```
package ovm_pkg;
```

```
include "ovm_macros.svh"

typedef class ovm_seq_item_pull_imp;
typedef class ovm_seq_item_pull_port;

include "dpi/uvm_dpi.svh"

include "base/base.svh"

include "tlm1/uvm_tlm.svh"

include "comps/comps.svh"

include "seq/seq.svh"

include "tlm2/tlm2.svh"

include "reg/uvm_reg_model.svh"

include "compatibility/ovm_compatibility.svh"

include "compatibility/urm_message.sv"

include "compatibility/legacy_compatibility.svh"
```

endpackage







## Mapping macros `uvm\_\* to `ovm\_\*

```
`include "uvm macros.svh"
`define ovm do callbacks(CB,T,METHOD CALL)
 `uvm do callbacks(T,CB,METHOD CALL)
`define ovm do callbacks exit on(CB,T,METHOD CALL,VAL)
  `uvm do callbacks exit on(T,CB,METHOD CALL,VAL)
`define ovm do task callbacks(CB,T,METHOD CALL)
  `uvm do task callbacks(T,CB,METHOD CALL)
`define ovm do obj callbacks(CB,T,OBJ,METHOD CALL)
  `uvm do obj callbacks(T,CB,OBJ,METHOD CALL)
`define ovm do obj callbacks exit on(CB,T,OBJ,METHOD CALL,VAL)
  `uvm do callbacks(T,CB,METHOD CALL)
`define ovm do obj task callbacks(CB,T,OBJ,METHOD CALL)
  `uvm do obj task callbacks(T,CB,OBJ,METHOD CALL)
`define ovm do ext callbacks(CB,T,OBJ,METHOD CALL)
  `uvm do ext callbacks(T,CB,OBJ,METHOD CALL)
`define ovm do ext callbacks exit on(CB,T,OBJ,METHOD CALL,VAL)
  'uvm do ext callbacks exit on (T,CB,OBJ,METHOD CALL,VAL)
`define ovm do ext task callbacks(CB,T,OBJ,METHOD CALL)
  `uvm do ext task callbacks(T,CB,OBJ,METHOD CALL)
`define ovm cb trace(OBJ,CB,OPER)
   `uvm cb trace(OBJ,CB,OPER)
```







#### Mapping classes uvm\_\* to ovm\_\*

```
// Typedefs: UVM->OVM
//
// Non-parameterized UVM classes can be simply typedefed to corresponding
// OVM types.
typedef uvm void
                                ovm void;
typedef uvm root
                                ovm root;
typedef uvm factory
                                ovm factory;
typedef uvm object
                                ovm object;
typedef uvm transaction
                                ovm transaction;
typedef uvm component
                                ovm component;
// Parameterized UVM classes cannot be simply typedefed to corresponding
// OVM types, have to extend from uvm equivalents and pass the right parameters
class ovm analysis port #(type T=int) extends uvm analysis port#(T);
  function new(string name, uvm component parent=null);
    super.new(name, parent);
  endfunction
endclass
```







#### **Mapping Enumerated Types**

```
typedef uvm active passive enum ovm active passive enum;
uvm active passive enum OVM PASSIVE = UVM PASSIVE;
uvm active passive enum OVM ACTIVE = UVM ACTIVE;
typedef uvm verbosity ovm verbosity;
parameter uvm verbosity OVM NONE
                                   = UVM NONE;
parameter uvm verbosity OVM LOW
                                   = UVM LOW;
parameter uvm verbosity OVM MEDIUM = UVM MEDIUM;
parameter uvm verbosity OVM HIGH
                                   = UVM HIGH;
parameter uvm verbosity OVM FULL
                                   = UVM FULL;
parameter uvm verbosity OVM DEBUG
                                   = UVM DEBUG;
typedef uvm severity ovm severity;
uvm severity OVM INFO
                         = UVM INFO;
uvm severity OVM WARNING = UVM WARNING;
uvm severity OVM ERROR
                        = UVM ERROR;
uvm severity OVM FATAL
                         = UVM FATAL;
```







### **UVM Source Code Change**

- With the compatibility layer we can get majority of the OVM based VIPs and environments to compile clean
- But there were still few UVM files we have to change to make it 100% back-ward compatible to our OVM usage
  - 1. uvm\_final/src/base/uvm\_component.svh
  - 2. uvm\_final/src/base/uvm\_factory.svh
  - 3. uvm\_final/src/base/uvm\_globals.svh
  - 4. uvm\_final/src/base/uvm\_root.svh
  - 5. uvm\_final/src/comps/uvm\_driver.svh
  - 6. uvm\_final/src/seq/uvm\_sequencer.svh
  - 7. uvm\_final/src/seq/uvm\_sequencer\_param\_base.svh
  - 8. uvm\_final/src/tlm1/sqr\_connections.svh







#### uvm\_component

Have to add pre\_run() and call it from start\_of\_simulation phase

```
function void uvm_component::start_of_simulation(); `ifdef OVM pre_run(); `endif
  return; endfunction
```

Have to add the ovm\_report\_\* functions into the uvm\_component







#### uvm\_factory

Have to add create\_object() function into uvm\_factory

Have to add set\_inst\_override function into uvm\_factory







#### uvm\_globals and uvm\_root

Have to add ovm\_test\_top to uvm\_globals.svh

```
`ifdef OVM
// This is set by run_test()
uvm_component ovm_test_top;
`endif
```

Have to set ovm\_test\_top in uvm\_root.svh

```
`ifdef OVM
     ovm_test_top = uvm_test_top;
`endif
```







#### Results

- Successfully migrated a OVM based SoC to UVM using the OVM compatibility layer
  - There were 25+ VIPs with complex bus interfaces like OCP/AHB/AXI and several I/Os like PCIE/USB/SDIO etc.
  - Have to add more code in the compatibility layer as few of the legacy IPs are even dependent on AVM compatibility layer in OVM
    - Ideally would be better to clean the IP source code to remove that legacy, but preferred to add support in compatibility layer as proof-of-concept
  - Managed to get all level-0 regressions containing 100+ tests passing
    - Took ~3 person weeks to enhance the compatibility layer
    - Took ~2 person weeks to run regressions and achieve same results as the reference
  - Filed several Mantis items on UVM source code based on the issues/bugs observed
    - e.g. print\_topology() was crawling in UVM compared to OVM, simulator enhancements were needed to match OVM performance







#### **Summary**

- Ideally, it would be nice to start from a clean code and not create a compatibility layer
- In our case this is not possible because of:
  - The amount of OVM code that we have which needs to be converted and tested
  - The IPs that we get from all over the place internally and externally.
- It will takes us years to use the clean code approach. Huge impact on
  - Our resources
  - Execution schedule
- Having the compatibility layer enables a SoC project to move to UVM when they
  decide and therefore without any effort or impact on execution schedule.
- This way a SoC can start developing new code in UVM and take opportunistic approach in converting old code as intercepts permit.







# A Reusable Verification Testbench Architecture Supporting C and UVM Mixed Tests

**Richard Tseng** 

Qualcomm, Boulder CO









### **Agenda**

- Introduction
- Testbench Features
- Testbench Architecture Overview
- Challenges & Solutions
- Summary
- -Q&A







### Introduction







#### Introduction

- UVM would be the ultimate methodology for design verification
  - It's the latest and greatest technology
  - Strong EDA vendor supports
  - VIPs and test sequences can be reusable
- In many applications, C was primarily used for testing legacy designs
  - Many scripts were written to support C tests
  - Thousand lines of C code have been evolved from many product generations
  - They are recognized as "golden regression suite"
- The goal is to build an UVM testbench
  - To re-use verification components
  - To re-use existing C and UVM test sequences







### **Testbench Features**







#### **Testbench Features**

- •The UVM testbench architecture allows us:
  - To reuse C and UVM tests in various platforms
  - To run C/UVM tests simultaneously
  - To reuse UVM verification components
  - To easily integrate UVM register layer







# Testbench Architecture Overview







#### **Testbench Architecture Overview**









- 1. Creating UVM sequences with API Tasks
- 2. Reusing high-level C and UVM tests and testbench components
- 3. Integrating UVM Register Layer







- 1. Creating UVM sequences with API Tasks
- 2. Reusing high-level C and UVM tests and testbench components
- 3. Integrating UVM Register Layer







#### Creating UVM sequences with API tasks

- The C tests are usually programmed with API tasks
- Typically in an UVM SVTB, UVM macros are used to create a test sequence,
  - ie, `uvm\_do(), `uvm\_do\_with(), and `uvm\_do\_on\_with()
- UVM "do" macros don't match the legacy C tests, using the API task is more consistent
- To reuse the C tests in an UVM testbench, C API tasks need to be mapped to UVM "do" macros
  - Constraints can be specified in API task arguments
  - Default constraints can be specified within the API task







#### UVM "do" macro V.S. API task

Transaction generated with "do" macro:

Equivalent API task being called:

```
// axi_master_write(address, data)
axi_master_write('h1000, 'h1234_5678);
```







#### **Create Bus Transaction with API task**

```
task axi master base seq::axi mst write(input bit [31:0] addr,
                                      input bit [31:0] data);
                                 Create a sequence item
  uvm create(req)
 assert (req.randomize() with {
     req.cmd type == AXI WR INC;
     req.address == addr;
     req.data == data
     req.burst length == 1;
                                Randomize with address,
     req.burst size == 4;
                                data, and default
     req.bready delay == 1;
                               constraints
     req.avalid delay == 0;
   }) else begin
     `uvm fatal(....)
    end
                               Send to sequencer
  `uvm send(req)
endtask: axi mst write
```







- 1. Creating UVM sequences with API Tasks
- 2. Reusing high-level C and UVM tests and testbench components
- 3. Integrating UVM Register Layer







## Map generic transactions to bus interface specific API tasks

- Portable tests are composed of generic API tasks
- GPB (Generic Primary Bus) tasks
  - Mapped to front-door register interface/primary interface transactions:
    - gpb\_write() => axi\_master\_write()
    - gpb\_read() => axi\_master\_read()
- GST (Generic Slave Transaction) tasks
  - Mapped to slave device's back-door transactions
    - gst\_bkdr\_write() => axi\_slave\_bkdr\_write()
    - gst\_bkdr\_read() => axi\_slave\_bkdr\_read()







## Map generic transactions to bus interface specific API tasks (cont'ed)

```
// generic test sequence
task my_test_seq::body():
   gpb_write(reg1, data);
   gpb_read(reg2, data);
   gpb_nop(10); // idle for 10 clocks
   gst_bkdr_write(32'h0, 32'h1234_5678);
   gst_bkdr_read(32'h5555_aaaa, read_data);
endtask
```

```
//bus interface specific sequence
task my_test_seq::body():
   axi_master_write(reg1, data);
   axi_master_read(reg2, data);
   axi_master_nop(10); // idle for 10 clocks
   axi_slave_bkdr_write(32'h0, 32'h1234_5678);
   axi_slave_bkdr_read(32'h5555_aaaa, read_data);
endtask
```







#### Make C and UVM tests identical!

```
class sv_main_seq extends prj_base_seq;
 task body();
   bit[31:0] read data;
    `uvm_info(get_type_name(), "Test starts", UVM_MEDIUM)
   gpb write(control reg, 32'h0000 3204);
   gpb read(status reg, read data);
   gst bkdr write('h400, 32'hA5);
   gst bkdr read('h400, read data);
 endtask: body
                                                         Same UVM macro
endclass: sv_main_seq
void c main seq(void) {
 unsigned int read data; // 32 bit unsigned integer
uvm info( func , "Test starts", UVM MEDIUM)
 gpb write(control reg, 0x00003204);
 gpb_read(status_reg, &read_data);
 gst bkdr write(0x400, 0x000000A5);
 gst bkdr read(0x400, &read data);
```





SYSTEMS INITIATIVE

### Using UVM reporting macros in C

- UVM has a good reporting service
  - Can specify the message verbosity level and severity
  - Generate messages which show where and when they are called during simulation

•Macros include `uvm\_info(), `uvm\_warning(), `uvm\_error(), `uvm\_fatal()







#### Implement UVM reporting macros in C

```
// export uvm_rpt_info function
export "DPI-C" function uvm_rpt_info;
```

```
// In C file, define verbosity level just as UVM definitions
#define UVM_LOW     100
#define UVM_MEDIUM 200
#define UVM_HIGH     300
```

```
// define C macros
#define uvm_info(id, message, verbosity) \
    uvm_rpt_info(id, message, verbosity);
```







#### Reuse high-level VIPs



- 1. Creating UVM sequences with API Tasks
- 2. Reusing high-level C and UVM tests and testbench components
- 3. Integrating UVM Register Layer







### **UVM** Register Layer integration

- Register models are auto generated by customized scripts or EDA vendor provided tools
- Explicit prediction mechanism is used in our example
  - It's recommended prediction mechanism
  - Register model will be updated
    - Register sequences issued from register sequencer (auto prediction)
    - Bus transactions issued from all other bus agents (passive prediction)
- Required additional verification components to be added, so that the followings can be reused across different platforms:
  - Register sequences
  - Predictor







#### **UVM** Register Layer Integration



**SYSTEMS INITIATIVE** 

### Summary







#### **Summary**

- In our applications, test sequences and the VIPs were reused in multiple testbenches
  - Modem core–level testbench
    - GPB => QSB AXI master
    - GST => QSB AXI salve
  - Modem block-level testbenches
    - GPB, GST => proprietary bus interfaces
  - Modem emulation platform testbench
    - GPB => AHB Master
    - GST => Off-chip ZBT memory
  - Regressions have been run with multiple simulators
- The testbench architecture extends the reusability beyond the scope of the UVM technology, and across the C and SV language boundary







### Q&A







## UVM to the Rescue – Path to Robust Verification

**Asad Khan** 

**Design Verification Lead** 

Texas Instruments, Inc.









### **Agenda**

- 1 Our Challenges Before...
- 2 Adopted Solutions and Roadblocks
- 3 UVM to the Rescue Key Issues Addressed
- 4 Issues Detailed & UVM Solution
- 5 UVM in AMS Simulation
- 6 Conclusions
- 7 Q/A







### Our Challenges Before...

Lack of <u>constrained</u> random stimulus

Looks of several verification environments patched together

Lack of <u>functional</u> <u>coverage driven</u> methodology

Lack of provisioning for verification IP (VIP) support



Lack <u>automated</u> <u>checking</u> aspects

Lack <u>structured</u> <u>methodology</u> and aspects of <u>reusability</u>

Lack of block level to top level test back compatibility

Bulky in nature with <u>directed test</u> overload – a <u>management nightmare</u>







### **Adopted Solutions and Roadblocks**



Lack of Full LRM
Support – Not all
features supported

Simulator Dependent – Compatibility issues with Major Options

Needed Wrapper
Development to build a
Test Environment

Language Specific – Needing Expertise



Not Open Source – Requiring Support, Licenses and Customizations

Lack of Base Class Functions – A lot of coding e.g. deep copy etc

Lack of Phases and Several Roadblocks in Methodology

VIP Solutions Lacking Standardization – Everyone Seemed to have their own methodology







#### UVM to the Rescue – Key Issues Addressed

**Packet Class Issues** 

<u>Issues due to rewrite</u> of the entire task/function for enhancing it for the newly added data member of the packet class – <u>no automation</u>

**End of Test Issues** 

Waiting for pending items & drain time/delays caused end of test issues

**Inconsistency** in Test Development

Issues e.g. errors, directed test overload due to lack of base test class

Stimulus Control Issues

<u>Controlling multiple DUT Interfaces</u> through Verification Components resulted in complex and non reusable scheme <u>due to Callbacks</u>

BFM/Monitor Reuse Issues

<u>Lack of schemes</u> to <u>reuse BFMs/Monitors</u> from legacy test environments

Lack of Debug Messaging Support

Lack of or <u>Non reusable custom environment printers</u> during runtime caused environment debug issues including <u>complex simulation logs</u>

Miscellaneous Testbench Issues

Issues in writing more tasks to process <u>channel data</u>, repetitive and non-reusable, <u>lack of phasing options</u>, and <u>extensive coding of register models</u>







#### **Packet Class Issues**

From Issue to Solution

We have faced issues in <u>rewriting functions/tasks</u> to override/cast e.g. for <u>copy</u>, <u>display</u> and all others, however, using <u>UVM</u> <u>object utils the data item automation is built in</u> with field level flags

#### **Example: Copy Function Issue**

```
class ahb master trans extends vmm data;
 //DATA members
 rand integer unsigned NumBytes = 0;
 rand integer unsigned NumBeats = 1;
 rand integer busy;
 rand byte data [$];
 rand bit [('AHB HADDR WIDTH -1):0] address = 'h0;
 function vmm data copy(vmm data to = null);
  ahb master trans cpy;
  super.copy data(cpy);
  cpy.NumBytes = this.NumBytes;
  cpy.NumBeats
                   = this.NumBeats;
  cpy.busy
                   = this.busy;
  cpy.address
                   = this.address:
  for(int i = 0; i<this.NumBytes; i++) begin
    cpy.data[i] = this.data[i];
  end
  copy = cpy;
 endfunction:copy
endclass
```



#### **UVM: Built-in Automation**

```
class ahb_master_trans extends uvm_transaction;

//DATA members

rand integer unsigned NumBytes = 0;

rand integer unsigned NumBeats = 1;

rand integer busy;

rand byte data [$];

rand bit [(`AHB_HADDR_WIDTH -1):0] address = 'h0;

...

`uvm_object_utils_begin(ahb_transfer)

`uvm_field_int(NumBytes , UVM_ALL)

`uvm_field_int(NumBeats , UVM_ALL)

//Similarly for other fields

.......

uvm_object_utils_end

....

endclass

UVM_NOCOMPARE, UVM_NOPRINT etc

Data Item Automation in Place!!
```







#### **End-of-Test Issues**

From Issue to Solution

We have always struggled with the issue on how to achieve "end of test".

Logically when all activity completes should be the end of test, however,

checkers could still be checking or testbench components can be busy - how
do we reach true end-of-test?

#### **Problematic End-of-Test Approaches**

- In cases where <u>simulation had to be stopped</u> based on multiple interacting components, <u>custom approaches</u> were used e.g. <u>detecting occurrence of a condition</u> and using events – This was <u>complex</u>, <u>cumbersome</u> and not reusable.
- <u>Callbacks</u> were used in VMM <u>for specific end of test</u> conditions such as monitor observing pass/fail condition. Several times <u>debug time</u> had to be put in when the <u>callback was missed from vmm\_env::build()</u>, and this approach made env <u>non-reusable</u> since test conditions changed per test scenarios.
- A <u>crude "drain time"</u> implementation was used to terminate tests after a certain delay but this resulted in <u>different end of test times for sims at different</u> corners/conditions.





#### **UVM: uvm\_test\_done**

- Using several techniques in UVM we were able to achieve uniform and highly reusable end of test solution:
  - Inside <u>tests</u>, <u>sequences/virtual</u> <u>sequences</u> & <u>uvm components</u>:
    - uvm\_test\_done.<u>raise\_objection</u> (this);
    - uvm\_test\_done.drop\_objection (this);
  - Inside <u>particular phase</u>
    - phase.<u>raise objection(this);</u>
    - phase.<u>drop\_objection(this);</u>
  - Inside <u>run\_phase</u>
    - global stop request()
  - For <u>debug</u> and <u>status of objections</u>:
    - phase.<u>phase\_done.display\_obj</u> ections();







### **Inconsistency in Test Development**

From Issue to Solution

Tests created as program blocks always resulted in <u>inconsistent DUT initializations</u>, <u>redundant code</u> with a <u>test becoming a testbench</u>, and in some cases incorrect sequences because of multiple test writers using independent approach – <u>Test Maintenance Nightmare!</u>

#### program block test issues

```
program my test(myInterface.TxRx my host if);
 'include "my vmm include.sv"
 vmm log log = new("My Env", "main-class");
 my env env;
 initial begin
  // I don't know but I am configuring wrong
   dut register base = 12'hBAD;
  // I am sending good packets to wrong place
   send link pkt();
   while (state!==0) begin // Wait For Complete
     read32(dut register base), tmp data[31:0]);
     state = tmp data[11:8];
   end
   $display("Device has reached U0 successfully\n");
   $display("Test Passed \n");
 end
endprogram
```

False Pass since "state" for this bad address is non-zero by default where expectation was that after sending link packet the correct address would have become non zero



#### **UVM: uvm\_test Class**

```
class test base extends uvm test;
 top env env;
 'uvm component utils(test base)
 virtual function void end of elaboration phase(uvm phase phase);
    dut register base = 12'hAAA;
 endfunction:end of elaboration phase
endclass
class my test extends test base;
  'uvm component utils(my test)
 task run phase(uvm phase phase);
  send link pkt();
  while (state!==0) begin // Wait For Complete
     read32(dut register base), tmp data[31:0]);
     state = tmp data[11:8];
  $display("Device has reached U0 successfully\n");
  $display("Test Passed \n");
endtask:run phase
endclass
```







#### **Stimulus Control Issues**

Issue at Hand?

DUT with <u>multiple interfaces</u> needed <u>interactive and interdependent</u> <u>control of stimulus and response</u> to incorporate feedback from the simulation to direct the transaction generation. This was <u>implemented using Callbacks</u> and became a critical issue from test to test.

#### **Verification Challenge**

- Four interfaces A, B, C and D that need to be exercised.
- Interface <u>A configures</u> whereas <u>B, C and D</u> can either be used to <u>read or write</u>.
- Interfaces <u>B</u>, <u>C</u> and <u>D</u> have to <u>wait until</u> <u>configuration</u> is complete.
- <u>Testing full and empty</u> conditions <u>involve</u> <u>coordination</u> between individual interfaces.
- Reconfiguration on A can happen anytime making B, C and D read or write or vice versa.









### **Stimulus Control Issues (Cont.)**

Example Case 1

- Configure DUT using interface A so interface C and D can both Read & Write
- Configure Interface <u>B to be Write only</u>
- Using interface <u>B</u>, <u>C</u> and <u>D</u> make <u>two iterations of DUT Full to Empty</u>









## **Stimulus Control Issues (Cont.)**

Example Case 2

- Configure DUT through interface A so that interface <u>C</u> is write, <u>D</u> is read, and <u>B</u> is write only
- Reconfigure from A to make B read and D write only









## **Stimulus Control Issues (Cont.)**

From Issue to Solution

Using <u>virtual sequences</u> and <u>nested sequences</u> in UVM we were <u>able to</u> <u>create a randomized approach to configure interface B, C and D</u> as either read, write or both and <u>also do full and empty sequencing</u>.

#### **Example Code Snippet of Issue**

```
program dut_test(..); //beginning of testcase
 //callback for controlling and generating transfers
 class write read extends int master callback;
   virtual task write(master trans transaction = null);
   begin
     if(transaction.int == `INTF A) begin
       //wait for the configuration write to complete
     end else begin
     if(transaction.int == `INTF_B && env.DIR == 1'b0)
       env.chan.put(wr trans); end
   endtask
   virtual task read(master_trans transaction = null);
     env.intf cl chan.put(rd trans); endtask
 endclass
initial begin
   env.build();
   env.intf_a_xactor.append_callback(w0 full clbk);
   env.intf_c_xactor.append_callback(w0 full clbk);
   env.intf_d_xactor.append_callback(w0 full clbk);
   env.run();
end endprogram
                    //end of testcase
```



#### **UVM: Nested and Virtual Sequences**

```
class intf sequence extends virtual sequence base;
 'uvm object utils(intf sequence)
 virtual task body();
  super.body();
   'uvm_do(cfg_init) // Configure using intf A per cfg_init_intf_*
  for(int i = 1;i<=iterations;i++)
    `uvm do on with(intf seg, intf seguencer, {
               recfg == `INT.reconfigure;
               tr1init == `INT.tr1init order:
 endtask
endclass
constraint int config::test specific {
 reconfigure == TRUE;
                                  iterations
                                                 == 1:
 tr1 init order == C FULL;
                                  tr1 final order == C EMPTY;
 tr2 init order == D EMPTY;
                                  tr2 final order == D FULL;
                                  tr3 final order == B_EMPTY;
 tr3_init_order == B_FULL;
 cfg init intf B == RD OR WR; cfg final intf B == RD ONLY;
 cfg_init_intf_C == RD_OR_WR; cfg_final_intf_C == RD_ONLY;
 cfg init intf D == RD ONLY;
                                  cfg final intf D == WR ONLY; }
```







#### **BFM/Monitor Reuse Issues**

From Issue to Solution

Our <u>BFM/Monitor reuse</u> from project to project has always resulted in to a <u>non-structured code</u> with several redundancies. This always <u>involved re-structuring and recode on every project</u> – <u>time wasted!</u>







## Lack of Debug Messaging Support

From Issue to Solution

We have always faced issues because our test environment <u>lacked</u> <u>debug friendly features</u>. <u>Additional code</u> had to be written to <u>make logs</u> <u>meaningful</u>, <u>print more info</u> on the test environment components and configurations.

#### **Legacy TB Debug Support** Case-1: if (condition happens) \$display("Print Details"); `vmm\_note(this.log,\$psprintf("ADDR: %h",address)); Case-2: `define WARNING \$write("WRN: %t %m",\$time); \$display 'define **ERROR** \$write("ERR: %t %m",\$time); \$display 'define **DEBUG** if(debug) \$write("DBG: %t %m",\$time); if(debug) \$display Compile with +DEBUG Case-3: 'ifdef DEBUG \$display("Print Details"); if(\$test\$plusargs("DEBUG")) debug = 1; `vmm\_fatal(this.log,"Data Mismatch"); `WARNING("Issue a Warning"); `ERROR("Issue an Error"); `DEBUG("Debug Debug Debug");









## Lack of Debug Messaging Support (Cont.)

Hierarchy Print & Report Server

Using <u>UVM messaging</u> support we were able to <u>print the TB hierarchy</u> and also <u>customize the report server</u> per our requirements.

```
uvm report server my rpt;
 virtual function void report phase(uvm phase phase);
    int error cnt, fatal cnt, warning cnt;
    my rpt = global reporter.get report server();
    error cnt = my rpt.get severity count(UVM ERROR);
    fatal cnt = my rpt.get severity count(UVM FATAL);
    warning cnt = my rpt.get severity count(UVM WARNING);
    if(error cnt != 0 || fatal cnt != 0) begin
      `uvm info(get type name(), "\n SIM FAILED \n", UVM NONE);
    end else begin
      `uvm_info(get_type_name(), "\n SIM PASSED \n", UVM NONE);
    end
 endfunction: report phase
                                            Customizing report server
task run phase(uvm phase phase);
  printer.knobs.depth = 5;
  `uvm info("ENV TOP",
```









#### Miscellaneous Testbench Issues

Issue at Hand?

We saw several problems in implementation using <u>data channels</u> where <u>additional code</u> had to be written <u>for data handling</u>, and also in cases where various <u>simulation phases had to be handled</u>.

#### **Data Transactions Between Blocks**



Needed to write tasks for data processing from vmm\_channel



TLMs were reusable, available functions, and simplified implementation

#### **Synchronizing Verification Components**



#### **BEFORE**



#### AFTER

 Phasing
 support in UVM supported synchronization

 Reset
 Configure

 Reset
 Configure







## Miscellaneous Testbench Issues (Cont.)

Issue to Solution

We were always <u>manually writing our configuration register models</u> that was <u>time consuming and full of errors</u>. Using SPIRIT (IPXACT) scripts we were able to <u>automate our register model code</u> generation for UVM.



java -jar \$UVM\_RGM\_HOME/builder/ipxact/uvmrgm\_ipxact2sv\_parser.jar -input project\_reg\_map.spirit -ov -ve internal

```
Generating output file with default settings
Input file: project rgm.spirit
Output file: project rgm.sv
Logfile: ipxact2sv.log
Vendor Extensions usage: 'internal vendor extensions
Class Definition postfix: type
Parsing input file...
Normalizing XML file tags...
Generating java model from input file ...
Writing SystemVerilog file...
Number of registers extracted = 988
Number of regFiles extracted = 1
Number of addrMaps extracted = 1
Parser is exiting ...
Output file has been generated at: project rgm.sv
 Execution complete
```

```
class DEVICE IDO type extends uvm rgm sized register #(8);
 typedef struct packed {
   logic [7:0] DEVICE IDO;
 } pkd flds s;
  'uvm object utils(DEVICE IDO type)
endclass : DEVICE IDO type
class project register db extends uvm rgm db;
 rand project address map type project address map;
  'uvm component utils(project register db)
 virtual function void build();
   super.build();
   // Create the address map
   project_address_map = project_address_map_type::type_id::create("project_address_map", this);
   add addr map(project address map);
 endfunction : build
endclass : project register db
```







## **UVM Usage in AMS Simulations**

Before

We faced <u>problems using constrained-random</u> top-level testbench in AMS environment because of <u>compile problems</u> and <u>inconsistent methodologies</u>, and had to rely on non-standard and non-reusable verification.









## **UVM Usage in AMS Simulations (Cont.)**

After

Using <u>UVM infrastructure</u> we were able to <u>re-use the same</u> <u>testbench and test suite</u> that was created for RTL/GATE level also <u>for AMS simulation</u> allowing us to <u>run an end-to-end simulation</u> with packet traffic and protocol.



#### **Technical Contributors**

- Paul Howard
- Ravi Makam
- Jim Skidmore
- Chuck Branch
- Shyam Narayan
- Arun Mohan
- Pradeep Hanumansetty
- Ronnie Koh







#### **Conclusions**

- UVM cleanly addressed our critical issues that were causing significant slowdown and down time due to code re-write
- UVM development goals align with our verification strategy/roadmap
- We did see some conversion effort in going from UVMEA1.0 to UVM1.1 but this effort was minimal
- We found UVM helpful in following ways:
  - Getting started with UVM was easy lots of trainings and guidance
  - We were able to develop complex test environments quickly
  - We found that available VIPs following UVM make integration and usability easier
- We are today using UVM actively in our Digital and Mixed signal verification, and plan to use in Analog verification also







# Q/A







## **OVM to UVM Migration**

or There and Back Again, a Consultant's Tale

**Mark Litterick** 

**Verification Consultant** 

Verilab GmbH, Munich, Germany









#### **Transition from OVM to UVM**



support ongoing OVM

provide OVCs to UVM

enable **new UVM** projects







## **Two Stage Evaluation**

#### Starting point

- clean OVM-2.1.2 OVCs
- no AVM or URM legacy
- different protocols but common style & development team
- => scripts are more effective, but less general purpose!

#### First attempt – early UVM translation

- similar to Mentor Graphics' Verification Academy flow
- goal : check for gotcha's in code, proof of concept, project running in UVM

#### Second attempt – late UVM translation

- optimized flow to do most effort in live OVM project
- automatic and repeatable translation for delivery to UVM







## **Early UVM Translation**









#### **Push Back to OVM**

- Features already in OVM-2.1.2 source:
  - only **objection handling** used for end-of-test
  - improved message macros used instead of methods
  - using back-ported uvm\_reg in OVM environments
- Key improvements that can be done in OVM:
  - virtual interface configuration
  - deprecate sequence utils and OVM sequence library







## **Example: Interface Configuration**

#### Improve virtual interface configuration

- better interface container specifically to help OVM to UVM translation
- container set and get methods similar to uvm\_config\_db











## **Example: Interface Configuration**

#### Improve virtual interface configuration

- better interface container specifically to help OVM to UVM translation
- container set and get methods similar to uvm\_config\_db









### **Example: Deprecated UVM**

- Sequencer & sequence utils deprecated in UVM
  - OVM sequence library not required in UVM or OVM
  - automatic script & manual repair (for reactive slaves)

```
class my_seq extends ovm_sequence #(my_seq_item);
`ovm_sequence_utils(my_seq, my_sequencer)
```

```
class my_sequencer extends ovm_sequencer #(my_seq_item);
`ovm_sequencer_utils(my_sequencer)
`ovm_update_sequence_lib_and_item(my_seq_item)
```

```
class my_env extends ovm_env;
set_config_int("*.my_sequencer", "count", 0);
set_config_string("*.my_sequencer", "default_sequence", "my_seq");
```

```
`ovm_do_on(my_seq, my_env.my_sequencer)
my_seq.start(my_env.my_sequencer);
```







### **Example: Deprecated UVM**

can be done in **OVM or UVM** 

- Sequencer & sequence utils deprecated in UVM
  - OVM sequence library not required in UVM or OVM

do once in OVM

automatic script & manual repair (for reactive slaves)

```
class my seg extends ovm sequence #(my seg item);
                                        `ovm object utils(my seq)
class my_seq extends ovm_sequence #(m
                                       `ovm declare_p_sequencer(my sequencer)
 `ovm_sequence_utils(my_seq, my_sequ
                                      class my sequencer extends ovm sequencer #(my seq item);
class my sequencer extends ovm sequen
                                        'ovm component utils(my sequencer)
 `ovm_sequencer_utils(my_sequencer)
 'ovm update sequence lib and item(my seg item)
                                      class my env extends ovm env;
class my_env extends ovm_env;
 set_config_int("*.my_sequencer", "count", 0);
 set_config_string("*.my_sequencer","default_sequence","my_seq");
                                       fovm_do_on(my seq, my env.my sequencer)
`ovm_do_on(my_seq, my_env.my_sequer
                                      my seq.start(my env.my sequencer);
my_seq.start(my_env.my_sequencer);
```







#### **Late UVM Translation**









#### **Final Translation Process**

- Prepare source OVC for translation once
- Continue development of OVC in live OVM project
- Release OVC versions to UVM when appropriate
- Automatic translate to UVM as part of VIP release

| MVO    | Audit                    | <ul> <li>audit OVM code</li> <li>execute audit script</li> <li>translate OVM to UVM</li> </ul>                                                                                | Automatic?<br>Find Fix<br>☑ ☑ |
|--------|--------------------------|-------------------------------------------------------------------------------------------------------------------------------------------------------------------------------|-------------------------------|
| M<br>N | O-to-U<br>Convert<br>New | <ul> <li>execute ovm2uvm script</li> <li>convert to UVM style</li> <li>execute convert script</li> <li>use new UVM features</li> <li>use improved UVM command args</li> </ul> |                               |







## **UVM to OVM Back-Porting**

- Slim OVM to UVM conversion supports reverse translation
- Valid when UVM transition period expected to endure
- Translate new UVC to OVC for ongoing OVM projects
- UVM limitations (hard to back-port)

still no industry consensus

- avoid **run-time phases** 

use sequence-based phasing

- avoid TLM-2.0 <

localize TLM2 if really required

- Other considerations (easier to back-port)
  - modified objection handling

OK if no run-time phases

updated phase methods

normally OK

- config db changes

command line processor

goal is not to cripple UVM but enable reuse in OVM







#### Conclusion

#### Goal is move to UVM

- transition period could endure for some time
- considerable **OVM legacy** and many ongoing projects
- new UVM projects need OVC libraries
- ongoing OVM projects may need new UVCs
- Presented an overview of migration process
  - prepare OVM for easier translation
  - slim automatic translation process
  - translation process is reversible



- applied to multiple projects @ different clients









## IBM Recommendations for OVM → UVM Migration

Wes Queen
Verification Manager, IBM









## Migrating from OVM to UVM

 Motivation: UVM API beginning to diverge from OVM as new features are added to UVM

Challenge: large code base in multiple projects

General approach: Convert code base using scripts







## **OVM Development Started in 2009**

Open source ran on multiple simulators

Methodology met verification team requirements for reuse

Initial development followed user guide

OVM\_RGM register package adopted

OVM use rapidly spread to multiple groups worldwide







## **Block Diagram HSS OVM Environment**



## Prepare Register Package

#### Install updated parser from Cadence

- Allows for OVM\_RGM, UVM\_RGM, and UVM\_REG generation
- OVM\_RGM and UVM\_RGM usage is identical
- UVM\_REG targets the Accellera package

#### Generate OVM and UVM register models

 Internal script used to generate file names and headers for new parser to match previous parser version

#### Install OVM\_RGM 2.5 to align with new UVM parser

Rerun OVM environment to be sure results match before proceeding







## **Run UVM Conversion Script**

- Download conversion guide from UVMWorld
  - <a href="http://www.uvmworld.org/contributions-">http://www.uvmworld.org/contributions-</a> details.php?id=107&keywords=Appnote: Migrating from OVM to UVM-1.0
  - Posted by John Rose on May 9, 2011 if you are navigating to find it
- Install UVM conversion script
  - Available within latest UVM kits on Accellera.org or within Cadence installation'
- Move any directories out of code tree that should not be converted
  - OVM\_RGM directory and/or legacy code
- Run conversion script







## Remove Deprecated Code and Compile DPI

- Change any deprecated code which wouldn't compile (OVM 1.0, typically)
  - Add\_seq\_cons\_if artifact code from OVM 1.0 that needs to be removed
  - Review conversion guide for other deprecated code

- Compile uvm\_dpi.cc (libdpi.so) in 32 bit or 64 bit
  - New requirement for UVM







## **Golden Test and Further Updates**

#### Run simulations to test conversion

Include +UVM\_USE\_OVM\_RUN\_SEMANTIC in simulation command

#### On-going clean-up

Remove other deprecated OVM calls (mostly super.build or straight build calls)

#### Adopt new UVM features

- Phases, sequences, UVM\_REG, etc.







#### Results

#### Conversion process has been used successfully in multiple groups

- Current 4 projects have converted over the last year.

#### Effort was relatively low

- Lowest risk is to do the conversion between projects
- Effort to convert took one day by single resource. 100K lines of code on single project.

#### Motivation to be on UVM is real

- New UVM features are valuable UVM\_REG, phasing, sequences, etc.
- New UVM features can impact backward compatibility







## FPGA chip verification using UVM

Ravi Ram

Principal Verification Engineer

**Altera Corp** 



**Charles Zhang** 

**Verification Architect** 

**Paradigm Works** 









#### **Outline**

- Overview
  - Verilog based verification environment
  - Why UVM?
  - New UVM based verification environment
  - FPGA chip verification flow
- Some of the challenges and solutions
  - Generic programmable logic
  - Legacy code integration.
  - Programmable core & IO connection
  - VIP integration(external and internal)







## Verilog based Verification Env

- Traditional Verilog based verification environment
- Multiple test benches for multiple modes of operation
  - PP, PS, SPI, USERMODE, etc.
- Recompilation for each test
- No object oriented programming (reuse = copy and change)
- Maintainability and scalability are poor (large number of tests, etc.)
- Easier for designer to understand and modify







# Why UVM?

- Supported and released by Accellera
- Supported by all major EDA vendors
- Object orient programming
- Reusability (vertical and horizontal)
- Well defined base class library
- Industry standard makes integration of third party or home grown VIP easier
- Good online documentation + UVM forums etc
- Little bit harder for designer to understand







#### **UVM** based Verification Env Overview









#### **UVM-based verification Env overview**

- Architected from scratch
- One environment supports multiple operating mode
  - PP, PS, SPI, USERMODE, etc.
- Significantly reduced number of tests by inheritance, configuration setting, etc
  - The current UVM based tests is about 1/3 of the tests of Module based ENV
- Simulation performance improved by compile once and run multiple tests
- Improved compile, run and regression flow
  - With UVM, cmd line processor is built-in and free







#### **FPGA Verification Flow**

- Configuration (Programming the FPGA).
  - Support multiple programming interfaces
  - Data compression and encryption
  - Front door and back door loading configuration
  - Verification goal: make sure the programmed image matches the expected image
- User Mode (Running programmed user logic)
  - Tests include testing all core logic blocks and all the IO systems
  - Considerable effort is on creating configurable verification environment
  - Verification goal: verify all the core blocks and I/O systems to be functioning and connected properly







## Generic programmable logic

- Programmable nature of FPGA calls for programmable verification environment
- Core logic interface UVC is a highly programmable verification component.
  - Allows user to decide on which pins to drive using UVM configuration
  - The monitor extended by user to implement any checking mechanism using UVM factory override.
  - Test based on sequences and transactions without worry about pin connection and toggling.
  - Compile once and run all tests.
- Used by the software group to verify real customer design.









## Legacy code integration



- Still need Verilog based verification environment to coexist with UVM verification environment
- Interface file used as bridge between UVM verification environment and module based verification environment
- Interfaces bound to its physical interface signals
- Virtual interface in UVC set by getting the instance from resource database
- Assertions implemented in interface binds to module or physical interface signals





SYSTEMS INITIATIVE

## Programmable core & IO connection

- FPGA core is programmable
- All hard IP is configurable
- Lots of different interfaces and VIPs
- Register access from reg UVC to configure FPGA
  - Thousands of configurations in FPGA. UVM Reg model is already > 20G for handling 30 to 40% of the FPGA configurations. So this is not scalable and not practical to use
- Hundreds of configurable registers which UVM reg based testing cannot handle
  - Use home grown resource allocator plus configuration settings
- Register access from reg UVC to configure FPGA
- Seamless integration of resource allocator(internal tool) with internal developed tools for unidirectional and bidirectional connections







# **VIP** integration

- Lots of VIPs to address hard IP in FPGA(1G/10G..., PCle plus other serial protocols, Altera Avalon VIP, different memory VIP for different memory protocols)
- Flexibility to configure and select VIPs in UVM test
- Use constraints to select the connections and VIPs
- Use on the fly point-to-point connections to connect VIP to the fabric
  - Turn off unused VIPs
- Same environment for integrating different vendor VIPs
- Environment setup for proliferation products for same FPGA family
- VIP interface easily portable to future FPGA families







**Avalon VIP Integration** 



- Integrate Avalon BFM in UVM environment
- Use of the existing bfm with a wrapper on top to make it a UVC
- VIP developed internally in Altera and is made available for use by all groups
- The configuration object generated for each instance of the VIP with a unique hdl Path which has a reference of the interface.
- The user provides the parameters for the VIP and the hdl Path in his test-bench hierarchy

## **Summary**

- Altera's first verification project adopting UVM
- Addressed critical challenges
  - Programmable user logic and io
  - Explosive configuration spaces, etc.
- Adopted pragmatic view of the methodology
  - Re-architected the whole environment using UVM
  - Reused and integrated both internal and external VIPs
- UVM provides ideal way to create configurable, reusable verification components and environment







**Q & A** 

# Thank You!

**Contributor: Manish Mahajan** 





