Commit a0d78193 authored by Jason Gunthorpe's avatar Jason Gunthorpe Committed by Greg Kroah-Hartman

IB/ucm: utilize new cdev_device_add helper function

The use after free is not triggerable here because the cdev holds
the module lock and the only device_unregister is only triggered by
module unload, however make the change for consistency.

To make this work the cdev_del needs to move out of the struct device
release function.

This cleans up the error path significantly and thus also fixes a minor
bug where the devnum would not be released if cdev_add failed.
Signed-off-by: default avatarJason Gunthorpe <jgunthorpe@obsidianresearch.com>
Signed-off-by: default avatarLogan Gunthorpe <logang@deltatee.com>
Reviewed-by: default avatarLogan Gunthorpe <logang@deltatee.com>
Reviewed-by: default avatarLeon Romanovsky <leonro@mellanox.com>
Signed-off-by: default avatarGreg Kroah-Hartman <gregkh@linuxfoundation.org>
parent 1c1d152c
...@@ -1205,12 +1205,15 @@ static void ib_ucm_release_dev(struct device *dev) ...@@ -1205,12 +1205,15 @@ static void ib_ucm_release_dev(struct device *dev)
struct ib_ucm_device *ucm_dev; struct ib_ucm_device *ucm_dev;
ucm_dev = container_of(dev, struct ib_ucm_device, dev); ucm_dev = container_of(dev, struct ib_ucm_device, dev);
cdev_del(&ucm_dev->cdev); kfree(ucm_dev);
}
static void ib_ucm_free_dev(struct ib_ucm_device *ucm_dev)
{
if (ucm_dev->devnum < IB_UCM_MAX_DEVICES) if (ucm_dev->devnum < IB_UCM_MAX_DEVICES)
clear_bit(ucm_dev->devnum, dev_map); clear_bit(ucm_dev->devnum, dev_map);
else else
clear_bit(ucm_dev->devnum - IB_UCM_MAX_DEVICES, overflow_map); clear_bit(ucm_dev->devnum - IB_UCM_MAX_DEVICES, overflow_map);
kfree(ucm_dev);
} }
static const struct file_operations ucm_fops = { static const struct file_operations ucm_fops = {
...@@ -1266,7 +1269,9 @@ static void ib_ucm_add_one(struct ib_device *device) ...@@ -1266,7 +1269,9 @@ static void ib_ucm_add_one(struct ib_device *device)
if (!ucm_dev) if (!ucm_dev)
return; return;
device_initialize(&ucm_dev->dev);
ucm_dev->ib_dev = device; ucm_dev->ib_dev = device;
ucm_dev->dev.release = ib_ucm_release_dev;
devnum = find_first_zero_bit(dev_map, IB_UCM_MAX_DEVICES); devnum = find_first_zero_bit(dev_map, IB_UCM_MAX_DEVICES);
if (devnum >= IB_UCM_MAX_DEVICES) { if (devnum >= IB_UCM_MAX_DEVICES) {
...@@ -1286,16 +1291,14 @@ static void ib_ucm_add_one(struct ib_device *device) ...@@ -1286,16 +1291,14 @@ static void ib_ucm_add_one(struct ib_device *device)
cdev_init(&ucm_dev->cdev, &ucm_fops); cdev_init(&ucm_dev->cdev, &ucm_fops);
ucm_dev->cdev.owner = THIS_MODULE; ucm_dev->cdev.owner = THIS_MODULE;
kobject_set_name(&ucm_dev->cdev.kobj, "ucm%d", ucm_dev->devnum); kobject_set_name(&ucm_dev->cdev.kobj, "ucm%d", ucm_dev->devnum);
if (cdev_add(&ucm_dev->cdev, base, 1))
goto err;
ucm_dev->dev.class = &cm_class; ucm_dev->dev.class = &cm_class;
ucm_dev->dev.parent = device->dev.parent; ucm_dev->dev.parent = device->dev.parent;
ucm_dev->dev.devt = ucm_dev->cdev.dev; ucm_dev->dev.devt = base;
ucm_dev->dev.release = ib_ucm_release_dev;
dev_set_name(&ucm_dev->dev, "ucm%d", ucm_dev->devnum); dev_set_name(&ucm_dev->dev, "ucm%d", ucm_dev->devnum);
if (device_register(&ucm_dev->dev)) if (cdev_device_add(&ucm_dev->cdev, &ucm_dev->dev))
goto err_cdev; goto err_devnum;
if (device_create_file(&ucm_dev->dev, &dev_attr_ibdev)) if (device_create_file(&ucm_dev->dev, &dev_attr_ibdev))
goto err_dev; goto err_dev;
...@@ -1304,15 +1307,11 @@ static void ib_ucm_add_one(struct ib_device *device) ...@@ -1304,15 +1307,11 @@ static void ib_ucm_add_one(struct ib_device *device)
return; return;
err_dev: err_dev:
device_unregister(&ucm_dev->dev); cdev_device_del(&ucm_dev->cdev, &ucm_dev->dev);
err_cdev: err_devnum:
cdev_del(&ucm_dev->cdev); ib_ucm_free_dev(ucm_dev);
if (ucm_dev->devnum < IB_UCM_MAX_DEVICES)
clear_bit(devnum, dev_map);
else
clear_bit(devnum, overflow_map);
err: err:
kfree(ucm_dev); put_device(&ucm_dev->dev);
return; return;
} }
...@@ -1323,7 +1322,9 @@ static void ib_ucm_remove_one(struct ib_device *device, void *client_data) ...@@ -1323,7 +1322,9 @@ static void ib_ucm_remove_one(struct ib_device *device, void *client_data)
if (!ucm_dev) if (!ucm_dev)
return; return;
device_unregister(&ucm_dev->dev); cdev_device_del(&ucm_dev->cdev, &ucm_dev->dev);
ib_ucm_free_dev(ucm_dev);
put_device(&ucm_dev->dev);
} }
static CLASS_ATTR_STRING(abi_version, S_IRUGO, static CLASS_ATTR_STRING(abi_version, S_IRUGO,
......
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