aboutsummaryrefslogtreecommitdiffstats
path: root/net/dsa
diff options
context:
space:
mode:
authorFlorian Fainelli <f.fainelli@gmail.com>2014-08-27 20:04:46 -0400
committerDavid S. Miller <davem@davemloft.net>2014-08-28 01:59:39 -0400
commit3e8a72d1dae374cf6fc1dba97cec663585845ff9 (patch)
tree83a2ebc590d0dc90d27515dc15bf6323cb2a6c1b /net/dsa
parent8663dc2002b02abfe5dfb0fb7e544b81982ad95b (diff)
net: dsa: reduce number of protocol hooks
DSA is currently registering one packet_type function per EtherType it needs to intercept in the receive path of a DSA-enabled Ethernet device. Right now we have three of them: trailer, DSA and eDSA, and there might be more in the future, this will not scale to the addition of new protocols. This patch proceeds with adding a new layer of abstraction and two new functions: dsa_switch_rcv() which will dispatch into the tag-protocol specific receive function implemented by net/dsa/tag_*.c dsa_slave_xmit() which will dispatch into the tag-protocol specific transmit function implemented by net/dsa/tag_*.c When we do create the per-port slave network devices, we iterate over the switch protocol to assign the DSA-specific receive and transmit operations. A new fake ethertype value is used: ETH_P_XDSA to illustrate the fact that this is no longer going to look like ETH_P_DSA or ETH_P_TRAILER like it used to be. This allows us to greatly simplify the check in eth_type_trans() and always override the skb->protocol with ETH_P_XDSA for Ethernet switches tagged protocol, while also reducing the number repetitive slave netdevice_ops assignments. Signed-off-by: Florian Fainelli <f.fainelli@gmail.com> Signed-off-by: David S. Miller <davem@davemloft.net>
Diffstat (limited to 'net/dsa')
-rw-r--r--net/dsa/dsa.c39
-rw-r--r--net/dsa/dsa_priv.h9
-rw-r--r--net/dsa/slave.c45
-rw-r--r--net/dsa/tag_dsa.c8
-rw-r--r--net/dsa/tag_edsa.c8
-rw-r--r--net/dsa/tag_trailer.c8
6 files changed, 50 insertions, 67 deletions
diff --git a/net/dsa/dsa.c b/net/dsa/dsa.c
index 0a49632fac47..92e71d2a2ccd 100644
--- a/net/dsa/dsa.c
+++ b/net/dsa/dsa.c
@@ -608,6 +608,24 @@ static void dsa_shutdown(struct platform_device *pdev)
608{ 608{
609} 609}
610 610
611static int dsa_switch_rcv(struct sk_buff *skb, struct net_device *dev,
612 struct packet_type *pt, struct net_device *orig_dev)
613{
614 struct dsa_switch_tree *dst = dev->dsa_ptr;
615
616 if (unlikely(dst == NULL)) {
617 kfree_skb(skb);
618 return 0;
619 }
620
621 return dst->ops->rcv(skb, dev, pt, orig_dev);
622}
623
624struct packet_type dsa_pack_type __read_mostly = {
625 .type = cpu_to_be16(ETH_P_XDSA),
626 .func = dsa_switch_rcv,
627};
628
611static const struct of_device_id dsa_of_match_table[] = { 629static const struct of_device_id dsa_of_match_table[] = {
612 { .compatible = "marvell,dsa", }, 630 { .compatible = "marvell,dsa", },
613 {} 631 {}
@@ -633,30 +651,15 @@ static int __init dsa_init_module(void)
633 if (rc) 651 if (rc)
634 return rc; 652 return rc;
635 653
636#ifdef CONFIG_NET_DSA_TAG_DSA 654 dev_add_pack(&dsa_pack_type);
637 dev_add_pack(&dsa_packet_type); 655
638#endif
639#ifdef CONFIG_NET_DSA_TAG_EDSA
640 dev_add_pack(&edsa_packet_type);
641#endif
642#ifdef CONFIG_NET_DSA_TAG_TRAILER
643 dev_add_pack(&trailer_packet_type);
644#endif
645 return 0; 656 return 0;
646} 657}
647module_init(dsa_init_module); 658module_init(dsa_init_module);
648 659
649static void __exit dsa_cleanup_module(void) 660static void __exit dsa_cleanup_module(void)
650{ 661{
651#ifdef CONFIG_NET_DSA_TAG_TRAILER 662 dev_remove_pack(&dsa_pack_type);
652 dev_remove_pack(&trailer_packet_type);
653#endif
654#ifdef CONFIG_NET_DSA_TAG_EDSA
655 dev_remove_pack(&edsa_packet_type);
656#endif
657#ifdef CONFIG_NET_DSA_TAG_DSA
658 dev_remove_pack(&dsa_packet_type);
659#endif
660 platform_driver_unregister(&dsa_driver); 663 platform_driver_unregister(&dsa_driver);
661} 664}
662module_exit(dsa_cleanup_module); 665module_exit(dsa_cleanup_module);
diff --git a/net/dsa/dsa_priv.h b/net/dsa/dsa_priv.h
index d4cf5cc747e3..218d75d16f6f 100644
--- a/net/dsa/dsa_priv.h
+++ b/net/dsa/dsa_priv.h
@@ -45,16 +45,13 @@ struct net_device *dsa_slave_create(struct dsa_switch *ds,
45 int port, char *name); 45 int port, char *name);
46 46
47/* tag_dsa.c */ 47/* tag_dsa.c */
48netdev_tx_t dsa_xmit(struct sk_buff *skb, struct net_device *dev); 48extern const struct dsa_device_ops dsa_netdev_ops;
49extern struct packet_type dsa_packet_type;
50 49
51/* tag_edsa.c */ 50/* tag_edsa.c */
52netdev_tx_t edsa_xmit(struct sk_buff *skb, struct net_device *dev); 51extern const struct dsa_device_ops edsa_netdev_ops;
53extern struct packet_type edsa_packet_type;
54 52
55/* tag_trailer.c */ 53/* tag_trailer.c */
56netdev_tx_t trailer_xmit(struct sk_buff *skb, struct net_device *dev); 54extern const struct dsa_device_ops trailer_netdev_ops;
57extern struct packet_type trailer_packet_type;
58 55
59 56
60#endif 57#endif
diff --git a/net/dsa/slave.c b/net/dsa/slave.c
index 45a1e34c89e0..ad1a913533aa 100644
--- a/net/dsa/slave.c
+++ b/net/dsa/slave.c
@@ -171,6 +171,14 @@ static int dsa_slave_ioctl(struct net_device *dev, struct ifreq *ifr, int cmd)
171 return -EOPNOTSUPP; 171 return -EOPNOTSUPP;
172} 172}
173 173
174static netdev_tx_t dsa_slave_xmit(struct sk_buff *skb, struct net_device *dev)
175{
176 struct dsa_slave_priv *p = netdev_priv(dev);
177 struct dsa_switch_tree *dst = p->parent->dst;
178
179 return dst->ops->xmit(skb, dev);
180}
181
174 182
175/* ethtool operations *******************************************************/ 183/* ethtool operations *******************************************************/
176static int 184static int
@@ -293,42 +301,16 @@ static const struct ethtool_ops dsa_slave_ethtool_ops = {
293 .get_sset_count = dsa_slave_get_sset_count, 301 .get_sset_count = dsa_slave_get_sset_count,
294}; 302};
295 303
296#ifdef CONFIG_NET_DSA_TAG_DSA 304static const struct net_device_ops dsa_slave_netdev_ops = {
297static const struct net_device_ops dsa_netdev_ops = {
298 .ndo_init = dsa_slave_init, 305 .ndo_init = dsa_slave_init,
299 .ndo_open = dsa_slave_open, 306 .ndo_open = dsa_slave_open,
300 .ndo_stop = dsa_slave_close, 307 .ndo_stop = dsa_slave_close,
301 .ndo_start_xmit = dsa_xmit, 308 .ndo_start_xmit = dsa_slave_xmit,
302 .ndo_change_rx_flags = dsa_slave_change_rx_flags, 309 .ndo_change_rx_flags = dsa_slave_change_rx_flags,
303 .ndo_set_rx_mode = dsa_slave_set_rx_mode, 310 .ndo_set_rx_mode = dsa_slave_set_rx_mode,
304 .ndo_set_mac_address = dsa_slave_set_mac_address, 311 .ndo_set_mac_address = dsa_slave_set_mac_address,
305 .ndo_do_ioctl = dsa_slave_ioctl, 312 .ndo_do_ioctl = dsa_slave_ioctl,
306}; 313};
307#endif
308#ifdef CONFIG_NET_DSA_TAG_EDSA
309static const struct net_device_ops edsa_netdev_ops = {
310 .ndo_init = dsa_slave_init,
311 .ndo_open = dsa_slave_open,
312 .ndo_stop = dsa_slave_close,
313 .ndo_start_xmit = edsa_xmit,
314 .ndo_change_rx_flags = dsa_slave_change_rx_flags,
315 .ndo_set_rx_mode = dsa_slave_set_rx_mode,
316 .ndo_set_mac_address = dsa_slave_set_mac_address,
317 .ndo_do_ioctl = dsa_slave_ioctl,
318};
319#endif
320#ifdef CONFIG_NET_DSA_TAG_TRAILER
321static const struct net_device_ops trailer_netdev_ops = {
322 .ndo_init = dsa_slave_init,
323 .ndo_open = dsa_slave_open,
324 .ndo_stop = dsa_slave_close,
325 .ndo_start_xmit = trailer_xmit,
326 .ndo_change_rx_flags = dsa_slave_change_rx_flags,
327 .ndo_set_rx_mode = dsa_slave_set_rx_mode,
328 .ndo_set_mac_address = dsa_slave_set_mac_address,
329 .ndo_do_ioctl = dsa_slave_ioctl,
330};
331#endif
332 314
333/* slave device setup *******************************************************/ 315/* slave device setup *******************************************************/
334struct net_device * 316struct net_device *
@@ -349,21 +331,22 @@ dsa_slave_create(struct dsa_switch *ds, struct device *parent,
349 slave_dev->ethtool_ops = &dsa_slave_ethtool_ops; 331 slave_dev->ethtool_ops = &dsa_slave_ethtool_ops;
350 eth_hw_addr_inherit(slave_dev, master); 332 eth_hw_addr_inherit(slave_dev, master);
351 slave_dev->tx_queue_len = 0; 333 slave_dev->tx_queue_len = 0;
334 slave_dev->netdev_ops = &dsa_slave_netdev_ops;
352 335
353 switch (ds->dst->tag_protocol) { 336 switch (ds->dst->tag_protocol) {
354#ifdef CONFIG_NET_DSA_TAG_DSA 337#ifdef CONFIG_NET_DSA_TAG_DSA
355 case htons(ETH_P_DSA): 338 case htons(ETH_P_DSA):
356 slave_dev->netdev_ops = &dsa_netdev_ops; 339 ds->dst->ops = &dsa_netdev_ops;
357 break; 340 break;
358#endif 341#endif
359#ifdef CONFIG_NET_DSA_TAG_EDSA 342#ifdef CONFIG_NET_DSA_TAG_EDSA
360 case htons(ETH_P_EDSA): 343 case htons(ETH_P_EDSA):
361 slave_dev->netdev_ops = &edsa_netdev_ops; 344 ds->dst->ops = &edsa_netdev_ops;
362 break; 345 break;
363#endif 346#endif
364#ifdef CONFIG_NET_DSA_TAG_TRAILER 347#ifdef CONFIG_NET_DSA_TAG_TRAILER
365 case htons(ETH_P_TRAILER): 348 case htons(ETH_P_TRAILER):
366 slave_dev->netdev_ops = &trailer_netdev_ops; 349 ds->dst->ops = &trailer_netdev_ops;
367 break; 350 break;
368#endif 351#endif
369 default: 352 default:
diff --git a/net/dsa/tag_dsa.c b/net/dsa/tag_dsa.c
index cacce1e22f9c..d7dbc5bda5c0 100644
--- a/net/dsa/tag_dsa.c
+++ b/net/dsa/tag_dsa.c
@@ -16,7 +16,7 @@
16 16
17#define DSA_HLEN 4 17#define DSA_HLEN 4
18 18
19netdev_tx_t dsa_xmit(struct sk_buff *skb, struct net_device *dev) 19static netdev_tx_t dsa_xmit(struct sk_buff *skb, struct net_device *dev)
20{ 20{
21 struct dsa_slave_priv *p = netdev_priv(dev); 21 struct dsa_slave_priv *p = netdev_priv(dev);
22 u8 *dsa_header; 22 u8 *dsa_header;
@@ -186,7 +186,7 @@ out:
186 return 0; 186 return 0;
187} 187}
188 188
189struct packet_type dsa_packet_type __read_mostly = { 189const struct dsa_device_ops dsa_netdev_ops = {
190 .type = cpu_to_be16(ETH_P_DSA), 190 .xmit = dsa_xmit,
191 .func = dsa_rcv, 191 .rcv = dsa_rcv,
192}; 192};
diff --git a/net/dsa/tag_edsa.c b/net/dsa/tag_edsa.c
index e70c43c25e64..6b30abe89183 100644
--- a/net/dsa/tag_edsa.c
+++ b/net/dsa/tag_edsa.c
@@ -17,7 +17,7 @@
17#define DSA_HLEN 4 17#define DSA_HLEN 4
18#define EDSA_HLEN 8 18#define EDSA_HLEN 8
19 19
20netdev_tx_t edsa_xmit(struct sk_buff *skb, struct net_device *dev) 20static netdev_tx_t edsa_xmit(struct sk_buff *skb, struct net_device *dev)
21{ 21{
22 struct dsa_slave_priv *p = netdev_priv(dev); 22 struct dsa_slave_priv *p = netdev_priv(dev);
23 u8 *edsa_header; 23 u8 *edsa_header;
@@ -205,7 +205,7 @@ out:
205 return 0; 205 return 0;
206} 206}
207 207
208struct packet_type edsa_packet_type __read_mostly = { 208const struct dsa_device_ops edsa_netdev_ops = {
209 .type = cpu_to_be16(ETH_P_EDSA), 209 .xmit = edsa_xmit,
210 .func = edsa_rcv, 210 .rcv = edsa_rcv,
211}; 211};
diff --git a/net/dsa/tag_trailer.c b/net/dsa/tag_trailer.c
index 94bc260d015d..5fe9444842c5 100644
--- a/net/dsa/tag_trailer.c
+++ b/net/dsa/tag_trailer.c
@@ -14,7 +14,7 @@
14#include <linux/slab.h> 14#include <linux/slab.h>
15#include "dsa_priv.h" 15#include "dsa_priv.h"
16 16
17netdev_tx_t trailer_xmit(struct sk_buff *skb, struct net_device *dev) 17static netdev_tx_t trailer_xmit(struct sk_buff *skb, struct net_device *dev)
18{ 18{
19 struct dsa_slave_priv *p = netdev_priv(dev); 19 struct dsa_slave_priv *p = netdev_priv(dev);
20 struct sk_buff *nskb; 20 struct sk_buff *nskb;
@@ -114,7 +114,7 @@ out:
114 return 0; 114 return 0;
115} 115}
116 116
117struct packet_type trailer_packet_type __read_mostly = { 117const struct dsa_device_ops trailer_netdev_ops = {
118 .type = cpu_to_be16(ETH_P_TRAILER), 118 .xmit = trailer_xmit,
119 .func = trailer_rcv, 119 .rcv = trailer_rcv,
120}; 120};