huangcm
2025-09-01 53d8e046ac1bf2ebe94f671983e3d3be059df91a
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
#include <linux/kernel.h>
#include <linux/list.h>
#include <linux/netdevice.h>
#include <linux/rtnetlink.h>
#include <linux/skbuff.h>
#include <net/switchdev.h>
 
#include "br_private.h"
 
static int br_switchdev_mark_get(struct net_bridge *br, struct net_device *dev)
{
   struct net_bridge_port *p;
 
   /* dev is yet to be added to the port list. */
   list_for_each_entry(p, &br->port_list, list) {
       if (switchdev_port_same_parent_id(dev, p->dev))
           return p->offload_fwd_mark;
   }
 
   return ++br->offload_fwd_mark;
}
 
int nbp_switchdev_mark_set(struct net_bridge_port *p)
{
   struct switchdev_attr attr = {
       .orig_dev = p->dev,
       .id = SWITCHDEV_ATTR_ID_PORT_PARENT_ID,
   };
   int err;
 
   ASSERT_RTNL();
 
   err = switchdev_port_attr_get(p->dev, &attr);
   if (err) {
       if (err == -EOPNOTSUPP)
           return 0;
       return err;
   }
 
   p->offload_fwd_mark = br_switchdev_mark_get(p->br, p->dev);
 
   return 0;
}
 
void nbp_switchdev_frame_mark(const struct net_bridge_port *p,
                 struct sk_buff *skb)
{
   if (skb->offload_fwd_mark && !WARN_ON_ONCE(!p->offload_fwd_mark))
       BR_INPUT_SKB_CB(skb)->offload_fwd_mark = p->offload_fwd_mark;
}
 
bool nbp_switchdev_allowed_egress(const struct net_bridge_port *p,
                 const struct sk_buff *skb)
{
   return !skb->offload_fwd_mark ||
          BR_INPUT_SKB_CB(skb)->offload_fwd_mark != p->offload_fwd_mark;
}