]> www.infradead.org Git - users/willy/linux.git/commitdiff
xfrm: Switch to skb_dstref_steal to clear dst_entry
authorStanislav Fomichev <sdf@fomichev.me>
Mon, 18 Aug 2025 15:40:27 +0000 (08:40 -0700)
committerJakub Kicinski <kuba@kernel.org>
Wed, 20 Aug 2025 00:54:17 +0000 (17:54 -0700)
Going forward skb_dst_set will assert that skb dst_entry
is empty during skb_dst_set. skb_dstref_steal is added to reset
existing entry without doing refcnt. Switch to skb_dstref_steal
in __xfrm_route_forward and add a comment on why it's safe
to skip skb_dstref_restore.

Signed-off-by: Stanislav Fomichev <sdf@fomichev.me>
Link: https://patch.msgid.link/20250818154032.3173645-3-sdf@fomichev.me
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/xfrm/xfrm_policy.c

index c5035a9bc3bb28e9a33ab100bf6f91a6bc413a78..7111184eef5957b62143ea666d65be32f15e7ee3 100644 (file)
@@ -3881,12 +3881,18 @@ int __xfrm_route_forward(struct sk_buff *skb, unsigned short family)
        }
 
        skb_dst_force(skb);
-       if (!skb_dst(skb)) {
+       dst = skb_dst(skb);
+       if (!dst) {
                XFRM_INC_STATS(net, LINUX_MIB_XFRMFWDHDRERROR);
                return 0;
        }
 
-       dst = xfrm_lookup(net, skb_dst(skb), &fl, NULL, XFRM_LOOKUP_QUEUE);
+       /* ignore return value from skb_dstref_steal, xfrm_lookup takes
+        * care of dropping the refcnt if needed.
+        */
+       skb_dstref_steal(skb);
+
+       dst = xfrm_lookup(net, dst, &fl, NULL, XFRM_LOOKUP_QUEUE);
        if (IS_ERR(dst)) {
                res = 0;
                dst = NULL;