From 166781608941be963d7b0ffd40c4a785cea5a97f Mon Sep 17 00:00:00 2001 From: Boris Protopopov Date: Wed, 9 Aug 2017 14:10:47 -0400 Subject: [PATCH] zv_suspend_lock in zvol_open()/zvol_release() Acquire zv_suspend_lock on first open and last close only. Reviewed-by: Brian Behlendorf Signed-off-by: Boris Protopopov Closes #6342 --- module/zfs/zvol.c | 64 ++++++++++++++++++++++++++++++----------------- 1 file changed, 41 insertions(+), 23 deletions(-) diff --git a/module/zfs/zvol.c b/module/zfs/zvol.c index 3e7059b34..ffa5fac7c 100644 --- a/module/zfs/zvol.c +++ b/module/zfs/zvol.c @@ -1347,9 +1347,9 @@ zvol_open(struct block_device *bdev, fmode_t flag) { zvol_state_t *zv; int error = 0; - boolean_t drop_suspend = B_FALSE; + boolean_t drop_suspend = B_TRUE; - ASSERT(!mutex_owned(&zvol_state_lock)); + ASSERT(!MUTEX_HELD(&zvol_state_lock)); mutex_enter(&zvol_state_lock); /* @@ -1364,23 +1364,31 @@ zvol_open(struct block_device *bdev, fmode_t flag) return (SET_ERROR(-ENXIO)); } - /* take zv_suspend_lock before zv_state_lock */ - rw_enter(&zv->zv_suspend_lock, RW_READER); - mutex_enter(&zv->zv_state_lock); - /* * make sure zvol is not suspended during first open - * (hold zv_suspend_lock), otherwise, drop the lock + * (hold zv_suspend_lock) and respect proper lock acquisition + * ordering - zv_suspend_lock before zv_state_lock */ if (zv->zv_open_count == 0) { - drop_suspend = B_TRUE; + if (!rw_tryenter(&zv->zv_suspend_lock, RW_READER)) { + mutex_exit(&zv->zv_state_lock); + rw_enter(&zv->zv_suspend_lock, RW_READER); + mutex_enter(&zv->zv_state_lock); + /* check to see if zv_suspend_lock is needed */ + if (zv->zv_open_count != 0) { + rw_exit(&zv->zv_suspend_lock); + drop_suspend = B_FALSE; + } + } } else { - rw_exit(&zv->zv_suspend_lock); + drop_suspend = B_FALSE; } - mutex_exit(&zvol_state_lock); + ASSERT(MUTEX_HELD(&zv->zv_state_lock)); + ASSERT(zv->zv_open_count != 0 || RW_READ_HELD(&zv->zv_suspend_lock)); + if (zv->zv_open_count == 0) { error = zvol_first_open(zv); if (error) @@ -1417,28 +1425,38 @@ static int zvol_release(struct gendisk *disk, fmode_t mode) { zvol_state_t *zv; - boolean_t drop_suspend = B_FALSE; + boolean_t drop_suspend = B_TRUE; - ASSERT(!mutex_owned(&zvol_state_lock)); + ASSERT(!MUTEX_HELD(&zvol_state_lock)); mutex_enter(&zvol_state_lock); zv = disk->private_data; - ASSERT(zv && zv->zv_open_count > 0); - - /* take zv_suspend_lock before zv_state_lock */ - rw_enter(&zv->zv_suspend_lock, RW_READER); mutex_enter(&zv->zv_state_lock); - mutex_exit(&zvol_state_lock); - + ASSERT(zv->zv_open_count > 0); /* * make sure zvol is not suspended during last close - * (hold zv_suspend_lock), otherwise, drop the lock + * (hold zv_suspend_lock) and respect proper lock acquisition + * ordering - zv_suspend_lock before zv_state_lock */ - if (zv->zv_open_count == 1) - drop_suspend = B_TRUE; - else - rw_exit(&zv->zv_suspend_lock); + if (zv->zv_open_count == 1) { + if (!rw_tryenter(&zv->zv_suspend_lock, RW_READER)) { + mutex_exit(&zv->zv_state_lock); + rw_enter(&zv->zv_suspend_lock, RW_READER); + mutex_enter(&zv->zv_state_lock); + /* check to see if zv_suspend_lock is needed */ + if (zv->zv_open_count != 1) { + rw_exit(&zv->zv_suspend_lock); + drop_suspend = B_FALSE; + } + } + } else { + drop_suspend = B_FALSE; + } + mutex_exit(&zvol_state_lock); + + ASSERT(MUTEX_HELD(&zv->zv_state_lock)); + ASSERT(zv->zv_open_count != 1 || RW_READ_HELD(&zv->zv_suspend_lock)); zv->zv_open_count--; if (zv->zv_open_count == 0)