aboutsummaryrefslogtreecommitdiffstats
path: root/fs/ext4
diff options
context:
space:
mode:
authorTheodore Ts'o <tytso@mit.edu>2014-07-30 22:17:17 -0400
committerTheodore Ts'o <tytso@mit.edu>2014-07-30 22:17:17 -0400
commit86f0afd463215fc3e58020493482faa4ac3a4d69 (patch)
tree977290d56ae0893d296a3b17e112cd0c65f9ab58 /fs/ext4
parentee98fa3a8b148a234600a20f7cdc2b4b37f38083 (diff)
ext4: fix ext4_discard_allocated_blocks() if we can't allocate the pa struct
If there is a failure while allocating the preallocation structure, a number of blocks can end up getting marked in the in-memory buddy bitmap, and then not getting released. This can result in the following corruption getting reported by the kernel: EXT4-fs error (device sda3): ext4_mb_generate_buddy:758: group 1126, 12793 clusters in bitmap, 12729 in gd In that case, we need to release the blocks using mb_free_blocks(). Tested: fs smoke test; also demonstrated that with injected errors, the file system is no longer getting corrupted Google-Bug-Id: 16657874 Signed-off-by: "Theodore Ts'o" <tytso@mit.edu> Cc: stable@vger.kernel.org
Diffstat (limited to 'fs/ext4')
-rw-r--r--fs/ext4/mballoc.c21
1 files changed, 20 insertions, 1 deletions
diff --git a/fs/ext4/mballoc.c b/fs/ext4/mballoc.c
index 0e9466f9e767..956027711faf 100644
--- a/fs/ext4/mballoc.c
+++ b/fs/ext4/mballoc.c
@@ -3217,8 +3217,27 @@ static void ext4_mb_collect_stats(struct ext4_allocation_context *ac)
3217static void ext4_discard_allocated_blocks(struct ext4_allocation_context *ac) 3217static void ext4_discard_allocated_blocks(struct ext4_allocation_context *ac)
3218{ 3218{
3219 struct ext4_prealloc_space *pa = ac->ac_pa; 3219 struct ext4_prealloc_space *pa = ac->ac_pa;
3220 struct ext4_buddy e4b;
3221 int err;
3220 3222
3221 if (pa && pa->pa_type == MB_INODE_PA) 3223 if (pa == NULL) {
3224 err = ext4_mb_load_buddy(ac->ac_sb, ac->ac_f_ex.fe_group, &e4b);
3225 if (err) {
3226 /*
3227 * This should never happen since we pin the
3228 * pages in the ext4_allocation_context so
3229 * ext4_mb_load_buddy() should never fail.
3230 */
3231 WARN(1, "mb_load_buddy failed (%d)", err);
3232 return;
3233 }
3234 ext4_lock_group(ac->ac_sb, ac->ac_f_ex.fe_group);
3235 mb_free_blocks(ac->ac_inode, &e4b, ac->ac_f_ex.fe_start,
3236 ac->ac_f_ex.fe_len);
3237 ext4_unlock_group(ac->ac_sb, ac->ac_f_ex.fe_group);
3238 return;
3239 }
3240 if (pa->pa_type == MB_INODE_PA)
3222 pa->pa_free += ac->ac_b_ex.fe_len; 3241 pa->pa_free += ac->ac_b_ex.fe_len;
3223} 3242}
3224 3243