Commit e6161d64 authored by Alexandre Bounine's avatar Alexandre Bounine Committed by Linus Torvalds

rapidio/rionet: rework driver initialization and removal

Rework probe/remove routines to prevent rionet driver from monopolizing
target RapidIO devices.  Fix conflict with modular RapidIO switch drivers.

Using one of RapidIO messaging channels rionet driver provides a service
layer common to all endpoint devices in a system's RapidIO network.  These
devices may also require their own specific device driver which will be
blocked from attaching to the target device by rionet (or block rionet if
loaded earlier).  To avoid conflict with device-specific drivers, the
rionet driver is reworked to be registered as a subsystem interface on the
RapidIO bus.

The reworked rio_remove_dev() and rionet_exit() routines also include
handling of individual rionet peer device removal which was not supported
before.
Signed-off-by: default avatarAlexandre Bounine <alexandre.bounine@idt.com>
Cc: Matt Porter <mporter@kernel.crashing.org>
Cc: Li Yang <leoli@freescale.com>
Cc: Kumar Gala <galak@kernel.crashing.org>
Cc: "David S. Miller" <davem@davemloft.net>
Cc: Andre van Herk <andre.van.herk@Prodrive.nl>
Cc: Micha Nelissen <micha.nelissen@Prodrive.nl>
Cc: Stef van Os <stef.van.os@Prodrive.nl>
Cc: Jean Delvare <jdelvare@suse.de>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 2ec3ba69
...@@ -208,6 +208,17 @@ static int rionet_start_xmit(struct sk_buff *skb, struct net_device *ndev) ...@@ -208,6 +208,17 @@ static int rionet_start_xmit(struct sk_buff *skb, struct net_device *ndev)
if (nets[rnet->mport->id].active[destid]) if (nets[rnet->mport->id].active[destid])
rionet_queue_tx_msg(skb, ndev, rionet_queue_tx_msg(skb, ndev,
nets[rnet->mport->id].active[destid]); nets[rnet->mport->id].active[destid]);
else {
/*
* If the target device was removed from the list of
* active peers but we still have TX packets targeting
* it just report sending a packet to the target
* (without actual packet transfer).
*/
dev_kfree_skb_any(skb);
ndev->stats.tx_packets++;
ndev->stats.tx_bytes += skb->len;
}
} }
spin_unlock_irqrestore(&rnet->tx_lock, flags); spin_unlock_irqrestore(&rnet->tx_lock, flags);
...@@ -385,24 +396,28 @@ static int rionet_close(struct net_device *ndev) ...@@ -385,24 +396,28 @@ static int rionet_close(struct net_device *ndev)
return 0; return 0;
} }
static void rionet_remove(struct rio_dev *rdev) static int rionet_remove_dev(struct device *dev, struct subsys_interface *sif)
{ {
struct net_device *ndev = rio_get_drvdata(rdev); struct rio_dev *rdev = to_rio_dev(dev);
unsigned char netid = rdev->net->hport->id; unsigned char netid = rdev->net->hport->id;
struct rionet_peer *peer, *tmp; struct rionet_peer *peer, *tmp;
unregister_netdev(ndev); if (dev_rionet_capable(rdev)) {
free_pages((unsigned long)nets[netid].active, get_order(sizeof(void *) *
RIO_MAX_ROUTE_ENTRIES(rdev->net->hport->sys_size)));
nets[netid].active = NULL;
list_for_each_entry_safe(peer, tmp, &nets[netid].peers, node) { list_for_each_entry_safe(peer, tmp, &nets[netid].peers, node) {
if (peer->rdev == rdev) {
if (nets[netid].active[rdev->destid]) {
nets[netid].active[rdev->destid] = NULL;
nets[netid].nact--;
}
list_del(&peer->node); list_del(&peer->node);
kfree(peer); kfree(peer);
break;
}
}
} }
free_netdev(ndev); return 0;
} }
static void rionet_get_drvinfo(struct net_device *ndev, static void rionet_get_drvinfo(struct net_device *ndev,
...@@ -503,12 +518,13 @@ static int rionet_setup_netdev(struct rio_mport *mport, struct net_device *ndev) ...@@ -503,12 +518,13 @@ static int rionet_setup_netdev(struct rio_mport *mport, struct net_device *ndev)
static unsigned long net_table[RIONET_MAX_NETS/sizeof(unsigned long) + 1]; static unsigned long net_table[RIONET_MAX_NETS/sizeof(unsigned long) + 1];
static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id) static int rionet_add_dev(struct device *dev, struct subsys_interface *sif)
{ {
int rc = -ENODEV; int rc = -ENODEV;
u32 lsrc_ops, ldst_ops; u32 lsrc_ops, ldst_ops;
struct rionet_peer *peer; struct rionet_peer *peer;
struct net_device *ndev = NULL; struct net_device *ndev = NULL;
struct rio_dev *rdev = to_rio_dev(dev);
unsigned char netid = rdev->net->hport->id; unsigned char netid = rdev->net->hport->id;
int oldnet; int oldnet;
...@@ -518,8 +534,9 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id) ...@@ -518,8 +534,9 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id)
oldnet = test_and_set_bit(netid, net_table); oldnet = test_and_set_bit(netid, net_table);
/* /*
* First time through, make sure local device is rionet * If first time through this net, make sure local device is rionet
* capable, setup netdev (will be skipped on later probes) * capable and setup netdev (this step will be skipped in later probes
* on the same net).
*/ */
if (!oldnet) { if (!oldnet) {
rio_local_read_config_32(rdev->net->hport, RIO_SRC_OPS_CAR, rio_local_read_config_32(rdev->net->hport, RIO_SRC_OPS_CAR,
...@@ -541,6 +558,12 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id) ...@@ -541,6 +558,12 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id)
} }
nets[netid].ndev = ndev; nets[netid].ndev = ndev;
rc = rionet_setup_netdev(rdev->net->hport, ndev); rc = rionet_setup_netdev(rdev->net->hport, ndev);
if (rc) {
printk(KERN_ERR "%s: failed to setup netdev (rc=%d)\n",
DRV_NAME, rc);
goto out;
}
INIT_LIST_HEAD(&nets[netid].peers); INIT_LIST_HEAD(&nets[netid].peers);
nets[netid].nact = 0; nets[netid].nact = 0;
} else if (nets[netid].ndev == NULL) } else if (nets[netid].ndev == NULL)
...@@ -559,31 +582,61 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id) ...@@ -559,31 +582,61 @@ static int rionet_probe(struct rio_dev *rdev, const struct rio_device_id *id)
list_add_tail(&peer->node, &nets[netid].peers); list_add_tail(&peer->node, &nets[netid].peers);
} }
rio_set_drvdata(rdev, nets[netid].ndev); return 0;
out:
out:
return rc; return rc;
} }
#ifdef MODULE
static struct rio_device_id rionet_id_table[] = { static struct rio_device_id rionet_id_table[] = {
{RIO_DEVICE(RIO_ANY_ID, RIO_ANY_ID)} {RIO_DEVICE(RIO_ANY_ID, RIO_ANY_ID)},
{ 0, } /* terminate list */
}; };
static struct rio_driver rionet_driver = { MODULE_DEVICE_TABLE(rapidio, rionet_id_table);
#endif
static struct subsys_interface rionet_interface = {
.name = "rionet", .name = "rionet",
.id_table = rionet_id_table, .subsys = &rio_bus_type,
.probe = rionet_probe, .add_dev = rionet_add_dev,
.remove = rionet_remove, .remove_dev = rionet_remove_dev,
}; };
static int __init rionet_init(void) static int __init rionet_init(void)
{ {
return rio_register_driver(&rionet_driver); return subsys_interface_register(&rionet_interface);
} }
static void __exit rionet_exit(void) static void __exit rionet_exit(void)
{ {
rio_unregister_driver(&rionet_driver); struct rionet_private *rnet;
struct net_device *ndev;
struct rionet_peer *peer, *tmp;
int i;
for (i = 0; i < RIONET_MAX_NETS; i++) {
if (nets[i].ndev != NULL) {
ndev = nets[i].ndev;
rnet = netdev_priv(ndev);
unregister_netdev(ndev);
list_for_each_entry_safe(peer,
tmp, &nets[i].peers, node) {
list_del(&peer->node);
kfree(peer);
}
free_pages((unsigned long)nets[i].active,
get_order(sizeof(void *) *
RIO_MAX_ROUTE_ENTRIES(rnet->mport->sys_size)));
nets[i].active = NULL;
free_netdev(ndev);
}
}
subsys_interface_unregister(&rionet_interface);
} }
late_initcall(rionet_init); late_initcall(rionet_init);
......
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