ravenoc

RaveNoC Testbench

On this folder we have all the tests that are developed for the RaveNoC, so far at this moment we have 8 tests which cover different aspects of the NoC behavior. Inside the folder common_noc we have 2 base classes that are used along all the tests to create packets and the testbench itself. All the tests are implemented using cocotb framework, thus in python along with Verilator as the simulation engine.

├── common_noc
│   ├── __init__.py
│   ├── constants.py
│   ├── ravenoc_pkt.py
│   └── testbench.py
├── test_all_buffers.py
├── test_irqs.py
├── test_max_data.py
├── test_noc_csr.py
├── test_ravenoc_basic.py
├── test_throughput.py
├── test_virt_chn_qos.py
└── test_wrong_ops.py

RaveNoC Wrapper

One of the issues during the development stage is that Verilator 4.106 (till now it is the only one that does not break cocotb VPI) cannot handle easy structs/arrays in the top level, thus RaveNoC is encapsulated into a wrapper which exports 2 AXI-Slave I/Fs that are used by the testbench. Along with the AXIs, we have some other signals that are driven by the tests to setup the different set of scenarios.

RaveNoC Wrapper

In the image above, it is possible to view all the connections of ravenoc_wrapper with the tests. Here is the list of signals and for what they are used:

The axi_sel_* signals should never be equal if both act_* are set. When cdc is not required, the user should set to 0 the top array parameter AXI_CDC_REQ for the respective router and not use bypass_cdc input. This is the recommended procedure because it forces the instantiation of async_gp_fifo. During standard IP usage, ravenoc_wrapper must not be part of filelist once it only exists because of the Verilator 4.106 limitations.

Hardware flavors

As RaveNoC has several system verilog macros that changes the hardware configuration, some set of values are chosen and called as flavors. Each flavor represents a different hardware where all the tests described in the sections below will be executed. The list of flavors and the values are described on the following table:

Parameters/Flavors Vanilla Coffee Liquorice
FLIT_DATA_WIDTH 32 64 64
ROUTING_ALG XYAlg YXAlg XYAlg
NOC_CFG_SZ_ROWS 2 4 8
NOC_CFG_SZ_COLS 2 4 8
FLIT_BUFF 1 2 4
MAX_SZ_PKT 256 256 256
N_VIRT_CHN 2 3 4
H_PRIORITY ZeroHighPrior ZeroLowPrior ZeroHighPrior

Test list

The following test list tries to cover different aspects of operation of the NoC, from the basic test #1 till a long test like the #5 which writes/reads to all buffers of all routers. Inside each python script there is a better description of the functional behavior of each individual test.

Test ID Test name Short description
1 test_ravenoc_basic Basic test that sends a packet over the NoC and checks it
2 test_wrong_ops Checks if the AXI-S/NoC is capable of throwing an errors when illegal operations are executed
3 test_max_data Test if the NoC is capable to transfer a pkt with the max. size
4 test_throughput Test to compute the max throughput of the NoC
5 test_all_buffers Test if all buffers of all routers are able to transfer flits
6 test_virt_chn_qos Test the QoS of VCs in the NoC
7 test_noc_csr Check all WR/RD CSRs inside the NoC
8 test_irqs Test that checks the IRQs modes inside the NoC

Test corners

All the tests are implemented using the TestFactory feature from cocotb. This class will generate a set of tests based on the different permutation of all the options passed. It is used as well an AXI master simulation model that implements the AXI4 protocol driving both AXI I/F (noc_in, noc_out), these objects are created in the testbench class explained late. In the RaveNoC the set of test options implemented are:

  1. Clock configuration - AXI clk slower/equal/higher than the NoC;
  2. With the AXI Master VIP inserting idle cycles on M->S flow channels (AW-write addr,AR-read addr,W-write data) during the transfers;
  3. With the AXI Master VIP inserting back pressure on S->M flow channels (B-write response,R-read data);

With the options above, for instance, let’s consider the test id #1:

