dsa_priv.h 7.6 KB
Newer Older
1
/* SPDX-License-Identifier: GPL-2.0-or-later */
2 3
/*
 * net/dsa/dsa_priv.h - Hardware switch handling
4
 * Copyright (c) 2008-2009 Marvell Semiconductor
5 6 7 8 9
 */

#ifndef __DSA_PRIV_H
#define __DSA_PRIV_H

10
#include <linux/if_bridge.h>
11
#include <linux/phy.h>
12
#include <linux/netdevice.h>
13
#include <linux/netpoll.h>
14
#include <net/dsa.h>
15
#include <net/gro_cells.h>
16

17
enum {
18
	DSA_NOTIFIER_AGEING_TIME,
19 20
	DSA_NOTIFIER_BRIDGE_JOIN,
	DSA_NOTIFIER_BRIDGE_LEAVE,
V
Vivien Didelot 已提交
21 22
	DSA_NOTIFIER_FDB_ADD,
	DSA_NOTIFIER_FDB_DEL,
V
Vivien Didelot 已提交
23 24
	DSA_NOTIFIER_MDB_ADD,
	DSA_NOTIFIER_MDB_DEL,
V
Vivien Didelot 已提交
25 26
	DSA_NOTIFIER_VLAN_ADD,
	DSA_NOTIFIER_VLAN_DEL,
27
	DSA_NOTIFIER_MTU,
28 29
};

30 31 32 33 34 35
/* DSA_NOTIFIER_AGEING_TIME */
struct dsa_notifier_ageing_time_info {
	struct switchdev_trans *trans;
	unsigned int ageing_time;
};

36 37 38
/* DSA_NOTIFIER_BRIDGE_* */
struct dsa_notifier_bridge_info {
	struct net_device *br;
39
	int tree_index;
40 41 42 43
	int sw_index;
	int port;
};

V
Vivien Didelot 已提交
44 45 46 47
/* DSA_NOTIFIER_FDB_* */
struct dsa_notifier_fdb_info {
	int sw_index;
	int port;
48 49
	const unsigned char *addr;
	u16 vid;
V
Vivien Didelot 已提交
50 51
};

V
Vivien Didelot 已提交
52 53 54 55 56 57 58 59
/* DSA_NOTIFIER_MDB_* */
struct dsa_notifier_mdb_info {
	const struct switchdev_obj_port_mdb *mdb;
	struct switchdev_trans *trans;
	int sw_index;
	int port;
};

V
Vivien Didelot 已提交
60 61 62 63 64 65 66 67
/* DSA_NOTIFIER_VLAN_* */
struct dsa_notifier_vlan_info {
	const struct switchdev_obj_port_vlan *vlan;
	struct switchdev_trans *trans;
	int sw_index;
	int port;
};

68 69 70 71 72 73 74 75
/* DSA_NOTIFIER_MTU */
struct dsa_notifier_mtu_info {
	bool propagate_upstream;
	int sw_index;
	int port;
	int mtu;
};

76
struct dsa_slave_priv {
V
Vivien Didelot 已提交
77
	/* Copy of CPU port xmit for faster access in slave transmit hot path */
78
	struct sk_buff *	(*xmit)(struct sk_buff *skb,
79
					struct net_device *dev);
80

81
	struct pcpu_sw_netstats	__percpu *stats64;
82

83 84
	struct gro_cells	gcells;

85 86
	/* DSA port data, such as switch, port index, etc. */
	struct dsa_port		*dp;
87

88 89 90
#ifdef CONFIG_NET_POLL_CONTROLLER
	struct netpoll		*netpoll;
#endif
91 92 93

	/* TC context */
	struct list_head	mall_tc_list;
94 95 96
};

/* dsa.c */
97
const struct dsa_device_ops *dsa_tag_driver_get(int tag_protocol);
98
void dsa_tag_driver_put(const struct dsa_device_ops *ops);
99

100
bool dsa_schedule_work(struct work_struct *work);
101
const char *dsa_tag_protocol_to_str(const struct dsa_device_ops *ops);
102

