diff options
| -rw-r--r-- | drivers/md/dm.c | 27 |
1 files changed, 19 insertions, 8 deletions
diff --git a/drivers/md/dm.c b/drivers/md/dm.c index fe7c56e10435..3668b170ea68 100644 --- a/drivers/md/dm.c +++ b/drivers/md/dm.c | |||
| @@ -1116,7 +1116,8 @@ static int __bind(struct mapped_device *md, struct dm_table *t) | |||
| 1116 | if (size != get_capacity(md->disk)) | 1116 | if (size != get_capacity(md->disk)) |
| 1117 | memset(&md->geometry, 0, sizeof(md->geometry)); | 1117 | memset(&md->geometry, 0, sizeof(md->geometry)); |
| 1118 | 1118 | ||
| 1119 | __set_size(md, size); | 1119 | if (md->suspended_bdev) |
| 1120 | __set_size(md, size); | ||
| 1120 | if (size == 0) | 1121 | if (size == 0) |
| 1121 | return 0; | 1122 | return 0; |
| 1122 | 1123 | ||
| @@ -1264,6 +1265,11 @@ int dm_swap_table(struct mapped_device *md, struct dm_table *table) | |||
| 1264 | if (!dm_suspended(md)) | 1265 | if (!dm_suspended(md)) |
| 1265 | goto out; | 1266 | goto out; |
| 1266 | 1267 | ||
| 1268 | /* without bdev, the device size cannot be changed */ | ||
| 1269 | if (!md->suspended_bdev) | ||
| 1270 | if (get_capacity(md->disk) != dm_table_get_size(table)) | ||
| 1271 | goto out; | ||
| 1272 | |||
| 1267 | __unbind(md); | 1273 | __unbind(md); |
| 1268 | r = __bind(md, table); | 1274 | r = __bind(md, table); |
| 1269 | 1275 | ||
| @@ -1341,11 +1347,14 @@ int dm_suspend(struct mapped_device *md, unsigned suspend_flags) | |||
| 1341 | /* This does not get reverted if there's an error later. */ | 1347 | /* This does not get reverted if there's an error later. */ |
| 1342 | dm_table_presuspend_targets(map); | 1348 | dm_table_presuspend_targets(map); |
| 1343 | 1349 | ||
| 1344 | md->suspended_bdev = bdget_disk(md->disk, 0); | 1350 | /* bdget() can stall if the pending I/Os are not flushed */ |
| 1345 | if (!md->suspended_bdev) { | 1351 | if (!noflush) { |
| 1346 | DMWARN("bdget failed in dm_suspend"); | 1352 | md->suspended_bdev = bdget_disk(md->disk, 0); |
| 1347 | r = -ENOMEM; | 1353 | if (!md->suspended_bdev) { |
| 1348 | goto flush_and_out; | 1354 | DMWARN("bdget failed in dm_suspend"); |
| 1355 | r = -ENOMEM; | ||
| 1356 | goto flush_and_out; | ||
| 1357 | } | ||
| 1349 | } | 1358 | } |
| 1350 | 1359 | ||
| 1351 | /* | 1360 | /* |
| @@ -1473,8 +1482,10 @@ int dm_resume(struct mapped_device *md) | |||
| 1473 | 1482 | ||
| 1474 | unlock_fs(md); | 1483 | unlock_fs(md); |
| 1475 | 1484 | ||
| 1476 | bdput(md->suspended_bdev); | 1485 | if (md->suspended_bdev) { |
| 1477 | md->suspended_bdev = NULL; | 1486 | bdput(md->suspended_bdev); |
| 1487 | md->suspended_bdev = NULL; | ||
| 1488 | } | ||
| 1478 | 1489 | ||
| 1479 | clear_bit(DMF_SUSPENDED, &md->flags); | 1490 | clear_bit(DMF_SUSPENDED, &md->flags); |
| 1480 | 1491 | ||
