]> www.infradead.org Git - users/dwmw2/linux.git/commitdiff
net: ipv6: fix dst refleaks in rpl, seg6 and ioam6 lwtunnels
authorJakub Kicinski <kuba@kernel.org>
Thu, 30 Jan 2025 03:15:18 +0000 (19:15 -0800)
committerJakub Kicinski <kuba@kernel.org>
Sun, 2 Feb 2025 00:55:30 +0000 (16:55 -0800)
dst_cache_get() gives us a reference, we need to release it.

Discovered by the ioam6.sh test, kmemleak was recently fixed
to catch per-cpu memory leaks.

Fixes: 985ec6f5e623 ("net: ipv6: rpl_iptunnel: mitigate 2-realloc issue")
Fixes: 40475b63761a ("net: ipv6: seg6_iptunnel: mitigate 2-realloc issue")
Fixes: dce525185bc9 ("net: ipv6: ioam6_iptunnel: mitigate 2-realloc issue")
Reviewed-by: Justin Iurman <justin.iurman@uliege.be>
Reviewed-by: Simon Horman <horms@kernel.org>
Link: https://patch.msgid.link/20250130031519.2716843-1-kuba@kernel.org
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/ipv6/ioam6_iptunnel.c
net/ipv6/rpl_iptunnel.c
net/ipv6/seg6_iptunnel.c

index 28e5a89dc2557493ff8b50f040bd5f1e73522a5f..3936c137a5727cd7e2689d1b6bd48babe6676982 100644 (file)
@@ -336,7 +336,7 @@ static int ioam6_do_encap(struct net *net, struct sk_buff *skb,
 
 static int ioam6_output(struct net *net, struct sock *sk, struct sk_buff *skb)
 {
-       struct dst_entry *dst = skb_dst(skb), *cache_dst;
+       struct dst_entry *dst = skb_dst(skb), *cache_dst = NULL;
        struct in6_addr orig_daddr;
        struct ioam6_lwt *ilwt;
        int err = -EINVAL;
@@ -407,7 +407,6 @@ do_encap:
                cache_dst = ip6_route_output(net, NULL, &fl6);
                if (cache_dst->error) {
                        err = cache_dst->error;
-                       dst_release(cache_dst);
                        goto drop;
                }
 
@@ -426,8 +425,10 @@ do_encap:
                return dst_output(net, sk, skb);
        }
 out:
+       dst_release(cache_dst);
        return dst->lwtstate->orig_output(net, sk, skb);
 drop:
+       dst_release(cache_dst);
        kfree_skb(skb);
        return err;
 }
index 7ba22d2f2bfefa73a9fcaed0fab50d8a5afc1cad..9b7d035631154b6fd51544fa1d3b17b7dbd1fc2d 100644 (file)
@@ -232,7 +232,6 @@ static int rpl_output(struct net *net, struct sock *sk, struct sk_buff *skb)
                dst = ip6_route_output(net, NULL, &fl6);
                if (dst->error) {
                        err = dst->error;
-                       dst_release(dst);
                        goto drop;
                }
 
@@ -251,6 +250,7 @@ static int rpl_output(struct net *net, struct sock *sk, struct sk_buff *skb)
        return dst_output(net, sk, skb);
 
 drop:
+       dst_release(dst);
        kfree_skb(skb);
        return err;
 }
@@ -269,8 +269,10 @@ static int rpl_input(struct sk_buff *skb)
        local_bh_enable();
 
        err = rpl_do_srh(skb, rlwt, dst);
-       if (unlikely(err))
+       if (unlikely(err)) {
+               dst_release(dst);
                goto drop;
+       }
 
        if (!dst) {
                ip6_route_input(skb);
index 4bf937bfc2633c6abad8249e6ffae6c67070d7a9..eacc4e91b48efd6762d352f1dc10d0e4cdd5748c 100644 (file)
@@ -482,8 +482,10 @@ static int seg6_input_core(struct net *net, struct sock *sk,
        local_bh_enable();
 
        err = seg6_do_srh(skb, dst);
-       if (unlikely(err))
+       if (unlikely(err)) {
+               dst_release(dst);
                goto drop;
+       }
 
        if (!dst) {
                ip6_route_input(skb);
@@ -571,7 +573,6 @@ static int seg6_output_core(struct net *net, struct sock *sk,
                dst = ip6_route_output(net, NULL, &fl6);
                if (dst->error) {
                        err = dst->error;
-                       dst_release(dst);
                        goto drop;
                }
 
@@ -593,6 +594,7 @@ static int seg6_output_core(struct net *net, struct sock *sk,
 
        return dst_output(net, sk, skb);
 drop:
+       dst_release(dst);
        kfree_skb(skb);
        return err;
 }