103 104 105
int dsa_legacy_fdb_add(struct ndmsg *ndm, struct nlattr *tb[],
		       struct net_device *dev,
		       const unsigned char *addr, u16 vid,
106 107
		       u16 flags,
		       struct netlink_ext_ack *extack);
108 109 110
int dsa_legacy_fdb_del(struct ndmsg *ndm, struct nlattr *tb[],
		       struct net_device *dev,
		       const unsigned char *addr, u16 vid);
V
Vivien Didelot 已提交
111

112
/* master.c */
113 114
int dsa_master_setup(struct net_device *dev, struct dsa_port *cpu_dp);
void dsa_master_teardown(struct net_device *dev);
115

116 117
static inline struct net_device *dsa_master_find_slave(struct net_device *dev,
						       int device, int port)
118
{
119 120
	struct dsa_port *cpu_dp = dev->dsa_ptr;
	struct dsa_switch_tree *dst = cpu_dp->dst;
121
	struct dsa_port *dp;
122

123 124 125 126
	list_for_each_entry(dp, &dst->ports, list)
		if (dp->ds->index == device && dp->index == port &&
		    dp->type == DSA_PORT_TYPE_USER)
			return dp->slave;
127

128
	return NULL;
129 130
}

131 132 133
/* port.c */
int dsa_port_set_state(struct dsa_port *dp, u8 state,
		       struct switchdev_trans *trans);
134
int dsa_port_enable_rt(struct dsa_port *dp, struct phy_device *phy);
135
int dsa_port_enable(struct dsa_port *dp, struct phy_device *phy);
136
void dsa_port_disable_rt(struct dsa_port *dp);
137
void dsa_port_disable(struct dsa_port *dp);
138 139
int dsa_port_bridge_join(struct dsa_port *dp, struct net_device *br);
void dsa_port_bridge_leave(struct dsa_port *dp, struct net_device *br);
140 141
int dsa_port_vlan_filtering(struct dsa_port *dp, bool vlan_filtering,
			    struct switchdev_trans *trans);
142
bool dsa_port_skip_vlan_configuration(struct dsa_port *dp);
143 144
int dsa_port_ageing_time(struct dsa_port *dp, clock_t ageing_clock,
			 struct switchdev_trans *trans);
145 146
int dsa_port_mtu_change(struct dsa_port *dp, int new_mtu,
			bool propagate_upstream);
147 148 149 150
int dsa_port_fdb_add(struct dsa_port *dp, const unsigned char *addr,
		     u16 vid);
int dsa_port_fdb_del(struct dsa_port *dp, const unsigned char *addr,
		     u16 vid);
V
Vivien Didelot 已提交
151
int dsa_port_fdb_dump(struct dsa_port *dp, dsa_fdb_dump_cb_t *cb, void *data);
A
Andrew Lunn 已提交
152
int dsa_port_mdb_add(const struct dsa_port *dp,
V
Vivien Didelot 已提交
153 154
		     const struct switchdev_obj_port_mdb *mdb,
		     struct switchdev_trans *trans);
A
Andrew Lunn 已提交
155
int dsa_port_mdb_del(const struct dsa_port *dp,
V
Vivien Didelot 已提交
156
		     const struct switchdev_obj_port_mdb *mdb);
157 158
int dsa_port_pre_bridge_flags(const struct dsa_port *dp, unsigned long flags,
			      struct switchdev_trans *trans);
159 160
int dsa_port_bridge_flags(const struct dsa_port *dp, unsigned long flags,
			  struct switchdev_trans *trans);
161 162
int dsa_port_mrouter(struct dsa_port *dp, bool mrouter,
		     struct switchdev_trans *trans);
V
Vivien Didelot 已提交
163 164 165 166 167
int dsa_port_vlan_add(struct dsa_port *dp,
		      const struct switchdev_obj_port_vlan *vlan,
		      struct switchdev_trans *trans);
