dsa.c 5.8 KB
Newer Older
1 2
/*
 * net/dsa/dsa.c - Hardware switch handling
3
 * Copyright (c) 2008-2009 Marvell Semiconductor
4
 * Copyright (c) 2013 Florian Fainelli <florian@openwrt.org>
5 6 7 8 9 10 11
 *
 * This program is free software; you can redistribute it and/or modify
 * it under the terms of the GNU General Public License as published by
 * the Free Software Foundation; either version 2 of the License, or
 * (at your option) any later version.
 */

12
#include <linux/device.h>
13 14
#include <linux/list.h>
#include <linux/platform_device.h>
15
#include <linux/slab.h>
16
#include <linux/module.h>
17 18 19
#include <linux/of.h>
#include <linux/of_mdio.h>
#include <linux/of_platform.h>
20
#include <linux/of_net.h>
21
#include <linux/of_gpio.h>
22
#include <linux/netdevice.h>
23
#include <linux/sysfs.h>
24
#include <linux/phy_fixed.h>
25
#include <linux/gpio/consumer.h>
26
#include <linux/etherdevice.h>
27

28 29
#include "dsa_priv.h"

30 31 32 33 34 35 36 37 38 39 40 41 42
static struct sk_buff *dsa_slave_notag_xmit(struct sk_buff *skb,
					    struct net_device *dev)
{
	/* Just return the original SKB */
	return skb;
}

static const struct dsa_device_ops none_ops = {
	.xmit	= dsa_slave_notag_xmit,
	.rcv	= NULL,
};

const struct dsa_device_ops *dsa_device_ops[DSA_TAG_LAST] = {
43 44 45
#ifdef CONFIG_NET_DSA_TAG_BRCM
	[DSA_TAG_PROTO_BRCM] = &brcm_netdev_ops,
#endif
46 47 48 49 50 51
#ifdef CONFIG_NET_DSA_TAG_DSA
	[DSA_TAG_PROTO_DSA] = &dsa_netdev_ops,
#endif
#ifdef CONFIG_NET_DSA_TAG_EDSA
	[DSA_TAG_PROTO_EDSA] = &edsa_netdev_ops,
#endif
52 53 54
#ifdef CONFIG_NET_DSA_TAG_KSZ
	[DSA_TAG_PROTO_KSZ] = &ksz_netdev_ops,
#endif
55 56
#ifdef CONFIG_NET_DSA_TAG_LAN9303
	[DSA_TAG_PROTO_LAN9303] = &lan9303_netdev_ops,
57
#endif
58 59
#ifdef CONFIG_NET_DSA_TAG_MTK
	[DSA_TAG_PROTO_MTK] = &mtk_netdev_ops,
60 61 62
#endif
#ifdef CONFIG_NET_DSA_TAG_QCA
	[DSA_TAG_PROTO_QCA] = &qca_netdev_ops,
63
#endif
64 65
#ifdef CONFIG_NET_DSA_TAG_TRAILER
	[DSA_TAG_PROTO_TRAILER] = &trailer_netdev_ops,
66 67 68
#endif
	[DSA_TAG_PROTO_NONE] = &none_ops,
};
69

70
int dsa_cpu_dsa_setup(struct dsa_switch *ds, struct device *dev,
71
		      struct dsa_port *dport, int port)
72
{
73
	struct device_node *port_dn = dport->dn;
74
	struct phy_device *phydev;
75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91
	int ret, mode;

	if (of_phy_is_fixed_link(port_dn)) {
		ret = of_phy_register_fixed_link(port_dn);
		if (ret) {
			dev_err(dev, "failed to register fixed PHY\n");
			return ret;
		}
		phydev = of_phy_find_device(port_dn);

		mode = of_get_phy_mode(port_dn);
		if (mode < 0)
			mode = PHY_INTERFACE_MODE_NA;
		phydev->interface = mode;

		genphy_config_init(phydev);
		genphy_read_status(phydev);
92 93
		if (ds->ops->adjust_link)
			ds->ops->adjust_link(ds, port, phydev);
94 95

		put_device(&phydev->mdio.dev);
96 97 98 99 100
	}

	return 0;
}

101 102 103 104 105 106 107 108 109 110 111 112 113 114
const struct dsa_device_ops *dsa_resolve_tag_protocol(int tag_protocol)
{
	const struct dsa_device_ops *ops;

	if (tag_protocol >= DSA_TAG_LAST)
		return ERR_PTR(-EINVAL);
	ops = dsa_device_ops[tag_protocol];

	if (!ops)
		return ERR_PTR(-ENOPROTOOPT);

	return ops;
}