... Lines 47-51
if cocotb.SIM_NAME:
    factory = TestFactory(run_test)
    factory.add_option("config_clk", ["AXI_slwT_NoC", "NoC_slwT_AXI", "NoC_equal_AXI"])
    factory.add_option("idle_inserter", [None, cycle_pause])
    factory.add_option("backpressure_inserter", [None, cycle_pause])
    factory.generate_tests()

This will generate the following set of tests:

1.  AXI_slwT_NoC_no_idle_no_backpressure
2.  AXI_slwT_NoC_no_idle_w_backpressure
3.  AXI_slwT_NoC_w_idle_no_backpressure
4.  AXI_slwT_NoC_w_idle_w_backpressure
5.  NoC_equal_AXI_no_idle_no_backpressure
6.  NoC_equal_AXI_no_idle_w_backpressure
7.  NoC_equal_AXI_w_idle_no_backpressure
8.  NoC_equal_AXI_w_idle_w_backpressure
9.  NoC_slwT_AXI_no_idle_no_backpressure
10. NoC_slwT_AXI_no_idle_w_backpressure
11. NoC_slwT_AXI_w_idle_no_backpressure
12. NoC_slwT_AXI_w_idle_w_backpressure

Although we have three corners to explore on every test, for a functional point of view, does not make sense to run some corners depending of what is intended to explore. For instance in the throughput test #4, having the back pressure / idle cycles corner it is unnecessary, once the intention is to explore the maximum bandwidth of the NoC. Due to this aspect, the table below shows for every test which corners are being exercised or not.

Test ID Test name AXI sl. NoC NoC sl. AXI NoC equal AXI No idle cycles W/ idle cycles No backpressure W/ backpressure
1 test_ravenoc_basic Y Y Y Y Y Y Y
2 test_wrong_ops Y Y Y Y Y Y Y
3 test_max_data Y Y Y Y Y Y Y
4 test_throughput Y Y Y - - - -
5 test_all_buffers Y Y Y Y Y Y Y
6 test_virt_chn_qos Y Y Y - - - -
7 test_noc_csr Y Y Y Y Y Y Y
8 test_irqs Y Y Y Y Y Y Y

All the corners mentioned will be executed for all the flavors listed previously. So in summary, we have a total of 144 tests running in the full regression.

> 6 tests running all the 7 corners = 42
> 2 tests running 3 corners = 6
= 48 test per flavor
------------------
> 3 flavors (vanilla, coffee, liquorice) - 144 tests

Logs

Each test will run the same sequence described in run_test function. For every corner there is a specific log file created (every time a new Tb obj is created) although the waveforms file is shared between all corners of each test, i.e all the set of tests above will dump the waves to the same dump.fst file. Because of this detail, the logs will have the relative timestamp when they run, see the one below.

140.00ns INFO     ..I_slwT_NoC_no_idle_w_backpressure        testbench.py:35   in __init__                        ------------[LOG - 01_Apr_2021_22h_53m_25s_1617314005.172996]------------
140.00ns INFO     ..I_slwT_NoC_no_idle_w_backpressure        testbench.py:36   in __init__                        SEED: 1617314004
140.00ns INFO     ..I_slwT_NoC_no_idle_w_backpressure        testbench.py:37   in __init__                        Log file: sim_AXI_slwT_NoC_no_idle_w_backpressure
...

As it is possible to check, the first statement logged there happens at 140ns due to the fact this is the corner (2.) for the test running. Log files will be located inside the run_dir folder created during the simulation where a symbolic link will point to the latest one of each corner (this symlink starts with latest_sim…).

In the log files there is also the random seed used to generate the random values and the NoC configuration with the selected parameters.

Base classes

To compose all the tests of the Noc, the class Tb and RaveNoC_pkt are implemented and use along all the set. In the Tb class, a constructor initializes the testbench structure by connecting the dut pins to the 2x AXI Master VIPs, set to zero the axi muxes and creates the log files. Inside this class it is available 2 methods to send send/receive packets to/from the NoC (write_pkt/read_pkt), and 2 other methods to do single read/write to the AXI interfaces. It also implemented methods for checking the packet integrity (check_pkt) by comparing a RaveNoC packet payload with an argument input data.

