Using non-discoverable devices connected to I/O virtualization block

Overview

The reference design platforms that support a IO Virtualization block as part of the compute subsystem allow connecting non-discoverable devices (non-PCIe) to it. The I/O virtualization block includes SMMUv3 to translate address and provide device isolation security, GIC-ITS to support MSI interrupts and NI-700 inter-connect to route transactions in and out of x16, x8, x4_1, and x4_0 ports.

The non-discoverable devices that are connected to the Io Virtualization block include - two PL011 UART, two PL330 DMA controllers and six regions of SRAM memory. The reference design software stack includes support to configure and use these devices and memory regions.

This document describes how to build the Neoverse reference design platform software stack and use it to test non-PCI devices that are connected on I/O virtualization blocks. Busybox Boot is used on the Neoverse RD FVP and tests are run from the command line to validate the devices.

NOTE: These tests are supported only on reference design platforms that designate one of the IO virtualization block for connecting non-discoverable devices.

Download the platform software

Skip this section if the required sources have been downloaded.

To obtain the required sources for the platform, follow the steps listed on the Setup Workspace page. Ensure that the platform software is downloaded before proceeding with the steps listed below. Also, note the host machine requirements listed on that page which is essential to build and execute the platform software stack.

Build the platform software

Refer to the Busybox Boot page to build the reference design platform software stack and boot into busybox on the Neoverse RD FVP.

Running tests for non-PCI devices on busybox

To begin the tests with the non-discoverable devices connected to the IO Virtualization block, boot to busybox using the command mentioned below (refer to Busybox Boot guide for details on the parameters).

./boot.sh -p <platform name> -a <additional_params> -n [true|false]

PL011 UART

There are two PL011 UART controllers connected to the non-discoverable IO Virtualization block. These UART controllers are initialized by edk2 firmware before booting to the Linux kernel.

These UART peripherals are enumerated by the Linux Kernel as Serial ports and can be tested by writing to the corresponding tty device.

  • After booting into busybox, verify that the two PL011 UART controllers are enumerated. The command provided below will list the serial ports detected by the Linux kernel.

    # dmesg | grep tty
    [    0.034995] ARMH0011:00: ttyAMA0 at MMIO 0x1080000000 (irq = 14, base_baud = 0) is a SBSA
    [    0.035195] ARMH0011:01: ttyAMA1 at MMIO 0x10a0000000 (irq = 15, base_baud = 0) is a SBSA
    [    0.035595] ARMH0011:02: ttyAMA2 at MMIO 0xef70000 (irq = 36, base_baud = 0) is a SBSA
    [    0.037095] printk: console [ttyAMA2] enabled
    
  • Here, ttyAMA0 and ttyAMA1 are the PL011 UART peripherals that are connected to the I/O virtualization block.

  • Now test the PL011 UART peripherals by writing to the corresponding tty device files using echo command:

    # echo "test message 0" > /dev/ttyAMA0
    # echo "test message 1" > /dev/ttyAMA1
    
  • The above commands print the message on the FVP iomacro_terminal_0 and FVP iomacro_terminal_1 terminals respectively.

PL330 DMA

There are two PL330 DMA controllers connected to the non-discoverable IO Virtualization block. Each of these controllers support 8 data channels and one instruction channel.