int dsa_port_vlan_del(struct dsa_port *dp,
		      const struct switchdev_obj_port_vlan *vlan);
168 169
int dsa_port_link_register_of(struct dsa_port *dp);
void dsa_port_link_unregister_of(struct dsa_port *dp);
170
extern const struct phylink_mac_ops dsa_port_phylink_mac_ops;
171

172
/* slave.c */
173
extern const struct dsa_device_ops notag_netdev_ops;
174
void dsa_slave_mii_bus_init(struct dsa_switch *ds);
175
int dsa_slave_create(struct dsa_port *dp);
176
void dsa_slave_destroy(struct net_device *slave_dev);
177
bool dsa_slave_dev_check(const struct net_device *dev);
178 179
int dsa_slave_suspend(struct net_device *slave_dev);
int dsa_slave_resume(struct net_device *slave_dev);
180 181
int dsa_slave_register_notifier(void);
void dsa_slave_unregister_notifier(void);
182

183 184 185 186 187 188 189
static inline struct dsa_port *dsa_slave_to_port(const struct net_device *dev)
{
	struct dsa_slave_priv *p = netdev_priv(dev);

	return p->dp;
}

190 191 192 193 194
static inline struct net_device *
dsa_slave_to_master(const struct net_device *dev)
{
	struct dsa_port *dp = dsa_slave_to_port(dev);

195
	return dp->cpu_dp->master;
196 197
}

198 199 200 201 202 203 204 205 206 207 208 209 210 211 212 213 214 215 216 217 218
/* If under a bridge with vlan_filtering=0, make sure to send pvid-tagged
 * frames as untagged, since the bridge will not untag them.
 */
static inline struct sk_buff *dsa_untag_bridge_pvid(struct sk_buff *skb)
{
	struct dsa_port *dp = dsa_slave_to_port(skb->dev);
	struct net_device *br = dp->bridge_dev;
	struct net_device *dev = skb->dev;
	struct net_device *upper_dev;
	struct list_head *iter;
	u16 vid, pvid, proto;
	int err;

	if (!br || br_vlan_enabled(br))
		return skb;

	err = br_vlan_get_proto(br, &proto);
	if (err)
		return skb;

	/* Move VLAN tag from data to hwaccel */
219
	if (!skb_vlan_tag_present(skb) && skb->protocol == htons(proto)) {
220 221 222 223 224 225 226 227 228 229 230 231 232 233 234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261
		skb = skb_vlan_untag(skb);
		if (!skb)
			return NULL;
	}

	if (!skb_vlan_tag_present(skb))
		return skb;

	vid = skb_vlan_tag_get_id(skb);

	/* We already run under an RCU read-side critical section since
	 * we are called from netif_receive_skb_list_internal().
	 */
	err = br_vlan_get_pvid_rcu(dev, &pvid);
	if (err)
		return skb;

	if (vid != pvid)
		return skb;

	/* The sad part about attempting to untag from DSA is that we
	 * don't know, unless we check, if the skb will end up in
	 * the bridge's data path - br_allowed_ingress() - or not.
	 * For example, there might be an 8021q upper for the
	 * default_pvid of the bridge, which will steal VLAN-tagged traffic
	 * from the bridge's data path. This is a configuration that DSA
	 * supports because vlan_filtering is 0. In that case, we should
	 * definitely keep the tag, to make sure it keeps working.
	 */
	netdev_for_each_upper_dev_rcu(dev, upper_dev, iter) {
		if (!is_vlan_dev(upper_dev))
			continue;

		if (vid == vlan_dev_vlan_id(upper_dev))
			return skb;
	}

	__vlan_hwaccel_clear_tag(skb);

	return skb;
}

V
Vivien Didelot 已提交
262 263 264
/* switch.c */
int dsa_switch_register_notifier(struct dsa_switch *ds);
void dsa_switch_unregister_notifier(struct dsa_switch *ds);
265 266 267 268

/* dsa2.c */
extern struct list_head dsa_tree_list;

269
#endif