diff --git a/tests/sys/net/Makefile b/tests/sys/net/Makefile index 40b3a5e5fff6..4771040816b7 100644 --- a/tests/sys/net/Makefile +++ b/tests/sys/net/Makefile @@ -1,33 +1,36 @@ # $FreeBSD$ PACKAGE= tests TESTSDIR= ${TESTSBASE}/sys/net BINDIR= ${TESTSDIR} ATF_TESTS_C+= if_epair ATF_TESTS_SH+= if_bridge_test TEST_METADATA.if_bridge_test+= required_programs="python" ATF_TESTS_SH+= if_clone_test ATF_TESTS_SH+= if_lagg_test ATF_TESTS_SH+= if_tun_test ATF_TESTS_SH+= if_vlan TESTS_SUBDIRS+= routing # The tests are written to be run in parallel, but doing so leads to random # panics. I think it's because the kernel's list of interfaces isn't properly # locked. TEST_METADATA+= is_exclusive=true ${PACKAGE}FILES+= \ + dhclient_pcp.conf \ + pcp.py \ stp.py +${PACKAGE}FILESMODE_pcp.py= 0555 ${PACKAGE}FILESMODE_stp.py= 0555 MAN= PROGS+= randsleep CFLAGS+= -I${.CURDIR:H:H} .include diff --git a/tests/sys/net/dhclient_pcp.conf b/tests/sys/net/dhclient_pcp.conf new file mode 100644 index 000000000000..fbd86e5bd0f8 --- /dev/null +++ b/tests/sys/net/dhclient_pcp.conf @@ -0,0 +1 @@ +vlan-pcp 6; diff --git a/tests/sys/net/if_vlan.sh b/tests/sys/net/if_vlan.sh index 2edcb16eab88..517206cb55c7 100755 --- a/tests/sys/net/if_vlan.sh +++ b/tests/sys/net/if_vlan.sh @@ -1,220 +1,267 @@ # $FreeBSD$ . $(atf_get_srcdir)/../common/vnet.subr atf_test_case "basic" "cleanup" basic_head() { atf_set descr 'Basic VLAN test' atf_set require.user root } basic_body() { vnet_init epair_vlan=$(vnet_mkepair) vnet_mkjail alcatraz ${epair_vlan}a vnet_mkjail singsing ${epair_vlan}b vlan0=$(jexec alcatraz ifconfig vlan create vlandev ${epair_vlan}a \ vlan 42) jexec alcatraz ifconfig ${epair_vlan}a up jexec alcatraz ifconfig ${vlan0} 10.0.0.1/24 up vlan1=$(jexec singsing ifconfig vlan create vlandev ${epair_vlan}b \ vlan 42) jexec singsing ifconfig ${epair_vlan}b up jexec singsing ifconfig ${vlan1} 10.0.0.2/24 up atf_check -s exit:0 -o ignore jexec singsing ping -c 1 10.0.0.1 } basic_cleanup() { vnet_cleanup } # Simple Q-in-Q (802.1Q over 802.1ad) atf_test_case "qinq_simple" "cleanup" qinq_simple_head() { atf_set descr 'Simple Q-in-Q test (802.1Q over 802.1ad)' atf_set require.user root } qinq_simple_body() { vnet_init epair_qinq=$(vnet_mkepair) vnet_mkjail jqinq0 ${epair_qinq}a vnet_mkjail jqinq1 ${epair_qinq}b vlan5a=$(jexec jqinq0 ifconfig vlan create \ vlandev ${epair_qinq}a vlan 5 vlanproto 802.1ad) vlan42a=$(jexec jqinq0 ifconfig vlan create \ vlandev ${vlan5a} vlan 42 vlanproto 802.1q) jexec jqinq0 ifconfig ${epair_qinq}a up jexec jqinq0 ifconfig ${vlan5a} up jexec jqinq0 ifconfig ${vlan42a} 10.5.42.1/24 up vlan5b=$(jexec jqinq1 ifconfig vlan create \ vlandev ${epair_qinq}b vlan 5 vlanproto 802.1ad) vlan42b=$(jexec jqinq1 ifconfig vlan create \ vlandev ${vlan5b} vlan 42 vlanproto 802.1q) jexec jqinq1 ifconfig ${epair_qinq}b up jexec jqinq1 ifconfig ${vlan5b} up jexec jqinq1 ifconfig ${vlan42b} 10.5.42.2/24 up atf_check -s exit:0 -o ignore jexec jqinq1 ping -c 1 10.5.42.1 } qinq_simple_cleanup() { vnet_cleanup } # Deep Q-in-Q (802.1Q over 802.1ad over 802.1ad) atf_test_case "qinq_deep" "cleanup" qinq_deep_head() { atf_set descr 'Deep Q-in-Q test (802.1Q over 802.1ad over 802.1ad)' atf_set require.user root } qinq_deep_body() { vnet_init epair_qinq=$(vnet_mkepair) vnet_mkjail jqinq2 ${epair_qinq}a vnet_mkjail jqinq3 ${epair_qinq}b vlan5a=$(jexec jqinq2 ifconfig vlan create \ vlandev ${epair_qinq}a vlan 5 vlanproto 802.1ad) vlan6a=$(jexec jqinq2 ifconfig vlan create \ vlandev ${vlan5a} vlan 6 vlanproto 802.1ad) vlan42a=$(jexec jqinq2 ifconfig vlan create \ vlandev ${vlan6a} vlan 42 vlanproto 802.1q) jexec jqinq2 ifconfig ${epair_qinq}a up jexec jqinq2 ifconfig ${vlan5a} up jexec jqinq2 ifconfig ${vlan6a} up jexec jqinq2 ifconfig ${vlan42a} 10.6.42.1/24 up vlan5b=$(jexec jqinq3 ifconfig vlan create \ vlandev ${epair_qinq}b vlan 5 vlanproto 802.1ad) vlan6b=$(jexec jqinq3 ifconfig vlan create \ vlandev ${vlan5b} vlan 6 vlanproto 802.1ad) vlan42b=$(jexec jqinq3 ifconfig vlan create \ vlandev ${vlan6b} vlan 42 vlanproto 802.1q) jexec jqinq3 ifconfig ${epair_qinq}b up jexec jqinq3 ifconfig ${vlan5b} up jexec jqinq3 ifconfig ${vlan6b} up jexec jqinq3 ifconfig ${vlan42b} 10.6.42.2/24 up atf_check -s exit:0 -o ignore jexec jqinq3 ping -c 1 10.6.42.1 } qinq_deep_cleanup() { vnet_cleanup } # Legacy Q-in-Q (802.1Q over 802.1Q) atf_test_case "qinq_legacy" "cleanup" qinq_legacy_head() { atf_set descr 'Legacy Q-in-Q test (802.1Q over 802.1Q)' atf_set require.user root } qinq_legacy_body() { vnet_init epair_qinq=$(vnet_mkepair) vnet_mkjail jqinq4 ${epair_qinq}a vnet_mkjail jqinq5 ${epair_qinq}b vlan5a=$(jexec jqinq4 ifconfig vlan create \ vlandev ${epair_qinq}a vlan 5) vlan42a=$(jexec jqinq4 ifconfig vlan create \ vlandev ${vlan5a} vlan 42) jexec jqinq4 ifconfig ${epair_qinq}a up jexec jqinq4 ifconfig ${vlan5a} up jexec jqinq4 ifconfig ${vlan42a} 10.5.42.1/24 up vlan5b=$(jexec jqinq5 ifconfig vlan create \ vlandev ${epair_qinq}b vlan 5) vlan42b=$(jexec jqinq5 ifconfig vlan create \ vlandev ${vlan5b} vlan 42) jexec jqinq5 ifconfig ${epair_qinq}b up jexec jqinq5 ifconfig ${vlan5b} up jexec jqinq5 ifconfig ${vlan42b} 10.5.42.2/24 up atf_check -s exit:0 -o ignore jexec jqinq5 ping -c 1 10.5.42.1 } qinq_legacy_cleanup() { vnet_cleanup } # Simple Q-in-Q with dot notation atf_test_case "qinq_dot" "cleanup" qinq_dot_head() { atf_set descr 'Simple Q-in-Q test with dot notation' atf_set require.user root } qinq_dot_body() { vnet_init epair_qinq=$(vnet_mkepair) vnet_mkjail jqinq6 ${epair_qinq}a vnet_mkjail jqinq7 ${epair_qinq}b jexec jqinq6 ifconfig vlan5 create \ vlandev ${epair_qinq}a vlan 5 vlanproto 802.1ad jexec jqinq6 ifconfig vlan5.42 create \ vlanproto 802.1q jexec jqinq6 ifconfig ${epair_qinq}a up jexec jqinq6 ifconfig vlan5 up jexec jqinq6 ifconfig vlan5.42 10.5.42.1/24 up vlan5b=$(jexec jqinq7 ifconfig vlan create \ vlandev ${epair_qinq}b vlan 5 vlanproto 802.1ad) vlan42b=$(jexec jqinq7 ifconfig vlan create \ vlandev ${vlan5b} vlan 42 vlanproto 802.1q) jexec jqinq7 ifconfig ${epair_qinq}b up jexec jqinq7 ifconfig ${vlan5b} up jexec jqinq7 ifconfig ${vlan42b} 10.5.42.2/24 up atf_check -s exit:0 -o ignore jexec jqinq7 ping -c 1 10.5.42.1 } qinq_dot_cleanup() { vnet_cleanup } +atf_test_case "bpf_pcp" "cleanup" +bpf_pcp_head() +{ + atf_set descr 'Set VLAN PCP through BPF' + atf_set require.config 'allow_sysctl_side_effects' + atf_set require.user root + atf_set require.progs scapy +} + +bpf_pcp_body() +{ + vnet_init + + epair=$(vnet_mkepair) + + ifconfig ${epair}a up + + vnet_mkjail alcatraz ${epair}b + vlan=$(jexec alcatraz ifconfig vlan create) + jexec alcatraz ifconfig ${vlan} vlan 42 vlandev ${epair}b + jexec alcatraz ifconfig ${vlan} up + jexec alcatraz ifconfig ${epair}b up + + sysctl net.link.vlan.mtag_pcp=1 + + jexec alcatraz dhclient ${vlan} & + atf_check -s exit:1 -o ignore -e ignore $(atf_get_srcdir)/pcp.py \ + --expect-pcp 6 \ + --recvif ${epair}a + + jexec alcatraz killall dhclient + sleep 1 + + jexec alcatraz dhclient -c $(atf_get_srcdir)/dhclient_pcp.conf ${vlan} & + atf_check -s exit:0 -o ignore -e ignore $(atf_get_srcdir)/pcp.py \ + --expect-pcp 6 \ + --recvif ${epair}a +} + +bpf_pcp_cleanup() +{ + sysctl net.link.vlan.mtag_pcp=0 + jexec alcatraz killall dhclient + vnet_cleanup +} + atf_init_test_cases() { atf_add_test_case "basic" atf_add_test_case "qinq_simple" atf_add_test_case "qinq_deep" atf_add_test_case "qinq_legacy" atf_add_test_case "qinq_dot" + atf_add_test_case "bpf_pcp" } diff --git a/tests/sys/netpfil/common/sniffer.py b/tests/sys/net/pcp.py similarity index 53% copy from tests/sys/netpfil/common/sniffer.py copy to tests/sys/net/pcp.py index 200ac750dd7f..cea88faaf438 100644 --- a/tests/sys/netpfil/common/sniffer.py +++ b/tests/sys/net/pcp.py @@ -1,54 +1,74 @@ -# $FreeBSD$ +#!/usr/bin/env python3 # -# SPDX-License-Identifier: BSD-2-Clause-FreeBSD +# SPDX-License-Identifier: BSD-2-Clause # -# Copyright (c) 2017 Kristof Provost +# Copyright (c) 2021 Rubicon Communications, LLC (Netgate). # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF # SUCH DAMAGE. -# -import threading +import argparse +import logging +logging.getLogger("scapy").setLevel(logging.CRITICAL) import scapy.all as sp +import sys +import os +curdir = os.path.dirname(os.path.realpath(__file__)) +netpfil_common = curdir + "/../netpfil/common" +sys.path.append(netpfil_common) +from sniffer import Sniffer + +def check_pcp(args, packet): + vlan = packet.getlayer(sp.Dot1Q) + + if vlan is None: + return False + + if not packet.getlayer(sp.BOOTP): + return False + + if vlan.prio == int(args.expect_pcp[0]): + return True + + return False + +def main(): + parser = argparse.ArgumentParser("pcp.py", + description="PCP test tool") + parser.add_argument('--recvif', nargs=1, + required=True, + help='The interface where to look for packets to check') + parser.add_argument('--expect-pcp', nargs=1, + help='The expected PCP value on VLAN packets') + + args = parser.parse_args() + + sniffer = Sniffer(args, check_pcp, recvif=args.recvif[0], timeout=20) + + sniffer.join() + + if sniffer.foundCorrectPacket: + sys.exit(0) + + sys.exit(1) -class Sniffer(threading.Thread): - def __init__(self, args, check_function, recvif=None): - threading.Thread.__init__(self) - - self._args = args - if recvif is not None: - self._recvif = recvif - else: - self._recvif = args.recvif[0] - self._check_function = check_function - self.foundCorrectPacket = False - - self.start() - - def _checkPacket(self, packet): - ret = self._check_function(self._args, packet) - if ret: - self.foundCorrectPacket = True - return ret - - def run(self): - self.packets = sp.sniff(iface=self._recvif, - stop_filter=self._checkPacket, timeout=3) +if __name__ == '__main__': + main() diff --git a/tests/sys/netpfil/common/sniffer.py b/tests/sys/netpfil/common/sniffer.py index 200ac750dd7f..a7ebfc122573 100644 --- a/tests/sys/netpfil/common/sniffer.py +++ b/tests/sys/netpfil/common/sniffer.py @@ -1,54 +1,60 @@ # $FreeBSD$ # # SPDX-License-Identifier: BSD-2-Clause-FreeBSD # # Copyright (c) 2017 Kristof Provost # # Redistribution and use in source and binary forms, with or without # modification, are permitted provided that the following conditions # are met: # 1. Redistributions of source code must retain the above copyright # notice, this list of conditions and the following disclaimer. # 2. Redistributions in binary form must reproduce the above copyright # notice, this list of conditions and the following disclaimer in the # documentation and/or other materials provided with the distribution. # # THIS SOFTWARE IS PROVIDED BY THE AUTHOR AND CONTRIBUTORS ``AS IS'' AND # ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE # IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE # ARE DISCLAIMED. IN NO EVENT SHALL THE AUTHOR OR CONTRIBUTORS BE LIABLE # FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL # DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS # OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) # HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT # LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY # OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF # SUCH DAMAGE. # import threading import scapy.all as sp +import sys class Sniffer(threading.Thread): - def __init__(self, args, check_function, recvif=None): + def __init__(self, args, check_function, recvif=None, timeout=3): threading.Thread.__init__(self) self._args = args + self._timeout = timeout if recvif is not None: self._recvif = recvif else: self._recvif = args.recvif[0] self._check_function = check_function self.foundCorrectPacket = False self.start() def _checkPacket(self, packet): ret = self._check_function(self._args, packet) if ret: self.foundCorrectPacket = True return ret def run(self): - self.packets = sp.sniff(iface=self._recvif, - stop_filter=self._checkPacket, timeout=3) + self.packets = [] + try: + self.packets = sp.sniff(iface=self._recvif, + stop_filter=self._checkPacket, timeout=self._timeout) + except Exception as e: + print(e, file=sys.stderr)