diff options
Diffstat (limited to 'drivers/md/dm.c')
-rw-r--r-- | drivers/md/dm.c | 51 |
1 files changed, 25 insertions, 26 deletions
diff --git a/drivers/md/dm.c b/drivers/md/dm.c index 9e8c1edd89dd..fc335e09d072 100644 --- a/drivers/md/dm.c +++ b/drivers/md/dm.c | |||
@@ -97,7 +97,7 @@ struct mapped_device { | |||
97 | * freeze/thaw support require holding onto a super block | 97 | * freeze/thaw support require holding onto a super block |
98 | */ | 98 | */ |
99 | struct super_block *frozen_sb; | 99 | struct super_block *frozen_sb; |
100 | struct block_device *frozen_bdev; | 100 | struct block_device *suspended_bdev; |
101 | }; | 101 | }; |
102 | 102 | ||
103 | #define MIN_IOS 256 | 103 | #define MIN_IOS 256 |
@@ -836,9 +836,9 @@ static void __set_size(struct mapped_device *md, sector_t size) | |||
836 | { | 836 | { |
837 | set_capacity(md->disk, size); | 837 | set_capacity(md->disk, size); |
838 | 838 | ||
839 | down(&md->frozen_bdev->bd_inode->i_sem); | 839 | down(&md->suspended_bdev->bd_inode->i_sem); |
840 | i_size_write(md->frozen_bdev->bd_inode, (loff_t)size << SECTOR_SHIFT); | 840 | i_size_write(md->suspended_bdev->bd_inode, (loff_t)size << SECTOR_SHIFT); |
841 | up(&md->frozen_bdev->bd_inode->i_sem); | 841 | up(&md->suspended_bdev->bd_inode->i_sem); |
842 | } | 842 | } |
843 | 843 | ||
844 | static int __bind(struct mapped_device *md, struct dm_table *t) | 844 | static int __bind(struct mapped_device *md, struct dm_table *t) |
@@ -1010,43 +1010,27 @@ out: | |||
1010 | */ | 1010 | */ |
1011 | static int lock_fs(struct mapped_device *md) | 1011 | static int lock_fs(struct mapped_device *md) |
1012 | { | 1012 | { |
1013 | int r = -ENOMEM; | 1013 | int r; |
1014 | |||
1015 | md->frozen_bdev = bdget_disk(md->disk, 0); | ||
1016 | if (!md->frozen_bdev) { | ||
1017 | DMWARN("bdget failed in lock_fs"); | ||
1018 | goto out; | ||
1019 | } | ||
1020 | 1014 | ||
1021 | WARN_ON(md->frozen_sb); | 1015 | WARN_ON(md->frozen_sb); |
1022 | 1016 | ||
1023 | md->frozen_sb = freeze_bdev(md->frozen_bdev); | 1017 | md->frozen_sb = freeze_bdev(md->suspended_bdev); |
1024 | if (IS_ERR(md->frozen_sb)) { | 1018 | if (IS_ERR(md->frozen_sb)) { |
1025 | r = PTR_ERR(md->frozen_sb); | 1019 | r = PTR_ERR(md->frozen_sb); |
1026 | goto out_bdput; | 1020 | md->frozen_sb = NULL; |
1021 | return r; | ||
1027 | } | 1022 | } |
1028 | 1023 | ||
1029 | /* don't bdput right now, we don't want the bdev | 1024 | /* don't bdput right now, we don't want the bdev |
1030 | * to go away while it is locked. We'll bdput | 1025 | * to go away while it is locked. |
1031 | * in unlock_fs | ||
1032 | */ | 1026 | */ |
1033 | return 0; | 1027 | return 0; |
1034 | |||
1035 | out_bdput: | ||
1036 | bdput(md->frozen_bdev); | ||
1037 | md->frozen_sb = NULL; | ||
1038 | md->frozen_bdev = NULL; | ||
1039 | out: | ||
1040 | return r; | ||
1041 | } | 1028 | } |
1042 | 1029 | ||
1043 | static void unlock_fs(struct mapped_device *md) | 1030 | static void unlock_fs(struct mapped_device *md) |
1044 | { | 1031 | { |
1045 | thaw_bdev(md->frozen_bdev, md->frozen_sb); | 1032 | thaw_bdev(md->suspended_bdev, md->frozen_sb); |
1046 | bdput(md->frozen_bdev); | ||
1047 | |||
1048 | md->frozen_sb = NULL; | 1033 | md->frozen_sb = NULL; |
1049 | md->frozen_bdev = NULL; | ||
1050 | } | 1034 | } |
1051 | 1035 | ||
1052 | /* | 1036 | /* |
@@ -1072,6 +1056,13 @@ int dm_suspend(struct mapped_device *md) | |||
1072 | /* This does not get reverted if there's an error later. */ | 1056 | /* This does not get reverted if there's an error later. */ |
1073 | dm_table_presuspend_targets(map); | 1057 | dm_table_presuspend_targets(map); |
1074 | 1058 | ||
1059 | md->suspended_bdev = bdget_disk(md->disk, 0); | ||
1060 | if (!md->suspended_bdev) { | ||
1061 | DMWARN("bdget failed in dm_suspend"); | ||
1062 | r = -ENOMEM; | ||
1063 | goto out; | ||
1064 | } | ||
1065 | |||
1075 | /* Flush I/O to the device. */ | 1066 | /* Flush I/O to the device. */ |
1076 | r = lock_fs(md); | 1067 | r = lock_fs(md); |
1077 | if (r) | 1068 | if (r) |
@@ -1124,6 +1115,11 @@ int dm_suspend(struct mapped_device *md) | |||
1124 | r = 0; | 1115 | r = 0; |
1125 | 1116 | ||
1126 | out: | 1117 | out: |
1118 | if (r && md->suspended_bdev) { | ||
1119 | bdput(md->suspended_bdev); | ||
1120 | md->suspended_bdev = NULL; | ||
1121 | } | ||
1122 | |||
1127 | dm_table_put(map); | 1123 | dm_table_put(map); |
1128 | up(&md->suspend_lock); | 1124 | up(&md->suspend_lock); |
1129 | return r; | 1125 | return r; |
@@ -1154,6 +1150,9 @@ int dm_resume(struct mapped_device *md) | |||
1154 | 1150 | ||
1155 | unlock_fs(md); | 1151 | unlock_fs(md); |
1156 | 1152 | ||
1153 | bdput(md->suspended_bdev); | ||
1154 | md->suspended_bdev = NULL; | ||
1155 | |||
1157 | clear_bit(DMF_SUSPENDED, &md->flags); | 1156 | clear_bit(DMF_SUSPENDED, &md->flags); |
1158 | 1157 | ||
1159 | dm_table_unplug_all(map); | 1158 | dm_table_unplug_all(map); |