* [PATCH v1 0/3] dts: add test suite for dual VLANs
@ 2024-07-15 19:58 jspewock
2024-07-15 19:58 ` [PATCH v1 1/3] dts: fix Testpmd function for resetting VLAN insertion jspewock
` (4 more replies)
0 siblings, 5 replies; 21+ messages in thread
From: jspewock @ 2024-07-15 19:58 UTC (permalink / raw)
To: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
This series ports over the implementation of the dual_vlan test suite in
old DTS and refactors it, dropping some duplicated functionality as well
as some features that are specific to certain NICs.
One thing to note about this series is that it is tested and fully
working on a Mellanox NIC running the mlx5_core driver, but in testing I
did notice some stranger behavior on a NIC running the bnxt_en driver.
The broadcom NIC worked for all test cases except for those involving
VLAN insertion. In the presence of 2 VLAN headers it seems that the
bnxt_en NIC drops the packet completely if you attempt to insert a 3rd.
I originally thought this might be an MTU issue, but with MTUs of 2000
on the DUT and 9000 on the traffic generator the packet was still
dropped. I believe VLAN insertion in the presence of no other VLAN
headers works on this same NIC was tested by Dean Marx.
Jeremy Spewock (3):
dts: fix Testpmd function for resetting VLAN insertion
dts: add dual_vlan testing suite
dts: add dual_vlan test suite to the yaml schema
dts/framework/config/conf_yaml_schema.json | 3 +-
dts/framework/remote_session/testpmd_shell.py | 2 +-
dts/tests/TestSuite_dual_vlan.py | 281 ++++++++++++++++++
3 files changed, 284 insertions(+), 2 deletions(-)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v1 1/3] dts: fix Testpmd function for resetting VLAN insertion
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
@ 2024-07-15 19:58 ` jspewock
2024-07-15 19:58 ` [PATCH v1 2/3] dts: add dual_vlan testing suite jspewock
` (3 subsequent siblings)
4 siblings, 0 replies; 21+ messages in thread
From: jspewock @ 2024-07-15 19:58 UTC (permalink / raw)
To: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro
Cc: dev, Jeremy Spewock, dmarx
From: Jeremy Spewock <jspewock@iol.unh.edu>
The previous method would send the command `tx_vlan set <port_id>` when
the correct command is `tx_vlan reset <port_id>`.
Fixes: a49d9da1e9a5 ("dts: add VLAN methods to testpmd shell")
Cc: dmarx@iol.unh.edu
depends-on: patch-142103 ("dts: add VLAN methods to testpmd shell")
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/framework/remote_session/testpmd_shell.py | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/dts/framework/remote_session/testpmd_shell.py b/dts/framework/remote_session/testpmd_shell.py
index 09d3bda5d6..a8b6a054b5 100644
--- a/dts/framework/remote_session/testpmd_shell.py
+++ b/dts/framework/remote_session/testpmd_shell.py
@@ -994,7 +994,7 @@ def tx_vlan_reset(self, port: int, verify: bool = True):
InteractiveCommandExecutionError: If `verify` is :data:`True` and the insertion
tag is not reset.
"""
- vlan_insert_output = self.send_command(f"tx_vlan set {port}")
+ vlan_insert_output = self.send_command(f"tx_vlan reset {port}")
if verify:
if "Please stop port" in vlan_insert_output or "Invalid port" in vlan_insert_output:
self._logger.debug(
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v1 2/3] dts: add dual_vlan testing suite
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
2024-07-15 19:58 ` [PATCH v1 1/3] dts: fix Testpmd function for resetting VLAN insertion jspewock
@ 2024-07-15 19:58 ` jspewock
2024-07-22 17:38 ` Dean Marx
2024-07-15 19:58 ` [PATCH v1 3/3] dts: add dual_vlan test suite to the yaml schema jspewock
` (2 subsequent siblings)
4 siblings, 1 reply; 21+ messages in thread
From: jspewock @ 2024-07-15 19:58 UTC (permalink / raw)
To: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
This patch ports over the functionality of the dual_vlan suite from old
DTS to the new framework. This test suite exists to test the
functionality of VLAN functions such as stripping, inserting, and
filerting in the presence of two VLAN headers.
There are some test cases which were left out in this refactored version
including test cases that test the functionality of VLAN functions on a
packet with only one VLAN header, as this is something that is tested in
another test suite which is currently in development. Additionally,
this series does not include test cases for testing the adjustment of
TPID or extended VLAN ranges, as these things were included in the old
test suite specifically for testing on Intel hardware and they are not
universally supported on every NIC. There could be further reason to add
these test cases in the future once the capabilities feature is fully
implemented. Extended mode for VLANs seems to be exposed through offload
capabilities of the port, but there doesn't seem to be anything as
obvious for TPID modification.
depends-on: patch-142103 ("dts: add VLAN methods to testpmd shell")
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/tests/TestSuite_dual_vlan.py | 281 +++++++++++++++++++++++++++++++
1 file changed, 281 insertions(+)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
diff --git a/dts/tests/TestSuite_dual_vlan.py b/dts/tests/TestSuite_dual_vlan.py
new file mode 100644
index 0000000000..095e57bc56
--- /dev/null
+++ b/dts/tests/TestSuite_dual_vlan.py
@@ -0,0 +1,281 @@
+# SPDX-License-Identifier: BSD-3-Clause
+# Copyright(c) 2024 University of New Hampshire
+
+"""Dual VLAN functionality testing suite.
+
+The main objective of this test suite is to ensure that standard VLAN functions such as stripping,
+filtering, and inserting all still carry out their expected behavior in the presence of a packet
+which contains two VLAN headers. These functions should carry out said behavior not just in
+isolation, but also when other VLAN functions are configured on the same port. In addition to this,
+the priority attributes of VLAN headers should be unchanged in the case of multiple VLAN headers
+existing on a single packet.
+"""
+import time
+from enum import Flag, auto
+from typing import ClassVar
+
+from scapy.layers.l2 import Dot1Q, Ether # type: ignore[import-untyped]
+from scapy.packet import Packet, Raw # type: ignore[import-untyped]
+
+from framework.params.testpmd import SimpleForwardingModes
+from framework.remote_session.testpmd_shell import TestPmdShell
+from framework.test_suite import TestSuite
+
+
+class TestDualVlan(TestSuite):
+ """DPDK Dual VLAN test suite.
+
+ This suite tests the behavior of VLAN functions and properties in the presence of two VLAN
+ headers. All VLAN functions which are tested in this suite are specified using the inner class
+ :class:`TestCaseOptions` and should have cases for configuring them in
+ :meth:`configure_testpmd` as well as cases for testing their behavior in
+ :meth:`verify_vlan_functions`. Every combination of VLAN functions being enabled should be
+ tested. Additionally, attributes of VLAN headers, such as priority, are tested to ensure they
+ are not modified in the case of two VLAN headers.
+ """
+
+ class TestCaseOptions(Flag):
+ """Flag for specifying which VLAN functions to configure."""
+
+ #:
+ VLAN_STRIP = auto()
+ #:
+ VLAN_FILTER_INNER = auto()
+ #:
+ VLAN_FILTER_OUTER = auto()
+ #:
+ VLAN_INSERT = auto()
+
+ #: ID to set on inner VLAN tags.
+ inner_vlan_tag: ClassVar[int] = 2
+ #: ID to set on outer VLAN tags.
+ outer_vlan_tag: ClassVar[int] = 1
+ #: ID to use when inserting VLAN tags.
+ vlan_insert_tag: ClassVar[int] = 3
+ #:
+ rx_port: ClassVar[int] = 0
+ #:
+ tx_port: ClassVar[int] = 1
+
+ def is_relevant_packet(self, pkt: Packet) -> bool:
+ """Check if a packet was sent by functions in this suite.
+
+ All functions in this test suite send packets with a payload that is packed with 20 "X"
+ characters. This method, therefore, can determine if the packet was sent by this test suite
+ by just checking to see if this payload exists on the received packet.
+
+ Args:
+ pkt: Packet to check for relevancy.
+
+ Returns:
+ :data:`True` if the packet contains the expected payload, :data:`False` otherwise.
+ """
+ return hasattr(pkt, "load") and "X" * 20 in str(pkt.load)
+
+ def pkt_payload_contains_layers(self, pkt: Packet, *expected_layers: Dot1Q) -> bool:
+ """Verify that the payload of the packet matches `expected_layers`.
+
+ The layers in the payload of `pkt` must match the type and the user-defined fields of the
+ layers in `expected_layers` in order.
+
+ Args:
+ pkt: Packet to check the payload of.
+ *expected_layers: Layers expected to be in the payload of `pkt`.
+
+ Returns:
+ :data:`True` if the payload of `pkt` matches the layers in `expected_layers` in order,
+ :data:`False` otherwise.
+ """
+ current_pkt_layer = pkt.payload
+ ret = True
+ for layer in expected_layers:
+ ret &= isinstance(current_pkt_layer, type(layer))
+ if not ret:
+ break
+ for field, val in layer.fields.items():
+ ret &= (
+ hasattr(current_pkt_layer, field) and getattr(current_pkt_layer, field) == val
+ )
+ current_pkt_layer = current_pkt_layer.payload
+ return ret
+
+ def verify_vlan_functions(self, send_packet: Packet, options: TestCaseOptions) -> None:
+ """Send packet and verify the received packet has the expected structure.
+
+ Expected structure is defined by `options` according to the following table:
+ +----------------------------------------------+-----------------------+
+ | Configure setting | Result |
+ +=======+=======+========+============+========+=======+=======+=======+
+ | Outer | Inner | Vlan | Vlan | Vlan | Pass/ | Outer | Inner |
+ | vlan | vlan | strip | filter | insert | Drop | vlan | vlan |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | no | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | no | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x2 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1,0x2| no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x2 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1,0x2| no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | no | yes | pass | 0x3 | 0x1 |
+ | | | | | | | | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | no | yes | pass | 0x3 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1 | yes | pass | 0x3 | 0x1 |
+ | | | | | | | | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x2 | yes | pass | 0x3 | 0x1 |
+ | | | | | | | | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1,0x2| yes | pass | 0x3 | 0x1 |
+ | | | | | | | | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1 | yes | pass | 0x3 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x2 | yes | pass | 0x3 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1,0x2| yes | pass | 0x3 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+
+ Args:
+ send_packet: Packet to send for testing.
+ options: Flag which defines the currents configured settings in testpmd.
+ """
+ recv = self.send_packet_and_capture(send_packet)
+ recv = list(filter(self.is_relevant_packet, recv))
+ expected_layers: list[Packet] = []
+
+ if self.TestCaseOptions.VLAN_INSERT in options:
+ expected_layers.append(Dot1Q(vlan=self.vlan_insert_tag))
+
+ if self.TestCaseOptions.VLAN_STRIP not in options:
+ expected_layers.append(Dot1Q(vlan=self.outer_vlan_tag))
+ expected_layers.append(Dot1Q(vlan=self.inner_vlan_tag))
+
+ self.verify(
+ len(recv) > 0,
+ f"Expected to receive packet with the payload {expected_layers} but got nothing.",
+ )
+
+ for pkt in recv:
+ self.verify(
+ self.pkt_payload_contains_layers(pkt, *expected_layers),
+ f"Received packet ({pkt.summary()}) did not match the expected sequence of layers "
+ f"{expected_layers} with options {options}.",
+ )
+
+ def configure_testpmd(self, shell: TestPmdShell, options: TestCaseOptions, add: bool) -> None:
+ """Configure VLAN functions in testpmd based on `options`.
+
+ Args:
+ shell: Testpmd session to configure the settings on. Expected to already be running
+ with all ports stopped before being passed into this function.
+ options: Settings to modify in `shell`.
+ add: If :data:`True`, turn the settings in `options` on, otherwise turn them off.
+ """
+ if (
+ self.TestCaseOptions.VLAN_FILTER_INNER in options
+ or self.TestCaseOptions.VLAN_FILTER_OUTER in options
+ ):
+ if add:
+ # If we are adding a filter, filtering has to be enabled first
+ shell.vlan_filter_set(self.rx_port, True)
+
+ if self.TestCaseOptions.VLAN_FILTER_INNER in options:
+ shell.rx_vlan(self.inner_vlan_tag, self.rx_port, add)
+ if self.TestCaseOptions.VLAN_FILTER_OUTER in options:
+ shell.rx_vlan(self.outer_vlan_tag, self.rx_port, add)
+
+ if not add:
+ # If we are removing filters then we need to remove the filters before we can
+ # disable filtering.
+ shell.vlan_filter_set(self.rx_port, False)
+ if self.TestCaseOptions.VLAN_INSERT in options:
+ if add:
+ shell.tx_vlan_set(self.tx_port, self.vlan_insert_tag)
+ else:
+ shell.tx_vlan_reset(self.tx_port)
+ if self.TestCaseOptions.VLAN_STRIP in options:
+ shell.vlan_strip_set(self.rx_port, add)
+
+ def test_vlan_functions(self) -> None:
+ """Test that all combinations of :class:`TestCaseOptions` behave as expected.
+
+ To test this, the base case is tested first, ensuring that a packet with two VLANs is
+ unchanged without the VLAN modification functions enabled. Then the same Testpmd shell is
+ modified to enable all necessary VLAN functions, followed by verification that the
+ functions work as expected, and finally the functions are disabled to allow for a clean
+ environment for the next test.
+ """
+ send_pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag)
+ / Dot1Q(vlan=self.inner_vlan_tag)
+ / Raw("X" * 20)
+ )
+ testpmd = TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac)
+ testpmd.start()
+ recv = self.send_packet_and_capture(send_pakt)
+ self.verify(len(recv) > 0, "Unmodified packet was not received.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, *[Dot1Q(vlan=self.outer_vlan_tag), Dot1Q(vlan=self.inner_vlan_tag)]
+ )
+ for p in recv
+ ),
+ "Packet was modified without any VLAN functions applied.",
+ )
+ testpmd.stop()
+ testpmd.port_stop_all()
+ for i in range(2 ** len(self.TestCaseOptions)):
+ options = self.TestCaseOptions(i)
+ self.configure_testpmd(testpmd, options, True)
+ testpmd.port_start_all()
+ testpmd.start()
+ self.verify_vlan_functions(send_pakt, options)
+ testpmd.stop()
+ testpmd.port_stop_all()
+ self.configure_testpmd(testpmd, options, False)
+
+ testpmd.close()
+ time.sleep(5)
+
+ def test_maintains_priority(self) -> None:
+ """Test that priorities of multiple VLAN tags are preserved by the PMD."""
+ pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag, prio=1)
+ / Dot1Q(vlan=self.inner_vlan_tag, prio=2)
+ / Raw("X" * 20)
+ )
+ testpmd = TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac)
+ testpmd.start()
+ recv = self.send_packet_and_capture(pakt)
+ self.verify(len(recv) > 0, "Did not receive and packets when testing VLAN priority.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p,
+ *[
+ Dot1Q(vlan=self.outer_vlan_tag, prio=1),
+ Dot1Q(vlan=self.inner_vlan_tag, prio=2),
+ ],
+ )
+ for p in recv
+ ),
+ "Vlan headers did not maintain their priorities.",
+ )
+ testpmd.close()
+ time.sleep(5)
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v1 3/3] dts: add dual_vlan test suite to the yaml schema
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
2024-07-15 19:58 ` [PATCH v1 1/3] dts: fix Testpmd function for resetting VLAN insertion jspewock
2024-07-15 19:58 ` [PATCH v1 2/3] dts: add dual_vlan testing suite jspewock
@ 2024-07-15 19:58 ` jspewock
2024-07-16 21:24 ` [PATCH v2 0/2] dts: add test suite for dual VLANs jspewock
2024-07-24 17:13 ` [PATCH v3 0/2] dts: add test suite for dual VLANs jspewock
4 siblings, 0 replies; 21+ messages in thread
From: jspewock @ 2024-07-15 19:58 UTC (permalink / raw)
To: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
Adds the test suite name to the yaml schema to allow for it to be run.
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/framework/config/conf_yaml_schema.json | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/dts/framework/config/conf_yaml_schema.json b/dts/framework/config/conf_yaml_schema.json
index f02a310bb5..b8ad5b37b3 100644
--- a/dts/framework/config/conf_yaml_schema.json
+++ b/dts/framework/config/conf_yaml_schema.json
@@ -187,7 +187,8 @@
"enum": [
"hello_world",
"os_udp",
- "pmd_buffer_scatter"
+ "pmd_buffer_scatter",
+ "dual_vlan"
]
},
"test_target": {
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v2 0/2] dts: add test suite for dual VLANs
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
` (2 preceding siblings ...)
2024-07-15 19:58 ` [PATCH v1 3/3] dts: add dual_vlan test suite to the yaml schema jspewock
@ 2024-07-16 21:24 ` jspewock
2024-07-16 21:24 ` [PATCH v2 1/2] dts: add dual_vlan testing suite jspewock
2024-07-16 21:24 ` [PATCH v2 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
2024-07-24 17:13 ` [PATCH v3 0/2] dts: add test suite for dual VLANs jspewock
4 siblings, 2 replies; 21+ messages in thread
From: jspewock @ 2024-07-16 21:24 UTC (permalink / raw)
To: thomas, wathsala.vithanage, Honnappa.Nagarahalli, yoan.picchi,
npratte, probb, paul.szczepanek, juraj.linkes, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
v2:
* remove test cases that verify the ability to have 3 VLAN headers as
this is a less practical case than only having 2.
* add a test case that verifies a packet with only 1 VLAN header can
have another inserted into it.
* remove patch that fixes tx_vlan_reset method in testpmd as it is no
longer used in this series.
Jeremy Spewock (2):
dts: add dual_vlan testing suite
dts: add dual_vlan test suite to the yaml schema
dts/framework/config/conf_yaml_schema.json | 3 +-
dts/tests/TestSuite_dual_vlan.py | 276 +++++++++++++++++++++
2 files changed, 278 insertions(+), 1 deletion(-)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v2 1/2] dts: add dual_vlan testing suite
2024-07-16 21:24 ` [PATCH v2 0/2] dts: add test suite for dual VLANs jspewock
@ 2024-07-16 21:24 ` jspewock
2024-08-07 19:07 ` Dean Marx
2024-07-16 21:24 ` [PATCH v2 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
1 sibling, 1 reply; 21+ messages in thread
From: jspewock @ 2024-07-16 21:24 UTC (permalink / raw)
To: thomas, wathsala.vithanage, Honnappa.Nagarahalli, yoan.picchi,
npratte, probb, paul.szczepanek, juraj.linkes, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
This patch ports over the functionality of the dual_vlan suite from old
DTS to the new framework. This test suite exists to test the
functionality of VLAN functions such as stripping, inserting, and
filerting in the presence of two VLAN headers.
There are some test cases which were left out in this refactored version
including test cases that test the functionality of VLAN functions on a
packet with only one VLAN header, as this is something that is tested in
another test suite which is currently in development. Additionally,
this series does not include test cases for testing the adjustment of
TPID or extended VLAN ranges, as these things were included in the old
test suite specifically for testing on Intel hardware and they are not
universally supported on every NIC. There could be further reason to add
these test cases in the future once the capabilities feature is fully
implemented. Extended mode for VLANs seems to be exposed through offload
capabilities of the port, but there doesn't seem to be anything as
obvious for TPID modification.
depends-on: patch-142103 ("dts: add VLAN methods to testpmd shell")
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/tests/TestSuite_dual_vlan.py | 276 +++++++++++++++++++++++++++++++
1 file changed, 276 insertions(+)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
diff --git a/dts/tests/TestSuite_dual_vlan.py b/dts/tests/TestSuite_dual_vlan.py
new file mode 100644
index 0000000000..623231c9b5
--- /dev/null
+++ b/dts/tests/TestSuite_dual_vlan.py
@@ -0,0 +1,276 @@
+# SPDX-License-Identifier: BSD-3-Clause
+# Copyright(c) 2024 University of New Hampshire
+
+"""Dual VLAN functionality testing suite.
+
+The main objective of this test suite is to ensure that standard VLAN functions such as stripping
+and filtering both still carry out their expected behavior in the presence of a packet which
+contains two VLAN headers. These functions should carry out said behavior not just in isolation,
+but also when other VLAN functions are configured on the same port. In addition to this, the
+priority attributes of VLAN headers should be unchanged in the case of multiple VLAN headers
+existing on a single packet, and a packet with only a single VLAN header should be able to have one
+additional VLAN inserted into it.
+"""
+import time
+from enum import Flag, auto
+from typing import ClassVar
+
+from scapy.layers.l2 import Dot1Q, Ether # type: ignore[import-untyped]
+from scapy.packet import Packet, Raw # type: ignore[import-untyped]
+
+from framework.params.testpmd import SimpleForwardingModes
+from framework.remote_session.testpmd_shell import TestPmdShell
+from framework.test_suite import TestSuite
+
+
+class TestDualVlan(TestSuite):
+ """DPDK Dual VLAN test suite.
+
+ This suite tests the behavior of VLAN functions and properties in the presence of two VLAN
+ headers. All VLAN functions which are tested in this suite are specified using the inner class
+ :class:`TestCaseOptions` and should have cases for configuring them in
+ :meth:`configure_testpmd` as well as cases for testing their behavior in
+ :meth:`verify_vlan_functions`. Every combination of VLAN functions being enabled should be
+ tested. Additionally, attributes of VLAN headers, such as priority, are tested to ensure they
+ are not modified in the case of two VLAN headers.
+ """
+
+ class TestCaseOptions(Flag):
+ """Flag for specifying which VLAN functions to configure."""
+
+ #:
+ VLAN_STRIP = auto()
+ #:
+ VLAN_FILTER_INNER = auto()
+ #:
+ VLAN_FILTER_OUTER = auto()
+
+ #: ID to set on inner VLAN tags.
+ inner_vlan_tag: ClassVar[int] = 2
+ #: ID to set on outer VLAN tags.
+ outer_vlan_tag: ClassVar[int] = 1
+ #: ID to use when inserting VLAN tags.
+ vlan_insert_tag: ClassVar[int] = 3
+ #:
+ rx_port: ClassVar[int] = 0
+ #:
+ tx_port: ClassVar[int] = 1
+
+ def is_relevant_packet(self, pkt: Packet) -> bool:
+ """Check if a packet was sent by functions in this suite.
+
+ All functions in this test suite send packets with a payload that is packed with 20 "X"
+ characters. This method, therefore, can determine if the packet was sent by this test suite
+ by just checking to see if this payload exists on the received packet.
+
+ Args:
+ pkt: Packet to check for relevancy.
+
+ Returns:
+ :data:`True` if the packet contains the expected payload, :data:`False` otherwise.
+ """
+ return hasattr(pkt, "load") and "X" * 20 in str(pkt.load)
+
+ def pkt_payload_contains_layers(self, pkt: Packet, *expected_layers: Dot1Q) -> bool:
+ """Verify that the payload of the packet matches `expected_layers`.
+
+ The layers in the payload of `pkt` must match the type and the user-defined fields of the
+ layers in `expected_layers` in order.
+
+ Args:
+ pkt: Packet to check the payload of.
+ *expected_layers: Layers expected to be in the payload of `pkt`.
+
+ Returns:
+ :data:`True` if the payload of `pkt` matches the layers in `expected_layers` in order,
+ :data:`False` otherwise.
+ """
+ current_pkt_layer = pkt.payload
+ ret = True
+ for layer in expected_layers:
+ ret &= isinstance(current_pkt_layer, type(layer))
+ if not ret:
+ break
+ for field, val in layer.fields.items():
+ ret &= (
+ hasattr(current_pkt_layer, field) and getattr(current_pkt_layer, field) == val
+ )
+ current_pkt_layer = current_pkt_layer.payload
+ return ret
+
+ def verify_vlan_functions(self, send_packet: Packet, options: TestCaseOptions) -> None:
+ """Send packet and verify the received packet has the expected structure.
+
+ Expected structure is defined by `options` according to the following table:
+ +----------------------------------------------+-----------------------+
+ | Configure setting | Result |
+ +=======+=======+========+============+========+=======+=======+=======+
+ | Outer | Inner | Vlan | Vlan | Vlan | Pass/ | Outer | Inner |
+ | vlan | vlan | strip | filter | insert | Drop | vlan | vlan |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | no | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | no | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x2 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1,0x2| no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x2 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1,0x2| no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+
+ Args:
+ send_packet: Packet to send for testing.
+ options: Flag which defines the currents configured settings in testpmd.
+ """
+ recv = self.send_packet_and_capture(send_packet)
+ recv = list(filter(self.is_relevant_packet, recv))
+ expected_layers: list[Packet] = []
+
+ if self.TestCaseOptions.VLAN_STRIP not in options:
+ expected_layers.append(Dot1Q(vlan=self.outer_vlan_tag))
+ expected_layers.append(Dot1Q(vlan=self.inner_vlan_tag))
+
+ self.verify(
+ len(recv) > 0,
+ f"Expected to receive packet with the payload {expected_layers} but got nothing.",
+ )
+
+ for pkt in recv:
+ self.verify(
+ self.pkt_payload_contains_layers(pkt, *expected_layers),
+ f"Received packet ({pkt.summary()}) did not match the expected sequence of layers "
+ f"{expected_layers} with options {options}.",
+ )
+
+ def configure_testpmd(self, shell: TestPmdShell, options: TestCaseOptions, add: bool) -> None:
+ """Configure VLAN functions in testpmd based on `options`.
+
+ Args:
+ shell: Testpmd session to configure the settings on. Expected to already be running
+ with all ports stopped before being passed into this function.
+ options: Settings to modify in `shell`.
+ add: If :data:`True`, turn the settings in `options` on, otherwise turn them off.
+ """
+ if (
+ self.TestCaseOptions.VLAN_FILTER_INNER in options
+ or self.TestCaseOptions.VLAN_FILTER_OUTER in options
+ ):
+ if add:
+ # If we are adding a filter, filtering has to be enabled first
+ shell.vlan_filter_set(self.rx_port, True)
+
+ if self.TestCaseOptions.VLAN_FILTER_INNER in options:
+ shell.rx_vlan(self.inner_vlan_tag, self.rx_port, add)
+ if self.TestCaseOptions.VLAN_FILTER_OUTER in options:
+ shell.rx_vlan(self.outer_vlan_tag, self.rx_port, add)
+
+ if not add:
+ # If we are removing filters then we need to remove the filters before we can
+ # disable filtering.
+ shell.vlan_filter_set(self.rx_port, False)
+ if self.TestCaseOptions.VLAN_STRIP in options:
+ shell.vlan_strip_set(self.rx_port, add)
+
+ def test_insert_second_vlan(self) -> None:
+ """Test that a packet with a single VLAN can have an additional one inserted into it."""
+ testpmd = TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac)
+ testpmd.port_stop_all()
+ testpmd.tx_vlan_set(self.tx_port, self.vlan_insert_tag)
+ testpmd.port_start_all()
+ testpmd.start()
+ recv = self.send_packet_and_capture(
+ Ether() / Dot1Q(vlan=self.outer_vlan_tag) / Raw("X" * 20)
+ )
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN insertion.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, *[Dot1Q(vlan=self.vlan_insert_tag), Dot1Q(vlan=self.outer_vlan_tag)]
+ )
+ for p in recv
+ ),
+ "Packet was unable to insert a second VLAN tag.",
+ )
+ testpmd.close()
+ time.sleep(5)
+
+ def test_all_vlan_functions(self) -> None:
+ """Test that all combinations of :class:`TestCaseOptions` behave as expected.
+
+ To test this, the base case is tested first, ensuring that a packet with two VLANs is
+ unchanged without the VLAN modification functions enabled. Then the same Testpmd shell is
+ modified to enable all necessary VLAN functions, followed by verification that the
+ functions work as expected, and finally the functions are disabled to allow for a clean
+ environment for the next test.
+ """
+ send_pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag)
+ / Dot1Q(vlan=self.inner_vlan_tag)
+ / Raw("X" * 20)
+ )
+ testpmd = TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac)
+ testpmd.start()
+ recv = self.send_packet_and_capture(send_pakt)
+ self.verify(len(recv) > 0, "Unmodified packet was not received.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, *[Dot1Q(vlan=self.outer_vlan_tag), Dot1Q(vlan=self.inner_vlan_tag)]
+ )
+ for p in recv
+ ),
+ "Packet was modified without any VLAN functions applied.",
+ )
+ testpmd.stop()
+ testpmd.port_stop_all()
+ for i in range(2 ** len(self.TestCaseOptions)):
+ options = self.TestCaseOptions(i)
+ self.configure_testpmd(testpmd, options, True)
+ testpmd.port_start_all()
+ testpmd.start()
+ self.verify_vlan_functions(send_pakt, options)
+ testpmd.stop()
+ testpmd.port_stop_all()
+ self.configure_testpmd(testpmd, options, False)
+
+ testpmd.close()
+ time.sleep(5)
+
+ def test_maintains_priority(self) -> None:
+ """Test that priorities of multiple VLAN tags are preserved by the PMD."""
+ pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag, prio=1)
+ / Dot1Q(vlan=self.inner_vlan_tag, prio=2)
+ / Raw("X" * 20)
+ )
+ testpmd = TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac)
+ testpmd.start()
+ recv = self.send_packet_and_capture(pakt)
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN priority.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p,
+ *[
+ Dot1Q(vlan=self.outer_vlan_tag, prio=1),
+ Dot1Q(vlan=self.inner_vlan_tag, prio=2),
+ ],
+ )
+ for p in recv
+ ),
+ "Vlan headers did not maintain their priorities.",
+ )
+ testpmd.close()
+ time.sleep(5)
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v2 2/2] dts: add dual_vlan test suite to the yaml schema
2024-07-16 21:24 ` [PATCH v2 0/2] dts: add test suite for dual VLANs jspewock
2024-07-16 21:24 ` [PATCH v2 1/2] dts: add dual_vlan testing suite jspewock
@ 2024-07-16 21:24 ` jspewock
1 sibling, 0 replies; 21+ messages in thread
From: jspewock @ 2024-07-16 21:24 UTC (permalink / raw)
To: thomas, wathsala.vithanage, Honnappa.Nagarahalli, yoan.picchi,
npratte, probb, paul.szczepanek, juraj.linkes, Luca.Vizzarro
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
Adds the test suite name to the yaml schema to allow for it to be run.
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/framework/config/conf_yaml_schema.json | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/dts/framework/config/conf_yaml_schema.json b/dts/framework/config/conf_yaml_schema.json
index f02a310bb5..b8ad5b37b3 100644
--- a/dts/framework/config/conf_yaml_schema.json
+++ b/dts/framework/config/conf_yaml_schema.json
@@ -187,7 +187,8 @@
"enum": [
"hello_world",
"os_udp",
- "pmd_buffer_scatter"
+ "pmd_buffer_scatter",
+ "dual_vlan"
]
},
"test_target": {
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v1 2/3] dts: add dual_vlan testing suite
2024-07-15 19:58 ` [PATCH v1 2/3] dts: add dual_vlan testing suite jspewock
@ 2024-07-22 17:38 ` Dean Marx
2024-07-25 13:56 ` Jeremy Spewock
0 siblings, 1 reply; 21+ messages in thread
From: Dean Marx @ 2024-07-22 17:38 UTC (permalink / raw)
To: jspewock
Cc: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro, dev
[-- Attachment #1: Type: text/plain, Size: 309 bytes --]
>
> <snip>
> + recv = self.send_packet_and_capture(pakt)
> + self.verify(len(recv) > 0, "Did not receive and packets when
> testing VLAN priority.")
>
I'm assuming this should say "any" instead of "and." Other than that,
everything looks good to me.
Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
[-- Attachment #2: Type: text/html, Size: 644 bytes --]
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v3 0/2] dts: add test suite for dual VLANs
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
` (3 preceding siblings ...)
2024-07-16 21:24 ` [PATCH v2 0/2] dts: add test suite for dual VLANs jspewock
@ 2024-07-24 17:13 ` jspewock
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
2024-07-24 17:13 ` [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
4 siblings, 2 replies; 21+ messages in thread
From: jspewock @ 2024-07-24 17:13 UTC (permalink / raw)
To: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, probb
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
v3:
* rebase on rc3
Jeremy Spewock (2):
dts: add dual_vlan testing suite
dts: add dual_vlan test suite to the yaml schema
dts/framework/config/conf_yaml_schema.json | 3 +-
dts/tests/TestSuite_dual_vlan.py | 268 +++++++++++++++++++++
2 files changed, 270 insertions(+), 1 deletion(-)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v3 1/2] dts: add dual_vlan testing suite
2024-07-24 17:13 ` [PATCH v3 0/2] dts: add test suite for dual VLANs jspewock
@ 2024-07-24 17:13 ` jspewock
2024-08-13 19:47 ` Dean Marx
` (2 more replies)
2024-07-24 17:13 ` [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
1 sibling, 3 replies; 21+ messages in thread
From: jspewock @ 2024-07-24 17:13 UTC (permalink / raw)
To: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, probb
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
This patch ports over the functionality of the dual_vlan suite from old
DTS to the new framework. This test suite exists to test the
functionality of VLAN functions such as stripping, inserting, and
filerting in the presence of two VLAN headers.
There are some test cases which were left out in this refactored version
including test cases that test the functionality of VLAN functions on a
packet with only one VLAN header, as this is something that is tested in
another test suite which is currently in development. Additionally,
this series does not include test cases for testing the adjustment of
TPID or extended VLAN ranges, as these things were included in the old
test suite specifically for testing on Intel hardware and they are not
universally supported on every NIC. There could be further reason to add
these test cases in the future once the capabilities feature is fully
implemented. Extended mode for VLANs seems to be exposed through offload
capabilities of the port, but there doesn't seem to be anything as
obvious for TPID modification.
depends-on: patch-142103 ("dts: add VLAN methods to testpmd shell")
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/tests/TestSuite_dual_vlan.py | 268 +++++++++++++++++++++++++++++++
1 file changed, 268 insertions(+)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
diff --git a/dts/tests/TestSuite_dual_vlan.py b/dts/tests/TestSuite_dual_vlan.py
new file mode 100644
index 0000000000..3a8a52afeb
--- /dev/null
+++ b/dts/tests/TestSuite_dual_vlan.py
@@ -0,0 +1,268 @@
+# SPDX-License-Identifier: BSD-3-Clause
+# Copyright(c) 2024 University of New Hampshire
+
+"""Dual VLAN functionality testing suite.
+
+The main objective of this test suite is to ensure that standard VLAN functions such as stripping
+and filtering both still carry out their expected behavior in the presence of a packet which
+contains two VLAN headers. These functions should carry out said behavior not just in isolation,
+but also when other VLAN functions are configured on the same port. In addition to this, the
+priority attributes of VLAN headers should be unchanged in the case of multiple VLAN headers
+existing on a single packet, and a packet with only a single VLAN header should be able to have one
+additional VLAN inserted into it.
+"""
+from enum import Flag, auto
+from typing import ClassVar
+
+from scapy.layers.l2 import Dot1Q, Ether # type: ignore[import-untyped]
+from scapy.packet import Packet, Raw # type: ignore[import-untyped]
+
+from framework.params.testpmd import SimpleForwardingModes
+from framework.remote_session.testpmd_shell import TestPmdShell
+from framework.test_suite import TestSuite
+
+
+class TestDualVlan(TestSuite):
+ """DPDK Dual VLAN test suite.
+
+ This suite tests the behavior of VLAN functions and properties in the presence of two VLAN
+ headers. All VLAN functions which are tested in this suite are specified using the inner class
+ :class:`TestCaseOptions` and should have cases for configuring them in
+ :meth:`configure_testpmd` as well as cases for testing their behavior in
+ :meth:`verify_vlan_functions`. Every combination of VLAN functions being enabled should be
+ tested. Additionally, attributes of VLAN headers, such as priority, are tested to ensure they
+ are not modified in the case of two VLAN headers.
+ """
+
+ class TestCaseOptions(Flag):
+ """Flag for specifying which VLAN functions to configure."""
+
+ #:
+ VLAN_STRIP = auto()
+ #:
+ VLAN_FILTER_INNER = auto()
+ #:
+ VLAN_FILTER_OUTER = auto()
+
+ #: ID to set on inner VLAN tags.
+ inner_vlan_tag: ClassVar[int] = 2
+ #: ID to set on outer VLAN tags.
+ outer_vlan_tag: ClassVar[int] = 1
+ #: ID to use when inserting VLAN tags.
+ vlan_insert_tag: ClassVar[int] = 3
+ #:
+ rx_port: ClassVar[int] = 0
+ #:
+ tx_port: ClassVar[int] = 1
+
+ def is_relevant_packet(self, pkt: Packet) -> bool:
+ """Check if a packet was sent by functions in this suite.
+
+ All functions in this test suite send packets with a payload that is packed with 20 "X"
+ characters. This method, therefore, can determine if the packet was sent by this test suite
+ by just checking to see if this payload exists on the received packet.
+
+ Args:
+ pkt: Packet to check for relevancy.
+
+ Returns:
+ :data:`True` if the packet contains the expected payload, :data:`False` otherwise.
+ """
+ return hasattr(pkt, "load") and "X" * 20 in str(pkt.load)
+
+ def pkt_payload_contains_layers(self, pkt: Packet, *expected_layers: Dot1Q) -> bool:
+ """Verify that the payload of the packet matches `expected_layers`.
+
+ The layers in the payload of `pkt` must match the type and the user-defined fields of the
+ layers in `expected_layers` in order.
+
+ Args:
+ pkt: Packet to check the payload of.
+ *expected_layers: Layers expected to be in the payload of `pkt`.
+
+ Returns:
+ :data:`True` if the payload of `pkt` matches the layers in `expected_layers` in order,
+ :data:`False` otherwise.
+ """
+ current_pkt_layer = pkt.payload
+ ret = True
+ for layer in expected_layers:
+ ret &= isinstance(current_pkt_layer, type(layer))
+ if not ret:
+ break
+ for field, val in layer.fields.items():
+ ret &= (
+ hasattr(current_pkt_layer, field) and getattr(current_pkt_layer, field) == val
+ )
+ current_pkt_layer = current_pkt_layer.payload
+ return ret
+
+ def verify_vlan_functions(self, send_packet: Packet, options: TestCaseOptions) -> None:
+ """Send packet and verify the received packet has the expected structure.
+
+ Expected structure is defined by `options` according to the following table:
+ +----------------------------------------------+-----------------------+
+ | Configure setting | Result |
+ +=======+=======+========+============+========+=======+=======+=======+
+ | Outer | Inner | Vlan | Vlan | Vlan | Pass/ | Outer | Inner |
+ | vlan | vlan | strip | filter | insert | Drop | vlan | vlan |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | no | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | no | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x2 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1,0x2| no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x2 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1,0x2| no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+
+ Args:
+ send_packet: Packet to send for testing.
+ options: Flag which defines the currents configured settings in testpmd.
+ """
+ recv = self.send_packet_and_capture(send_packet)
+ recv = list(filter(self.is_relevant_packet, recv))
+ expected_layers: list[Packet] = []
+
+ if self.TestCaseOptions.VLAN_STRIP not in options:
+ expected_layers.append(Dot1Q(vlan=self.outer_vlan_tag))
+ expected_layers.append(Dot1Q(vlan=self.inner_vlan_tag))
+
+ self.verify(
+ len(recv) > 0,
+ f"Expected to receive packet with the payload {expected_layers} but got nothing.",
+ )
+
+ for pkt in recv:
+ self.verify(
+ self.pkt_payload_contains_layers(pkt, *expected_layers),
+ f"Received packet ({pkt.summary()}) did not match the expected sequence of layers "
+ f"{expected_layers} with options {options}.",
+ )
+
+ def configure_testpmd(self, shell: TestPmdShell, options: TestCaseOptions, add: bool) -> None:
+ """Configure VLAN functions in testpmd based on `options`.
+
+ Args:
+ shell: Testpmd session to configure the settings on. Expected to already be running
+ with all ports stopped before being passed into this function.
+ options: Settings to modify in `shell`.
+ add: If :data:`True`, turn the settings in `options` on, otherwise turn them off.
+ """
+ if (
+ self.TestCaseOptions.VLAN_FILTER_INNER in options
+ or self.TestCaseOptions.VLAN_FILTER_OUTER in options
+ ):
+ if add:
+ # If we are adding a filter, filtering has to be enabled first
+ shell.vlan_filter_set(self.rx_port, True)
+
+ if self.TestCaseOptions.VLAN_FILTER_INNER in options:
+ shell.rx_vlan(self.inner_vlan_tag, self.rx_port, add)
+ if self.TestCaseOptions.VLAN_FILTER_OUTER in options:
+ shell.rx_vlan(self.outer_vlan_tag, self.rx_port, add)
+
+ if not add:
+ # If we are removing filters then we need to remove the filters before we can
+ # disable filtering.
+ shell.vlan_filter_set(self.rx_port, False)
+ if self.TestCaseOptions.VLAN_STRIP in options:
+ shell.vlan_strip_set(self.rx_port, add)
+
+ def test_insert_second_vlan(self) -> None:
+ """Test that a packet with a single VLAN can have an additional one inserted into it."""
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.port_stop_all()
+ testpmd.tx_vlan_set(self.tx_port, self.vlan_insert_tag)
+ testpmd.port_start_all()
+ testpmd.start()
+ recv = self.send_packet_and_capture(
+ Ether() / Dot1Q(vlan=self.outer_vlan_tag) / Raw("X" * 20)
+ )
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN insertion.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, *[Dot1Q(vlan=self.vlan_insert_tag), Dot1Q(vlan=self.outer_vlan_tag)]
+ )
+ for p in recv
+ ),
+ "Packet was unable to insert a second VLAN tag.",
+ )
+
+ def test_all_vlan_functions(self) -> None:
+ """Test that all combinations of :class:`TestCaseOptions` behave as expected.
+
+ To test this, the base case is tested first, ensuring that a packet with two VLANs is
+ unchanged without the VLAN modification functions enabled. Then the same Testpmd shell is
+ modified to enable all necessary VLAN functions, followed by verification that the
+ functions work as expected, and finally the functions are disabled to allow for a clean
+ environment for the next test.
+ """
+ send_pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag)
+ / Dot1Q(vlan=self.inner_vlan_tag)
+ / Raw("X" * 20)
+ )
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.start()
+ recv = self.send_packet_and_capture(send_pakt)
+ self.verify(len(recv) > 0, "Unmodified packet was not received.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, *[Dot1Q(vlan=self.outer_vlan_tag), Dot1Q(vlan=self.inner_vlan_tag)]
+ )
+ for p in recv
+ ),
+ "Packet was modified without any VLAN functions applied.",
+ )
+ testpmd.stop()
+ testpmd.port_stop_all()
+ for i in range(2 ** len(self.TestCaseOptions)):
+ options = self.TestCaseOptions(i)
+ self.configure_testpmd(testpmd, options, True)
+ testpmd.port_start_all()
+ testpmd.start()
+ self.verify_vlan_functions(send_pakt, options)
+ testpmd.stop()
+ testpmd.port_stop_all()
+ self.configure_testpmd(testpmd, options, False)
+
+ def test_maintains_priority(self) -> None:
+ """Test that priorities of multiple VLAN tags are preserved by the PMD."""
+ pakt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag, prio=1)
+ / Dot1Q(vlan=self.inner_vlan_tag, prio=2)
+ / Raw("X" * 20)
+ )
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.start()
+ recv = self.send_packet_and_capture(pakt)
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN priority.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p,
+ *[
+ Dot1Q(vlan=self.outer_vlan_tag, prio=1),
+ Dot1Q(vlan=self.inner_vlan_tag, prio=2),
+ ],
+ )
+ for p in recv
+ ),
+ "Vlan headers did not maintain their priorities.",
+ )
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema
2024-07-24 17:13 ` [PATCH v3 0/2] dts: add test suite for dual VLANs jspewock
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
@ 2024-07-24 17:13 ` jspewock
2024-08-13 19:48 ` Dean Marx
1 sibling, 1 reply; 21+ messages in thread
From: jspewock @ 2024-07-24 17:13 UTC (permalink / raw)
To: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, probb
Cc: dev, Jeremy Spewock
From: Jeremy Spewock <jspewock@iol.unh.edu>
Adds the test suite name to the yaml schema to allow for it to be run.
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
---
dts/framework/config/conf_yaml_schema.json | 3 ++-
1 file changed, 2 insertions(+), 1 deletion(-)
diff --git a/dts/framework/config/conf_yaml_schema.json b/dts/framework/config/conf_yaml_schema.json
index f02a310bb5..b8ad5b37b3 100644
--- a/dts/framework/config/conf_yaml_schema.json
+++ b/dts/framework/config/conf_yaml_schema.json
@@ -187,7 +187,8 @@
"enum": [
"hello_world",
"os_udp",
- "pmd_buffer_scatter"
+ "pmd_buffer_scatter",
+ "dual_vlan"
]
},
"test_target": {
--
2.45.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v1 2/3] dts: add dual_vlan testing suite
2024-07-22 17:38 ` Dean Marx
@ 2024-07-25 13:56 ` Jeremy Spewock
2024-07-31 13:54 ` Jeremy Spewock
0 siblings, 1 reply; 21+ messages in thread
From: Jeremy Spewock @ 2024-07-25 13:56 UTC (permalink / raw)
To: Dean Marx
Cc: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro, dev
On Mon, Jul 22, 2024 at 1:38 PM Dean Marx <dmarx@iol.unh.edu> wrote:
>>
>> <snip>
>> + recv = self.send_packet_and_capture(pakt)
>> + self.verify(len(recv) > 0, "Did not receive and packets when testing VLAN priority.")
>
>
> I'm assuming this should say "any" instead of "and." Other than that, everything looks good to me.
>
Good catch, I'll fix this in the next version.
> Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v1 2/3] dts: add dual_vlan testing suite
2024-07-25 13:56 ` Jeremy Spewock
@ 2024-07-31 13:54 ` Jeremy Spewock
0 siblings, 0 replies; 21+ messages in thread
From: Jeremy Spewock @ 2024-07-31 13:54 UTC (permalink / raw)
To: Dean Marx
Cc: probb, juraj.linkes, thomas, wathsala.vithanage, paul.szczepanek,
npratte, yoan.picchi, Honnappa.Nagarahalli, Luca.Vizzarro, dev
On Thu, Jul 25, 2024 at 9:56 AM Jeremy Spewock <jspewock@iol.unh.edu> wrote:
>
> On Mon, Jul 22, 2024 at 1:38 PM Dean Marx <dmarx@iol.unh.edu> wrote:
> >>
> >> <snip>
> >> + recv = self.send_packet_and_capture(pakt)
> >> + self.verify(len(recv) > 0, "Did not receive and packets when testing VLAN priority.")
> >
> >
> > I'm assuming this should say "any" instead of "and." Other than that, everything looks good to me.
> >
>
> Good catch, I'll fix this in the next version.
Circling back on this, it looks like this is fixed in the current version.
>
> > Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v2 1/2] dts: add dual_vlan testing suite
2024-07-16 21:24 ` [PATCH v2 1/2] dts: add dual_vlan testing suite jspewock
@ 2024-08-07 19:07 ` Dean Marx
0 siblings, 0 replies; 21+ messages in thread
From: Dean Marx @ 2024-08-07 19:07 UTC (permalink / raw)
To: jspewock
Cc: thomas, wathsala.vithanage, Honnappa.Nagarahalli, yoan.picchi,
npratte, probb, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
dev
[-- Attachment #1: Type: text/plain, Size: 43 bytes --]
Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
[-- Attachment #2: Type: text/html, Size: 109 bytes --]
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v3 1/2] dts: add dual_vlan testing suite
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
@ 2024-08-13 19:47 ` Dean Marx
2024-09-12 3:46 ` Patrick Robb
2025-01-09 18:01 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Paul Szczepanek
2 siblings, 0 replies; 21+ messages in thread
From: Dean Marx @ 2024-08-13 19:47 UTC (permalink / raw)
To: jspewock
Cc: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, probb,
dev
[-- Attachment #1: Type: text/plain, Size: 43 bytes --]
Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
[-- Attachment #2: Type: text/html, Size: 109 bytes --]
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema
2024-07-24 17:13 ` [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
@ 2024-08-13 19:48 ` Dean Marx
0 siblings, 0 replies; 21+ messages in thread
From: Dean Marx @ 2024-08-13 19:48 UTC (permalink / raw)
To: jspewock
Cc: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, probb,
dev
[-- Attachment #1: Type: text/plain, Size: 43 bytes --]
Reviewed-by: Dean Marx <dmarx@iol.unh.edu>
[-- Attachment #2: Type: text/html, Size: 109 bytes --]
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v3 1/2] dts: add dual_vlan testing suite
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
2024-08-13 19:47 ` Dean Marx
@ 2024-09-12 3:46 ` Patrick Robb
2025-01-09 18:01 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Paul Szczepanek
2 siblings, 0 replies; 21+ messages in thread
From: Patrick Robb @ 2024-09-12 3:46 UTC (permalink / raw)
To: jspewock
Cc: yoan.picchi, paul.szczepanek, juraj.linkes, Luca.Vizzarro,
Honnappa.Nagarahalli, thomas, npratte, wathsala.vithanage, dev
Looks good other than updating the testpmd_shell method names
Reviewed-by: Patrick Robb <probb@iol.unh.edu>
On Wed, Jul 24, 2024 at 1:13 PM <jspewock@iol.unh.edu> wrote:
> +class TestDualVlan(TestSuite):
> + """DPDK Dual VLAN test suite.
> +
> + This suite tests the behavior of VLAN functions and properties in the presence of two VLAN
> + headers. All VLAN functions which are tested in this suite are specified using the inner class
> + :class:`TestCaseOptions` and should have cases for configuring them in
> + :meth:`configure_testpmd` as well as cases for testing their behavior in
> + :meth:`verify_vlan_functions`. Every combination of VLAN functions being enabled should be
> + tested. Additionally, attributes of VLAN headers, such as priority, are tested to ensure they
> + are not modified in the case of two VLAN headers.
> + """
> +
> + class TestCaseOptions(Flag):
> + """Flag for specifying which VLAN functions to configure."""
> +
> + #:
> + VLAN_STRIP = auto()
> + #:
> + VLAN_FILTER_INNER = auto()
> + #:
> + VLAN_FILTER_OUTER = auto()
I think this can now be dropped and we can import VLANOffloadFlag from
testpmd_shell in its place.
> +
> + #: ID to set on inner VLAN tags.
> + inner_vlan_tag: ClassVar[int] = 2
> + #: ID to set on outer VLAN tags.
> + outer_vlan_tag: ClassVar[int] = 1
> + #: ID to use when inserting VLAN tags.
> + vlan_insert_tag: ClassVar[int] = 3
> + #:
> + rx_port: ClassVar[int] = 0
> + #:
> + tx_port: ClassVar[int] = 1
> +
> + def is_relevant_packet(self, pkt: Packet) -> bool:
> + """Check if a packet was sent by functions in this suite.
> +
> + All functions in this test suite send packets with a payload that is packed with 20 "X"
> + characters. This method, therefore, can determine if the packet was sent by this test suite
> + by just checking to see if this payload exists on the received packet.
> +
> + Args:
> + pkt: Packet to check for relevancy.
> +
> + Returns:
> + :data:`True` if the packet contains the expected payload, :data:`False` otherwise.
> + """
> + return hasattr(pkt, "load") and "X" * 20 in str(pkt.load)
> +
> + def pkt_payload_contains_layers(self, pkt: Packet, *expected_layers: Dot1Q) -> bool:
> + """Verify that the payload of the packet matches `expected_layers`.
> +
> + The layers in the payload of `pkt` must match the type and the user-defined fields of the
> + layers in `expected_layers` in order.
> +
> + Args:
> + pkt: Packet to check the payload of.
> + *expected_layers: Layers expected to be in the payload of `pkt`.
> +
> + Returns:
> + :data:`True` if the payload of `pkt` matches the layers in `expected_layers` in order,
> + :data:`False` otherwise.
> + """
> + current_pkt_layer = pkt.payload
> + ret = True
> + for layer in expected_layers:
> + ret &= isinstance(current_pkt_layer, type(layer))
> + if not ret:
> + break
> + for field, val in layer.fields.items():
> + ret &= (
> + hasattr(current_pkt_layer, field) and getattr(current_pkt_layer, field) == val
> + )
> + current_pkt_layer = current_pkt_layer.payload
> + return ret
> +
> + def verify_vlan_functions(self, send_packet: Packet, options: TestCaseOptions) -> None:
> + """Send packet and verify the received packet has the expected structure.
> +
> + Expected structure is defined by `options` according to the following table:
> + +----------------------------------------------+-----------------------+
> + | Configure setting | Result |
> + +=======+=======+========+============+========+=======+=======+=======+
> + | Outer | Inner | Vlan | Vlan | Vlan | Pass/ | Outer | Inner |
> + | vlan | vlan | strip | filter | insert | Drop | vlan | vlan |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | no | no | no | pass | 0x1 | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | yes | no | no | pass | no | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | no | yes,0x1 | no | pass | 0x1 | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | no | yes,0x2 | no | pass | 0x1 | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | no | yes,0x1,0x2| no | pass | 0x1 | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | yes | yes,0x1 | no | pass | no | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | yes | yes,0x2 | no | pass | no | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> + | 0x1 | 0x2 | yes | yes,0x1,0x2| no | pass | no | 0x2 |
> + +-------+-------+--------+------------+--------+-------+-------+-------+
> +
> + Args:
> + send_packet: Packet to send for testing.
> + options: Flag which defines the currents configured settings in testpmd.
> + """
> + recv = self.send_packet_and_capture(send_packet)
> + recv = list(filter(self.is_relevant_packet, recv))
> + expected_layers: list[Packet] = []
> +
> + if self.TestCaseOptions.VLAN_STRIP not in options:
> + expected_layers.append(Dot1Q(vlan=self.outer_vlan_tag))
> + expected_layers.append(Dot1Q(vlan=self.inner_vlan_tag))
> +
> + self.verify(
> + len(recv) > 0,
> + f"Expected to receive packet with the payload {expected_layers} but got nothing.",
> + )
> +
> + for pkt in recv:
> + self.verify(
> + self.pkt_payload_contains_layers(pkt, *expected_layers),
> + f"Received packet ({pkt.summary()}) did not match the expected sequence of layers "
> + f"{expected_layers} with options {options}.",
> + )
> +
> + def configure_testpmd(self, shell: TestPmdShell, options: TestCaseOptions, add: bool) -> None:
> + """Configure VLAN functions in testpmd based on `options`.
> +
> + Args:
> + shell: Testpmd session to configure the settings on. Expected to already be running
> + with all ports stopped before being passed into this function.
Instead of having this requirement that all ports be stopped for the
testpmd session, we can just use testpmd_shell.requires_stopped_ports
decorator now.
> + options: Settings to modify in `shell`.
> + add: If :data:`True`, turn the settings in `options` on, otherwise turn them off.
> + """
> + if (
> + self.TestCaseOptions.VLAN_FILTER_INNER in options
> + or self.TestCaseOptions.VLAN_FILTER_OUTER in options
> + ):
> + if add:
> + # If we are adding a filter, filtering has to be enabled first
> + shell.vlan_filter_set(self.rx_port, True)
> +
> + if self.TestCaseOptions.VLAN_FILTER_INNER in options:
> + shell.rx_vlan(self.inner_vlan_tag, self.rx_port, add)
> + if self.TestCaseOptions.VLAN_FILTER_OUTER in options:
> + shell.rx_vlan(self.outer_vlan_tag, self.rx_port, add)
> +
> + if not add:
> + # If we are removing filters then we need to remove the filters before we can
> + # disable filtering.
> + shell.vlan_filter_set(self.rx_port, False)
> + if self.TestCaseOptions.VLAN_STRIP in options:
> + shell.vlan_strip_set(self.rx_port, add)
> +
> + def test_insert_second_vlan(self) -> None:
> + """Test that a packet with a single VLAN can have an additional one inserted into it."""
> + with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
> + testpmd.port_stop_all()
Should be testpmd.stop_all_ports() now per luca's pktegen/testpmd patch.
> + testpmd.tx_vlan_set(self.tx_port, self.vlan_insert_tag)
> + testpmd.port_start_all()
start_all_ports()
> + testpmd.start()
> + recv = self.send_packet_and_capture(
> + Ether() / Dot1Q(vlan=self.outer_vlan_tag) / Raw("X" * 20)
> + )
> + self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN insertion.")
> + self.verify(
> + any(
> + self.is_relevant_packet(p)
> + and self.pkt_payload_contains_layers(
> + p, *[Dot1Q(vlan=self.vlan_insert_tag), Dot1Q(vlan=self.outer_vlan_tag)]
> + )
> + for p in recv
> + ),
> + "Packet was unable to insert a second VLAN tag.",
> + )
> +
> + def test_all_vlan_functions(self) -> None:
> + """Test that all combinations of :class:`TestCaseOptions` behave as expected.
> +
> + To test this, the base case is tested first, ensuring that a packet with two VLANs is
> + unchanged without the VLAN modification functions enabled. Then the same Testpmd shell is
> + modified to enable all necessary VLAN functions, followed by verification that the
> + functions work as expected, and finally the functions are disabled to allow for a clean
> + environment for the next test.
> + """
> + send_pakt = (
> + Ether()
> + / Dot1Q(vlan=self.outer_vlan_tag)
> + / Dot1Q(vlan=self.inner_vlan_tag)
> + / Raw("X" * 20)
> + )
> + with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
> + testpmd.start()
> + recv = self.send_packet_and_capture(send_pakt)
> + self.verify(len(recv) > 0, "Unmodified packet was not received.")
> + self.verify(
> + any(
> + self.is_relevant_packet(p)
> + and self.pkt_payload_contains_layers(
> + p, *[Dot1Q(vlan=self.outer_vlan_tag), Dot1Q(vlan=self.inner_vlan_tag)]
> + )
> + for p in recv
> + ),
> + "Packet was modified without any VLAN functions applied.",
> + )
> + testpmd.stop()
> + testpmd.port_stop_all()
stop_all_ports(). I'll stop mentioning this below. :)
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
2024-08-13 19:47 ` Dean Marx
2024-09-12 3:46 ` Patrick Robb
@ 2025-01-09 18:01 ` Paul Szczepanek
2025-01-09 18:01 ` [PATCH v4 2/2] dts: add dual VLAN test suite Paul Szczepanek
2025-01-10 11:26 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Luca Vizzarro
2 siblings, 2 replies; 21+ messages in thread
From: Paul Szczepanek @ 2025-01-09 18:01 UTC (permalink / raw)
To: dev; +Cc: Paul Szczepanek
Testpmd functions manipulating the VLAN options require
the ports to be stopped when applying changes.
Signed-off-by: Paul Szczepanek <paul.szczepanek@arm.com>
---
dts/framework/remote_session/testpmd_shell.py | 3 +++
1 file changed, 3 insertions(+)
diff --git a/dts/framework/remote_session/testpmd_shell.py b/dts/framework/remote_session/testpmd_shell.py
index c01ee74b21..aa55bd91d3 100644
--- a/dts/framework/remote_session/testpmd_shell.py
+++ b/dts/framework/remote_session/testpmd_shell.py
@@ -1909,6 +1909,7 @@ def extract_verbose_output(output: str) -> list[TestPmdVerbosePacket]:
out.append(TestPmdVerbosePacket.parse(f"{prev_header}\n{match.group('PACKET')}"))
return out
+ @requires_stopped_ports
def set_vlan_filter(self, port: int, enable: bool, verify: bool = True) -> None:
"""Set vlan filter on.
@@ -1935,6 +1936,7 @@ def set_vlan_filter(self, port: int, enable: bool, verify: bool = True) -> None:
filter on port {port}"""
)
+ @requires_stopped_ports
def rx_vlan(self, vlan: int, port: int, add: bool, verify: bool = True) -> None:
"""Add specified vlan tag to the filter list on a port. Requires vlan filter to be on.
@@ -1964,6 +1966,7 @@ def rx_vlan(self, vlan: int, port: int, add: bool, verify: bool = True) -> None:
f"Testpmd failed to {'add' if add else 'remove'} tag {vlan} on port {port}."
)
+ @requires_stopped_ports
def set_vlan_strip(self, port: int, enable: bool, verify: bool = True) -> None:
"""Enable or disable vlan stripping on the specified port.
--
2.39.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* [PATCH v4 2/2] dts: add dual VLAN test suite
2025-01-09 18:01 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Paul Szczepanek
@ 2025-01-09 18:01 ` Paul Szczepanek
2025-01-10 11:30 ` Luca Vizzarro
2025-01-10 11:26 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Luca Vizzarro
1 sibling, 1 reply; 21+ messages in thread
From: Paul Szczepanek @ 2025-01-09 18:01 UTC (permalink / raw)
To: dev; +Cc: Jeremy Spewock, Paul Szczepanek
From: Jeremy Spewock <jspewock@iol.unh.edu>
Test functionality of VLAN functions such as stripping,
inserting and filtering with two tags present.
Signed-off-by: Jeremy Spewock <jspewock@iol.unh.edu>
Signed-off-by: Paul Szczepanek <paul.szczepanek@arm.com>
---
dts/tests/TestSuite_dual_vlan.py | 287 +++++++++++++++++++++++++++++++
1 file changed, 287 insertions(+)
create mode 100644 dts/tests/TestSuite_dual_vlan.py
diff --git a/dts/tests/TestSuite_dual_vlan.py b/dts/tests/TestSuite_dual_vlan.py
new file mode 100644
index 0000000000..bdbee7e8d1
--- /dev/null
+++ b/dts/tests/TestSuite_dual_vlan.py
@@ -0,0 +1,287 @@
+# SPDX-License-Identifier: BSD-3-Clause
+# Copyright(c) 2024 University of New Hampshire
+
+"""Dual VLAN functionality testing suite.
+
+The main objective of this test suite is to ensure that standard VLAN functions such as stripping
+and filtering both still carry out their expected behavior in the presence of a packet which
+contains two VLAN headers. These functions should carry out said behavior not just in isolation,
+but also when other VLAN functions are configured on the same port. In addition to this, the
+priority attributes of VLAN headers should be unchanged in the case of multiple VLAN headers
+existing on a single packet, and a packet with only a single VLAN header should be able to have one
+additional VLAN inserted into it.
+"""
+
+from enum import Flag, auto
+from typing import ClassVar
+
+from scapy.layers.l2 import Dot1Q, Ether
+from scapy.packet import Packet, Raw
+
+from framework.params.testpmd import SimpleForwardingModes
+from framework.remote_session.testpmd_shell import TestPmdShell
+from framework.test_suite import TestSuite, func_test
+
+
+class TestDualVlan(TestSuite):
+ """DPDK Dual VLAN test suite.
+
+ This suite tests the behavior of VLAN functions and properties in the presence of two VLAN
+ headers. All VLAN functions which are tested in this suite are specified using the inner class
+ :class:`TestCaseOptions` and should have cases for configuring them in
+ :meth:`configure_testpmd` as well as cases for testing their behavior in
+ :meth:`verify_vlan_functions`. Every combination of VLAN functions being enabled should be
+ tested. Additionally, attributes of VLAN headers, such as priority, are tested to ensure they
+ are not modified in the case of two VLAN headers.
+ """
+
+ class TestCaseOptions(Flag):
+ """Flag for specifying which VLAN functions to configure.
+
+ These flags are specific to configuring this testcase and are used to
+ create a matrix of all configuration combinations.
+ """
+
+ #:
+ VLAN_STRIP = auto()
+ #:
+ VLAN_FILTER_INNER = auto()
+ #:
+ VLAN_FILTER_OUTER = auto()
+
+ #: ID to set on inner VLAN tags.
+ inner_vlan_tag: ClassVar[int] = 2
+ #: ID to set on outer VLAN tags.
+ outer_vlan_tag: ClassVar[int] = 1
+ #: ID to use when inserting VLAN tags.
+ vlan_insert_tag: ClassVar[int] = 3
+ #:
+ rx_port: ClassVar[int] = 0
+ #:
+ tx_port: ClassVar[int] = 1
+
+ def is_relevant_packet(self, pkt: Packet) -> bool:
+ """Check if a packet was sent by functions in this suite.
+
+ All functions in this test suite send packets with a payload that is packed with 20 "X"
+ characters. This method, therefore, can determine if the packet was sent by this test suite
+ by just checking to see if this payload exists on the received packet.
+
+ Args:
+ pkt: Packet to check for relevancy.
+
+ Returns:
+ :data:`True` if the packet contains the expected payload, :data:`False` otherwise.
+ """
+ return hasattr(pkt, "load") and "X" * 20 in str(pkt.load)
+
+ def pkt_payload_contains_layers(self, pkt: Packet, *expected_layers: Packet) -> bool:
+ """Verify that the payload of the packet matches `expected_layers`.
+
+ The layers in the payload of `pkt` must match the type and the user-defined fields of the
+ layers in `expected_layers` in order.
+
+ Args:
+ pkt: Packet to check the payload of.
+ *expected_layers: Layers expected to be in the payload of `pkt`.
+
+ Returns:
+ :data:`True` if the payload of `pkt` matches the layers in `expected_layers` in order,
+ :data:`False` otherwise.
+ """
+ current_pkt_layer = pkt.payload
+ ret = True
+ for layer in expected_layers:
+ ret &= isinstance(current_pkt_layer, type(layer))
+ if not ret:
+ break
+ for field, val in layer.fields.items():
+ ret &= (
+ hasattr(current_pkt_layer, field) and getattr(current_pkt_layer, field) == val
+ )
+ current_pkt_layer = current_pkt_layer.payload
+ return ret
+
+ def verify_vlan_functions(self, send_packet: Packet, options: TestCaseOptions) -> None:
+ """Send packet and verify the received packet has the expected structure.
+
+ Expected structure is defined by `options` according to the following table:
+ +----------------------------------------------+-----------------------+
+ | Configure setting | Result |
+ +=======+=======+========+============+========+=======+=======+=======+
+ | Outer | Inner | VLAN | VLAN | VLAN | Pass/ | Outer | Inner |
+ | vlan | vlan | strip | filter | insert | Drop | vlan | vlan |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | no | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | no | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x2 | no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | no | yes,0x1,0x2| no | pass | 0x1 | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x2 | no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+ | 0x1 | 0x2 | yes | yes,0x1,0x2| no | pass | no | 0x2 |
+ +-------+-------+--------+------------+--------+-------+-------+-------+
+
+ Args:
+ send_packet: Packet to send for testing.
+ options: Flag which defines the currents configured settings in testpmd.
+ """
+ recv = self.send_packet_and_capture(send_packet)
+ recv = list(filter(self.is_relevant_packet, recv))
+ expected_layers: list[Packet] = []
+
+ if self.TestCaseOptions.VLAN_STRIP not in options:
+ expected_layers.append(Dot1Q(vlan=self.outer_vlan_tag))
+ expected_layers.append(Dot1Q(vlan=self.inner_vlan_tag))
+
+ self.verify(
+ len(recv) > 0,
+ f"Expected to receive packet with the payload {expected_layers} but got nothing.",
+ )
+
+ for pkt in recv:
+ self.verify(
+ self.pkt_payload_contains_layers(pkt, *expected_layers),
+ f"Received packet ({pkt.summary()}) did not match the expected sequence of layers "
+ f"{expected_layers} with options {options}.",
+ )
+
+ def configure_testpmd(self, shell: TestPmdShell, options: TestCaseOptions, add: bool) -> None:
+ """Configure VLAN functions in testpmd based on `options`.
+
+ Args:
+ shell: Testpmd session to configure the settings on.
+ options: Settings to modify in `shell`.
+ add: If :data:`True`, turn the settings in `options` on, otherwise turn them off.
+ """
+ if (
+ self.TestCaseOptions.VLAN_FILTER_INNER in options
+ or self.TestCaseOptions.VLAN_FILTER_OUTER in options
+ ):
+ if add:
+ # If we are adding a filter, filtering has to be enabled first
+ shell.set_vlan_filter(self.rx_port, True)
+
+ if self.TestCaseOptions.VLAN_FILTER_INNER in options:
+ shell.rx_vlan(self.inner_vlan_tag, self.rx_port, add)
+ if self.TestCaseOptions.VLAN_FILTER_OUTER in options:
+ shell.rx_vlan(self.outer_vlan_tag, self.rx_port, add)
+
+ if not add:
+ # If we are removing filters then we need to remove the filters before we can
+ # disable filtering.
+ shell.set_vlan_filter(self.rx_port, False)
+ if self.TestCaseOptions.VLAN_STRIP in options:
+ shell.set_vlan_strip(self.rx_port, add)
+
+ @func_test
+ def insert_second_vlan(self) -> None:
+ """Test that a packet with a single VLAN can have an additional one inserted into it.
+
+ Steps:
+ Set VLAN tag on the tx port.
+ Create a packet with a VLAN tag.
+ Send and receive the packet.
+ Verify:
+ Packets are received.
+ Packet contains two VLAN tags.
+ """
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.tx_vlan_set(port=self.tx_port, enable=True, vlan=self.vlan_insert_tag)
+ testpmd.start()
+ recv = self.send_packet_and_capture(
+ Ether() / Dot1Q(vlan=self.outer_vlan_tag) / Raw(b"X" * 20)
+ )
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN insertion.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, Dot1Q(vlan=self.vlan_insert_tag), Dot1Q(vlan=self.outer_vlan_tag)
+ )
+ for p in recv
+ ),
+ "Packet was unable to insert a second VLAN tag.",
+ )
+
+ @func_test
+ def all_vlan_functions(self) -> None:
+ """Test that all combinations of :class:`TestCaseOptions` behave as expected.
+
+ Steps:
+ Send packets with VLAN functions disabled.
+ Send packets with a set of combinations of VLAN functions enabled.
+ Disable VLAN function to allow for a clean environment for the next test.
+ Verify:
+ Packet with two VLANs is unchanged without the VLAN modification functions enabled.
+ VLAN functions work as expected.
+ """
+ send_pkt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag)
+ / Dot1Q(vlan=self.inner_vlan_tag)
+ / Raw(b"X" * 20)
+ )
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.start()
+ recv = self.send_packet_and_capture(send_pkt)
+ self.verify(len(recv) > 0, "Unmodified packet was not received.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p, Dot1Q(vlan=self.outer_vlan_tag), Dot1Q(vlan=self.inner_vlan_tag)
+ )
+ for p in recv
+ ),
+ "Packet was modified without any VLAN functions applied.",
+ )
+ testpmd.stop()
+ for i in range(2 ** len(self.TestCaseOptions)):
+ options = self.TestCaseOptions(i)
+ self.configure_testpmd(testpmd, options, True)
+ testpmd.start()
+ self.verify_vlan_functions(send_pkt, options)
+ testpmd.stop()
+ self.configure_testpmd(testpmd, options, False)
+
+ @func_test
+ def maintains_priority(self) -> None:
+ """Test that priorities of multiple VLAN tags are preserved by the PMD.
+
+ Steps:
+ Create packets with VLAN tags with priorities.
+ Send and receive packets.
+ Verify:
+ Packets are received.
+ Priorities are unchanged.
+ """
+ pkt = (
+ Ether()
+ / Dot1Q(vlan=self.outer_vlan_tag, prio=1)
+ / Dot1Q(vlan=self.inner_vlan_tag, prio=2)
+ / Raw(b"X" * 20)
+ )
+ with TestPmdShell(self.sut_node, forward_mode=SimpleForwardingModes.mac) as testpmd:
+ testpmd.start()
+ recv = self.send_packet_and_capture(pkt)
+ self.verify(len(recv) > 0, "Did not receive any packets when testing VLAN priority.")
+ self.verify(
+ any(
+ self.is_relevant_packet(p)
+ and self.pkt_payload_contains_layers(
+ p,
+ Dot1Q(vlan=self.outer_vlan_tag, prio=1),
+ Dot1Q(vlan=self.inner_vlan_tag, prio=2),
+ )
+ for p in recv
+ ),
+ "VLAN headers did not maintain their priorities.",
+ )
--
2.39.2
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions
2025-01-09 18:01 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Paul Szczepanek
2025-01-09 18:01 ` [PATCH v4 2/2] dts: add dual VLAN test suite Paul Szczepanek
@ 2025-01-10 11:26 ` Luca Vizzarro
1 sibling, 0 replies; 21+ messages in thread
From: Luca Vizzarro @ 2025-01-10 11:26 UTC (permalink / raw)
To: Paul Szczepanek, dev
Reviewed-by: Luca Vizzarro <luca.vizzarro@arm.com>
^ permalink raw reply [flat|nested] 21+ messages in thread
* Re: [PATCH v4 2/2] dts: add dual VLAN test suite
2025-01-09 18:01 ` [PATCH v4 2/2] dts: add dual VLAN test suite Paul Szczepanek
@ 2025-01-10 11:30 ` Luca Vizzarro
0 siblings, 0 replies; 21+ messages in thread
From: Luca Vizzarro @ 2025-01-10 11:30 UTC (permalink / raw)
To: Paul Szczepanek, dev; +Cc: Jeremy Spewock
Reviewed-by: Luca Vizzarro <luca.vizzarro@arm.com>
^ permalink raw reply [flat|nested] 21+ messages in thread
end of thread, other threads:[~2025-01-10 11:30 UTC | newest]
Thread overview: 21+ messages (download: mbox.gz / follow: Atom feed)
-- links below jump to the message on this page --
2024-07-15 19:58 [PATCH v1 0/3] dts: add test suite for dual VLANs jspewock
2024-07-15 19:58 ` [PATCH v1 1/3] dts: fix Testpmd function for resetting VLAN insertion jspewock
2024-07-15 19:58 ` [PATCH v1 2/3] dts: add dual_vlan testing suite jspewock
2024-07-22 17:38 ` Dean Marx
2024-07-25 13:56 ` Jeremy Spewock
2024-07-31 13:54 ` Jeremy Spewock
2024-07-15 19:58 ` [PATCH v1 3/3] dts: add dual_vlan test suite to the yaml schema jspewock
2024-07-16 21:24 ` [PATCH v2 0/2] dts: add test suite for dual VLANs jspewock
2024-07-16 21:24 ` [PATCH v2 1/2] dts: add dual_vlan testing suite jspewock
2024-08-07 19:07 ` Dean Marx
2024-07-16 21:24 ` [PATCH v2 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
2024-07-24 17:13 ` [PATCH v3 0/2] dts: add test suite for dual VLANs jspewock
2024-07-24 17:13 ` [PATCH v3 1/2] dts: add dual_vlan testing suite jspewock
2024-08-13 19:47 ` Dean Marx
2024-09-12 3:46 ` Patrick Robb
2025-01-09 18:01 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Paul Szczepanek
2025-01-09 18:01 ` [PATCH v4 2/2] dts: add dual VLAN test suite Paul Szczepanek
2025-01-10 11:30 ` Luca Vizzarro
2025-01-10 11:26 ` [PATCH v4 1/2] dts: add stop ports decorator for VLAN functions Luca Vizzarro
2024-07-24 17:13 ` [PATCH v3 2/2] dts: add dual_vlan test suite to the yaml schema jspewock
2024-08-13 19:48 ` Dean Marx
This is a public inbox, see mirroring instructions
for how to clone and mirror all data and code used for this inbox;
as well as URLs for NNTP newsgroup(s).