Skip to content

Commit

Permalink
rbd: don't clean up watch in device release function
Browse files Browse the repository at this point in the history
Currently, a watch on an rbd device header object gets torn down
when its final Linux device reference gets dropped.  Instead, tear
it down when removing the device.  If an error occurs cleaning up
the watch event when unmapping, abort the unmap request.

All images (including parents) still get watch requests set up, so
tear these down also, in rbd_dev_remove_parent().  For now, ignore
any errors that occur in this case.

Get rid of local variable "rc" in rbd_remove(); use "ret" instead
(they both somehow ended up defined in the function and only one is
needed).

Signed-off-by: Alex Elder <elder@inktank.com>
Reviewed-by: Josh Durgin <josh.durgin@inktank.com>
  • Loading branch information
Alex Elder authored and Sage Weil committed May 2, 2013
1 parent 332bb12 commit 0d8189e
Showing 1 changed file with 23 additions and 8 deletions.
31 changes: 23 additions & 8 deletions drivers/block/rbd.c
Original file line number Diff line number Diff line change
Expand Up @@ -4729,6 +4729,7 @@ static int rbd_dev_probe_parent(struct rbd_device *rbd_dev)
static int rbd_dev_probe_finish(struct rbd_device *rbd_dev)
{
int ret;
int tmp;

ret = rbd_dev_header_watch_sync(rbd_dev, 1);
if (ret)
Expand Down Expand Up @@ -4780,6 +4781,9 @@ static int rbd_dev_probe_finish(struct rbd_device *rbd_dev)
unregister_blkdev(rbd_dev->major, rbd_dev->name);
err_out_id:
rbd_dev_id_put(rbd_dev);
tmp = rbd_dev_header_watch_sync(rbd_dev, 0);
if (tmp)
rbd_warn(rbd_dev, "failed to cancel watch event (%d)\n", ret);
rbd_dev_mapping_clear(rbd_dev);

return ret;
Expand Down Expand Up @@ -4975,9 +4979,6 @@ static void rbd_dev_release(struct device *dev)
{
struct rbd_device *rbd_dev = dev_to_rbd_dev(dev);

if (rbd_dev->watch_event)
rbd_dev_header_watch_sync(rbd_dev, 0);

/* clean up and free blkdev */
rbd_free_disk(rbd_dev);
unregister_blkdev(rbd_dev->major, rbd_dev->name);
Expand All @@ -5003,6 +5004,7 @@ static void rbd_dev_remove_parent(struct rbd_device *rbd_dev)
struct rbd_device *first = rbd_dev;
struct rbd_device *second = first->parent;
struct rbd_device *third;
int ret;

/*
* Follow to the parent with no grandparent and
Expand All @@ -5013,6 +5015,10 @@ static void rbd_dev_remove_parent(struct rbd_device *rbd_dev)
second = third;
}
rbd_assert(second);
ret = rbd_dev_header_watch_sync(rbd_dev, 0);
if (ret)
rbd_warn(rbd_dev,
"failed to cancel watch event (%d)\n", ret);
rbd_remove_all_snaps(second);
rbd_bus_del_dev(second);
first->parent = NULL;
Expand All @@ -5029,13 +5035,13 @@ static ssize_t rbd_remove(struct bus_type *bus,
size_t count)
{
struct rbd_device *rbd_dev = NULL;
int target_id, rc;
int target_id;
unsigned long ul;
int ret = count;
int ret;

rc = strict_strtoul(buf, 10, &ul);
if (rc)
return rc;
ret = strict_strtoul(buf, 10, &ul);
if (ret)
return ret;

/* convert to int; abort if we lost anything in the conversion */
target_id = (int) ul;
Expand All @@ -5059,6 +5065,15 @@ static ssize_t rbd_remove(struct bus_type *bus,
if (ret < 0)
goto done;

ret = rbd_dev_header_watch_sync(rbd_dev, 0);
if (ret) {
rbd_warn(rbd_dev, "failed to cancel watch event (%d)\n", ret);
clear_bit(RBD_DEV_FLAG_REMOVING, &rbd_dev->flags);
smp_mb();
return ret;
}
ret = count;

rbd_dev_remove_parent(rbd_dev);

rbd_remove_all_snaps(rbd_dev);
Expand Down

0 comments on commit 0d8189e

Please sign in to comment.