

#### Single Source Register Sequencing Toolkit and Methodology for FW and Verification Co-Development

Josh Loo, Anthony Cabrera, Kiel Boyle, Scott R. Nelson Non-Volatile Memory Solutions Group Intel Corporation







## Agenda







#### Cross-Domain Register Access Handshaking



#### Results











The Problem:

- Modern SoC productization requires FW for production releases and RAL sequences for pre-silicon verification
- FW and verification teams commonly access registers using different techniques and source languages (ex. C++ vs. SystemVerilog/UVM)
- This leads to effort duplication and discrepancies between sequences make it difficult to replicate exact behavior on different test platforms





#### The Solution:

- Single Source Register Sequencing (SSRS)
  - A toolkit that includes a pair of support libraries with associated execution kernels, one in C++ and the other in SystemVerilog
  - Enables FW C++ register sequences to be run against RTL simulations without needing a CPU simulation model
  - C++ register sequences are executed on an x86 thread that runs parallel to the simulator





#### The Benefits:

- Earlier FW/HW integration testing against standalone IP modules
- Faster simulations due to less RTL
- Verbatim re-use of FW
- Pre-silicon verification scoped to actual FW use cases
- Accurate behavior replication on all test platforms







| class ssrs_test extends uvm_test;                                         |  |  |  |  |  |
|---------------------------------------------------------------------------|--|--|--|--|--|
| • • •                                                                     |  |  |  |  |  |
| fork                                                                      |  |  |  |  |  |
| <pre>m_env.m_uvc.m_svek.run_c_sequence("path/to/cpp_collateral.so",</pre> |  |  |  |  |  |
| "function1");                                                             |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| <pre>m uvm sequence.start(m sequencer);</pre>                             |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| m env.m uvc.m svek.run c sequence("path/to/cpp collateral.so",            |  |  |  |  |  |
| "function2");                                                             |  |  |  |  |  |
| join                                                                      |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| endclass                                                                  |  |  |  |  |  |

















• C++ Sequences – orchestrate driver calls to perform a series of register accesses







• Drivers – stateless FW modules that implement accessors to registers







 Register Headers – contain data type definitions that describe the address map and access policies of registers







 Register Monitoring Classes – class objects for registers are allocated in memory and operator overloading is used to monitor register accesses by drivers











| class ssrs_test extends uvm_test;                                         |  |  |  |  |  |
|---------------------------------------------------------------------------|--|--|--|--|--|
| •••                                                                       |  |  |  |  |  |
| fork                                                                      |  |  |  |  |  |
| <pre>m_env.m_uvc.m_svek.run_c_sequence("path/to/cpp_collateral.so",</pre> |  |  |  |  |  |
| "function1");                                                             |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| <pre>m_uvm_sequence.start(m_sequencer);</pre>                             |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| m env.m uvc.m svek.run c sequence("path/to/cpp collateral.so",            |  |  |  |  |  |
| "function2");                                                             |  |  |  |  |  |
| join                                                                      |  |  |  |  |  |
|                                                                           |  |  |  |  |  |
| endclass                                                                  |  |  |  |  |  |
|                                                                           |  |  |  |  |  |





















Initiating a Register Access from C++

- Driver accesses to reg objects in memory are observed by reg monitoring classes
- Overloading read and write operators triggers the CEK to convert the memory address of a register being accessed to its equivalent address from the RAL
- Register Access Message (C++  $\rightarrow$  SV)
  - RAL address of register being accessed
  - Access type (e.g. read or write)
  - Value to update DUT register with (writes only)











- Completion Message (SV  $\rightarrow$  C++)
  - Value to return to C++:
    - Read accesses return the value read from the DUT











**Bidirectional Communication Channel** 

- Encapsulates two message queues: C++  $\rightarrow$  SV and SV  $\rightarrow$  C++
- Implemented in C++ and allocated in memory by CEK
- Message queues hold entries of type message\_t, a user-defined struct that is identically defined in both language domains

| Message Queue<br>Accessor Function | Action                             | Blocking? | Accessible via DPI<br>Import? |
|------------------------------------|------------------------------------|-----------|-------------------------------|
| put()                              | En-queues a<br>message             | No        | Yes                           |
| get()                              | De-queues a<br>message             | Yes       | Yes                           |
| tryget()                           | Attempts to de-<br>queue a message | Νο        | Yes                           |

















Portability

• SSRS is easily portable to existing UVM testbenches due to its RAL interface

Multi-Platform Code Re-use

- Scope of pre-silicon verification was focused on actual FW use cases as defined by the common sequences
- Sharing common FW drivers and sequences reduced code redundancy and ensured behavior replicability across different test platforms





Flushing Out FW/HW Interaction Bugs Earlier

- Example:
  - FW sequence appeared to be passing in post-silicon testing
  - When re-running the FW sequence in an RTL simulation, we noticed IO glitches that could trigger undefined behavior on external components

**Cross-Team Collaboration** 

• Verification experts should be able to easily notice errors in FW drivers or sequences since they are experts in SoC design specifications





Continuous Integration

- Changes to shared FW code triggered rigorous reviews across multiple disciplines resulting in higher quality code
- Modifications were automatically regressed against different test platforms
- Regressing FW in simulation provided prompt quality feedback to FW developers without requiring detailed knowledge about UVM environments
- SSRS provided greater debug visibility into FW interactions with the RTL by reproducing behavior in simulation





### **Extended Verification Features**





# **Extended Verification Features**

**Constrained Random Parameter Generation** 

- Existing constraint constructs within a UVM test environment can be used to generate constrained-random values for C++ Sequences
- Effectively reduces manual effort for test creation while increasing functional coverage of the DUT and FW source code
- Currently exploring how to share SystemVerilog generated parameters with other test platforms





# **Extended Verification Features**

Inter-Environment Communication

- Existing message passing mechanisms can also be used to trigger custom SystemVerilog or UVM functionality from C++
- Example use-cases:
  - Converting assertion and print statements in C++ to UVM report macros
  - Generating backdoor register accesses to accelerate run time
  - Generating burst register accesses in simulation





# Conclusion

Single Source Register Sequencing (SSRS)

- Enables FW execution against RTL simulations without a CPU model
- Portable to lower-level verification environments due to RAL interface
- Provides developers with greater debug visibility into FW/RTL interactions
- Verbatim reuse of FW enables trivial behavior replication in RTL simulation
- Allows for extension to trigger custom functionality between C++ and SystemVerilog





### **Questions?**

