This is a simple description of the procedure to install the traffic generator TRex. The detailed documentation for TRex can be found here. This is intended as a preparation for measuring the performance of open source UPFs.
- Simple Overview of TRex and DUT (UPF)
- Install TRex
- Setup TRex
- Run TRex
- Sample Configurations
- Changelog (summary)
This describes a simple configuration of TRex and DUT (UPF). Note that this configuration is implemented with Proxmox VE VMs.
The following minimum configuration was set as a condition.
- One TRex and DUT (UPF)
The built simulation environment is as follows.
The TRex used is as follows.
- TRex v3.06 (2024.09.17) - https://github.com/cisco-system-traffic-generator/trex-core
- Scapy v2.6.1 (2024.11.05) - https://github.com/secdev/scapy
Each VMs are as follows.
VM | SW & Role | IP address | OS | CPU (Min) |
Mem (Min) |
HDD (Min) |
---|---|---|---|---|---|---|
VM-TG | TRex Traffic Generator |
192.168.0.131/24 | Ubuntu 22.04 | 3 | 8GB | 20GB |
VM-DUT | each UPF DUT (Device Under Test) |
192.168.0.151/24 | Ubuntu 24.04 or 22.04 |
2 | 8GB | 20GB |
The network interfaces of each VM are as follows.
VM | Device | Model | Linux Bridge | IP address | Interface | Under DPDK |
---|---|---|---|---|---|---|
VM-TG | ens18 | VirtIO | vmbr1 | 10.0.0.131/24 | (NAPT NW) | -- |
ens19 | VirtIO | mgbr0 | 192.168.0.131/24 | (Mgmt NW) | -- | |
ens20 | VirtIO | vmbr3 | 192.168.13.131/24 | N3 | x | |
ens21 | VirtIO | vmbr6 | 192.168.16.152/24 | N6 | x | |
VM-DUT | -- | |||||
ens19 | VirtIO | mgbr0 | 192.168.0.151/24 | (Mgmt NW) | -- | |
ens20 | VirtIO | vmbr3 | 192.168.13.151/24 | N3 | -- | |
ens21 | VirtIO | vmbr4 | 192.168.14.151/24 | N4 | -- | |
ens22 | VirtIO | vmbr6 | 192.168.16.151/24 | N6 | -- |
Linux Bridges of Proxmox VE are as follows.
Linux Bridge | Network CIDR | Interface |
---|---|---|
vmbr1 | 10.0.0.0/24 | NAPT NW |
mgbr0 | 192.168.0.0/24 | Mgmt NW |
vmbr3 | 192.168.13.0/24 | N3 |
vmbr4 | 192.168.14.0/24 | N4 |
vmbr6 | 192.168.16.0/24 | N6 |
UE IP address and TEID are as follows.
UE IP address | TEID |
---|---|
10.45.0.2/24 | 0x00000001 |
Please refer to the following for installing TRex.
- TRex v3.06 (2024.09.17) - https://github.com/cisco-system-traffic-generator/trex-core/wiki
This section explains how to build and install TRex v3.06 to /opt/trex
directory.
This time, build TRex using Scapy v2.6.1 instead of v2.4.3 included in TRex v3.06.
# apt install build-essential zlib1g-dev
First, git clone TRex v3.06 in an appropriate directory.
# cd ~/
# git clone https://github.com/cisco-system-traffic-generator/trex-core
Next, download Scapy v2.6.1 and extract it to the appropriate directory as follows.
# cd ~/
# wget https://github.com/secdev/scapy/archive/refs/tags/v2.6.1.tar.gz
# tar -C ~/trex-core/scripts/external_libs -zxvf v2.6.1.tar.gz
Next, apply this patch that changes Scapy v2.4.3 to v2.6.1.
# cd ~/
# wget https://raw.githubusercontent.com/s5uishida/install_trex/refs/heads/main/patches/update-scapy.patch
# cd trex-core
# patch -p1 < ../update-scapy.patch
Then, build TRex.
# cd ~/trex-core/linux_dpdk
# ./b configure
# ./b build
Finally, install the built TRex to /opt/trex
.
# cd ~/trex-core
# cp -prL scripts /opt/trex
First, check the network devices and bus information of the installation machine.
# lshw -c network -businfo
Bus info Device Class Description
====================================================
pci@0000:00:12.0 network Virtio network device
virtio@1 ens18 network Ethernet interface
pci@0000:00:13.0 network Virtio network device
virtio@2 ens19 network Ethernet interface
pci@0000:00:14.0 network Virtio network device
virtio@3 ens20 network Ethernet interface
pci@0000:00:15.0 network Virtio network device
virtio@4 ens21 network Ethernet interface
In my environment, based on this information, configure pci@0000:00:14.0
of ens20
(N3) and pci@0000:00:15.0
of ens21
(N6).
Use the following dpdk_setup_ports.py
script to create /etc/trex_cfg.yaml
with the IP settings for the ports used by DPDK.
# cd /opt/trex
# ./dpdk_setup_ports.py -i
...
I edited this file created as follows.
/etc/trex_cfg.yaml
### Config file generated by dpdk_setup_ports.py ###
- version: 2
interfaces: ['00:14.0', '00:15.0']
port_info:
- ip: 192.168.13.131
default_gw: 192.168.13.151
- ip: 192.168.16.152
default_gw: 192.168.16.151
platform:
master_thread_id: 0
latency_thread_id: 1
dual_if:
- socket: 0
threads: [2]
I am using the following for the TRex load profile. Also, the payload size is set to 1400 bytes and the QFI is set to 1.
/opt/trex/stl/gtp_1pkt_simple.py
from trex_stl_lib.api import *
from scapy.contrib.gtp import GTP_U_Header, GTPPDUSessionContainer
import argparse
class STLS1(object):
def create_stream (self):
return STLStream(
packet =
STLPktBuilder(
pkt = Ether()/IP(src="192.168.13.131",dst="192.168.13.151",version=4)/
UDP(dport=2152,sport=2152)/
GTP_U_Header(teid=0x00000001)/
GTPPDUSessionContainer(type=1,QFI=1)/
IP(src="10.45.0.2",dst="192.168.16.152",version=4)/
UDP()/
(1400*'x')
),
mode = STLTXCont())
def get_streams (self, tunables, **kwargs):
parser = argparse.ArgumentParser(description='Argparser for {}'.format(os.path.basename(__file__)),
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
args = parser.parse_args(tunables)
# create 1 stream
return [ self.create_stream() ]
def register():
return STLS1()
/opt/trex/stl/udp_1pkt_simple.py
from trex_stl_lib.api import *
import argparse
class STLS1(object):
def create_stream (self):
return STLStream(
packet =
STLPktBuilder(
pkt = Ether()/IP(src="192.168.16.152",dst="10.45.0.2",version=4)/
UDP()/
(1400*'x')
),
mode = STLTXCont())
def get_streams (self, tunables, **kwargs):
parser = argparse.ArgumentParser(description='Argparser for {}'.format(os.path.basename(__file__)),
formatter_class=argparse.ArgumentDefaultsHelpFormatter)
args = parser.parse_args(tunables)
# create 1 stream
return [ self.create_stream() ]
def register():
return STLS1()
Set vm.nr_hugepages
to 2048.
# sysctl vm.nr_hugepages=2048
Please configure and start the DUT (UPF in my environment) in advance.
For information on how to use the TRex console, see here.
First, start the TRex server.
# cd /opt/trex
# ./t-rex-64 -i --no-scapy-server
Then, open another terminal and connect to the TRex server to check the status of each port.
# cd /opt/trex
# ./trex-console
...
trex>
trex>portattr
Port Status
port | 0 | 1
----------------+----------------------+---------------------
driver | net_virtio | net_virtio
description | Virtio network dev | Virtio network dev
link status | UP | UP
link speed | 200 Gb/s | 200 Gb/s
port status | IDLE | IDLE
promiscuous | off | off
multicast | off | off
flow ctrl | N/A | N/A
vxlan fs | - | -
-- | |
layer mode | IPv4 | IPv4
src IPv4 | 192.168.13.131 | 192.168.16.152
IPv6 | off | off
src MAC | bc:24:11:6b:bf:13 | bc:24:11:5c:31:7b
--- | |
Destination | 192.168.13.151 | 192.168.16.151
ARP Resolution | bc:24:11:74:fe:7b | bc:24:11:bf:d4:23
---- | |
VLAN | - | -
----- | |
PCI Address | 0000:00:14.0 | 0000:00:15.0
NUMA Node | -1 | -1
RX Filter Mode | hardware match | hardware match
RX Queueing | off | off
Grat ARP | every 120 seconds | every 120 seconds
------ | |
trex>
Switch ports: [0, 1]
into service mode and ping to the IP addresses (192.168.13.151
and 192.168.16.151
) of the DUT.
trex>service
Enabling service mode on port(s): [0, 1] [SUCCESS]
12.98 [ms]
trex(service)>
trex(service)>ping -p 0 -d 192.168.13.151 -n 3
Pinging 192.168.13.151 from port 0 with 64 bytes of data:
Reply from 192.168.13.151: bytes=64, time=46.84ms, TTL=64
Reply from 192.168.13.151: bytes=64, time=2.78ms, TTL=64
Reply from 192.168.13.151: bytes=64, time=1.59ms, TTL=64
trex(service)>
trex(service)>ping -p 1 -d 192.168.16.151 -n 3
Pinging 192.168.16.151 from port 1 with 64 bytes of data:
Reply from 192.168.16.151: bytes=64, time=9.11ms, TTL=64
Reply from 192.168.16.151: bytes=64, time=1.47ms, TTL=64
Reply from 192.168.16.151: bytes=64, time=2.66ms, TTL=64
trex(service)>
Once confirmed that ping works, the service mode exit.
trex(service)>service --off
Disabling service mode on port(s): [0, 1] [SUCCESS]
11.86 [ms]
trex>
Then, apply traffic to the DUT.
In the following example, use the load profile gtp_1pkt_simple.py
to apply GTP-U traffic to the DUT at 150 Kpps for 60 seconds via port 0.
trex>start -f stl/gtp_1pkt_simple.py -p 0 -m 150kpps -d 60
To check the traffic statistics, type tui
in the TRex console to switch the view.
trex>tui
Below are some sample statistics. According to this, 705.53 Mbps of 1.79 Gbps was dropped, and 1.08 Gbps was received.
Global Statistics
connection : localhost, Port 4501 total_tx_L2 : 1.79 Gbps
version : STL @ v3.06 total_tx_L1 : 1.81 Gbps
cpu_util. : 1.68% @ 1 cores (1 per dual port) total_rx : 1.08 Gbps
rx_cpu_util. : 0.18% / 93.65 Kpps total_pps : 150.07 Kpps
async_util. : 0% / 4.79 bps drop_rate : 705.53 Mbps
total_cps. : 0 cps queue_full : 12,034 pkts
Port Statistics
port | 0 | 1 | total
-----------+-------------------+-------------------+------------------
owner | root | root |
link | UP | UP |
state | TRANSMITTING | IDLE |
speed | 200 Gb/s | 200 Gb/s |
CPU util. | 1.68% | 0.0% |
-- | | |
Tx bps L2 | 1.79 Gbps | 0.09 bps | 1.79 Gbps
Tx bps L1 | 1.81 Gbps | 0.13 bps | 1.81 Gbps
Tx pps | 150.07 Kpps | 0 pps | 150.07 Kpps
Line Util. | 0.91 % | 0 % |
--- | | |
Rx bps | 0 bps | 1.08 Gbps | 1.08 Gbps
Rx pps | 0 pps | 93.65 Kpps | 93.65 Kpps
---- | | |
opackets | 2687829 | 1 | 2687830
ipackets | 0 | 1739130 | 1739130
obytes | 4004865210 | 46 | 4004865256
ibytes | 0 | 2514780580 | 2514780580
tx-pkts | 2.69 Mpkts | 1 pkts | 2.69 Mpkts
rx-pkts | 0 pkts | 1.74 Mpkts | 1.74 Mpkts
tx-bytes | 4 GB | 46 B | 4 GB
rx-bytes | 0 B | 2.51 GB | 2.51 GB
----- | | |
oerrors | 0 | 0 | 0
ierrors | 0 | 0 | 0
status: |
Press 'ESC' for navigation panel...
status:
tui>
In the following example, use the load profile udp_1pkt_simple.py
to apply UDP traffic to the DUT at 150 Kpps for 60 seconds via port 1.
trex>start -f stl/udp_1pkt_simple.py -p 1 -m 150kpps -d 60
Below are some sample statistics. According to this, 413.05 Mbps of 1.74 Gbps was dropped, and 1.33 Gbps was received.
Global Statistics
connection : localhost, Port 4501 total_tx_L2 : 1.74 Gbps
version : STL @ v3.06 total_tx_L1 : 1.76 Gbps
cpu_util. : 1.89% @ 1 cores (1 per dual port) total_rx : 1.33 Gbps
rx_cpu_util. : 0.21% / 111.87 Kpps total_pps : 150.36 Kpps
async_util. : 0% / 9.02 bps drop_rate : 413.05 Mbps
total_cps. : 0 cps queue_full : 3,817 pkts
Port Statistics
port | 0 | 1 | total
-----------+-------------------+-------------------+------------------
owner | root | root |
link | UP | UP |
state | IDLE | TRANSMITTING |
speed | 200 Gb/s | 200 Gb/s |
CPU util. | 0.0% | 1.89% |
-- | | |
Tx bps L2 | 1.41 bps | 1.74 Gbps | 1.74 Gbps
Tx bps L1 | 2.02 bps | 1.76 Gbps | 1.76 Gbps
Tx pps | 0 pps | 150.36 Kpps | 150.36 Kpps
Line Util. | 0 % | 0.88 % |
--- | | |
Rx bps | 1.33 Gbps | 0 bps | 1.33 Gbps
Rx pps | 111.87 Kpps | 0 pps | 111.87 Kpps
---- | | |
opackets | 1 | 2007396 | 2007397
ipackets | 1304349 | 0 | 1304349
obytes | 46 | 2902694616 | 2902694662
ibytes | 1933043782 | 0 | 1933043782
tx-pkts | 1 pkts | 2.01 Mpkts | 2.01 Mpkts
rx-pkts | 1.3 Mpkts | 0 pkts | 1.3 Mpkts
tx-bytes | 46 B | 2.9 GB | 2.9 GB
rx-bytes | 1.93 GB | 0 B | 1.93 GB
----- | | |
oerrors | 0 | 0 | 0
ierrors | 0 | 0 | 0
status: |
Press 'ESC' for navigation panel...
status:
tui>
With the above steps, TRex has been constructed. Although the settings differ for each UPF, I believe this has made some progress in preparing to measure open source UPFs performance using TRex. I would like to thank the excellent developers and all the contributors of TRex.
- [2025.01.16] Added the downlink measurement.
- [2024.12.08] Updated Scapy from v2.4.3 to v2.6.1 and built TRex for using the PDU Session container in the GTP-U packet header.
- [2024.11.03] Initial release.