]> git.ipfire.org Git - thirdparty/kernel/linux.git/commitdiff
bridge: br_nd_send: linearize skb before parsing ND options
authorYang Yang <n05ec@lzu.edu.cn>
Thu, 26 Mar 2026 03:44:39 +0000 (03:44 +0000)
committerJakub Kicinski <kuba@kernel.org>
Sat, 28 Mar 2026 03:37:14 +0000 (20:37 -0700)
br_nd_send() parses neighbour discovery options from ns->opt[] and
assumes that these options are in the linear part of request.

Its callers only guarantee that the ICMPv6 header and target address
are available, so the option area can still be non-linear. Parsing
ns->opt[] in that case can access data past the linear buffer.

Linearize request before option parsing and derive ns from the linear
network header.

Fixes: ed842faeb2bd ("bridge: suppress nd pkts on BR_NEIGH_SUPPRESS ports")
Reported-by: Yifan Wu <yifanwucs@gmail.com>
Reported-by: Juefei Pu <tomapufckgml@gmail.com>
Tested-by: Ao Zhou <n05ec@lzu.edu.cn>
Co-developed-by: Yuan Tan <tanyuan98@outlook.com>
Signed-off-by: Yuan Tan <tanyuan98@outlook.com>
Suggested-by: Xin Liu <bird@lzu.edu.cn>
Signed-off-by: Yang Yang <n05ec@lzu.edu.cn>
Reviewed-by: Ido Schimmel <idosch@nvidia.com>
Acked-by: Nikolay Aleksandrov <razor@blackwall.org>
Link: https://patch.msgid.link/20260326034441.2037420-2-n05ec@lzu.edu.cn
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/bridge/br_arp_nd_proxy.c

index 1e2b51769eec81c78aaddd739bb6c28d91c6f0bf..af3d1e33f50b83c6d5782aa5eeb23e24f6df2286 100644 (file)
@@ -251,12 +251,12 @@ struct nd_msg *br_is_nd_neigh_msg(const struct sk_buff *skb, struct nd_msg *msg)
 
 static void br_nd_send(struct net_bridge *br, struct net_bridge_port *p,
                       struct sk_buff *request, struct neighbour *n,
-                      __be16 vlan_proto, u16 vlan_tci, struct nd_msg *ns)
+                      __be16 vlan_proto, u16 vlan_tci)
 {
        struct net_device *dev = request->dev;
        struct net_bridge_vlan_group *vg;
+       struct nd_msg *na, *ns;
        struct sk_buff *reply;
-       struct nd_msg *na;
        struct ipv6hdr *pip6;
        int na_olen = 8; /* opt hdr + ETH_ALEN for target */
        int ns_olen;
@@ -264,7 +264,7 @@ static void br_nd_send(struct net_bridge *br, struct net_bridge_port *p,
        u8 *daddr;
        u16 pvid;
 
-       if (!dev)
+       if (!dev || skb_linearize(request))
                return;
 
        len = LL_RESERVED_SPACE(dev) + sizeof(struct ipv6hdr) +
@@ -281,6 +281,8 @@ static void br_nd_send(struct net_bridge *br, struct net_bridge_port *p,
        skb_set_mac_header(reply, 0);
 
        daddr = eth_hdr(request)->h_source;
+       ns = (struct nd_msg *)(skb_network_header(request) +
+                              sizeof(struct ipv6hdr));
 
        /* Do we need option processing ? */
        ns_olen = request->len - (skb_network_offset(request) +
@@ -472,9 +474,9 @@ void br_do_suppress_nd(struct sk_buff *skb, struct net_bridge *br,
                                if (vid != 0)
                                        br_nd_send(br, p, skb, n,
                                                   skb->vlan_proto,
-                                                  skb_vlan_tag_get(skb), msg);
+                                                  skb_vlan_tag_get(skb));
                                else
-                                       br_nd_send(br, p, skb, n, 0, 0, msg);
+                                       br_nd_send(br, p, skb, n, 0, 0);
                                replied = true;
                        }