]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
ipv4: account for fraggap on the paged allocation path
authorWongi Lee <qw3rtyp0@gmail.com>
Tue, 16 Jun 2026 13:38:29 +0000 (22:38 +0900)
committerJakub Kicinski <kuba@kernel.org>
Sun, 21 Jun 2026 22:24:49 +0000 (15:24 -0700)
In __ip_append_data(), when the paged-allocation branch is taken,
alloclen and pagedlen are computed as

alloclen = fragheaderlen + transhdrlen;
pagedlen = datalen - transhdrlen;

datalen already includes fraggap, but the fraggap bytes carried over
from the previous skb are copied into the new skb's linear area at
offset transhdrlen by the subsequent skb_copy_and_csum_bits(). The
linear area is therefore undersized by fraggap bytes while pagedlen is
overstated by the same amount.

The non-paged branch sets alloclen to fraglen, which already accounts
for fraggap because datalen does. Bring the paged branch in line by
adding fraggap to alloclen and subtracting it from pagedlen.

After this adjustment, copy no longer collapses to -fraggap on the
paged path, so remove the stale comment describing that old arithmetic.

Fixes: 8eb77cc73977 ("ipv4: avoid partial copy for zc")
Signed-off-by: Jungwoo Lee <jwlee2217@gmail.com>
Signed-off-by: Wongi Lee <qw3rtyp0@gmail.com>
Reviewed-by: Ido Schimmel <idosch@nvidia.com>
Link: https://patch.msgid.link/ajFR1eLAIs42TN3g@DESKTOP-19IMU7U.localdomain
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/ipv4/ip_output.c

index 3b4e9b8af044f19d76802abf519228ca6496e600..e6dd1e5b8c327bfa7b4201baca4ae01aeed941b3 100644 (file)
@@ -1116,8 +1116,8 @@ alloc_new_skb:
                                  !(rt->dst.dev->features & NETIF_F_SG)))
                                alloclen = fraglen;
                        else {
-                               alloclen = fragheaderlen + transhdrlen;
-                               pagedlen = datalen - transhdrlen;
+                               alloclen = fragheaderlen + transhdrlen + fraggap;
+                               pagedlen = datalen - transhdrlen - fraggap;
                        }
 
                        alloclen += alloc_extra;
@@ -1164,9 +1164,6 @@ alloc_new_skb:
                        }
 
                        copy = datalen - transhdrlen - fraggap - pagedlen;
-                       /* [!] NOTE: copy will be negative if pagedlen>0
-                        * because then the equation reduces to -fraggap.
-                        */
                        if (copy > 0 &&
                            INDIRECT_CALL_1(getfrag, ip_generic_getfrag,
                                            from, data + transhdrlen, offset,