]> git.ipfire.org Git - thirdparty/linux.git/commitdiff
bridge: provide lockless access to p->path_cost
authorEric Dumazet <edumazet@google.com>
Thu, 4 Jun 2026 14:13:37 +0000 (14:13 +0000)
committerJakub Kicinski <kuba@kernel.org>
Sat, 6 Jun 2026 00:46:17 +0000 (17:46 -0700)
Add READ_ONCE()/WRITE_ONCE() annotations around p->path_cost.

This is needed at least for sysfs show_path_cost(), BRCTL_GET_PORT_INFO
and upcoming RTNL avoidance in "ip link" dumps (cf br_port_fill_attrs()).

Signed-off-by: Eric Dumazet <edumazet@google.com>
Acked-by: Nikolay Aleksandrov <razor@blackwall.org>
Reviewed-by: Ido Schimmel <idosch@nvidia.com>
Link: https://patch.msgid.link/20260604141343.2124500-6-edumazet@google.com
Signed-off-by: Jakub Kicinski <kuba@kernel.org>
net/bridge/br_if.c
net/bridge/br_ioctl.c
net/bridge/br_netlink.c
net/bridge/br_stp.c
net/bridge/br_stp_if.c
net/bridge/br_sysfs_if.c

index 630ca7ac962ac1d10fdba6035476955525e2221c..463c3c7083dcdd4f9064b12375792b2961a9b674 100644 (file)
@@ -78,7 +78,7 @@ void br_port_carrier_check(struct net_bridge_port *p, bool *notified)
 
        if (!test_bit(BR_ADMIN_COST_BIT, &p->flags) &&
            netif_running(dev) && netif_oper_up(dev))
-               p->path_cost = port_cost(dev);
+               WRITE_ONCE(p->path_cost, port_cost(dev));
 
        *notified = false;
        if (!netif_running(br->dev))
index 766c43b327af830a80fba11c940ba34eb03c50c9..07cfcb821e27ebcb72901a365a3f06491f8b0bc3 100644 (file)
@@ -259,7 +259,7 @@ int br_dev_siocdevprivate(struct net_device *dev, struct ifreq *rq,
                memcpy(&p.designated_bridge, &pt->designated_bridge, 8);
                p.port_id = pt->port_id;
                p.designated_port = pt->designated_port;
-               p.path_cost = pt->path_cost;
+               p.path_cost = READ_ONCE(pt->path_cost);
                p.designated_cost = pt->designated_cost;
                p.state = pt->state;
                p.top_change_ack = pt->topology_change_ack;
index a104b25c871d263e4b5701f1be05cb17fbfce28f..08718ebe282aa1d089a127cdc7c24fda5a0a38a5 100644 (file)
@@ -240,7 +240,7 @@ static int br_port_fill_attrs(struct sk_buff *skb,
 
        if (nla_put_u8(skb, IFLA_BRPORT_STATE, p->state) ||
            nla_put_u16(skb, IFLA_BRPORT_PRIORITY, p->priority) ||
-           nla_put_u32(skb, IFLA_BRPORT_COST, p->path_cost) ||
+           nla_put_u32(skb, IFLA_BRPORT_COST, READ_ONCE(p->path_cost)) ||
            nla_put_u8(skb, IFLA_BRPORT_MODE, mode) ||
            nla_put_u8(skb, IFLA_BRPORT_GUARD, !!(p->flags & BR_BPDU_GUARD)) ||
            nla_put_u8(skb, IFLA_BRPORT_PROTECT,
index 024210f95468308066827ac2ae71f68f99fa77f5..2d9d0823aa56b63a62ed51c49df54055294e82ad 100644 (file)
@@ -102,8 +102,9 @@ struct net_bridge_port *br_get_port(struct net_bridge *br, u16 port_no)
 static int br_should_become_root_port(const struct net_bridge_port *p,
                                      u16 root_port)
 {
-       struct net_bridge *br;
+       u32 p_path_cost, rp_path_cost;
        struct net_bridge_port *rp;
+       struct net_bridge *br;
        int t;
 
        br = p->br;
@@ -125,11 +126,14 @@ static int br_should_become_root_port(const struct net_bridge_port *p,
        else if (t > 0)
                return 0;
 
-       if (p->designated_cost + p->path_cost <
-           rp->designated_cost + rp->path_cost)
+       p_path_cost = READ_ONCE(p->path_cost);
+       rp_path_cost = READ_ONCE(rp->path_cost);
+
+       if (p->designated_cost + p_path_cost <
+           rp->designated_cost + rp_path_cost)
                return 1;
-       else if (p->designated_cost + p->path_cost >
-                rp->designated_cost + rp->path_cost)
+       else if (p->designated_cost + p_path_cost >
+                rp->designated_cost + rp_path_cost)
                return 0;
 
        t = memcmp(&p->designated_bridge, &rp->designated_bridge, 8);
@@ -187,7 +191,8 @@ static void br_root_selection(struct net_bridge *br)
        } else {
                p = br_get_port(br, root_port);
                br->designated_root = p->designated_root;
-               br->root_path_cost = p->designated_cost + p->path_cost;
+               br->root_path_cost = p->designated_cost +
+                                    READ_ONCE(p->path_cost);
        }
 }
 
index b29dc97b9ad8a5b5c56517d34fe426abd00e2ad6..e5d43492d2dcf0615ee984bc190d7ce2264aaab0 100644 (file)
@@ -341,7 +341,7 @@ int br_stp_set_path_cost(struct net_bridge_port *p, unsigned long path_cost)
                return -ERANGE;
 
        set_bit(BR_ADMIN_COST_BIT, &p->flags);
-       p->path_cost = path_cost;
+       WRITE_ONCE(p->path_cost, path_cost);
        br_configuration_update(p->br);
        br_port_state_selection(p->br);
        return 0;
index d6df81fa0d13fe7e732b69f881b947da2066d9ef..23574476d7f701080452777ab2c23a42b4defba4 100644 (file)
@@ -83,7 +83,7 @@ static int store_flag(struct net_bridge_port *p, unsigned long v,
 
 static ssize_t show_path_cost(struct net_bridge_port *p, char *buf)
 {
-       return sysfs_emit(buf, "%d\n", p->path_cost);
+       return sysfs_emit(buf, "%d\n", READ_ONCE(p->path_cost));
 }
 
 static int store_path_cost(struct net_bridge_port *p, unsigned long v)