Commit 94735ec4044a6d318b83ad3c5794e931ed168d10
Committed by
David Woodhouse
1 parent
5c39c4c54c
Exists in
master
and in
4 other branches
mtd: mtd_blkdevs: fix error path in blktrans_open
The 'blktrans_open()' does not handle possible '__get_mtd_device()' failures because it does not check the error code. Moreover, the 'dev->tr->open()' failures are not handled correctly because in this case the function just goes ahead and gets the mtd device, then returns an error. But Instead, it should _not_ try to get the mtd device, then it should put back the module and the kref. This patch fixes the issue. Note, I only compile-tested it. This patch was inspired by a bug report about a similar issue in 2.6.34 kernels sent by Mike Turner <admin@islandsoftware.co.uk> to the MTD mailing list: http://lists.infradead.org/pipermail/linux-mtd/2011-April/034980.html Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@nokia.com> Signed-off-by: David Woodhouse <David.Woodhouse@intel.com>
Showing 1 changed file with 21 additions and 3 deletions Side-by-side Diff
drivers/mtd/mtd_blkdevs.c
... | ... | @@ -221,12 +221,30 @@ |
221 | 221 | kref_get(&dev->ref); |
222 | 222 | __module_get(dev->tr->owner); |
223 | 223 | |
224 | - if (dev->mtd) { | |
225 | - ret = dev->tr->open ? dev->tr->open(dev) : 0; | |
226 | - __get_mtd_device(dev->mtd); | |
224 | + if (!dev->mtd) | |
225 | + goto unlock; | |
226 | + | |
227 | + if (dev->tr->open) { | |
228 | + ret = dev->tr->open(dev); | |
229 | + if (ret) | |
230 | + goto error_put; | |
227 | 231 | } |
228 | 232 | |
233 | + ret = __get_mtd_device(dev->mtd); | |
234 | + if (ret) | |
235 | + goto error_release; | |
236 | + | |
229 | 237 | unlock: |
238 | + mutex_unlock(&dev->lock); | |
239 | + blktrans_dev_put(dev); | |
240 | + return ret; | |
241 | + | |
242 | +error_release: | |
243 | + if (dev->tr->release) | |
244 | + dev->tr->release(dev); | |
245 | +error_put: | |
246 | + module_put(dev->tr->owner); | |
247 | + kref_put(&dev->ref, blktrans_dev_release); | |
230 | 248 | mutex_unlock(&dev->lock); |
231 | 249 | blktrans_dev_put(dev); |
232 | 250 | return ret; |