115
int dsa_cpu_port_ethtool_setup(struct dsa_port *cpu_dp)
116
{
117
	struct dsa_switch *ds = cpu_dp->ds;
118 119 120 121 122 123 124 125 126 127 128 129 130 131 132 133 134 135 136 137 138 139
	struct net_device *master;
	struct ethtool_ops *cpu_ops;

	master = ds->dst->master_netdev;
	if (ds->master_netdev)
		master = ds->master_netdev;

	cpu_ops = devm_kzalloc(ds->dev, sizeof(*cpu_ops), GFP_KERNEL);
	if (!cpu_ops)
		return -ENOMEM;

	memcpy(&ds->dst->master_ethtool_ops, master->ethtool_ops,
	       sizeof(struct ethtool_ops));
	ds->dst->master_orig_ethtool_ops = master->ethtool_ops;
	memcpy(cpu_ops, &ds->dst->master_ethtool_ops,
	       sizeof(struct ethtool_ops));
	dsa_cpu_port_ethtool_init(cpu_ops);
	master->ethtool_ops = cpu_ops;

	return 0;
}

140
void dsa_cpu_port_ethtool_restore(struct dsa_port *cpu_dp)
141
{
142
	struct dsa_switch *ds = cpu_dp->ds;
143 144 145 146 147 148 149 150 151
	struct net_device *master;

	master = ds->dst->master_netdev;
	if (ds->master_netdev)
		master = ds->master_netdev;

	master->ethtool_ops = ds->dst->master_orig_ethtool_ops;
}

152
void dsa_cpu_dsa_destroy(struct dsa_port *port)
153
{
154 155
	struct device_node *port_dn = port->dn;

156 157
	if (of_phy_is_fixed_link(port_dn))
		of_phy_deregister_fixed_link(port_dn);
158 159
}

160 161 162 163 164 165 166 167 168 169 170 171 172 173 174 175 176 177
static int dev_is_class(struct device *dev, void *class)
{
	if (dev->class != NULL && !strcmp(dev->class->name, class))
		return 1;

	return 0;
}

static struct device *dev_find_class(struct device *parent, char *class)
{
	if (dev_is_class(parent, class)) {
		get_device(parent);
		return parent;
	}

	return device_find_child(parent, class, dev_is_class);
}

178
struct net_device *dsa_dev_to_net_device(struct device *dev)
179 180 181 182 183 184 185 186 187 188 189 190 191 192 193 194
{
	struct device *d;

	d = dev_find_class(dev, "net");
	if (d != NULL) {
		struct net_device *nd;

		nd = to_net_dev(d);
		dev_hold(nd);
		put_device(d);

		return nd;
	}

	return NULL;
}
195
EXPORT_SYMBOL_GPL(dsa_dev_to_net_device);
196

197 198 199 200
static int dsa_switch_rcv(struct sk_buff *skb, struct net_device *dev,
			  struct packet_type *pt, struct net_device *orig_dev)
{
	struct dsa_switch_tree *dst = dev->dsa_ptr;
201
	struct sk_buff *nskb = NULL;
202 203 204 205 206 207

	if (unlikely(dst == NULL)) {
		kfree_skb(skb);
		return 0;
	}

208 209 210 211
	skb = skb_unshare(skb, GFP_ATOMIC);
	if (!skb)
		return 0;

212 213 214 215 216 217 218 219 220 221 222 223 224 225 226 227 228
	nskb = dst->rcv(skb, dev, pt, orig_dev);
	if (!nskb) {
		kfree_skb(skb);
		return 0;
	}

	skb = nskb;
	skb_push(skb, ETH_HLEN);
	skb->pkt_type = PACKET_HOST;
	skb->protocol = eth_type_trans(skb, skb->dev);

	skb->dev->stats.rx_packets++;
	skb->dev->stats.rx_bytes += skb->len;

	netif_receive_skb(skb);

	return 0;
229 230
}

231
static struct packet_type dsa_pack_type __read_mostly = {
232 233 234 235
	.type	= cpu_to_be16(ETH_P_XDSA),
	.func	= dsa_switch_rcv,
};

236 237
static int __init dsa_init_module(void)
{
B
Ben Hutchings 已提交
238 239
	int rc;

240 241 242
	rc = dsa_slave_register_notifier();
	if (rc)
		return rc;
243

V
Vivien Didelot 已提交
244
	rc = dsa_legacy_register();
B
Ben Hutchings 已提交
245 246 247
	if (rc)
		return rc;

248 249
	dev_add_pack(&dsa_pack_type);

B
Ben Hutchings 已提交
250
	return 0;
251 252 253 254 255
}
module_init(dsa_init_module);

static void __exit dsa_cleanup_module(void)
{
256
	dsa_slave_unregister_notifier();
257
	dev_remove_pack(&dsa_pack_type);
V
Vivien Didelot 已提交
258
	dsa_legacy_unregister();
259 260 261
}
module_exit(dsa_cleanup_module);

262
MODULE_AUTHOR("Lennert Buytenhek <buytenh@wantstofly.org>");
263 264 265
MODULE_DESCRIPTION("Driver for Distributed Switch Architecture switch chips");
MODULE_LICENSE("GPL");
MODULE_ALIAS("platform:dsa");