



# UVM Testbench Automation for AMS Designs

Jonathan David – Innophase Inc.

Henry Chang – Designer's Guide Consulting, Inc.



# Outline

- The Analog TB generation problem
- Designing a standard TB
- A generic UVM agent design
- Automating TB construction
- Results
- Resources



# Why Long delays until first test of mixed-signal designs?

Per design:

- Select test approach
- Build components
  - stimulate inputs
  - observe outputs
- Assemble
  - DUT
  - test components
- Manage
  - Error reports
  - Test checking

Result:

- Wild variations
  - Per design
  - Per verifier or team
  - Per company
- More TB time => Less Testing
  - Test quality/quantity suffers

Importance:

- Early System level
- Control Sequencing

• A Faster way?





# Designing a Standard Test-bench

Select a standard framework

Design test-bench approach adaptable to any design.



# UVM: the Incumbent Framework (for digital design verification)

## Pro

- Available training material
- digital team (probably) uses
  - Possible help source
- Built in:
  - Messaging (Fatal, Err, Warn, Info)
  - Phasing
  - Sequence management
  - Randomization
  - Test parallelization

## Con

- Complicated
- Available training focused on digital bus transactions.
- Debugging also complicated

Alternatives?

# Models-in-Minutes for the System or Chip-Level Test-bench?

- MiM generates models
- MiM generates block-level testbenches
  - Model validation
  - SystemVerilog RNM / AMS model vs Schematic
- TB easy to use
  - small designs
  - model testing and validation
- Limited above block-level IP top, chip-level
- Poor fit where:
  - Many sub-function types
  - 100+ control signals
  - Multiple test scenarios
- Not selected as full design framework



# Approach: Manage Complexity

## Standardize, Abstract, Automate

- Standardize:
  - Reusable standard agents
  - Stimulus generation patterns
  - Observation collection patterns
- Abstract:
  - Single file to edit
  - “Simple” data structure
  - Available Editor Support
- Automate:
  - Python
  - Jinja template rendering
- Agents:
  - Autb\_generic agent
  - Autb\_csr\_agent
- Templated TB top with
  - Power, Bias, (other) Analog
  - Register, (other) Digital
- Python dictionary
  - NextedText format+ Python code and Jinja2 for code templates
- Python scripts
  - Jinja Templates



# Standard AUTB Construction





# A Generic UVM Agent Design

Since we select UVM can we abstract away some complexity?



# AUTB Generic IF

Interface -> sequence\_item -> driver, monitor, agent

## Flexibility: Key Requirement

```
1 // interface for the autb_generic agent.  
2 interface autb_generic_if();  
3     string settings[string], observations[string];  
4     event sample_trigger;  
5     string sequence_name, design_state_name;  
6 endinterface
```

- Associative arrays
- String functions for get put
  - String <->logic variable
  - String <-> real variables
- String index
  - easy access to values by name.
- Modify per case:
  - Sequences
  - Adapter module

# AUTB Generic Sequence Item (base)

**Contains: same associative arrays  
adds housekeeping variables**

```
1 // sends control values to tb in name, value pairs in a hash,  
2 // with values converted to string for sending  
3 > class autb_generic_seq_item extends uvm_sequence_item; ...  
4   `uvm_object_utils(autb_generic_seq_item) ...  
5   string settings[string];  
6   string observations[string];  
7   string design_state_name; ...  
8   extern function new(string name = "autb_generic_seq_item");  
9   extern function void do_copy(uvm_object rhs);  
10  extern function bit  
11    || do_compare(uvm_object rhs, uvm_comparer comparer);  
12  extern function string convert2string();  
13  extern function void do_print(uvm_printer printer);  
14  extern function void do_record(uvm_recorder recorder);  
15 endclass:autb_generic_seq_item
```

**Base: simple string comparison  
(similar for convert2string, copy etc)**

```
42 function bit autb_generic_seq_item::do_compare(  
43   uvm_object rhs, uvm_comparer comparer);  
44   autb_generic_seq_item rhs_;  
45   string i;  
46   bit eq = 1;  
47   if(!$cast(rhs_, rhs)) begin  
48     `uvm_error("do_compare", "cast of rhs object failed")  
49     return 0;  
50   end  
51   eq &= super.do_compare(rhs, comparer);  
52   foreach (rhs_.settings[i])  
53     eq &= settings[i] == rhs_.settings[i];  
54   foreach (rhs_.observations[i])  
55     eq &= observations[i] == rhs_.observations[i];  
56   eq &= design_state_name == rhs_.design_state_name;  
57   return eq;  
58 endfunction:do_compare
```

# AUTB Generic Driver

## Passes sequence item to the interface