To test these dma controllers, DMA test guide included in the Linux kernel documentation has to be followed. As mentioned in the guide, CONFIG_DMATEST has to be enabled in the Linux kernel.

  • After booting into busybox validate that the DMA PL330 controllers are probed fine and showing 8 channels on each dma controllers - dma0 and dma1.

    # ls /sys/class/dma
    dma0chan0  dma0chan3  dma0chan6  dma1chan1  dma1chan4  dma1chan7
    dma0chan1  dma0chan4  dma0chan7  dma1chan2  dma1chan5
    dma0chan2  dma0chan5  dma1chan0  dma1chan3  dma1chan6
    
  • Also verify that the two dma controllers are attached to SMMUv3 of I/O Virtualization block. An example of this shown below.

    # ls /sys/class/iommu/smmu3.0x0000000048000000/devices
    ARMH0330:00  ARMH0330:01
    
  • Following the DMA test guide, set the timeout and number of iterations. For example,

    # echo 2000 > /sys/module/dmatest/parameters/timeout
    # echo 1 > /sys/module/dmatest/parameters/iterations
    
  • Start the test for different channels. For example, to run on dma0chan0 use the following command:

    # echo dma0chan0 > /sys/module/dmatest/parameters/channel
    # echo 1 > /sys/module/dmatest/parameters/run
    
  • One can use loops to run the tests on all channels, for example for dma0:

    # for ch in 0 1 2 3 4 5 6 7; do echo dma0chan$ch > /sys/module/dmatest/parameters/channel; echo 1 > /sys/module/dmatest/parameters/run; sleep 1; done
    
    • Similarly, for other controller - dma1:

    # for ch in 0 1 2 3 4 5 6 7; do echo dma1chan$ch > /sys/module/dmatest/parameters/channel; echo 1 > /sys/module/dmatest/parameters/run; sleep 1; done
    
  • Test result: Test results are printed to the kernel log buffer with the format:

    "dmatest: result <channel>: <test id>: '<error msg>' with src_off=<val> dst_off=<val> len=<val> (<err code>)"
    
    • Below example if from running the test on all channels of dma0.

    # for ch in 0 1 2 3 4 5 6 7; do echo dma0chan$ch > /sys/module/dmatest/parameters/channel; echo 1 > /sys/module/dmatest/parameters/run; sleep 1; done
    [ 1375.118694] dmatest: Added 1 threads using dma0chan0
    [ 1375.118694] dmatest: Started 1 threads using dma0chan0
    [ 1375.123195] dmatest: dma0chan0-copy0: summary 1 tests, 0 failures 235.45 iops 3060 KB/s (0)
    [ 1376.119025] dmatest: Added 1 threads using dma0chan1
    [ 1376.119025] dmatest: Started 1 threads using dma0chan1
    [ 1376.120894] dmatest: dma0chan1-copy0: summary 1 tests, 0 failures 615.76 iops 3078 KB/s (0)
    [ 1377.119311] dmatest: Added 1 threads using dma0chan2
    [ 1377.119311] dmatest: Started 1 threads using dma0chan2
    [ 1377.123594] dmatest: dma0chan2-copy0: summary 1 tests, 0 failures 246.24 iops 2954 KB/s (0)
    
    ... and so on
    

SRAM Memory

There are six SRAM memory regions connected to the non-discoverable IO Virtualization block. Out of the six, two SRAM memory are connected to the the high bandwidth port of the I/O virtualization block and the remaining 4 are connected to the low bandwidth port. The size of each SRAM memory connected to the I/O virtualization block is 4MiB. The memory mapping for all the SRAM memory are listed in the below table:

SRAM that are connected to high bandwidth port:

Mem Name

Start Address

End Address

Size

MEM0

0x10_8001_0000

0x10_8001_FFFF

4MiB

MEM1

0x10_B002_0000

0x10_B002_FFFF

4MiB

SRAM that are connected to low bandwidth port:

iomacro_size = 0x2000000

iomacro_instance :
    RD-N2-Cfg1 = 1
    RD-N2      = 4

Mem Name

Start Address

End Address

Size

MEM2

0x4100_0000 + (iomacro_instance * iomacro_size)

0x413F_FFFF + (iomacro_instance * iomacro_size)

4MiB

MEM3

0x4140_0000 + (iomacro_instance * iomacro_size)

0x417F_FFFF + (iomacro_instance * iomacro_size)

4MiB

MEM4

0x4180_0000 + (iomacro_instance * iomacro_size)

0x41BF_FFFF + (iomacro_instance * iomacro_size)

4MiB

MEM5

0x41C0_0000 + (iomacro_instance * iomacro_size)

0x41FF_FFFF + (iomacro_instance * iomacro_size)

4MiB

  • The SRAM memory can be tested by using devmem busybox utility which can be used to read and write to physical memory using /dev/mem provided that the Linux Kernel is built with Kernel config option CONFIG_DEVMEM=y

  • After booting into busybox, use the following example to test the SRAM memory connected to the non-discoverable I/O virtualization block instance.

  • Type “devmem” to display the busybox devmem utility info

    # devmem
    BusyBox v1.33.0 (2021-08-10 13:24:14 IST) multi-call binary.
    
    Usage: devmem ADDRESS [WIDTH [VALUE]]
    
    Read/write from physical address
    
            ADDRESS Address to act upon
            WIDTH   Width (8/16/...)
            VALUE   Data to be written
    
  • Test SRAM memory write by using the following example.

    # devmem 0x1080010000 32 0xabcd1234
    
  • Similarly, test SRAM memory read by using the following example.

    # devmem 0x1080010000 32
    0xABCD1234
    

This completes the testing for non-PCI devices connected to the I/O virtualization block.


Copyright (c) 2022-2023, Arm Limited. All rights reserved.