| #!/bin/bash |
| # SPDX-License-Identifier: GPL-2.0 |
| |
| ALL_TESTS="standalone two_bridges one_bridge_two_pvids" |
| NUM_NETIFS=4 |
| |
| source lib.sh |
| |
| h1=${NETIFS[p1]} |
| h2=${NETIFS[p3]} |
| swp1=${NETIFS[p2]} |
| swp2=${NETIFS[p4]} |
| |
| H1_IPV4="192.0.2.1" |
| H2_IPV4="192.0.2.2" |
| H1_IPV6="2001:db8:1::1" |
| H2_IPV6="2001:db8:1::2" |
| |
| IPV4_ALLNODES="224.0.0.1" |
| IPV6_ALLNODES="ff02::1" |
| MACV4_ALLNODES="01:00:5e:00:00:01" |
| MACV6_ALLNODES="33:33:00:00:00:01" |
| NON_IP_MC="01:02:03:04:05:06" |
| NON_IP_PKT="00:04 48:45:4c:4f" |
| BC="ff:ff:ff:ff:ff:ff" |
| |
| # The full 4K VLAN space is too much to check, so strategically pick some |
| # values which should provide reasonable coverage |
| vids=(0 1 2 5 10 20 50 100 200 500 1000 1000 2000 4000 4094) |
| |
| send_non_ip() |
| { |
| local if_name=$1 |
| local smac=$2 |
| local dmac=$3 |
| |
| $MZ -q $if_name "$dmac $smac $NON_IP_PKT" |
| } |
| |
| send_uc_ipv4() |
| { |
| local if_name=$1 |
| local dmac=$2 |
| |
| ip neigh add $H2_IPV4 lladdr $dmac dev $if_name |
| ping_do $if_name $H2_IPV4 |
| ip neigh del $H2_IPV4 dev $if_name |
| } |
| |
| send_mc_ipv4() |
| { |
| local if_name=$1 |
| |
| ping_do $if_name $IPV4_ALLNODES "-I $if_name" |
| } |
| |
| send_uc_ipv6() |
| { |
| local if_name=$1 |
| local dmac=$2 |
| |
| ip -6 neigh add $H2_IPV6 lladdr $dmac dev $if_name |
| ping6_do $if_name $H2_IPV6 |
| ip -6 neigh del $H2_IPV6 dev $if_name |
| } |
| |
| send_mc_ipv6() |
| { |
| local if_name=$1 |
| |
| ping6_do $if_name $IPV6_ALLNODES%$if_name |
| } |
| |
| check_rcv() |
| { |
| local if_name=$1 |
| local type=$2 |
| local pattern=$3 |
| local should_fail=1 |
| |
| RET=0 |
| |
| tcpdump_show $if_name | grep -q "$pattern" |
| |
| check_err_fail "$should_fail" "$?" "reception" |
| |
| log_test "$type" |
| } |
| |
| run_test() |
| { |
| local test_name="$1" |
| local smac=$(mac_get $h1) |
| local dmac=$(mac_get $h2) |
| local h1_ipv6_lladdr=$(ipv6_lladdr_get $h1) |
| local vid= |
| |
| echo "$test_name: Sending packets" |
| |
| tcpdump_start $h2 |
| |
| send_non_ip $h1 $smac $dmac |
| send_non_ip $h1 $smac $NON_IP_MC |
| send_non_ip $h1 $smac $BC |
| send_uc_ipv4 $h1 $dmac |
| send_mc_ipv4 $h1 |
| send_uc_ipv6 $h1 $dmac |
| send_mc_ipv6 $h1 |
| |
| for vid in "${vids[@]}"; do |
| vlan_create $h1 $vid |
| simple_if_init $h1.$vid $H1_IPV4/24 $H1_IPV6/64 |
| |
| send_non_ip $h1.$vid $smac $dmac |
| send_non_ip $h1.$vid $smac $NON_IP_MC |
| send_non_ip $h1.$vid $smac $BC |
| send_uc_ipv4 $h1.$vid $dmac |
| send_mc_ipv4 $h1.$vid |
| send_uc_ipv6 $h1.$vid $dmac |
| send_mc_ipv6 $h1.$vid |
| |
| simple_if_fini $h1.$vid $H1_IPV4/24 $H1_IPV6/64 |
| vlan_destroy $h1 $vid |
| done |
| |
| sleep 1 |
| |
| echo "$test_name: Checking which packets were received" |
| |
| tcpdump_stop $h2 |
| |
| check_rcv $h2 "$test_name: Unicast non-IP untagged" \ |
| "$smac > $dmac, 802.3, length 4:" |
| |
| check_rcv $h2 "$test_name: Multicast non-IP untagged" \ |
| "$smac > $NON_IP_MC, 802.3, length 4:" |
| |
| check_rcv $h2 "$test_name: Broadcast non-IP untagged" \ |
| "$smac > $BC, 802.3, length 4:" |
| |
| check_rcv $h2 "$test_name: Unicast IPv4 untagged" \ |
| "$smac > $dmac, ethertype IPv4 (0x0800)" |
| |
| check_rcv $h2 "$test_name: Multicast IPv4 untagged" \ |
| "$smac > $MACV4_ALLNODES, ethertype IPv4 (0x0800).*: $H1_IPV4 > $IPV4_ALLNODES" |
| |
| check_rcv $h2 "$test_name: Unicast IPv6 untagged" \ |
| "$smac > $dmac, ethertype IPv6 (0x86dd).*8: $H1_IPV6 > $H2_IPV6" |
| |
| check_rcv $h2 "$test_name: Multicast IPv6 untagged" \ |
| "$smac > $MACV6_ALLNODES, ethertype IPv6 (0x86dd).*: $h1_ipv6_lladdr > $IPV6_ALLNODES" |
| |
| for vid in "${vids[@]}"; do |
| check_rcv $h2 "$test_name: Unicast non-IP VID $vid" \ |
| "$smac > $dmac, ethertype 802.1Q (0x8100).*vlan $vid,.*length 4" |
| |
| check_rcv $h2 "$test_name: Multicast non-IP VID $vid" \ |
| "$smac > $NON_IP_MC, ethertype 802.1Q (0x8100).*vlan $vid,.*length 4" |
| |
| check_rcv $h2 "$test_name: Broadcast non-IP VID $vid" \ |
| "$smac > $BC, ethertype 802.1Q (0x8100).*vlan $vid,.*length 4" |
| |
| check_rcv $h2 "$test_name: Unicast IPv4 VID $vid" \ |
| "$smac > $dmac, ethertype 802.1Q (0x8100).*vlan $vid,.*ethertype IPv4 (0x0800), $H1_IPV4 > $H2_IPV4" |
| |
| check_rcv $h2 "$test_name: Multicast IPv4 VID $vid" \ |
| "$smac > $MACV4_ALLNODES, ethertype 802.1Q (0x8100).*vlan $vid,.*ethertype IPv4 (0x0800), $H1_IPV4 > $IPV4_ALLNODES" |
| |
| check_rcv $h2 "$test_name: Unicast IPv6 VID $vid" \ |
| "$smac > $dmac, ethertype 802.1Q (0x8100).*vlan $vid,.*ethertype IPv6 (0x86dd), $H1_IPV6 > $H2_IPV6" |
| |
| check_rcv $h2 "$test_name: Multicast IPv6 VID $vid" \ |
| "$smac > $MACV6_ALLNODES, ethertype 802.1Q (0x8100).*vlan $vid,.*ethertype IPv6 (0x86dd), $h1_ipv6_lladdr > $IPV6_ALLNODES" |
| done |
| |
| tcpdump_cleanup $h2 |
| } |
| |
| standalone() |
| { |
| run_test "Standalone switch ports" |
| } |
| |
| two_bridges() |
| { |
| ip link add br0 type bridge && ip link set br0 up |
| ip link add br1 type bridge && ip link set br1 up |
| ip link set $swp1 master br0 |
| ip link set $swp2 master br1 |
| |
| run_test "Switch ports in different bridges" |
| |
| ip link del br1 |
| ip link del br0 |
| } |
| |
| one_bridge_two_pvids() |
| { |
| ip link add br0 type bridge vlan_filtering 1 vlan_default_pvid 0 |
| ip link set br0 up |
| ip link set $swp1 master br0 |
| ip link set $swp2 master br0 |
| |
| bridge vlan add dev $swp1 vid 1 pvid untagged |
| bridge vlan add dev $swp1 vid 2 pvid untagged |
| |
| run_test "Switch ports in VLAN-aware bridge with different PVIDs" |
| |
| ip link del br0 |
| } |
| |
| h1_create() |
| { |
| simple_if_init $h1 $H1_IPV4/24 $H1_IPV6/64 |
| } |
| |
| h1_destroy() |
| { |
| simple_if_fini $h1 $H1_IPV4/24 $H1_IPV6/64 |
| } |
| |
| h2_create() |
| { |
| simple_if_init $h2 $H2_IPV4/24 $H2_IPV6/64 |
| } |
| |
| h2_destroy() |
| { |
| simple_if_fini $h2 $H2_IPV4/24 $H2_IPV6/64 |
| } |
| |
| cleanup() |
| { |
| pre_cleanup |
| |
| h2_destroy |
| h1_destroy |
| |
| vrf_cleanup |
| } |
| |
| setup_prepare() |
| { |
| vrf_prepare |
| |
| h1_create |
| h2_create |
| # we call simple_if_init from the test itself, but setup_wait expects |
| # that we call it from here, and waits until the interfaces are up |
| ip link set dev $swp1 up |
| ip link set dev $swp2 up |
| } |
| |
| trap cleanup EXIT |
| |
| setup_prepare |
| setup_wait |
| |
| tests_run |
| |
| exit $EXIT_STATUS |