]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
net/sched: act_police: more accurate MTU policing
authorDavide Caratti <dcaratti@redhat.com>
Thu, 10 Feb 2022 17:56:08 +0000 (18:56 +0100)
committerGreg Kroah-Hartman <gregkh@linuxfoundation.org>
Wed, 22 Jun 2022 12:13:20 +0000 (14:13 +0200)
commit 4ddc844eb81da59bfb816d8d52089aba4e59e269 upstream.

in current Linux, MTU policing does not take into account that packets at
the TC ingress have the L2 header pulled. Thus, the same TC police action
(with the same value of tcfp_mtu) behaves differently for ingress/egress.
In addition, the full GSO size is compared to tcfp_mtu: as a consequence,
the policer drops GSO packets even when individual segments have the L2 +
L3 + L4 + payload length below the configured valued of tcfp_mtu.

Improve the accuracy of MTU policing as follows:
 - account for mac_len for non-GSO packets at TC ingress.
 - compare MTU threshold with the segmented size for GSO packets.
Also, add a kselftest that verifies the correct behavior.

Signed-off-by: Davide Caratti <dcaratti@redhat.com>
Reviewed-by: Marcelo Ricardo Leitner <marcelo.leitner@gmail.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
[dcaratti: fix conflicts due to lack of the following commits:
 - commit 2ffe0395288a ("net/sched: act_police: add support for
   packet-per-second policing")
 - commit 53b61f29367d ("selftests: forwarding: Add tc-police tests for
   packets per second")]
Link: https://lore.kernel.org/netdev/876d597a0ff55f6ba786f73c5a9fd9eb8d597a03.1644514748.git.dcaratti@redhat.com
Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
net/sched/act_police.c
tools/testing/selftests/net/forwarding/tc_police.sh

index 8d8452b1cdd4287448f38472a7676af63e16f879..380733588959060e1b9aff44d0fd5f4ce49e257f 100644 (file)
@@ -213,6 +213,20 @@ release_idr:
        return err;
 }
 
+static bool tcf_police_mtu_check(struct sk_buff *skb, u32 limit)
+{
+       u32 len;
+
+       if (skb_is_gso(skb))
+               return skb_gso_validate_mac_len(skb, limit);
+
+       len = qdisc_pkt_len(skb);
+       if (skb_at_tc_ingress(skb))
+               len += skb->mac_len;
+
+       return len <= limit;
+}
+
 static int tcf_police_act(struct sk_buff *skb, const struct tc_action *a,
                          struct tcf_result *res)
 {
@@ -235,7 +249,7 @@ static int tcf_police_act(struct sk_buff *skb, const struct tc_action *a,
                        goto inc_overlimits;
        }
 
-       if (qdisc_pkt_len(skb) <= p->tcfp_mtu) {
+       if (tcf_police_mtu_check(skb, p->tcfp_mtu)) {
                if (!p->rate_present) {
                        ret = p->tcfp_result;
                        goto end;
index 160f9cccdfb7944b4350d28a88283930e137339d..eb09acdcb3ff1c103d9c04acbfb7309384e682af 100755 (executable)
@@ -35,6 +35,8 @@ ALL_TESTS="
        police_shared_test
        police_rx_mirror_test
        police_tx_mirror_test
+       police_mtu_rx_test
+       police_mtu_tx_test
 "
 NUM_NETIFS=6
 source tc_common.sh
@@ -290,6 +292,56 @@ police_tx_mirror_test()
        police_mirror_common_test $rp2 egress "police tx and mirror"
 }
 
+police_mtu_common_test() {
+       RET=0
+
+       local test_name=$1; shift
+       local dev=$1; shift
+       local direction=$1; shift
+
+       tc filter add dev $dev $direction protocol ip pref 1 handle 101 flower \
+               dst_ip 198.51.100.1 ip_proto udp dst_port 54321 \
+               action police mtu 1042 conform-exceed drop/ok
+
+       # to count "conform" packets
+       tc filter add dev $h2 ingress protocol ip pref 1 handle 101 flower \
+               dst_ip 198.51.100.1 ip_proto udp dst_port 54321 \
+               action drop
+
+       mausezahn $h1 -a own -b $(mac_get $rp1) -A 192.0.2.1 -B 198.51.100.1 \
+               -t udp sp=12345,dp=54321 -p 1001 -c 10 -q
+
+       mausezahn $h1 -a own -b $(mac_get $rp1) -A 192.0.2.1 -B 198.51.100.1 \
+               -t udp sp=12345,dp=54321 -p 1000 -c 3 -q
+
+       tc_check_packets "dev $dev $direction" 101 13
+       check_err $? "wrong packet counter"
+
+       # "exceed" packets
+       local overlimits_t0=$(tc_rule_stats_get ${dev} 1 ${direction} .overlimits)
+       test ${overlimits_t0} = 10
+       check_err $? "wrong overlimits, expected 10 got ${overlimits_t0}"
+
+       # "conform" packets
+       tc_check_packets "dev $h2 ingress" 101 3
+       check_err $? "forwarding error"
+
+       tc filter del dev $h2 ingress protocol ip pref 1 handle 101 flower
+       tc filter del dev $dev $direction protocol ip pref 1 handle 101 flower
+
+       log_test "$test_name"
+}
+
+police_mtu_rx_test()
+{
+       police_mtu_common_test "police mtu (rx)" $rp1 ingress
+}
+
+police_mtu_tx_test()
+{
+       police_mtu_common_test "police mtu (tx)" $rp2 egress
+}
+
 setup_prepare()
 {
        h1=${NETIFS[p1]}