Commit a7b216b1 authored by David S. Miller's avatar David S. Miller

Merge branch 'dsa-port-parsing'

Vivien Didelot says:

====================
net: dsa: add port parsing functions

This patchset adds port parsing functions called early in the new
bindings parsing stage, which regroup all the fetching of static data
available at the port level, including the port's type, name and CPU
master interface.

This simplifies the rest of the code which does not need to dig into
device tree or platform data again in order to check a port's type or
name.
====================
Signed-off-by: default avatarDavid S. Miller <davem@davemloft.net>
parents 638f5b90 951259aa
...@@ -87,23 +87,17 @@ static void dsa_dst_del_ds(struct dsa_switch_tree *dst, ...@@ -87,23 +87,17 @@ static void dsa_dst_del_ds(struct dsa_switch_tree *dst,
*/ */
static bool dsa_port_is_valid(struct dsa_port *port) static bool dsa_port_is_valid(struct dsa_port *port)
{ {
return !!(port->dn || port->name); return port->type != DSA_PORT_TYPE_UNUSED;
} }
static bool dsa_port_is_dsa(struct dsa_port *port) static bool dsa_port_is_dsa(struct dsa_port *port)
{ {
if (port->name && !strcmp(port->name, "dsa")) return port->type == DSA_PORT_TYPE_DSA;
return true;
else
return !!of_parse_phandle(port->dn, "link", 0);
} }
static bool dsa_port_is_cpu(struct dsa_port *port) static bool dsa_port_is_cpu(struct dsa_port *port)
{ {
if (port->name && !strcmp(port->name, "cpu")) return port->type == DSA_PORT_TYPE_CPU;
return true;
else
return !!of_parse_phandle(port->dn, "ethernet", 0);
} }
static bool dsa_ds_find_port_dn(struct dsa_switch *ds, static bool dsa_ds_find_port_dn(struct dsa_switch *ds,
...@@ -183,8 +177,6 @@ static int dsa_ds_complete(struct dsa_switch_tree *dst, struct dsa_switch *ds) ...@@ -183,8 +177,6 @@ static int dsa_ds_complete(struct dsa_switch_tree *dst, struct dsa_switch *ds)
err = dsa_port_complete(dst, ds, port, index); err = dsa_port_complete(dst, ds, port, index);
if (err != 0) if (err != 0)
return err; return err;
port->type = DSA_PORT_TYPE_DSA;
} }
return 0; return 0;
...@@ -265,15 +257,9 @@ static void dsa_cpu_port_unapply(struct dsa_port *port) ...@@ -265,15 +257,9 @@ static void dsa_cpu_port_unapply(struct dsa_port *port)
static int dsa_user_port_apply(struct dsa_port *port) static int dsa_user_port_apply(struct dsa_port *port)
{ {
struct dsa_switch *ds = port->ds; struct dsa_switch *ds = port->ds;
const char *name = port->name;
int err; int err;
if (port->dn) err = dsa_slave_create(port);
name = of_get_property(port->dn, "label", NULL);
if (!name)
name = "eth%d";
err = dsa_slave_create(port, name);
if (err) { if (err) {
dev_warn(ds->dev, "Failed to create slave %d: %d\n", dev_warn(ds->dev, "Failed to create slave %d: %d\n",
port->index, err); port->index, err);
...@@ -478,29 +464,9 @@ static int dsa_cpu_parse(struct dsa_port *port, u32 index, ...@@ -478,29 +464,9 @@ static int dsa_cpu_parse(struct dsa_port *port, u32 index,
{ {
const struct dsa_device_ops *tag_ops; const struct dsa_device_ops *tag_ops;
enum dsa_tag_protocol tag_protocol; enum dsa_tag_protocol tag_protocol;
struct net_device *ethernet_dev;
struct device_node *ethernet;
if (port->dn) { if (!dst->cpu_dp)
ethernet = of_parse_phandle(port->dn, "ethernet", 0);
if (!ethernet)
return -EINVAL;
ethernet_dev = of_find_net_device_by_node(ethernet);
if (!ethernet_dev)
return -EPROBE_DEFER;
} else {
ethernet_dev = dsa_dev_to_net_device(ds->cd->netdev[index]);
if (!ethernet_dev)
return -EPROBE_DEFER;
dev_put(ethernet_dev);
}
if (!dst->cpu_dp) {
dst->cpu_dp = port; dst->cpu_dp = port;
dst->cpu_dp->master = ethernet_dev;
}
port->type = DSA_PORT_TYPE_CPU;
tag_protocol = ds->ops->get_tag_protocol(ds); tag_protocol = ds->ops->get_tag_protocol(ds);
tag_ops = dsa_resolve_tag_protocol(tag_protocol); tag_ops = dsa_resolve_tag_protocol(tag_protocol);
...@@ -534,8 +500,6 @@ static int dsa_ds_parse(struct dsa_switch_tree *dst, struct dsa_switch *ds) ...@@ -534,8 +500,6 @@ static int dsa_ds_parse(struct dsa_switch_tree *dst, struct dsa_switch *ds)
err = dsa_cpu_parse(port, index, dst, ds); err = dsa_cpu_parse(port, index, dst, ds);
if (err) if (err)
return err; return err;
} else {
port->type = DSA_PORT_TYPE_USER;
} }
} }
...@@ -590,11 +554,48 @@ static int dsa_dst_parse(struct dsa_switch_tree *dst) ...@@ -590,11 +554,48 @@ static int dsa_dst_parse(struct dsa_switch_tree *dst)
return 0; return 0;
} }
static int dsa_parse_ports_dn(struct device_node *ports, struct dsa_switch *ds) static int dsa_port_parse_of(struct dsa_port *dp, struct device_node *dn)
{ {
struct device_node *port; struct device_node *ethernet = of_parse_phandle(dn, "ethernet", 0);
int err; struct device_node *link = of_parse_phandle(dn, "link", 0);
const char *name = of_get_property(dn, "label", NULL);
if (ethernet) {
struct net_device *master;
master = of_find_net_device_by_node(ethernet);
if (!master)
return -EPROBE_DEFER;
dp->type = DSA_PORT_TYPE_CPU;
dp->master = master;
} else if (link) {
dp->type = DSA_PORT_TYPE_DSA;
} else {
if (!name)
name = "eth%d";
dp->type = DSA_PORT_TYPE_USER;
dp->name = name;
}
dp->dn = dn;
return 0;
}
static int dsa_parse_ports_of(struct device_node *dn, struct dsa_switch *ds)
{
struct device_node *ports, *port;
struct dsa_port *dp;
u32 reg; u32 reg;
int err;
ports = of_get_child_by_name(dn, "ports");
if (!ports) {
dev_err(ds->dev, "no ports child node found\n");
return -EINVAL;
}
for_each_available_child_of_node(ports, port) { for_each_available_child_of_node(ports, port) {
err = of_property_read_u32(port, "reg", &reg); err = of_property_read_u32(port, "reg", &reg);
...@@ -604,22 +605,62 @@ static int dsa_parse_ports_dn(struct device_node *ports, struct dsa_switch *ds) ...@@ -604,22 +605,62 @@ static int dsa_parse_ports_dn(struct device_node *ports, struct dsa_switch *ds)
if (reg >= ds->num_ports) if (reg >= ds->num_ports)
return -EINVAL; return -EINVAL;
ds->ports[reg].dn = port; dp = &ds->ports[reg];
err = dsa_port_parse_of(dp, port);
if (err)
return err;
}
return 0;
}
static int dsa_port_parse(struct dsa_port *dp, const char *name,
struct device *dev)
{
if (!strcmp(name, "cpu")) {
struct net_device *master;
master = dsa_dev_to_net_device(dev);
if (!master)
return -EPROBE_DEFER;
dev_put(master);
dp->type = DSA_PORT_TYPE_CPU;
dp->master = master;
} else if (!strcmp(name, "dsa")) {
dp->type = DSA_PORT_TYPE_DSA;
} else {
dp->type = DSA_PORT_TYPE_USER;
} }
dp->name = name;
return 0; return 0;
} }
static int dsa_parse_ports(struct dsa_chip_data *cd, struct dsa_switch *ds) static int dsa_parse_ports(struct dsa_chip_data *cd, struct dsa_switch *ds)
{ {
bool valid_name_found = false; bool valid_name_found = false;
struct dsa_port *dp;
struct device *dev;
const char *name;
unsigned int i; unsigned int i;
int err;
for (i = 0; i < DSA_MAX_PORTS; i++) { for (i = 0; i < DSA_MAX_PORTS; i++) {
if (!cd->port_names[i]) name = cd->port_names[i];
dev = cd->netdev[i];
dp = &ds->ports[i];
if (!name)
continue; continue;
ds->ports[i].name = cd->port_names[i]; err = dsa_port_parse(dp, name, dev);
if (err)
return err;
valid_name_found = true; valid_name_found = true;
} }
...@@ -665,26 +706,11 @@ static int dsa_parse_member(struct dsa_chip_data *pd, u32 *tree, u32 *index) ...@@ -665,26 +706,11 @@ static int dsa_parse_member(struct dsa_chip_data *pd, u32 *tree, u32 *index)
return 0; return 0;
} }
static struct device_node *dsa_get_ports(struct dsa_switch *ds,
struct device_node *np)
{
struct device_node *ports;
ports = of_get_child_by_name(np, "ports");
if (!ports) {
dev_err(ds->dev, "no ports child node found\n");
return ERR_PTR(-EINVAL);
}
return ports;
}
static int _dsa_register_switch(struct dsa_switch *ds) static int _dsa_register_switch(struct dsa_switch *ds)
{ {
struct dsa_chip_data *pdata = ds->dev->platform_data; struct dsa_chip_data *pdata = ds->dev->platform_data;
struct device_node *np = ds->dev->of_node; struct device_node *np = ds->dev->of_node;
struct dsa_switch_tree *dst; struct dsa_switch_tree *dst;
struct device_node *ports;
u32 tree, index; u32 tree, index;
int i, err; int i, err;
...@@ -693,11 +719,7 @@ static int _dsa_register_switch(struct dsa_switch *ds) ...@@ -693,11 +719,7 @@ static int _dsa_register_switch(struct dsa_switch *ds)
if (err) if (err)
return err; return err;
ports = dsa_get_ports(ds, np); err = dsa_parse_ports_of(np, ds);
if (IS_ERR(ports))
return PTR_ERR(ports);
err = dsa_parse_ports_dn(ports, ds);
if (err) if (err)
return err; return err;
} else { } else {
...@@ -748,14 +770,8 @@ static int _dsa_register_switch(struct dsa_switch *ds) ...@@ -748,14 +770,8 @@ static int _dsa_register_switch(struct dsa_switch *ds)
} }
err = dsa_dst_parse(dst); err = dsa_dst_parse(dst);
if (err) { if (err)
if (err == -EPROBE_DEFER) {
dsa_dst_del_ds(dst, ds, ds->index);
return err;
}
goto out_del_dst; goto out_del_dst;
}
err = dsa_dst_apply(dst); err = dsa_dst_apply(dst);
if (err) { if (err) {
......
...@@ -163,7 +163,7 @@ void dsa_port_fixed_link_unregister_of(struct dsa_port *dp); ...@@ -163,7 +163,7 @@ void dsa_port_fixed_link_unregister_of(struct dsa_port *dp);
/* slave.c */ /* slave.c */
extern const struct dsa_device_ops notag_netdev_ops; extern const struct dsa_device_ops notag_netdev_ops;
void dsa_slave_mii_bus_init(struct dsa_switch *ds); void dsa_slave_mii_bus_init(struct dsa_switch *ds);
int dsa_slave_create(struct dsa_port *port, const char *name); int dsa_slave_create(struct dsa_port *dp);
void dsa_slave_destroy(struct net_device *slave_dev); void dsa_slave_destroy(struct net_device *slave_dev);
int dsa_slave_suspend(struct net_device *slave_dev); int dsa_slave_suspend(struct net_device *slave_dev);
int dsa_slave_resume(struct net_device *slave_dev); int dsa_slave_resume(struct net_device *slave_dev);
......
...@@ -115,6 +115,7 @@ static int dsa_switch_setup_one(struct dsa_switch *ds, ...@@ -115,6 +115,7 @@ static int dsa_switch_setup_one(struct dsa_switch *ds,
name = cd->port_names[i]; name = cd->port_names[i];
if (name == NULL) if (name == NULL)
continue; continue;
dp->name = name;
if (!strcmp(name, "cpu")) { if (!strcmp(name, "cpu")) {
if (dst->cpu_dp) { if (dst->cpu_dp) {
...@@ -196,7 +197,7 @@ static int dsa_switch_setup_one(struct dsa_switch *ds, ...@@ -196,7 +197,7 @@ static int dsa_switch_setup_one(struct dsa_switch *ds,
if (dsa_is_user_port(ds, i)) if (dsa_is_user_port(ds, i))
continue; continue;
ret = dsa_slave_create(&ds->ports[i], cd->port_names[i]); ret = dsa_slave_create(&ds->ports[i]);
if (ret < 0) if (ret < 0)
netdev_err(master, "[%d]: can't create dsa slave device for port %d(%s): %d\n", netdev_err(master, "[%d]: can't create dsa slave device for port %d(%s): %d\n",
index, i, cd->port_names[i], ret); index, i, cd->port_names[i], ret);
......
...@@ -1142,11 +1142,12 @@ static void dsa_slave_notify(struct net_device *dev, unsigned long val) ...@@ -1142,11 +1142,12 @@ static void dsa_slave_notify(struct net_device *dev, unsigned long val)
call_dsa_notifiers(val, dev, &rinfo.info); call_dsa_notifiers(val, dev, &rinfo.info);
} }
int dsa_slave_create(struct dsa_port *port, const char *name) int dsa_slave_create(struct dsa_port *port)
{ {
struct dsa_port *cpu_dp = port->cpu_dp; struct dsa_port *cpu_dp = port->cpu_dp;
struct net_device *master = cpu_dp->master; struct net_device *master = cpu_dp->master;
struct dsa_switch *ds = port->ds; struct dsa_switch *ds = port->ds;
const char *name = port->name;
struct net_device *slave_dev; struct net_device *slave_dev;
struct dsa_slave_priv *p; struct dsa_slave_priv *p;
int ret; int ret;
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment