diff options
author | Brian Norris <computersforpeace@gmail.com> | 2011-11-07 18:51:05 -0500 |
---|---|---|
committer | David Woodhouse <David.Woodhouse@intel.com> | 2012-01-09 13:04:01 -0500 |
commit | 342ff28f5a2e5aa3236617bd2bddf6c749677ef2 (patch) | |
tree | 6c76de805597459768c25b9ffbb9ccfe27c39b13 /drivers/mtd/mtd_blkdevs.c | |
parent | 8c3423359644d01cfba3a401e403c549c3f88ac4 (diff) |
mtd: mtd_blkdevs: don't increase 'open' count on error path
Some error paths in mtd_blkdevs were fixed in the following commit:
commit 94735ec4044a6d318b83ad3c5794e931ed168d10
mtd: mtd_blkdevs: fix error path in blktrans_open
But on these error paths, the block device's `dev->open' count is
already incremented before we check for errors. This meant that, while
the error path was handled correctly on the first time through
blktrans_open(), the device is erroneously considered already open on
the second time through.
This problem can be seen, for instance, when a UBI volume is
simultaneously mounted as a UBIFS partition and read through its
corresponding gluebi mtdblockX device. This results in blktrans_open()
passing its error checks (with `dev->open > 0') without actually having
a handle on the device. Here's a summarized log of the actions and
results with nandsim:
# modprobe nandsim
# modprobe mtdblock
# modprobe gluebi
# modprobe ubifs
# ubiattach /dev/ubi_ctrl -m 0
...
# ubimkvol /dev/ubi0 -N test -s 16MiB
...
# mount -t ubifs ubi0:test /mnt
# ls /dev/mtdblock*
/dev/mtdblock0 /dev/mtdblock1
# cat /dev/mtdblock1 > /dev/null
cat: can't open '/dev/mtdblock4': Device or resource busy
# cat /dev/mtdblock1 > /dev/null
CPU 0 Unable to handle kernel paging request at virtual address
fffffff0, epc == 8031536c, ra == 8031f280
Oops[#1]:
...
Call Trace:
[<8031536c>] ubi_leb_read+0x14/0x164
[<8031f280>] gluebi_read+0xf0/0x148
[<802edba8>] mtdblock_readsect+0x64/0x198
[<802ecfe4>] mtd_blktrans_thread+0x330/0x3f4
[<8005be98>] kthread+0x88/0x90
[<8000bc04>] kernel_thread_helper+0x10/0x18
Cc: stable@kernel.org [3.0+]
Signed-off-by: Brian Norris <computersforpeace@gmail.com>
Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@linux.intel.com>
Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Diffstat (limited to 'drivers/mtd/mtd_blkdevs.c')
-rw-r--r-- | drivers/mtd/mtd_blkdevs.c | 3 |
1 files changed, 2 insertions, 1 deletions
diff --git a/drivers/mtd/mtd_blkdevs.c b/drivers/mtd/mtd_blkdevs.c index ed8b5e744b12..424ca5f93c6c 100644 --- a/drivers/mtd/mtd_blkdevs.c +++ b/drivers/mtd/mtd_blkdevs.c | |||
@@ -215,7 +215,7 @@ static int blktrans_open(struct block_device *bdev, fmode_t mode) | |||
215 | 215 | ||
216 | mutex_lock(&dev->lock); | 216 | mutex_lock(&dev->lock); |
217 | 217 | ||
218 | if (dev->open++) | 218 | if (dev->open) |
219 | goto unlock; | 219 | goto unlock; |
220 | 220 | ||
221 | kref_get(&dev->ref); | 221 | kref_get(&dev->ref); |
@@ -235,6 +235,7 @@ static int blktrans_open(struct block_device *bdev, fmode_t mode) | |||
235 | goto error_release; | 235 | goto error_release; |
236 | 236 | ||
237 | unlock: | 237 | unlock: |
238 | dev->open++; | ||
238 | mutex_unlock(&dev->lock); | 239 | mutex_unlock(&dev->lock); |
239 | blktrans_dev_put(dev); | 240 | blktrans_dev_put(dev); |
240 | return ret; | 241 | return ret; |