| #!/bin/bash |
| # SPDX-License-Identifier: GPL-2.0 |
| # Copyright 2020 NXP |
| |
| WAIT_TIME=1 |
| NUM_NETIFS=4 |
| STABLE_MAC_ADDRS=yes |
| lib_dir=$(dirname $0)/../../../net/forwarding |
| source $lib_dir/tc_common.sh |
| source $lib_dir/lib.sh |
| |
| require_command tcpdump |
| |
| h1=${NETIFS[p1]} |
| swp1=${NETIFS[p2]} |
| swp2=${NETIFS[p3]} |
| h2=${NETIFS[p4]} |
| |
| # Helpers to map a VCAP IS1 and VCAP IS2 lookup and policy to a chain number |
| # used by the kernel driver. The numbers are: |
| # VCAP IS1 lookup 0: 10000 |
| # VCAP IS1 lookup 1: 11000 |
| # VCAP IS1 lookup 2: 12000 |
| # VCAP IS2 lookup 0 policy 0: 20000 |
| # VCAP IS2 lookup 0 policy 1: 20001 |
| # VCAP IS2 lookup 0 policy 255: 20255 |
| # VCAP IS2 lookup 1 policy 0: 21000 |
| # VCAP IS2 lookup 1 policy 1: 21001 |
| # VCAP IS2 lookup 1 policy 255: 21255 |
| IS1() |
| { |
| local lookup=$1 |
| |
| echo $((10000 + 1000 * lookup)) |
| } |
| |
| IS2() |
| { |
| local lookup=$1 |
| local pag=$2 |
| |
| echo $((20000 + 1000 * lookup + pag)) |
| } |
| |
| ES0() |
| { |
| echo 0 |
| } |
| |
| # The Ocelot switches have a fixed ingress pipeline composed of: |
| # |
| # +----------------------------------------------+ +-----------------------------------------+ |
| # | VCAP IS1 | | VCAP IS2 | |
| # | | | | |
| # | +----------+ +----------+ +----------+ | | +----------+ +----------+ | |
| # | | Lookup 0 | | Lookup 1 | | Lookup 2 | | --+------> PAG 0: | Lookup 0 | -> | Lookup 1 | | |
| # | +----------+ -> +----------+ -> +----------+ | | | +----------+ +----------+ | |
| # | |key&action| |key&action| |key&action| | | | |key&action| |key&action| | |
| # | |key&action| |key&action| |key&action| | | | | .. | | .. | | |
| # | | .. | | .. | | .. | | | | +----------+ +----------+ | |
| # | +----------+ +----------+ +----------+ | | | | |
| # | selects PAG | | | +----------+ +----------+ | |
| # +----------------------------------------------+ +------> PAG 1: | Lookup 0 | -> | Lookup 1 | | |
| # | | +----------+ +----------+ | |
| # | | |key&action| |key&action| | |
| # | | | .. | | .. | | |
| # | | +----------+ +----------+ | |
| # | | ... | |
| # | | | |
| # | | +----------+ +----------+ | |
| # +----> PAG 254: | Lookup 0 | -> | Lookup 1 | | |
| # | | +----------+ +----------+ | |
| # | | |key&action| |key&action| | |
| # | | | .. | | .. | | |
| # | | +----------+ +----------+ | |
| # | | | |
| # | | +----------+ +----------+ | |
| # +----> PAG 255: | Lookup 0 | -> | Lookup 1 | | |
| # | +----------+ +----------+ | |
| # | |key&action| |key&action| | |
| # | | .. | | .. | | |
| # | +----------+ +----------+ | |
| # +-----------------------------------------+ |
| # |
| # Both the VCAP IS1 (Ingress Stage 1) and IS2 (Ingress Stage 2) are indexed |
| # (looked up) multiple times: IS1 3 times, and IS2 2 times. Each filter |
| # (key and action pair) can be configured to only match during the first, or |
| # second, etc, lookup. |
| # |
| # During one TCAM lookup, the filter processing stops at the first entry that |
| # matches, then the pipeline jumps to the next lookup. |
| # The driver maps each individual lookup of each individual ingress TCAM to a |
| # separate chain number. For correct rule offloading, it is mandatory that each |
| # filter installed in one TCAM is terminated by a non-optional GOTO action to |
| # the next lookup from the fixed pipeline. |
| # |
| # A chain can only be used if there is a GOTO action correctly set up from the |
| # prior lookup in the processing pipeline. Setting up all chains is not |
| # mandatory. |
| |
| # NOTE: VCAP IS1 currently uses only S1_NORMAL half keys and VCAP IS2 |
| # dynamically chooses between MAC_ETYPE, ARP, IP4_TCP_UDP, IP4_OTHER, which are |
| # all half keys as well. |
| |
| create_tcam_skeleton() |
| { |
| local eth=$1 |
| |
| tc qdisc add dev $eth clsact |
| |
| # VCAP IS1 is the Ingress Classification TCAM and can offload the |
| # following actions: |
| # - skbedit priority |
| # - vlan pop |
| # - vlan modify |
| # - goto (only in lookup 2, the last IS1 lookup) |
| tc filter add dev $eth ingress chain 0 pref 49152 flower \ |
| skip_sw action goto chain $(IS1 0) |
| tc filter add dev $eth ingress chain $(IS1 0) pref 49152 \ |
| flower skip_sw action goto chain $(IS1 1) |
| tc filter add dev $eth ingress chain $(IS1 1) pref 49152 \ |
| flower skip_sw action goto chain $(IS1 2) |
| tc filter add dev $eth ingress chain $(IS1 2) pref 49152 \ |
| flower skip_sw action goto chain $(IS2 0 0) |
| |
| # VCAP IS2 is the Security Enforcement ingress TCAM and can offload the |
| # following actions: |
| # - trap |
| # - drop |
| # - police |
| # The two VCAP IS2 lookups can be segmented into up to 256 groups of |
| # rules, called Policies. A Policy is selected through the Policy |
| # Association Group (PAG) action of VCAP IS1 (which is the |
| # GOTO offload). |
| tc filter add dev $eth ingress chain $(IS2 0 0) pref 49152 \ |
| flower skip_sw action goto chain $(IS2 1 0) |
| } |
| |
| setup_prepare() |
| { |
| ip link set $swp1 up |
| ip link set $swp2 up |
| ip link set $h2 up |
| ip link set $h1 up |
| |
| create_tcam_skeleton $swp1 |
| |
| ip link add br0 type bridge |
| ip link set $swp1 master br0 |
| ip link set $swp2 master br0 |
| ip link set br0 up |
| |
| ip link add link $h1 name $h1.100 type vlan id 100 |
| ip link set $h1.100 up |
| |
| ip link add link $h1 name $h1.200 type vlan id 200 |
| ip link set $h1.200 up |
| |
| tc filter add dev $swp1 ingress chain $(IS1 1) pref 1 \ |
| protocol 802.1Q flower skip_sw vlan_id 100 \ |
| action vlan pop \ |
| action goto chain $(IS1 2) |
| |
| tc filter add dev $swp1 egress chain $(ES0) pref 1 \ |
| flower skip_sw indev $swp2 \ |
| action vlan push protocol 802.1Q id 100 |
| |
| tc filter add dev $swp1 ingress chain $(IS1 0) pref 2 \ |
| protocol ipv4 flower skip_sw src_ip 10.1.1.2 \ |
| action skbedit priority 7 \ |
| action goto chain $(IS1 1) |
| |
| tc filter add dev $swp1 ingress chain $(IS2 0 0) pref 1 \ |
| protocol ipv4 flower skip_sw ip_proto udp dst_port 5201 \ |
| action police rate 50mbit burst 64k conform-exceed drop/pipe \ |
| action goto chain $(IS2 1 0) |
| } |
| |
| cleanup() |
| { |
| ip link del $h1.200 |
| ip link del $h1.100 |
| tc qdisc del dev $swp1 clsact |
| ip link del br0 |
| } |
| |
| test_vlan_pop() |
| { |
| local h1_mac=$(mac_get $h1) |
| local h2_mac=$(mac_get $h2) |
| |
| RET=0 |
| |
| tcpdump_start $h2 |
| |
| # Work around Mausezahn VLAN builder bug |
| # (https://github.com/netsniff-ng/netsniff-ng/issues/225) by using |
| # an 8021q upper |
| $MZ $h1.100 -q -c 1 -p 64 -a $h1_mac -b $h2_mac -t ip |
| |
| sleep 1 |
| |
| tcpdump_stop $h2 |
| |
| tcpdump_show $h2 | grep -q "$h1_mac > $h2_mac, ethertype IPv4" |
| check_err "$?" "untagged reception" |
| |
| tcpdump_cleanup $h2 |
| |
| log_test "VLAN pop" |
| } |
| |
| test_vlan_push() |
| { |
| local h1_mac=$(mac_get $h1) |
| local h2_mac=$(mac_get $h2) |
| |
| RET=0 |
| |
| tcpdump_start $h1.100 |
| |
| $MZ $h2 -q -c 1 -p 64 -a $h2_mac -b $h1_mac -t ip |
| |
| sleep 1 |
| |
| tcpdump_stop $h1.100 |
| |
| tcpdump_show $h1.100 | grep -q "$h2_mac > $h1_mac" |
| check_err "$?" "tagged reception" |
| |
| tcpdump_cleanup $h1.100 |
| |
| log_test "VLAN push" |
| } |
| |
| test_vlan_ingress_modify() |
| { |
| local h1_mac=$(mac_get $h1) |
| local h2_mac=$(mac_get $h2) |
| |
| RET=0 |
| |
| ip link set br0 type bridge vlan_filtering 1 |
| bridge vlan add dev $swp1 vid 200 |
| bridge vlan add dev $swp1 vid 300 |
| bridge vlan add dev $swp2 vid 300 |
| |
| tc filter add dev $swp1 ingress chain $(IS1 2) pref 3 \ |
| protocol 802.1Q flower skip_sw vlan_id 200 \ |
| action vlan modify id 300 \ |
| action goto chain $(IS2 0 0) |
| |
| tcpdump_start $h2 |
| |
| $MZ $h1.200 -q -c 1 -p 64 -a $h1_mac -b $h2_mac -t ip |
| |
| sleep 1 |
| |
| tcpdump_stop $h2 |
| |
| tcpdump_show $h2 | grep -q "$h1_mac > $h2_mac, .* vlan 300" |
| check_err "$?" "tagged reception" |
| |
| tcpdump_cleanup $h2 |
| |
| tc filter del dev $swp1 ingress chain $(IS1 2) pref 3 |
| |
| bridge vlan del dev $swp1 vid 200 |
| bridge vlan del dev $swp1 vid 300 |
| bridge vlan del dev $swp2 vid 300 |
| ip link set br0 type bridge vlan_filtering 0 |
| |
| log_test "Ingress VLAN modification" |
| } |
| |
| test_vlan_egress_modify() |
| { |
| local h1_mac=$(mac_get $h1) |
| local h2_mac=$(mac_get $h2) |
| |
| RET=0 |
| |
| tc qdisc add dev $swp2 clsact |
| |
| ip link set br0 type bridge vlan_filtering 1 |
| bridge vlan add dev $swp1 vid 200 |
| bridge vlan add dev $swp2 vid 200 |
| |
| tc filter add dev $swp2 egress chain $(ES0) pref 3 \ |
| protocol 802.1Q flower skip_sw vlan_id 200 vlan_prio 0 \ |
| action vlan modify id 300 priority 7 |
| |
| tcpdump_start $h2 |
| |
| $MZ $h1.200 -q -c 1 -p 64 -a $h1_mac -b $h2_mac -t ip |
| |
| sleep 1 |
| |
| tcpdump_stop $h2 |
| |
| tcpdump_show $h2 | grep -q "$h1_mac > $h2_mac, .* vlan 300" |
| check_err "$?" "tagged reception" |
| |
| tcpdump_cleanup $h2 |
| |
| tc filter del dev $swp2 egress chain $(ES0) pref 3 |
| tc qdisc del dev $swp2 clsact |
| |
| bridge vlan del dev $swp1 vid 200 |
| bridge vlan del dev $swp2 vid 200 |
| ip link set br0 type bridge vlan_filtering 0 |
| |
| log_test "Egress VLAN modification" |
| } |
| |
| test_skbedit_priority() |
| { |
| local h1_mac=$(mac_get $h1) |
| local h2_mac=$(mac_get $h2) |
| local num_pkts=100 |
| |
| before=$(ethtool_stats_get $swp1 'rx_green_prio_7') |
| |
| $MZ $h1 -q -c $num_pkts -p 64 -a $h1_mac -b $h2_mac -t ip -A 10.1.1.2 |
| |
| after=$(ethtool_stats_get $swp1 'rx_green_prio_7') |
| |
| if [ $((after - before)) = $num_pkts ]; then |
| RET=0 |
| else |
| RET=1 |
| fi |
| |
| log_test "Frame prioritization" |
| } |
| |
| trap cleanup EXIT |
| |
| ALL_TESTS=" |
| test_vlan_pop |
| test_vlan_push |
| test_vlan_ingress_modify |
| test_vlan_egress_modify |
| test_skbedit_priority |
| " |
| |
| setup_prepare |
| setup_wait |
| |
| tests_run |
| |
| exit $EXIT_STATUS |