atm: correct sysfs 'device' link creation and parent relationships

The ATM subsystem was incorrectly creating the 'device' link for ATM
nodes in sysfs.  This led to incorrect device/parent relationships
exposed by sysfs and udev.  Instead of rolling the 'device' link by hand
in the generic ATM code, pass each ATM driver's bus device down to the
sysfs code and let sysfs do this stuff correctly.

Signed-off-by: Dan Williams <dcbw@redhat.com>
Signed-off-by: David S. Miller <davem@davemloft.net>
This commit is contained in:
Dan Williams 2010-12-08 19:40:47 +00:00 committed by David S. Miller
parent fb890ae7d6
commit d9ca676bcb
19 changed files with 41 additions and 42 deletions

View File

@ -154,7 +154,7 @@ static int __init adummy_init(void)
err = -ENOMEM; err = -ENOMEM;
goto out; goto out;
} }
atm_dev = atm_dev_register(DEV_LABEL, &adummy_ops, -1, NULL); atm_dev = atm_dev_register(DEV_LABEL, NULL, &adummy_ops, -1, NULL);
if (!atm_dev) { if (!atm_dev) {
printk(KERN_ERR DEV_LABEL ": atm_dev_register() failed\n"); printk(KERN_ERR DEV_LABEL ": atm_dev_register() failed\n");
err = -ENODEV; err = -ENODEV;

View File

@ -2244,7 +2244,8 @@ static int __devinit amb_probe(struct pci_dev *pci_dev, const struct pci_device_
goto out_reset; goto out_reset;
} }
dev->atm_dev = atm_dev_register (DEV_LABEL, &amb_ops, -1, NULL); dev->atm_dev = atm_dev_register (DEV_LABEL, &pci_dev->dev, &amb_ops, -1,
NULL);
if (!dev->atm_dev) { if (!dev->atm_dev) {
PRINTD (DBG_ERR, "failed to register Madge ATM adapter"); PRINTD (DBG_ERR, "failed to register Madge ATM adapter");
err = -EINVAL; err = -EINVAL;

View File

@ -366,7 +366,7 @@ static int atmtcp_create(int itf,int persist,struct atm_dev **result)
if (!dev_data) if (!dev_data)
return -ENOMEM; return -ENOMEM;
dev = atm_dev_register(DEV_LABEL,&atmtcp_v_dev_ops,itf,NULL); dev = atm_dev_register(DEV_LABEL,NULL,&atmtcp_v_dev_ops,itf,NULL);
if (!dev) { if (!dev) {
kfree(dev_data); kfree(dev_data);
return itf == -1 ? -ENOMEM : -EBUSY; return itf == -1 ? -ENOMEM : -EBUSY;

View File

@ -2244,7 +2244,7 @@ static int __devinit eni_init_one(struct pci_dev *pci_dev,
&zeroes); &zeroes);
if (!cpu_zeroes) goto out1; if (!cpu_zeroes) goto out1;
} }
dev = atm_dev_register(DEV_LABEL,&ops,-1,NULL); dev = atm_dev_register(DEV_LABEL, &pci_dev->dev, &ops, -1, NULL);
if (!dev) goto out2; if (!dev) goto out2;
pci_set_drvdata(pci_dev, dev); pci_set_drvdata(pci_dev, dev);
eni_dev->pci_dev = pci_dev; eni_dev->pci_dev = pci_dev;

View File

@ -1911,7 +1911,7 @@ static int __devinit firestream_init_one (struct pci_dev *pci_dev,
fs_dev, sizeof (struct fs_dev)); fs_dev, sizeof (struct fs_dev));
if (!fs_dev) if (!fs_dev)
goto err_out; goto err_out;
atm_dev = atm_dev_register("fs", &ops, -1, NULL); atm_dev = atm_dev_register("fs", &pci_dev->dev, &ops, -1, NULL);
if (!atm_dev) if (!atm_dev)
goto err_out_free_fs_dev; goto err_out_free_fs_dev;

View File

@ -2567,14 +2567,14 @@ fore200e_load_and_start_fw(struct fore200e* fore200e)
static int __devinit static int __devinit
fore200e_register(struct fore200e* fore200e) fore200e_register(struct fore200e* fore200e, struct device *parent)
{ {
struct atm_dev* atm_dev; struct atm_dev* atm_dev;
DPRINTK(2, "device %s being registered\n", fore200e->name); DPRINTK(2, "device %s being registered\n", fore200e->name);
atm_dev = atm_dev_register(fore200e->bus->proc_name, &fore200e_ops, -1, atm_dev = atm_dev_register(fore200e->bus->proc_name, parent, &fore200e_ops,
NULL); -1, NULL);
if (atm_dev == NULL) { if (atm_dev == NULL) {
printk(FORE200E "unable to register device %s\n", fore200e->name); printk(FORE200E "unable to register device %s\n", fore200e->name);
return -ENODEV; return -ENODEV;
@ -2594,9 +2594,9 @@ fore200e_register(struct fore200e* fore200e)
static int __devinit static int __devinit
fore200e_init(struct fore200e* fore200e) fore200e_init(struct fore200e* fore200e, struct device *parent)
{ {
if (fore200e_register(fore200e) < 0) if (fore200e_register(fore200e, parent) < 0)
return -ENODEV; return -ENODEV;
if (fore200e->bus->configure(fore200e) < 0) if (fore200e->bus->configure(fore200e) < 0)
@ -2662,7 +2662,7 @@ static int __devinit fore200e_sba_probe(struct platform_device *op,
sprintf(fore200e->name, "%s-%d", bus->model_name, index); sprintf(fore200e->name, "%s-%d", bus->model_name, index);
err = fore200e_init(fore200e); err = fore200e_init(fore200e, &op->dev);
if (err < 0) { if (err < 0) {
fore200e_shutdown(fore200e); fore200e_shutdown(fore200e);
kfree(fore200e); kfree(fore200e);
@ -2740,7 +2740,7 @@ fore200e_pca_detect(struct pci_dev *pci_dev, const struct pci_device_id *pci_ent
sprintf(fore200e->name, "%s-%d", bus->model_name, index); sprintf(fore200e->name, "%s-%d", bus->model_name, index);
err = fore200e_init(fore200e); err = fore200e_init(fore200e, &pci_dev->dev);
if (err < 0) { if (err < 0) {
fore200e_shutdown(fore200e); fore200e_shutdown(fore200e);
goto out_free; goto out_free;

View File

@ -366,7 +366,7 @@ he_init_one(struct pci_dev *pci_dev, const struct pci_device_id *pci_ent)
goto init_one_failure; goto init_one_failure;
} }
atm_dev = atm_dev_register(DEV_LABEL, &he_ops, -1, NULL); atm_dev = atm_dev_register(DEV_LABEL, &pci_dev->dev, &he_ops, -1, NULL);
if (!atm_dev) { if (!atm_dev) {
err = -ENODEV; err = -ENODEV;
goto init_one_failure; goto init_one_failure;

View File

@ -2733,7 +2733,8 @@ static int __devinit hrz_probe(struct pci_dev *pci_dev, const struct pci_device_
PRINTD(DBG_INFO, "found Madge ATM adapter (hrz) at: IO %x, IRQ %u, MEM %p", PRINTD(DBG_INFO, "found Madge ATM adapter (hrz) at: IO %x, IRQ %u, MEM %p",
iobase, irq, membase); iobase, irq, membase);
dev->atm_dev = atm_dev_register(DEV_LABEL, &hrz_ops, -1, NULL); dev->atm_dev = atm_dev_register(DEV_LABEL, &pci_dev->dev, &hrz_ops, -1,
NULL);
if (!(dev->atm_dev)) { if (!(dev->atm_dev)) {
PRINTD(DBG_ERR, "failed to register Madge ATM adapter"); PRINTD(DBG_ERR, "failed to register Madge ATM adapter");
err = -EINVAL; err = -EINVAL;

View File

@ -3698,7 +3698,8 @@ idt77252_init_one(struct pci_dev *pcidev, const struct pci_device_id *id)
goto err_out_iounmap; goto err_out_iounmap;
} }
dev = atm_dev_register("idt77252", &idt77252_ops, -1, NULL); dev = atm_dev_register("idt77252", &pcidev->dev, &idt77252_ops, -1,
NULL);
if (!dev) { if (!dev) {
printk("%s: can't register atm device\n", card->name); printk("%s: can't register atm device\n", card->name);
err = -EIO; err = -EIO;

View File

@ -3172,7 +3172,7 @@ static int __devinit ia_init_one(struct pci_dev *pdev,
ret = -ENODEV; ret = -ENODEV;
goto err_out_free_iadev; goto err_out_free_iadev;
} }
dev = atm_dev_register(DEV_LABEL, &ops, -1, NULL); dev = atm_dev_register(DEV_LABEL, &pdev->dev, &ops, -1, NULL);
if (!dev) { if (!dev) {
ret = -ENOMEM; ret = -ENOMEM;
goto err_out_disable_dev; goto err_out_disable_dev;

View File

@ -2591,7 +2591,7 @@ static int __devinit lanai_init_one(struct pci_dev *pci,
return -ENOMEM; return -ENOMEM;
} }
atmdev = atm_dev_register(DEV_LABEL, &ops, -1, NULL); atmdev = atm_dev_register(DEV_LABEL, &pci->dev, &ops, -1, NULL);
if (atmdev == NULL) { if (atmdev == NULL) {
printk(KERN_ERR DEV_LABEL printk(KERN_ERR DEV_LABEL
": couldn't register atm device!\n"); ": couldn't register atm device!\n");

View File

@ -771,7 +771,8 @@ static int __devinit ns_init_card(int i, struct pci_dev *pcidev)
} }
/* Register device */ /* Register device */
card->atmdev = atm_dev_register("nicstar", &atm_ops, -1, NULL); card->atmdev = atm_dev_register("nicstar", &card->pcidev->dev, &atm_ops,
-1, NULL);
if (card->atmdev == NULL) { if (card->atmdev == NULL) {
printk("nicstar%d: can't register device.\n", i); printk("nicstar%d: can't register device.\n", i);
error = 17; error = 17;

View File

@ -166,7 +166,7 @@ static irqreturn_t solos_irq(int irq, void *dev_id);
static struct atm_vcc* find_vcc(struct atm_dev *dev, short vpi, int vci); static struct atm_vcc* find_vcc(struct atm_dev *dev, short vpi, int vci);
static int list_vccs(int vci); static int list_vccs(int vci);
static void release_vccs(struct atm_dev *dev); static void release_vccs(struct atm_dev *dev);
static int atm_init(struct solos_card *); static int atm_init(struct solos_card *, struct device *);
static void atm_remove(struct solos_card *); static void atm_remove(struct solos_card *);
static int send_command(struct solos_card *card, int dev, const char *buf, size_t size); static int send_command(struct solos_card *card, int dev, const char *buf, size_t size);
static void solos_bh(unsigned long); static void solos_bh(unsigned long);
@ -1210,7 +1210,7 @@ static int fpga_probe(struct pci_dev *dev, const struct pci_device_id *id)
if (db_firmware_upgrade) if (db_firmware_upgrade)
flash_upgrade(card, 3); flash_upgrade(card, 3);
err = atm_init(card); err = atm_init(card, &dev->dev);
if (err) if (err)
goto out_free_irq; goto out_free_irq;
@ -1233,7 +1233,7 @@ static int fpga_probe(struct pci_dev *dev, const struct pci_device_id *id)
return err; return err;
} }
static int atm_init(struct solos_card *card) static int atm_init(struct solos_card *card, struct device *parent)
{ {
int i; int i;
@ -1244,7 +1244,7 @@ static int atm_init(struct solos_card *card)
skb_queue_head_init(&card->tx_queue[i]); skb_queue_head_init(&card->tx_queue[i]);
skb_queue_head_init(&card->cli_queue[i]); skb_queue_head_init(&card->cli_queue[i]);
card->atmdev[i] = atm_dev_register("solos-pci", &fpga_ops, -1, NULL); card->atmdev[i] = atm_dev_register("solos-pci", parent, &fpga_ops, -1, NULL);
if (!card->atmdev[i]) { if (!card->atmdev[i]) {
dev_err(&card->dev->dev, "Could not register ATM device %d\n", i); dev_err(&card->dev->dev, "Could not register ATM device %d\n", i);
atm_remove(card); atm_remove(card);

View File

@ -1597,7 +1597,7 @@ static int __devinit zatm_init_one(struct pci_dev *pci_dev,
goto out; goto out;
} }
dev = atm_dev_register(DEV_LABEL, &ops, -1, NULL); dev = atm_dev_register(DEV_LABEL, &pci_dev->dev, &ops, -1, NULL);
if (!dev) if (!dev)
goto out_free; goto out_free;

View File

@ -951,7 +951,9 @@ static int usbatm_atm_init(struct usbatm_data *instance)
* condition: callbacks we register can be executed at once, before we have * condition: callbacks we register can be executed at once, before we have
* initialized the struct atm_dev. To protect against this, all callbacks * initialized the struct atm_dev. To protect against this, all callbacks
* abort if atm_dev->dev_data is NULL. */ * abort if atm_dev->dev_data is NULL. */
atm_dev = atm_dev_register(instance->driver_name, &usbatm_atm_devops, -1, NULL); atm_dev = atm_dev_register(instance->driver_name,
&instance->usb_intf->dev, &usbatm_atm_devops,
-1, NULL);
if (!atm_dev) { if (!atm_dev) {
usb_err(instance, "%s: failed to register ATM device!\n", __func__); usb_err(instance, "%s: failed to register ATM device!\n", __func__);
return -1; return -1;
@ -966,14 +968,6 @@ static int usbatm_atm_init(struct usbatm_data *instance)
/* temp init ATM device, set to 128kbit */ /* temp init ATM device, set to 128kbit */
atm_dev->link_rate = 128 * 1000 / 424; atm_dev->link_rate = 128 * 1000 / 424;
ret = sysfs_create_link(&atm_dev->class_dev.kobj,
&instance->usb_intf->dev.kobj, "device");
if (ret) {
atm_err(instance, "%s: sysfs_create_link failed: %d\n",
__func__, ret);
goto fail_sysfs;
}
if (instance->driver->atm_start && ((ret = instance->driver->atm_start(instance, atm_dev)) < 0)) { if (instance->driver->atm_start && ((ret = instance->driver->atm_start(instance, atm_dev)) < 0)) {
atm_err(instance, "%s: atm_start failed: %d!\n", __func__, ret); atm_err(instance, "%s: atm_start failed: %d!\n", __func__, ret);
goto fail; goto fail;
@ -992,8 +986,6 @@ static int usbatm_atm_init(struct usbatm_data *instance)
return 0; return 0;
fail: fail:
sysfs_remove_link(&atm_dev->class_dev.kobj, "device");
fail_sysfs:
instance->atm_dev = NULL; instance->atm_dev = NULL;
atm_dev_deregister(atm_dev); /* usbatm_atm_dev_close will eventually be called */ atm_dev_deregister(atm_dev); /* usbatm_atm_dev_close will eventually be called */
return ret; return ret;
@ -1329,7 +1321,6 @@ void usbatm_usb_disconnect(struct usb_interface *intf)
/* ATM finalize */ /* ATM finalize */
if (instance->atm_dev) { if (instance->atm_dev) {
sysfs_remove_link(&instance->atm_dev->class_dev.kobj, "device");
atm_dev_deregister(instance->atm_dev); atm_dev_deregister(instance->atm_dev);
instance->atm_dev = NULL; instance->atm_dev = NULL;
} }

View File

@ -427,8 +427,10 @@ extern rwlock_t vcc_sklist_lock;
#define ATM_SKB(skb) (((struct atm_skb_data *) (skb)->cb)) #define ATM_SKB(skb) (((struct atm_skb_data *) (skb)->cb))
struct atm_dev *atm_dev_register(const char *type,const struct atmdev_ops *ops, struct atm_dev *atm_dev_register(const char *type, struct device *parent,
int number,unsigned long *flags); /* number == -1: pick first available */ const struct atmdev_ops *ops,
int number, /* -1 == pick first available */
unsigned long *flags);
struct atm_dev *atm_dev_lookup(int number); struct atm_dev *atm_dev_lookup(int number);
void atm_dev_deregister(struct atm_dev *dev); void atm_dev_deregister(struct atm_dev *dev);

View File

@ -143,12 +143,13 @@ static struct class atm_class = {
.dev_uevent = atm_uevent, .dev_uevent = atm_uevent,
}; };
int atm_register_sysfs(struct atm_dev *adev) int atm_register_sysfs(struct atm_dev *adev, struct device *parent)
{ {
struct device *cdev = &adev->class_dev; struct device *cdev = &adev->class_dev;
int i, j, err; int i, j, err;
cdev->class = &atm_class; cdev->class = &atm_class;
cdev->parent = parent;
dev_set_drvdata(cdev, adev); dev_set_drvdata(cdev, adev);
dev_set_name(cdev, "%s%d", adev->type, adev->number); dev_set_name(cdev, "%s%d", adev->type, adev->number);

View File

@ -74,8 +74,9 @@ struct atm_dev *atm_dev_lookup(int number)
} }
EXPORT_SYMBOL(atm_dev_lookup); EXPORT_SYMBOL(atm_dev_lookup);
struct atm_dev *atm_dev_register(const char *type, const struct atmdev_ops *ops, struct atm_dev *atm_dev_register(const char *type, struct device *parent,
int number, unsigned long *flags) const struct atmdev_ops *ops, int number,
unsigned long *flags)
{ {
struct atm_dev *dev, *inuse; struct atm_dev *dev, *inuse;
@ -115,7 +116,7 @@ struct atm_dev *atm_dev_register(const char *type, const struct atmdev_ops *ops,
goto out_fail; goto out_fail;
} }
if (atm_register_sysfs(dev) < 0) { if (atm_register_sysfs(dev, parent) < 0) {
pr_err("atm_register_sysfs failed for dev %s\n", type); pr_err("atm_register_sysfs failed for dev %s\n", type);
atm_proc_dev_deregister(dev); atm_proc_dev_deregister(dev);
goto out_fail; goto out_fail;

View File

@ -42,6 +42,6 @@ static inline void atm_proc_dev_deregister(struct atm_dev *dev)
#endif /* CONFIG_PROC_FS */ #endif /* CONFIG_PROC_FS */
int atm_register_sysfs(struct atm_dev *adev); int atm_register_sysfs(struct atm_dev *adev, struct device *parent);
void atm_unregister_sysfs(struct atm_dev *adev); void atm_unregister_sysfs(struct atm_dev *adev);
#endif #endif