Commit fb951eb5 authored by Zongxun Wang's avatar Zongxun Wang Committed by Linus Torvalds

ocfs2: free allocated clusters if error occurs after ocfs2_claim_clusters

Even if using the same jbd2 handle, we cannot rollback a transaction.
So once some error occurs after successfully allocating clusters, the
allocated clusters will never be used and it means they are lost.  For
example, call ocfs2_claim_clusters successfully when expanding a file,
but failed in ocfs2_insert_extent.  So we need free the allocated
clusters if they are not used indeed.
Signed-off-by: default avatarZongxun Wang <wangzongxun@huawei.com>
Signed-off-by: default avatarJoseph Qi <joseph.qi@huawei.com>
Acked-by: default avatarJoel Becker <jlbec@evilplan.org>
Cc: Mark Fasheh <mfasheh@suse.com>
Cc: Li Zefan <lizefan@huawei.com>
Signed-off-by: default avatarAndrew Morton <akpm@linux-foundation.org>
Signed-off-by: default avatarLinus Torvalds <torvalds@linux-foundation.org>
parent 277cba1d
...@@ -4742,6 +4742,7 @@ int ocfs2_add_clusters_in_btree(handle_t *handle, ...@@ -4742,6 +4742,7 @@ int ocfs2_add_clusters_in_btree(handle_t *handle,
enum ocfs2_alloc_restarted *reason_ret) enum ocfs2_alloc_restarted *reason_ret)
{ {
int status = 0, err = 0; int status = 0, err = 0;
int need_free = 0;
int free_extents; int free_extents;
enum ocfs2_alloc_restarted reason = RESTART_NONE; enum ocfs2_alloc_restarted reason = RESTART_NONE;
u32 bit_off, num_bits; u32 bit_off, num_bits;
...@@ -4796,7 +4797,8 @@ int ocfs2_add_clusters_in_btree(handle_t *handle, ...@@ -4796,7 +4797,8 @@ int ocfs2_add_clusters_in_btree(handle_t *handle,
OCFS2_JOURNAL_ACCESS_WRITE); OCFS2_JOURNAL_ACCESS_WRITE);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; need_free = 1;
goto bail;
} }
block = ocfs2_clusters_to_blocks(osb->sb, bit_off); block = ocfs2_clusters_to_blocks(osb->sb, bit_off);
...@@ -4807,7 +4809,8 @@ int ocfs2_add_clusters_in_btree(handle_t *handle, ...@@ -4807,7 +4809,8 @@ int ocfs2_add_clusters_in_btree(handle_t *handle,
num_bits, flags, meta_ac); num_bits, flags, meta_ac);
if (status < 0) { if (status < 0) {
mlog_errno(status); mlog_errno(status);
goto leave; need_free = 1;
goto bail;
} }
ocfs2_journal_dirty(handle, et->et_root_bh); ocfs2_journal_dirty(handle, et->et_root_bh);
...@@ -4821,6 +4824,19 @@ int ocfs2_add_clusters_in_btree(handle_t *handle, ...@@ -4821,6 +4824,19 @@ int ocfs2_add_clusters_in_btree(handle_t *handle,
reason = RESTART_TRANS; reason = RESTART_TRANS;
} }
bail:
if (need_free) {
if (data_ac->ac_which == OCFS2_AC_USE_LOCAL)
ocfs2_free_local_alloc_bits(osb, handle, data_ac,
bit_off, num_bits);
else
ocfs2_free_clusters(handle,
data_ac->ac_inode,
data_ac->ac_bh,
ocfs2_clusters_to_blocks(osb->sb, bit_off),
num_bits);
}
leave: leave:
if (reason_ret) if (reason_ret)
*reason_ret = reason; *reason_ret = reason;
...@@ -6805,6 +6821,8 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6805,6 +6821,8 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
struct buffer_head *di_bh) struct buffer_head *di_bh)
{ {
int ret, i, has_data, num_pages = 0; int ret, i, has_data, num_pages = 0;
int need_free = 0;
u32 bit_off, num;
handle_t *handle; handle_t *handle;
u64 uninitialized_var(block); u64 uninitialized_var(block);
struct ocfs2_inode_info *oi = OCFS2_I(inode); struct ocfs2_inode_info *oi = OCFS2_I(inode);
...@@ -6850,7 +6868,6 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6850,7 +6868,6 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
} }
if (has_data) { if (has_data) {
u32 bit_off, num;
unsigned int page_end; unsigned int page_end;
u64 phys; u64 phys;
...@@ -6886,6 +6903,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6886,6 +6903,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
ret = ocfs2_grab_eof_pages(inode, 0, end, pages, &num_pages); ret = ocfs2_grab_eof_pages(inode, 0, end, pages, &num_pages);
if (ret) { if (ret) {
mlog_errno(ret); mlog_errno(ret);
need_free = 1;
goto out_commit; goto out_commit;
} }
...@@ -6896,6 +6914,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6896,6 +6914,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
ret = ocfs2_read_inline_data(inode, pages[0], di_bh); ret = ocfs2_read_inline_data(inode, pages[0], di_bh);
if (ret) { if (ret) {
mlog_errno(ret); mlog_errno(ret);
need_free = 1;
goto out_commit; goto out_commit;
} }
...@@ -6927,6 +6946,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6927,6 +6946,7 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
ret = ocfs2_insert_extent(handle, &et, 0, block, 1, 0, NULL); ret = ocfs2_insert_extent(handle, &et, 0, block, 1, 0, NULL);
if (ret) { if (ret) {
mlog_errno(ret); mlog_errno(ret);
need_free = 1;
goto out_commit; goto out_commit;
} }
...@@ -6938,6 +6958,18 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode, ...@@ -6938,6 +6958,18 @@ int ocfs2_convert_inline_data_to_extents(struct inode *inode,
dquot_free_space_nodirty(inode, dquot_free_space_nodirty(inode,
ocfs2_clusters_to_bytes(osb->sb, 1)); ocfs2_clusters_to_bytes(osb->sb, 1));
if (need_free) {
if (data_ac->ac_which == OCFS2_AC_USE_LOCAL)
ocfs2_free_local_alloc_bits(osb, handle, data_ac,
bit_off, num);
else
ocfs2_free_clusters(handle,
data_ac->ac_inode,
data_ac->ac_bh,
ocfs2_clusters_to_blocks(osb->sb, bit_off),
num);
}
ocfs2_commit_trans(osb, handle); ocfs2_commit_trans(osb, handle);
out_unlock: out_unlock:
......
...@@ -781,6 +781,48 @@ int ocfs2_claim_local_alloc_bits(struct ocfs2_super *osb, ...@@ -781,6 +781,48 @@ int ocfs2_claim_local_alloc_bits(struct ocfs2_super *osb,
return status; return status;
} }
int ocfs2_free_local_alloc_bits(struct ocfs2_super *osb,
handle_t *handle,
struct ocfs2_alloc_context *ac,
u32 bit_off,
u32 num_bits)
{
int status, start;
u32 clear_bits;
struct inode *local_alloc_inode;
void *bitmap;
struct ocfs2_dinode *alloc;
struct ocfs2_local_alloc *la;
BUG_ON(ac->ac_which != OCFS2_AC_USE_LOCAL);
local_alloc_inode = ac->ac_inode;
alloc = (struct ocfs2_dinode *) osb->local_alloc_bh->b_data;
la = OCFS2_LOCAL_ALLOC(alloc);
bitmap = la->la_bitmap;
start = bit_off - le32_to_cpu(la->la_bm_off);
clear_bits = num_bits;
status = ocfs2_journal_access_di(handle,
INODE_CACHE(local_alloc_inode),
osb->local_alloc_bh,
OCFS2_JOURNAL_ACCESS_WRITE);
if (status < 0) {
mlog_errno(status);
goto bail;
}
while (clear_bits--)
ocfs2_clear_bit(start++, bitmap);
le32_add_cpu(&alloc->id1.bitmap1.i_used, -num_bits);
ocfs2_journal_dirty(handle, osb->local_alloc_bh);
bail:
return status;
}
static u32 ocfs2_local_alloc_count_bits(struct ocfs2_dinode *alloc) static u32 ocfs2_local_alloc_count_bits(struct ocfs2_dinode *alloc)
{ {
u32 count; u32 count;
......
...@@ -55,6 +55,12 @@ int ocfs2_claim_local_alloc_bits(struct ocfs2_super *osb, ...@@ -55,6 +55,12 @@ int ocfs2_claim_local_alloc_bits(struct ocfs2_super *osb,
u32 *bit_off, u32 *bit_off,
u32 *num_bits); u32 *num_bits);
int ocfs2_free_local_alloc_bits(struct ocfs2_super *osb,
handle_t *handle,
struct ocfs2_alloc_context *ac,
u32 bit_off,
u32 num_bits);
void ocfs2_local_alloc_seen_free_bits(struct ocfs2_super *osb, void ocfs2_local_alloc_seen_free_bits(struct ocfs2_super *osb,
unsigned int num_clusters); unsigned int num_clusters);
void ocfs2_la_enable_worker(struct work_struct *work); void ocfs2_la_enable_worker(struct work_struct *work);
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment