| #!/bin/bash |
| # SPDX-License-Identifier: GPL-2.0 |
| |
| # This test sends one stream of traffic from H1 through a TBF shaper, to a RED |
| # within TBF shaper on $swp3. The two shapers have the same configuration, and |
| # thus the resulting stream should fill all available bandwidth on the latter |
| # shaper. A second stream is sent from H2 also via $swp3, and used to inject |
| # additional traffic. Since all available bandwidth is taken, this traffic has |
| # to go to backlog. |
| # |
| # +--------------------------+ +--------------------------+ |
| # | H1 | | H2 | |
| # | + $h1 | | + $h2 | |
| # | | 192.0.2.1/28 | | | 192.0.2.2/28 | |
| # | | TBF 10Mbps | | | | |
| # +-----|--------------------+ +-----|--------------------+ |
| # | | |
| # +-----|------------------------------------------------|--------------------+ |
| # | SW | | | |
| # | +--|------------------------------------------------|----------------+ | |
| # | | + $swp1 + $swp2 | | |
| # | | BR | | |
| # | | | | |
| # | | + $swp3 | | |
| # | | | TBF 10Mbps / RED | | |
| # | +--------------------------------|-----------------------------------+ | |
| # | | | |
| # +-----------------------------------|---------------------------------------+ |
| # | |
| # +-----|--------------------+ |
| # | H3 | | |
| # | + $h1 | |
| # | 192.0.2.3/28 | |
| # | | |
| # +--------------------------+ |
| |
| ALL_TESTS=" |
| ping_ipv4 |
| ecn_test |
| ecn_nodrop_test |
| red_test |
| red_qevent_test |
| ecn_qevent_test |
| " |
| |
| NUM_NETIFS=6 |
| CHECK_TC="yes" |
| source lib.sh |
| |
| BACKLOG=30000 |
| PKTSZ=1400 |
| |
| h1_create() |
| { |
| simple_if_init $h1 192.0.2.1/28 |
| mtu_set $h1 10000 |
| tc qdisc replace dev $h1 root handle 1: tbf \ |
| rate 10Mbit burst 10K limit 1M |
| } |
| |
| h1_destroy() |
| { |
| tc qdisc del dev $h1 root |
| mtu_restore $h1 |
| simple_if_fini $h1 192.0.2.1/28 |
| } |
| |
| h2_create() |
| { |
| simple_if_init $h2 192.0.2.2/28 |
| mtu_set $h2 10000 |
| } |
| |
| h2_destroy() |
| { |
| mtu_restore $h2 |
| simple_if_fini $h2 192.0.2.2/28 |
| } |
| |
| h3_create() |
| { |
| simple_if_init $h3 192.0.2.3/28 |
| mtu_set $h3 10000 |
| } |
| |
| h3_destroy() |
| { |
| mtu_restore $h3 |
| simple_if_fini $h3 192.0.2.3/28 |
| } |
| |
| switch_create() |
| { |
| ip link add dev br up type bridge |
| ip link set dev $swp1 up master br |
| ip link set dev $swp2 up master br |
| ip link set dev $swp3 up master br |
| |
| mtu_set $swp1 10000 |
| mtu_set $swp2 10000 |
| mtu_set $swp3 10000 |
| |
| tc qdisc replace dev $swp3 root handle 1: tbf \ |
| rate 10Mbit burst 10K limit 1M |
| ip link add name _drop_test up type dummy |
| } |
| |
| switch_destroy() |
| { |
| ip link del dev _drop_test |
| tc qdisc del dev $swp3 root |
| |
| mtu_restore $h3 |
| mtu_restore $h2 |
| mtu_restore $h1 |
| |
| ip link set dev $swp3 down nomaster |
| ip link set dev $swp2 down nomaster |
| ip link set dev $swp1 down nomaster |
| ip link del dev br |
| } |
| |
| setup_prepare() |
| { |
| h1=${NETIFS[p1]} |
| swp1=${NETIFS[p2]} |
| |
| h2=${NETIFS[p3]} |
| swp2=${NETIFS[p4]} |
| |
| swp3=${NETIFS[p5]} |
| h3=${NETIFS[p6]} |
| |
| h3_mac=$(mac_get $h3) |
| |
| vrf_prepare |
| |
| h1_create |
| h2_create |
| h3_create |
| switch_create |
| } |
| |
| cleanup() |
| { |
| pre_cleanup |
| |
| switch_destroy |
| h3_destroy |
| h2_destroy |
| h1_destroy |
| |
| vrf_cleanup |
| } |
| |
| ping_ipv4() |
| { |
| ping_test $h1 192.0.2.3 " from host 1" |
| ping_test $h2 192.0.2.3 " from host 2" |
| } |
| |
| get_qdisc_backlog() |
| { |
| qdisc_stats_get $swp3 11: .backlog |
| } |
| |
| get_nmarked() |
| { |
| qdisc_stats_get $swp3 11: .marked |
| } |
| |
| get_qdisc_npackets() |
| { |
| qdisc_stats_get $swp3 11: .packets |
| } |
| |
| get_nmirrored() |
| { |
| link_stats_get _drop_test tx packets |
| } |
| |
| send_packets() |
| { |
| local proto=$1; shift |
| local pkts=$1; shift |
| |
| $MZ $h2 -p $PKTSZ -a own -b $h3_mac -A 192.0.2.2 -B 192.0.2.3 -t $proto -q -c $pkts "$@" |
| } |
| |
| # This sends traffic in an attempt to build a backlog of $size. Returns 0 on |
| # success. After 10 failed attempts it bails out and returns 1. It dumps the |
| # backlog size to stdout. |
| build_backlog() |
| { |
| local size=$1; shift |
| local proto=$1; shift |
| |
| local i=0 |
| |
| while :; do |
| local cur=$(get_qdisc_backlog) |
| local diff=$((size - cur)) |
| local pkts=$(((diff + PKTSZ - 1) / PKTSZ)) |
| |
| if ((cur >= size)); then |
| echo $cur |
| return 0 |
| elif ((i++ > 10)); then |
| echo $cur |
| return 1 |
| fi |
| |
| send_packets $proto $pkts "$@" |
| sleep 1 |
| done |
| } |
| |
| check_marking() |
| { |
| local cond=$1; shift |
| |
| local npackets_0=$(get_qdisc_npackets) |
| local nmarked_0=$(get_nmarked) |
| sleep 5 |
| local npackets_1=$(get_qdisc_npackets) |
| local nmarked_1=$(get_nmarked) |
| |
| local nmarked_d=$((nmarked_1 - nmarked_0)) |
| local npackets_d=$((npackets_1 - npackets_0)) |
| local pct=$((100 * nmarked_d / npackets_d)) |
| |
| echo $pct |
| ((pct $cond)) |
| } |
| |
| check_mirroring() |
| { |
| local cond=$1; shift |
| |
| local npackets_0=$(get_qdisc_npackets) |
| local nmirrored_0=$(get_nmirrored) |
| sleep 5 |
| local npackets_1=$(get_qdisc_npackets) |
| local nmirrored_1=$(get_nmirrored) |
| |
| local nmirrored_d=$((nmirrored_1 - nmirrored_0)) |
| local npackets_d=$((npackets_1 - npackets_0)) |
| local pct=$((100 * nmirrored_d / npackets_d)) |
| |
| echo $pct |
| ((pct $cond)) |
| } |
| |
| ecn_test_common() |
| { |
| local name=$1; shift |
| local limit=$1; shift |
| local backlog |
| local pct |
| |
| # Build the below-the-limit backlog using UDP. We could use TCP just |
| # fine, but this way we get a proof that UDP is accepted when queue |
| # length is below the limit. The main stream is using TCP, and if the |
| # limit is misconfigured, we would see this traffic being ECN marked. |
| RET=0 |
| backlog=$(build_backlog $((2 * limit / 3)) udp) |
| check_err $? "Could not build the requested backlog" |
| pct=$(check_marking "== 0") |
| check_err $? "backlog $backlog / $limit Got $pct% marked packets, expected == 0." |
| log_test "$name backlog < limit" |
| |
| # Now push TCP, because non-TCP traffic would be early-dropped after the |
| # backlog crosses the limit, and we want to make sure that the backlog |
| # is above the limit. |
| RET=0 |
| backlog=$(build_backlog $((3 * limit / 2)) tcp tos=0x01) |
| check_err $? "Could not build the requested backlog" |
| pct=$(check_marking ">= 95") |
| check_err $? "backlog $backlog / $limit Got $pct% marked packets, expected >= 95." |
| log_test "$name backlog > limit" |
| } |
| |
| do_ecn_test() |
| { |
| local limit=$1; shift |
| local name=ECN |
| |
| $MZ $h1 -p $PKTSZ -A 192.0.2.1 -B 192.0.2.3 -c 0 \ |
| -a own -b $h3_mac -t tcp -q tos=0x01 & |
| sleep 1 |
| |
| ecn_test_common "$name" $limit |
| |
| # Up there we saw that UDP gets accepted when backlog is below the |
| # limit. Now that it is above, it should all get dropped, and backlog |
| # building should fail. |
| RET=0 |
| build_backlog $((2 * limit)) udp >/dev/null |
| check_fail $? "UDP traffic went into backlog instead of being early-dropped" |
| log_test "$name backlog > limit: UDP early-dropped" |
| |
| stop_traffic |
| sleep 1 |
| } |
| |
| do_ecn_nodrop_test() |
| { |
| local limit=$1; shift |
| local name="ECN nodrop" |
| |
| $MZ $h1 -p $PKTSZ -A 192.0.2.1 -B 192.0.2.3 -c 0 \ |
| -a own -b $h3_mac -t tcp -q tos=0x01 & |
| sleep 1 |
| |
| ecn_test_common "$name" $limit |
| |
| # Up there we saw that UDP gets accepted when backlog is below the |
| # limit. Now that it is above, in nodrop mode, make sure it goes to |
| # backlog as well. |
| RET=0 |
| build_backlog $((2 * limit)) udp >/dev/null |
| check_err $? "UDP traffic was early-dropped instead of getting into backlog" |
| log_test "$name backlog > limit: UDP not dropped" |
| |
| stop_traffic |
| sleep 1 |
| } |
| |
| do_red_test() |
| { |
| local limit=$1; shift |
| local backlog |
| local pct |
| |
| # Use ECN-capable TCP to verify there's no marking even though the queue |
| # is above limit. |
| $MZ $h1 -p $PKTSZ -A 192.0.2.1 -B 192.0.2.3 -c 0 \ |
| -a own -b $h3_mac -t tcp -q tos=0x01 & |
| |
| # Pushing below the queue limit should work. |
| RET=0 |
| backlog=$(build_backlog $((2 * limit / 3)) tcp tos=0x01) |
| check_err $? "Could not build the requested backlog" |
| pct=$(check_marking "== 0") |
| check_err $? "backlog $backlog / $limit Got $pct% marked packets, expected == 0." |
| log_test "RED backlog < limit" |
| |
| # Pushing above should not. |
| RET=0 |
| backlog=$(build_backlog $((3 * limit / 2)) tcp tos=0x01) |
| check_fail $? "Traffic went into backlog instead of being early-dropped" |
| pct=$(check_marking "== 0") |
| check_err $? "backlog $backlog / $limit Got $pct% marked packets, expected == 0." |
| log_test "RED backlog > limit" |
| |
| stop_traffic |
| sleep 1 |
| } |
| |
| do_red_qevent_test() |
| { |
| local limit=$1; shift |
| local backlog |
| local base |
| local now |
| local pct |
| |
| RET=0 |
| |
| $MZ $h1 -p $PKTSZ -A 192.0.2.1 -B 192.0.2.3 -c 0 \ |
| -a own -b $h3_mac -t udp -q & |
| sleep 1 |
| |
| tc filter add block 10 pref 1234 handle 102 matchall skip_hw \ |
| action mirred egress mirror dev _drop_test |
| |
| # Push to the queue until it's at the limit. The configured limit is |
| # rounded by the qdisc, so this is the best we can do to get to the real |
| # limit. |
| build_backlog $((3 * limit / 2)) udp >/dev/null |
| |
| base=$(get_nmirrored) |
| send_packets udp 100 |
| sleep 1 |
| now=$(get_nmirrored) |
| ((now >= base + 100)) |
| check_err $? "Dropped packets not observed: 100 expected, $((now - base)) seen" |
| |
| tc filter del block 10 pref 1234 handle 102 matchall |
| |
| base=$(get_nmirrored) |
| send_packets udp 100 |
| sleep 1 |
| now=$(get_nmirrored) |
| ((now == base)) |
| check_err $? "Dropped packets still observed: 0 expected, $((now - base)) seen" |
| |
| log_test "RED early_dropped packets mirrored" |
| |
| stop_traffic |
| sleep 1 |
| } |
| |
| do_ecn_qevent_test() |
| { |
| local limit=$1; shift |
| local name=ECN |
| |
| RET=0 |
| |
| $MZ $h1 -p $PKTSZ -A 192.0.2.1 -B 192.0.2.3 -c 0 \ |
| -a own -b $h3_mac -t tcp -q tos=0x01 & |
| sleep 1 |
| |
| tc filter add block 10 pref 1234 handle 102 matchall skip_hw \ |
| action mirred egress mirror dev _drop_test |
| |
| backlog=$(build_backlog $((2 * limit / 3)) tcp tos=0x01) |
| check_err $? "Could not build the requested backlog" |
| pct=$(check_mirroring "== 0") |
| check_err $? "backlog $backlog / $limit Got $pct% mirrored packets, expected == 0." |
| |
| backlog=$(build_backlog $((3 * limit / 2)) tcp tos=0x01) |
| check_err $? "Could not build the requested backlog" |
| pct=$(check_mirroring ">= 95") |
| check_err $? "backlog $backlog / $limit Got $pct% mirrored packets, expected >= 95." |
| |
| tc filter del block 10 pref 1234 handle 102 matchall |
| |
| log_test "ECN marked packets mirrored" |
| |
| stop_traffic |
| sleep 1 |
| } |
| |
| install_qdisc() |
| { |
| local -a args=("$@") |
| |
| tc qdisc replace dev $swp3 parent 1:1 handle 11: red \ |
| limit 1M avpkt $PKTSZ probability 1 \ |
| min $BACKLOG max $((BACKLOG + 1)) burst 38 "${args[@]}" |
| sleep 1 |
| } |
| |
| uninstall_qdisc() |
| { |
| tc qdisc del dev $swp3 parent 1:1 |
| } |
| |
| ecn_test() |
| { |
| install_qdisc ecn |
| xfail_on_slow do_ecn_test $BACKLOG |
| uninstall_qdisc |
| } |
| |
| ecn_nodrop_test() |
| { |
| install_qdisc ecn nodrop |
| xfail_on_slow do_ecn_nodrop_test $BACKLOG |
| uninstall_qdisc |
| } |
| |
| red_test() |
| { |
| install_qdisc |
| xfail_on_slow do_red_test $BACKLOG |
| uninstall_qdisc |
| } |
| |
| red_qevent_test() |
| { |
| install_qdisc qevent early_drop block 10 |
| xfail_on_slow do_red_qevent_test $BACKLOG |
| uninstall_qdisc |
| } |
| |
| ecn_qevent_test() |
| { |
| install_qdisc ecn qevent mark block 10 |
| xfail_on_slow do_ecn_qevent_test $BACKLOG |
| uninstall_qdisc |
| } |
| |
| trap cleanup EXIT |
| |
| setup_prepare |
| setup_wait |
| |
| tests_run |
| |
| exit $EXIT_STATUS |