As the Tb class has all the connections with the dut, the methods for setting up the clocks and resetting the NoC+AXI I/F are also there. And as the NoC is capable of generating IRQs, two other methods wait for IRQs to happens, one for any IRQ value (wait_irq) and one for a specific value (wait_irq_x).

Another base class is the Ravenoc_pkt, which is used to create packets that are send to the NoC. If not specified, every time a user creates a new object of this class, random values will be picked for the source, destination router, virtual channel to use and payload message. The class is responsible for assembling the header packet in the same encoding as explained in the uArch readme.

Extending the test set

To extend the test set, the list of steps that should be taken are the following ones:

  1. Create a new file in the tb folder with the prefix *test_** (should be on this format because this is the way pytest parse the tests);
  2. Add the template below on the bottom part of your test changing the proper labels. This will call cocotb-test to run our tests with pytest using all the available flavors;
  3. Implement the test considering the add_options that are required, for the TestFactory;
  4. Run the test with the following Tox command that will call pytest in the end;
    $ tox -- -k "test_MYTEST[vanilla]" -rP
    

    When running tox, the argument -- will forward any following options to the pytest command, -rP will output the cocotb command even if you pass during the test execution and [flavor] will execute the test considering only this hardware set of options.

Template:

@pytest.mark.parametrize("flavor",noc_const.regression_setup)
def test_MYTEST(flavor):
    """
    Short desc. of my test

    Test ID: X

    Description:
    Long desc. of the test.
    """
    module = os.path.splitext(os.path.basename(__file__))[0]
    SIM_BUILD = os.path.join(noc_const.TESTS_DIR,
            f"../../run_dir/sim_build_{noc_const.SIMULATOR}_{module}_{flavor}")
    noc_const.EXTRA_ENV['SIM_BUILD'] = SIM_BUILD
    noc_const.EXTRA_ENV['FLAVOR'] = flavor

    extra_args_sim = noc_const._get_cfg_args(flavor)

    run(
        python_search=[noc_const.TESTS_DIR],
        includes=noc_const.INC_DIR,
        verilog_sources=noc_const.VERILOG_SOURCES,
        toplevel=noc_const.TOPLEVEL,
        module=module,
        sim_build=SIM_BUILD,
        extra_env=noc_const.EXTRA_ENV,
        extra_args=extra_args_sim
    )

Adding a new hw flavor

New hardware flavors can be added easily only editing the tb/common_noc/constants.py file. The steps to add a new hardware where all the tests will run through:

  1. Add the flavor name on the variable regression_setup set at constants.py;
  2. Do a content copy from EXTRA_ARGS creating at the same time a variable to hold the extra arguments at constants.py [Lines 69-71];
  3. Create a new dictionary [Line 72] that represents this hardware configuration and populate with the values you want to change as well the ones you want to keep it i.e without changing from the default at src/include/ravenoc_define.svh;
  4. It is important to keep in mind that the line 139, where there is the for loop that appends -D in front of the keys/values, should convert all the options defined till there as part of system verilog macros. Thus impacts the hardware behavior, with that in mind, please be aware to define the keys max_nodes, x_w,… after this loop;
  5. Run the basic NoC test with the following command;
    $ tox -- -k "test_ravenoc_basic[my_new_flavor]" -rP
    

    Running the full regression

    To run the full regression, considering all the three flavors, execute the following commands:

    $ export FULL_REGRESSION=1 #Set this env variable to enable liquorice run
    $ tox -- -rP
    

    Please be aware that it might take a lot of time to run the liquorice flavor, because of the size of the generated hardware.

Modules & Versions

All the tests are written using the following versions of the dependencies (python modules). It might work or not with newer versions but in the worst case, try with the ones below used during the development.

Dependencies Version
cocotb-bus 0.1.1
cocotbext-axi 0.1.10
cocotb-test 0.2.0
cocotb 1.5.1
verilator 4.106