blob: d3f1a760746382f38a656191c22d533a2c29fa58 [file] [log] [blame]
Andrew Lunn83c0afa2016-06-04 21:17:07 +02001/*
2 * net/dsa/dsa2.c - Hardware switch handling, binding version 2
3 * Copyright (c) 2008-2009 Marvell Semiconductor
4 * Copyright (c) 2013 Florian Fainelli <florian@openwrt.org>
5 * Copyright (c) 2016 Andrew Lunn <andrew@lunn.ch>
6 *
7 * This program is free software; you can redistribute it and/or modify
8 * it under the terms of the GNU General Public License as published by
9 * the Free Software Foundation; either version 2 of the License, or
10 * (at your option) any later version.
11 */
12
13#include <linux/device.h>
14#include <linux/err.h>
15#include <linux/list.h>
Andrew Lunnc6e970a2017-03-28 23:45:06 +020016#include <linux/netdevice.h>
Andrew Lunn83c0afa2016-06-04 21:17:07 +020017#include <linux/slab.h>
18#include <linux/rtnetlink.h>
Andrew Lunn83c0afa2016-06-04 21:17:07 +020019#include <linux/of.h>
20#include <linux/of_net.h>
Vivien Didelotea5dd342017-05-17 15:46:03 -040021
Andrew Lunn83c0afa2016-06-04 21:17:07 +020022#include "dsa_priv.h"
23
24static LIST_HEAD(dsa_switch_trees);
25static DEFINE_MUTEX(dsa2_mutex);
26
Andrew Lunn96567d52017-03-28 23:45:07 +020027static const struct devlink_ops dsa_devlink_ops = {
28};
29
Vivien Didelot49463b72017-11-03 19:05:21 -040030static struct dsa_switch_tree *dsa_get_dst(unsigned int index)
Andrew Lunn83c0afa2016-06-04 21:17:07 +020031{
32 struct dsa_switch_tree *dst;
33
34 list_for_each_entry(dst, &dsa_switch_trees, list)
Vivien Didelot8e5bf972017-11-03 19:05:22 -040035 if (dst->index == index)
Andrew Lunn83c0afa2016-06-04 21:17:07 +020036 return dst;
Vivien Didelot8e5bf972017-11-03 19:05:22 -040037
Andrew Lunn83c0afa2016-06-04 21:17:07 +020038 return NULL;
39}
40
41static void dsa_free_dst(struct kref *ref)
42{
43 struct dsa_switch_tree *dst = container_of(ref, struct dsa_switch_tree,
44 refcount);
45
46 list_del(&dst->list);
47 kfree(dst);
48}
49
Vivien Didelot49463b72017-11-03 19:05:21 -040050static struct dsa_switch_tree *dsa_add_dst(unsigned int index)
Andrew Lunn83c0afa2016-06-04 21:17:07 +020051{
52 struct dsa_switch_tree *dst;
53
54 dst = kzalloc(sizeof(*dst), GFP_KERNEL);
55 if (!dst)
56 return NULL;
Vivien Didelot49463b72017-11-03 19:05:21 -040057 dst->index = index;
Andrew Lunn83c0afa2016-06-04 21:17:07 +020058 INIT_LIST_HEAD(&dst->list);
59 list_add_tail(&dsa_switch_trees, &dst->list);
Vivien Didelot8e5bf972017-11-03 19:05:22 -040060
61 /* Initialize the reference counter to the number of switches, not 1 */
Andrew Lunn83c0afa2016-06-04 21:17:07 +020062 kref_init(&dst->refcount);
Vivien Didelot8e5bf972017-11-03 19:05:22 -040063 refcount_set(&dst->refcount.refcount, 0);
Andrew Lunn83c0afa2016-06-04 21:17:07 +020064
65 return dst;
66}
67
68static void dsa_dst_add_ds(struct dsa_switch_tree *dst,
69 struct dsa_switch *ds, u32 index)
70{
71 kref_get(&dst->refcount);
72 dst->ds[index] = ds;
73}
74
75static void dsa_dst_del_ds(struct dsa_switch_tree *dst,
76 struct dsa_switch *ds, u32 index)
77{
78 dst->ds[index] = NULL;
79 kref_put(&dst->refcount, dsa_free_dst);
80}
81
Florian Fainelli71e0bbd2017-02-04 13:02:43 -080082/* For platform data configurations, we need to have a valid name argument to
83 * differentiate a disabled port from an enabled one
84 */
Florian Fainelli293784a2017-01-26 10:45:52 -080085static bool dsa_port_is_valid(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +020086{
Vivien Didelot6d4e5c52017-10-27 15:55:15 -040087 return port->type != DSA_PORT_TYPE_UNUSED;
Andrew Lunn83c0afa2016-06-04 21:17:07 +020088}
89
Florian Fainelli293784a2017-01-26 10:45:52 -080090static bool dsa_port_is_dsa(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +020091{
Vivien Didelot6d4e5c52017-10-27 15:55:15 -040092 return port->type == DSA_PORT_TYPE_DSA;
Florian Fainelli293784a2017-01-26 10:45:52 -080093}
94
95static bool dsa_port_is_cpu(struct dsa_port *port)
96{
Vivien Didelot6d4e5c52017-10-27 15:55:15 -040097 return port->type == DSA_PORT_TYPE_CPU;
Andrew Lunn83c0afa2016-06-04 21:17:07 +020098}
99
Florian Fainelli3512a8e2017-01-26 10:45:53 -0800100static bool dsa_ds_find_port_dn(struct dsa_switch *ds,
101 struct device_node *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200102{
103 u32 index;
104
Vivien Didelot26895e22017-01-27 15:29:37 -0500105 for (index = 0; index < ds->num_ports; index++)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200106 if (ds->ports[index].dn == port)
107 return true;
108 return false;
109}
110
Florian Fainelli3512a8e2017-01-26 10:45:53 -0800111static struct dsa_switch *dsa_dst_find_port_dn(struct dsa_switch_tree *dst,
112 struct device_node *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200113{
114 struct dsa_switch *ds;
115 u32 index;
116
117 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
118 ds = dst->ds[index];
119 if (!ds)
120 continue;
121
Florian Fainelli3512a8e2017-01-26 10:45:53 -0800122 if (dsa_ds_find_port_dn(ds, port))
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200123 return ds;
124 }
125
126 return NULL;
127}
128
129static int dsa_port_complete(struct dsa_switch_tree *dst,
130 struct dsa_switch *src_ds,
Florian Fainelli293784a2017-01-26 10:45:52 -0800131 struct dsa_port *port,
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200132 u32 src_port)
133{
134 struct device_node *link;
135 int index;
136 struct dsa_switch *dst_ds;
137
138 for (index = 0;; index++) {
Florian Fainelli293784a2017-01-26 10:45:52 -0800139 link = of_parse_phandle(port->dn, "link", index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200140 if (!link)
141 break;
142
Florian Fainelli3512a8e2017-01-26 10:45:53 -0800143 dst_ds = dsa_dst_find_port_dn(dst, link);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200144 of_node_put(link);
145
146 if (!dst_ds)
147 return 1;
148
149 src_ds->rtable[dst_ds->index] = src_port;
150 }
151
152 return 0;
153}
154
155/* A switch is complete if all the DSA ports phandles point to ports
156 * known in the tree. A return value of 1 means the tree is not
157 * complete. This is not an error condition. A value of 0 is
158 * success.
159 */
160static int dsa_ds_complete(struct dsa_switch_tree *dst, struct dsa_switch *ds)
161{
Florian Fainelli293784a2017-01-26 10:45:52 -0800162 struct dsa_port *port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200163 u32 index;
164 int err;
165
Vivien Didelot26895e22017-01-27 15:29:37 -0500166 for (index = 0; index < ds->num_ports; index++) {
Florian Fainelli293784a2017-01-26 10:45:52 -0800167 port = &ds->ports[index];
168 if (!dsa_port_is_valid(port))
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200169 continue;
170
171 if (!dsa_port_is_dsa(port))
172 continue;
173
174 err = dsa_port_complete(dst, ds, port, index);
175 if (err != 0)
176 return err;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200177 }
178
179 return 0;
180}
181
182/* A tree is complete if all the DSA ports phandles point to ports
183 * known in the tree. A return value of 1 means the tree is not
184 * complete. This is not an error condition. A value of 0 is
185 * success.
186 */
187static int dsa_dst_complete(struct dsa_switch_tree *dst)
188{
189 struct dsa_switch *ds;
190 u32 index;
191 int err;
192
193 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
194 ds = dst->ds[index];
195 if (!ds)
196 continue;
197
198 err = dsa_ds_complete(dst, ds);
199 if (err != 0)
200 return err;
201 }
202
203 return 0;
204}
205
Florian Fainellie41c1b52017-06-02 12:31:22 -0700206static int dsa_dsa_port_apply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200207{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700208 struct dsa_switch *ds = port->ds;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200209 int err;
210
Vivien Didelot57ab1ca2017-10-26 10:50:07 -0400211 err = dsa_port_fixed_link_register_of(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200212 if (err) {
213 dev_warn(ds->dev, "Failed to setup dsa port %d: %d\n",
Florian Fainellie41c1b52017-06-02 12:31:22 -0700214 port->index, err);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200215 return err;
216 }
217
Florian Fainellie41c1b52017-06-02 12:31:22 -0700218 memset(&port->devlink_port, 0, sizeof(port->devlink_port));
Andrew Lunn96567d52017-03-28 23:45:07 +0200219
Florian Fainellie41c1b52017-06-02 12:31:22 -0700220 return devlink_port_register(ds->devlink, &port->devlink_port,
221 port->index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200222}
223
Florian Fainellie41c1b52017-06-02 12:31:22 -0700224static void dsa_dsa_port_unapply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200225{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700226 devlink_port_unregister(&port->devlink_port);
Vivien Didelot57ab1ca2017-10-26 10:50:07 -0400227 dsa_port_fixed_link_unregister_of(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200228}
229
Florian Fainellie41c1b52017-06-02 12:31:22 -0700230static int dsa_cpu_port_apply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200231{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700232 struct dsa_switch *ds = port->ds;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200233 int err;
234
Vivien Didelot57ab1ca2017-10-26 10:50:07 -0400235 err = dsa_port_fixed_link_register_of(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200236 if (err) {
237 dev_warn(ds->dev, "Failed to setup cpu port %d: %d\n",
Florian Fainellie41c1b52017-06-02 12:31:22 -0700238 port->index, err);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200239 return err;
240 }
241
Florian Fainellie41c1b52017-06-02 12:31:22 -0700242 memset(&port->devlink_port, 0, sizeof(port->devlink_port));
243 err = devlink_port_register(ds->devlink, &port->devlink_port,
244 port->index);
Andrew Lunn96567d52017-03-28 23:45:07 +0200245 return err;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200246}
247
Florian Fainellie41c1b52017-06-02 12:31:22 -0700248static void dsa_cpu_port_unapply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200249{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700250 devlink_port_unregister(&port->devlink_port);
Vivien Didelot57ab1ca2017-10-26 10:50:07 -0400251 dsa_port_fixed_link_unregister_of(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200252}
253
Florian Fainellie41c1b52017-06-02 12:31:22 -0700254static int dsa_user_port_apply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200255{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700256 struct dsa_switch *ds = port->ds;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200257 int err;
258
Vivien Didelot951259aa2017-10-27 15:55:19 -0400259 err = dsa_slave_create(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200260 if (err) {
261 dev_warn(ds->dev, "Failed to create slave %d: %d\n",
Florian Fainellie41c1b52017-06-02 12:31:22 -0700262 port->index, err);
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400263 port->slave = NULL;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200264 return err;
265 }
266
Florian Fainellie41c1b52017-06-02 12:31:22 -0700267 memset(&port->devlink_port, 0, sizeof(port->devlink_port));
268 err = devlink_port_register(ds->devlink, &port->devlink_port,
269 port->index);
Andrew Lunn96567d52017-03-28 23:45:07 +0200270 if (err)
271 return err;
272
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400273 devlink_port_type_eth_set(&port->devlink_port, port->slave);
Andrew Lunn96567d52017-03-28 23:45:07 +0200274
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200275 return 0;
276}
277
Florian Fainellie41c1b52017-06-02 12:31:22 -0700278static void dsa_user_port_unapply(struct dsa_port *port)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200279{
Florian Fainellie41c1b52017-06-02 12:31:22 -0700280 devlink_port_unregister(&port->devlink_port);
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400281 if (port->slave) {
282 dsa_slave_destroy(port->slave);
283 port->slave = NULL;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200284 }
285}
286
287static int dsa_ds_apply(struct dsa_switch_tree *dst, struct dsa_switch *ds)
288{
Florian Fainelli293784a2017-01-26 10:45:52 -0800289 struct dsa_port *port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200290 u32 index;
291 int err;
292
Florian Fainelli6e830d82016-06-07 16:32:39 -0700293 /* Initialize ds->phys_mii_mask before registering the slave MDIO bus
Vivien Didelot9d490b42016-08-23 12:38:56 -0400294 * driver and before ops->setup() has run, since the switch drivers and
Florian Fainelli6e830d82016-06-07 16:32:39 -0700295 * the slave MDIO bus driver rely on these values for probing PHY
296 * devices or not
297 */
Vivien Didelot02bc6e52017-10-26 11:22:56 -0400298 ds->phys_mii_mask |= dsa_user_ports(ds);
Florian Fainelli6e830d82016-06-07 16:32:39 -0700299
Andrew Lunn96567d52017-03-28 23:45:07 +0200300 /* Add the switch to devlink before calling setup, so that setup can
301 * add dpipe tables
302 */
303 ds->devlink = devlink_alloc(&dsa_devlink_ops, 0);
304 if (!ds->devlink)
305 return -ENOMEM;
306
307 err = devlink_register(ds->devlink, ds->dev);
308 if (err)
309 return err;
310
Vivien Didelot9d490b42016-08-23 12:38:56 -0400311 err = ds->ops->setup(ds);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200312 if (err < 0)
313 return err;
314
Vivien Didelotf515f192017-02-03 13:20:20 -0500315 err = dsa_switch_register_notifier(ds);
316 if (err)
317 return err;
318
Vivien Didelot9d490b42016-08-23 12:38:56 -0400319 if (!ds->slave_mii_bus && ds->ops->phy_read) {
Florian Fainelli1eb59442016-06-07 16:32:40 -0700320 ds->slave_mii_bus = devm_mdiobus_alloc(ds->dev);
321 if (!ds->slave_mii_bus)
322 return -ENOMEM;
323
324 dsa_slave_mii_bus_init(ds);
325
326 err = mdiobus_register(ds->slave_mii_bus);
327 if (err < 0)
328 return err;
329 }
330
Vivien Didelot26895e22017-01-27 15:29:37 -0500331 for (index = 0; index < ds->num_ports; index++) {
Florian Fainelli293784a2017-01-26 10:45:52 -0800332 port = &ds->ports[index];
333 if (!dsa_port_is_valid(port))
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200334 continue;
335
336 if (dsa_port_is_dsa(port)) {
Florian Fainellie41c1b52017-06-02 12:31:22 -0700337 err = dsa_dsa_port_apply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200338 if (err)
339 return err;
340 continue;
341 }
342
343 if (dsa_port_is_cpu(port)) {
Florian Fainellie41c1b52017-06-02 12:31:22 -0700344 err = dsa_cpu_port_apply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200345 if (err)
346 return err;
347 continue;
348 }
349
Florian Fainellie41c1b52017-06-02 12:31:22 -0700350 err = dsa_user_port_apply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200351 if (err)
352 continue;
353 }
354
355 return 0;
356}
357
358static void dsa_ds_unapply(struct dsa_switch_tree *dst, struct dsa_switch *ds)
359{
Florian Fainelli293784a2017-01-26 10:45:52 -0800360 struct dsa_port *port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200361 u32 index;
362
Vivien Didelot26895e22017-01-27 15:29:37 -0500363 for (index = 0; index < ds->num_ports; index++) {
Florian Fainelli293784a2017-01-26 10:45:52 -0800364 port = &ds->ports[index];
365 if (!dsa_port_is_valid(port))
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200366 continue;
367
368 if (dsa_port_is_dsa(port)) {
Florian Fainellie41c1b52017-06-02 12:31:22 -0700369 dsa_dsa_port_unapply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200370 continue;
371 }
372
373 if (dsa_port_is_cpu(port)) {
Florian Fainellie41c1b52017-06-02 12:31:22 -0700374 dsa_cpu_port_unapply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200375 continue;
376 }
377
Florian Fainellie41c1b52017-06-02 12:31:22 -0700378 dsa_user_port_unapply(port);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200379 }
Florian Fainelli1eb59442016-06-07 16:32:40 -0700380
Vivien Didelot9d490b42016-08-23 12:38:56 -0400381 if (ds->slave_mii_bus && ds->ops->phy_read)
Florian Fainelli1eb59442016-06-07 16:32:40 -0700382 mdiobus_unregister(ds->slave_mii_bus);
Vivien Didelotf515f192017-02-03 13:20:20 -0500383
384 dsa_switch_unregister_notifier(ds);
Andrew Lunn96567d52017-03-28 23:45:07 +0200385
386 if (ds->devlink) {
387 devlink_unregister(ds->devlink);
388 devlink_free(ds->devlink);
389 ds->devlink = NULL;
390 }
391
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200392}
393
394static int dsa_dst_apply(struct dsa_switch_tree *dst)
395{
396 struct dsa_switch *ds;
397 u32 index;
398 int err;
399
400 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
401 ds = dst->ds[index];
402 if (!ds)
403 continue;
404
405 err = dsa_ds_apply(dst, ds);
406 if (err)
407 return err;
408 }
409
410 /* If we use a tagging format that doesn't have an ethertype
411 * field, make sure that all packets from this point on get
412 * sent to the tag format's receive function.
413 */
414 wmb();
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400415 dst->cpu_dp->master->dsa_ptr = dst->cpu_dp;
Vivien Didelot19435632017-09-19 11:56:59 -0400416
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400417 err = dsa_master_ethtool_setup(dst->cpu_dp->master);
Vivien Didelot19435632017-09-19 11:56:59 -0400418 if (err)
419 return err;
420
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200421 dst->applied = true;
422
423 return 0;
424}
425
426static void dsa_dst_unapply(struct dsa_switch_tree *dst)
427{
428 struct dsa_switch *ds;
429 u32 index;
430
431 if (!dst->applied)
432 return;
433
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400434 dsa_master_ethtool_restore(dst->cpu_dp->master);
Vivien Didelot19435632017-09-19 11:56:59 -0400435
Vivien Didelotf8b8b1c2017-10-16 11:12:18 -0400436 dst->cpu_dp->master->dsa_ptr = NULL;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200437
438 /* If we used a tagging format that doesn't have an ethertype
439 * field, make sure that all packets from this point get sent
440 * without the tag and go through the regular receive path.
441 */
442 wmb();
443
444 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
445 ds = dst->ds[index];
446 if (!ds)
447 continue;
448
449 dsa_ds_unapply(dst, ds);
450 }
451
Vivien Didelotcd8d7dd2017-09-19 11:56:58 -0400452 dst->cpu_dp = NULL;
Florian Fainelli0c73c522016-06-07 16:32:42 -0700453
Vivien Didelot49463b72017-11-03 19:05:21 -0400454 pr_info("DSA: tree %d unapplied\n", dst->index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200455 dst->applied = false;
456}
457
Florian Fainelli293784a2017-01-26 10:45:52 -0800458static int dsa_cpu_parse(struct dsa_port *port, u32 index,
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200459 struct dsa_switch_tree *dst,
460 struct dsa_switch *ds)
461{
Vivien Didelot62fc9582017-09-29 17:19:17 -0400462 const struct dsa_device_ops *tag_ops;
Andrew Lunn7b314362016-08-22 16:01:01 +0200463 enum dsa_tag_protocol tag_protocol;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200464
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400465 if (!dst->cpu_dp)
Vivien Didelot8b0d3ea2017-05-16 14:10:33 -0400466 dst->cpu_dp = port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200467
Florian Fainelli9f9e7722017-07-24 10:49:23 -0700468 tag_protocol = ds->ops->get_tag_protocol(ds);
Vivien Didelot62fc9582017-09-29 17:19:17 -0400469 tag_ops = dsa_resolve_tag_protocol(tag_protocol);
470 if (IS_ERR(tag_ops)) {
Florian Fainelli9f9e7722017-07-24 10:49:23 -0700471 dev_warn(ds->dev, "No tagger for this switch\n");
Vivien Didelot62fc9582017-09-29 17:19:17 -0400472 return PTR_ERR(tag_ops);
Florian Fainelli9f9e7722017-07-24 10:49:23 -0700473 }
474
Vivien Didelot15240242017-09-29 17:19:18 -0400475 dst->cpu_dp->tag_ops = tag_ops;
Vivien Didelot3e41f932017-09-29 17:19:19 -0400476
477 /* Make a few copies for faster access in master receive hot path */
478 dst->cpu_dp->rcv = dst->cpu_dp->tag_ops->rcv;
Vivien Didelot3e41f932017-09-29 17:19:19 -0400479 dst->cpu_dp->dst = dst;
Florian Fainelli9f9e7722017-07-24 10:49:23 -0700480
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200481 return 0;
482}
483
484static int dsa_ds_parse(struct dsa_switch_tree *dst, struct dsa_switch *ds)
485{
Florian Fainelli293784a2017-01-26 10:45:52 -0800486 struct dsa_port *port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200487 u32 index;
488 int err;
489
Vivien Didelot26895e22017-01-27 15:29:37 -0500490 for (index = 0; index < ds->num_ports; index++) {
Florian Fainelli293784a2017-01-26 10:45:52 -0800491 port = &ds->ports[index];
Florian Fainelli14be36c2017-06-02 12:31:23 -0700492 if (!dsa_port_is_valid(port) ||
493 dsa_port_is_dsa(port))
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200494 continue;
495
496 if (dsa_port_is_cpu(port)) {
497 err = dsa_cpu_parse(port, index, dst, ds);
498 if (err)
499 return err;
500 }
Florian Fainelli14be36c2017-06-02 12:31:23 -0700501
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200502 }
503
Vivien Didelot49463b72017-11-03 19:05:21 -0400504 pr_info("DSA: switch %d %d parsed\n", dst->index, ds->index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200505
506 return 0;
507}
508
509static int dsa_dst_parse(struct dsa_switch_tree *dst)
510{
511 struct dsa_switch *ds;
Vivien Didelote4b77782017-06-15 15:06:54 -0400512 struct dsa_port *dp;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200513 u32 index;
Vivien Didelote4b77782017-06-15 15:06:54 -0400514 int port;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200515 int err;
516
517 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
518 ds = dst->ds[index];
519 if (!ds)
520 continue;
521
522 err = dsa_ds_parse(dst, ds);
523 if (err)
524 return err;
525 }
526
Florian Fainellic7848392017-08-28 17:10:51 -0700527 if (!dst->cpu_dp) {
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200528 pr_warn("Tree has no master device\n");
529 return -EINVAL;
530 }
531
Vivien Didelote4b77782017-06-15 15:06:54 -0400532 /* Assign the default CPU port to all ports of the fabric */
533 for (index = 0; index < DSA_MAX_SWITCHES; index++) {
534 ds = dst->ds[index];
535 if (!ds)
536 continue;
537
538 for (port = 0; port < ds->num_ports; port++) {
539 dp = &ds->ports[port];
540 if (!dsa_port_is_valid(dp) ||
541 dsa_port_is_dsa(dp) ||
542 dsa_port_is_cpu(dp))
543 continue;
544
545 dp->cpu_dp = dst->cpu_dp;
546 }
547 }
548
Vivien Didelot49463b72017-11-03 19:05:21 -0400549 pr_info("DSA: tree %d parsed\n", dst->index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200550
551 return 0;
552}
553
Vivien Didelotfd223e22017-10-27 15:55:14 -0400554static int dsa_port_parse_of(struct dsa_port *dp, struct device_node *dn)
555{
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400556 struct device_node *ethernet = of_parse_phandle(dn, "ethernet", 0);
557 struct device_node *link = of_parse_phandle(dn, "link", 0);
Vivien Didelot1838fa82017-10-27 15:55:18 -0400558 const char *name = of_get_property(dn, "label", NULL);
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400559
560 if (ethernet) {
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400561 struct net_device *master;
562
563 master = of_find_net_device_by_node(ethernet);
564 if (!master)
565 return -EPROBE_DEFER;
566
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400567 dp->type = DSA_PORT_TYPE_CPU;
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400568 dp->master = master;
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400569 } else if (link) {
570 dp->type = DSA_PORT_TYPE_DSA;
571 } else {
Vivien Didelot1838fa82017-10-27 15:55:18 -0400572 if (!name)
573 name = "eth%d";
574
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400575 dp->type = DSA_PORT_TYPE_USER;
Vivien Didelot1838fa82017-10-27 15:55:18 -0400576 dp->name = name;
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400577 }
578
Vivien Didelotfd223e22017-10-27 15:55:14 -0400579 dp->dn = dn;
580
581 return 0;
582}
583
Vivien Didelot5b32fe02017-10-27 15:55:13 -0400584static int dsa_parse_ports_of(struct device_node *dn, struct dsa_switch *ds)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200585{
Vivien Didelot5b32fe02017-10-27 15:55:13 -0400586 struct device_node *ports, *port;
Vivien Didelotfd223e22017-10-27 15:55:14 -0400587 struct dsa_port *dp;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200588 u32 reg;
Vivien Didelot5b32fe02017-10-27 15:55:13 -0400589 int err;
590
591 ports = of_get_child_by_name(dn, "ports");
592 if (!ports) {
593 dev_err(ds->dev, "no ports child node found\n");
594 return -EINVAL;
595 }
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200596
597 for_each_available_child_of_node(ports, port) {
598 err = of_property_read_u32(port, "reg", &reg);
599 if (err)
600 return err;
601
Vivien Didelot26895e22017-01-27 15:29:37 -0500602 if (reg >= ds->num_ports)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200603 return -EINVAL;
604
Vivien Didelotfd223e22017-10-27 15:55:14 -0400605 dp = &ds->ports[reg];
606
607 err = dsa_port_parse_of(dp, port);
608 if (err)
609 return err;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200610 }
611
612 return 0;
613}
614
Vivien Didelotfd223e22017-10-27 15:55:14 -0400615static int dsa_port_parse(struct dsa_port *dp, const char *name,
616 struct device *dev)
617{
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400618 if (!strcmp(name, "cpu")) {
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400619 struct net_device *master;
620
621 master = dsa_dev_to_net_device(dev);
622 if (!master)
623 return -EPROBE_DEFER;
624
625 dev_put(master);
626
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400627 dp->type = DSA_PORT_TYPE_CPU;
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400628 dp->master = master;
Vivien Didelot6d4e5c52017-10-27 15:55:15 -0400629 } else if (!strcmp(name, "dsa")) {
630 dp->type = DSA_PORT_TYPE_DSA;
631 } else {
632 dp->type = DSA_PORT_TYPE_USER;
633 }
634
Vivien Didelotfd223e22017-10-27 15:55:14 -0400635 dp->name = name;
636
637 return 0;
638}
639
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800640static int dsa_parse_ports(struct dsa_chip_data *cd, struct dsa_switch *ds)
641{
642 bool valid_name_found = false;
Vivien Didelotfd223e22017-10-27 15:55:14 -0400643 struct dsa_port *dp;
644 struct device *dev;
645 const char *name;
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800646 unsigned int i;
Vivien Didelotfd223e22017-10-27 15:55:14 -0400647 int err;
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800648
649 for (i = 0; i < DSA_MAX_PORTS; i++) {
Vivien Didelotfd223e22017-10-27 15:55:14 -0400650 name = cd->port_names[i];
651 dev = cd->netdev[i];
652 dp = &ds->ports[i];
653
654 if (!name)
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800655 continue;
656
Vivien Didelotfd223e22017-10-27 15:55:14 -0400657 err = dsa_port_parse(dp, name, dev);
658 if (err)
659 return err;
660
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800661 valid_name_found = true;
662 }
663
664 if (!valid_name_found && i == DSA_MAX_PORTS)
665 return -EINVAL;
666
667 return 0;
668}
669
Florian Fainelli3512a8e2017-01-26 10:45:53 -0800670static int dsa_parse_member_dn(struct device_node *np, u32 *tree, u32 *index)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200671{
672 int err;
673
674 *tree = *index = 0;
675
676 err = of_property_read_u32_index(np, "dsa,member", 0, tree);
677 if (err) {
678 /* Does not exist, but it is optional */
679 if (err == -EINVAL)
680 return 0;
681 return err;
682 }
683
684 err = of_property_read_u32_index(np, "dsa,member", 1, index);
685 if (err)
686 return err;
687
688 if (*index >= DSA_MAX_SWITCHES)
689 return -EINVAL;
690
691 return 0;
692}
693
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800694static int dsa_parse_member(struct dsa_chip_data *pd, u32 *tree, u32 *index)
695{
696 if (!pd)
697 return -ENODEV;
698
699 /* We do not support complex trees with dsa_chip_data */
700 *tree = 0;
701 *index = 0;
702
703 return 0;
704}
705
Vivien Didelot23c9ee42017-05-26 18:12:51 -0400706static int _dsa_register_switch(struct dsa_switch *ds)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200707{
Vivien Didelot23c9ee42017-05-26 18:12:51 -0400708 struct dsa_chip_data *pdata = ds->dev->platform_data;
709 struct device_node *np = ds->dev->of_node;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200710 struct dsa_switch_tree *dst;
711 u32 tree, index;
Vivien Didelotd3902382016-07-06 20:03:54 -0400712 int i, err;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200713
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800714 if (np) {
715 err = dsa_parse_member_dn(np, &tree, &index);
716 if (err)
717 return err;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200718
Vivien Didelot5b32fe02017-10-27 15:55:13 -0400719 err = dsa_parse_ports_of(np, ds);
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800720 if (err)
721 return err;
722 } else {
723 err = dsa_parse_member(pdata, &tree, &index);
724 if (err)
725 return err;
726
727 err = dsa_parse_ports(pdata, ds);
728 if (err)
729 return err;
730 }
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200731
732 dst = dsa_get_dst(tree);
733 if (!dst) {
734 dst = dsa_add_dst(tree);
735 if (!dst)
736 return -ENOMEM;
737 }
738
Vivien Didelot8e5bf972017-11-03 19:05:22 -0400739 if (dst->ds[index])
740 return -EBUSY;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200741
742 ds->dst = dst;
743 ds->index = index;
Florian Fainelli71e0bbd2017-02-04 13:02:43 -0800744 ds->cd = pdata;
Vivien Didelotd3902382016-07-06 20:03:54 -0400745
746 /* Initialize the routing table */
747 for (i = 0; i < DSA_MAX_SWITCHES; ++i)
748 ds->rtable[i] = DSA_RTABLE_NONE;
749
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200750 dsa_dst_add_ds(dst, ds, index);
751
752 err = dsa_dst_complete(dst);
753 if (err < 0)
754 goto out_del_dst;
755
Vivien Didelot8e5bf972017-11-03 19:05:22 -0400756 /* Not all switches registered yet */
757 if (err == 1)
758 return 0;
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200759
760 if (dst->applied) {
761 pr_info("DSA: Disjoint trees?\n");
762 return -EINVAL;
763 }
764
765 err = dsa_dst_parse(dst);
Vivien Didelotcbabb0a2017-10-27 15:55:17 -0400766 if (err)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200767 goto out_del_dst;
768
769 err = dsa_dst_apply(dst);
770 if (err) {
771 dsa_dst_unapply(dst);
772 goto out_del_dst;
773 }
774
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200775 return 0;
776
777out_del_dst:
778 dsa_dst_del_ds(dst, ds, ds->index);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200779
780 return err;
781}
782
Vivien Didelota0c02162017-01-27 15:29:36 -0500783struct dsa_switch *dsa_switch_alloc(struct device *dev, size_t n)
784{
785 size_t size = sizeof(struct dsa_switch) + n * sizeof(struct dsa_port);
786 struct dsa_switch *ds;
Vivien Didelot818be842017-01-27 15:29:38 -0500787 int i;
Vivien Didelota0c02162017-01-27 15:29:36 -0500788
789 ds = devm_kzalloc(dev, size, GFP_KERNEL);
790 if (!ds)
791 return NULL;
792
793 ds->dev = dev;
794 ds->num_ports = n;
795
Vivien Didelot818be842017-01-27 15:29:38 -0500796 for (i = 0; i < ds->num_ports; ++i) {
797 ds->ports[i].index = i;
798 ds->ports[i].ds = ds;
799 }
800
Vivien Didelota0c02162017-01-27 15:29:36 -0500801 return ds;
802}
803EXPORT_SYMBOL_GPL(dsa_switch_alloc);
804
Vivien Didelot23c9ee42017-05-26 18:12:51 -0400805int dsa_register_switch(struct dsa_switch *ds)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200806{
807 int err;
808
809 mutex_lock(&dsa2_mutex);
Vivien Didelot23c9ee42017-05-26 18:12:51 -0400810 err = _dsa_register_switch(ds);
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200811 mutex_unlock(&dsa2_mutex);
812
813 return err;
814}
815EXPORT_SYMBOL_GPL(dsa_register_switch);
816
Wei Yongjun85c22ba2016-07-12 15:24:10 +0000817static void _dsa_unregister_switch(struct dsa_switch *ds)
Andrew Lunn83c0afa2016-06-04 21:17:07 +0200818{
819 struct dsa_switch_tree *dst = ds->dst;
820
821 dsa_dst_unapply(dst);
822
823 dsa_dst_del_ds(dst, ds, ds->index);
824}
825
826void dsa_unregister_switch(struct dsa_switch *ds)
827{
828 mutex_lock(&dsa2_mutex);
829 _dsa_unregister_switch(ds);
830 mutex_unlock(&dsa2_mutex);
831}
832EXPORT_SYMBOL_GPL(dsa_unregister_switch);