```
42 task autb_generic_driver::run_phase(uvm_phase phase);
43     autb_generic_seq_item req;
44     autb_generic_seq_item rsp;
45     int psel_index;
46     string setting;
47     forever
48     begin
49         seq_item_port.get_next_item(req);
50         `uvm_info($sformatf("%s_DRIVER",
51             this.get_full_name().toupper()),
52             $sformatf("Starting sequence %s",
53             req.get_name()),UVM_LOW)
54         foreach (req.settings[setting])
55             m_vif.settings[setting] = req.settings[setting];
56         m_vif.sequence_name = req.get_name();
57         seq_item_port.item_done();
58     end
59 endtask: run_phase
```

## Monitor is similar, and collects if changes, and puts them in an item.

- Here we chose to copy each element in the array to the if
- This allows unchanged elements in the interface to persist
- Separate sequences to manage subsets of variables simply.

# AUTB CSR Interface and Agent

- Similar to generic agent
- Associative arrays:
  - *int* indexed by *string*.
- Because Many registers
  - conversions (*int*<->*string*) might affect performance
- Adds delay function
  - mimics bus transaction delays.

```
1 //interface for the autb_csr agent
2 interface autb_csr_if();
3     int readonly[string], writable[string];
4     string sequence_name, design_state_name;
5     int delay_us, delay_ns;
6     > task delay(input int us_delay=1, ns_delay=0); ...
18 endinterface
```



# Automating Testbench Creation



# Automation Work Flow

Colored boxes show the inputs needed



# Example Ninja Template

Data structure(dictionary) passed to template render engine

Included: dictionaries, lists, variables with their values  
variety of functions for output interpolation

```
1 //SystemVerilog HDL for "{{library_name}}", "{{name}}" "svlog"
2 module {{ name }} ( {% set comma=joiner(', ') %}{% for port, portitem in ports.items() %}{{ comma() }} 
3 |   {{portitem['dir']}} interconnect {{if portitem['msb'] %}}[{{portitem['msb']}}]:{{portitem['lsb']}}]
4 {%- else %}     {%endif%}   {{port}} {% endfor %})
5 );
6 import uvm_pkg::uvm_config_db;
7 autb_generic_if pwr_if(); // interface contains an associative array of string with string index settings
8 initial begin
9     uvm_config_db#(virtual autb_generic_if)::set(null, "uvm_test_top", "power_vif", pwr_if);
10 end
11 {{name}}_adrive power_adrive({% set comma=joiner(', ') %}{% for port, portitem in ports.items() %}{{ comma() }} 
12 |   .{{port}}({{port}}){%endfor%});
13 bit sample_trigger;
```

The code is annotated with several callout boxes:

- A blue box labeled "Simple var" points to the variable {{name}}.
- A blue box labeled "Function defined" points to the function definition of joiner().
- A blue box labeled "Function used" points to the usage of joiner() in the code.
- A blue box labeled "loop" points to the for loop structure.
- A blue box labeled "test" points to the if condition.
- A blue box labeled "Dict var" points to the dictionary variable portitem.

# Demo Design Data Structure

NestedText format – some repeated sections are folded.

```
1 library: RFDV_scratch          23   clk_en:           115
2 name: simple_uvm_testcase      24     direction: input 116
3 ports:                         25     lsb: 0            117
4   AVDD1P8:                     26     msb: 3            118
5     direction: input           27     porttype: register 119 >
6     porttype: power           28     reg_info: ...
7     supply_nom: 1.8           35     clk_out_0: ...
8     supply_type: volts        39     clk_out_1: ...
9   AVSS:                        43     clk_out_2: ...
10    direction: input          47     clk_out_3:           123 >
11    porttype: power           48       direction: output 127 >
12    supply_type: ground       49       porttype: digital 131 tests:
13   bg_enable:                 50       digital_type: clock 132   sanity_test:
14     direction: input          51     dll_lock_status: ... 133   sequences:
15     porttype: register        61     mclk_in: ...
16     reg_info:                 67     mode_sel_0: ...
17       name: bg_enable         79     mode_sel_1: ...
18       blockid: top            91     mode_sel_2: ...
19       width: 1                103    mode_sel_3: ...
20       default: 'b0
21       on_value: 'b1
22       description: bias_blk enable
23
24
25
26
27
28 >
29
30
31
32
33
34
35 >
36
37
38
39 >
40
41
42
43 >
44
45
46
47
48
49
50
51 >
52
53
54
55
56
57
58
59
60
61 >
62
63
64
65
66
67 >
68
69
70
71
72
73
74
75
76
77
78
79 >
80
81
82
83
84
85
86
87
88
89
90
91 >
92
93
94
95
96
97
98
99
100
101
102
103 >
```

Simple Data modification here  
drives UVM file generation.



# Example Design Simulation

(Folded) DLL operation at

startup

locking

locked



# Results

Effort level of single person – seems to be productive use of time.

- First iteration
  - Initial development of flow, agents, templates and code
  - 3 weeks from dut netlist to first test working.
- Second iteration
  - Simplified and polished the flow
  - Another 3 weeks from DUT netlist to full startup test working.
- Third iteration (and additional)
  - Simple flow reuse with minor tweaks
  - 1 day to build a simple example design and testbench and test
  - ( testcase for simulator issue)



# Future Plans

- Package code for command line use
- Add results checking
  - Enabling randomization.
- Provide generalized support for dynamic analog signals.

# Resources

- Agent and template Code examples at  
<https://github.com/jbdavid-inno/analog-uvm-tb> .
- Example DUT with generated files are published at  
[https://github.com/jbdavid-inno/simple\\_uvm\\_testcase](https://github.com/jbdavid-inno/simple_uvm_testcase) .
- MiM ([designers-guide.com](http://designers-guide.com))
- Visual Studio Code(Microsoft)
- [Jinja.palletsprojects.com](https://jinja.palletsprojects.com)
- Verification Academy
- Google/Bing search
- NestedText.org
- Python.org
- IEEE.org



# Questions?

Jonathan's Email: [jdavid@innophaseinc.com](mailto:jdavid@innophaseinc.com)

Henry's Email: [henry@designers-guide.com](mailto:henry@designers-guide.com)

