Skip to content

Commit ad34d9c

Browse files
tytsogregkh
authored andcommitted
ext4: fix FS_IOC_GETFSMAP handling
commit 4a622e4 upstream. The original implementation ext4's FS_IOC_GETFSMAP handling only worked when the range of queried blocks included at least one free (unallocated) block range. This is because how the metadata blocks were emitted was as a side effect of ext4_mballoc_query_range() calling ext4_getfsmap_datadev_helper(), and that function was only called when a free block range was identified. As a result, this caused generic/365 to fail. Fix this by creating a new function ext4_getfsmap_meta_helper() which gets called so that blocks before the first free block range in a block group can get properly reported. Signed-off-by: Theodore Ts'o <tytso@mit.edu> Cc: stable@vger.kernel.org Signed-off-by: Greg Kroah-Hartman <gregkh@linuxfoundation.org>
1 parent ec56ada commit ad34d9c

File tree

3 files changed

+68
-5
lines changed

3 files changed

+68
-5
lines changed

fs/ext4/fsmap.c

+53-1
Original file line numberDiff line numberDiff line change
@@ -185,6 +185,56 @@ static inline ext4_fsblk_t ext4_fsmap_next_pblk(struct ext4_fsmap *fmr)
185185
return fmr->fmr_physical + fmr->fmr_length;
186186
}
187187

188+
static int ext4_getfsmap_meta_helper(struct super_block *sb,
189+
ext4_group_t agno, ext4_grpblk_t start,
190+
ext4_grpblk_t len, void *priv)
191+
{
192+
struct ext4_getfsmap_info *info = priv;
193+
struct ext4_fsmap *p;
194+
struct ext4_fsmap *tmp;
195+
struct ext4_sb_info *sbi = EXT4_SB(sb);
196+
ext4_fsblk_t fsb, fs_start, fs_end;
197+
int error;
198+
199+
fs_start = fsb = (EXT4_C2B(sbi, start) +
200+
ext4_group_first_block_no(sb, agno));
201+
fs_end = fs_start + EXT4_C2B(sbi, len);
202+
203+
/* Return relevant extents from the meta_list */
204+
list_for_each_entry_safe(p, tmp, &info->gfi_meta_list, fmr_list) {
205+
if (p->fmr_physical < info->gfi_next_fsblk) {
206+
list_del(&p->fmr_list);
207+
kfree(p);
208+
continue;
209+
}
210+
if (p->fmr_physical <= fs_start ||
211+
p->fmr_physical + p->fmr_length <= fs_end) {
212+
/* Emit the retained free extent record if present */
213+
if (info->gfi_lastfree.fmr_owner) {
214+
error = ext4_getfsmap_helper(sb, info,
215+
&info->gfi_lastfree);
216+
if (error)
217+
return error;
218+
info->gfi_lastfree.fmr_owner = 0;
219+
}
220+
error = ext4_getfsmap_helper(sb, info, p);
221+
if (error)
222+
return error;
223+
fsb = p->fmr_physical + p->fmr_length;
224+
if (info->gfi_next_fsblk < fsb)
225+
info->gfi_next_fsblk = fsb;
226+
list_del(&p->fmr_list);
227+
kfree(p);
228+
continue;
229+
}
230+
}
231+
if (info->gfi_next_fsblk < fsb)
232+
info->gfi_next_fsblk = fsb;
233+
234+
return 0;
235+
}
236+
237+
188238
/* Transform a blockgroup's free record into a fsmap */
189239
static int ext4_getfsmap_datadev_helper(struct super_block *sb,
190240
ext4_group_t agno, ext4_grpblk_t start,
@@ -539,6 +589,7 @@ static int ext4_getfsmap_datadev(struct super_block *sb,
539589
error = ext4_mballoc_query_range(sb, info->gfi_agno,
540590
EXT4_B2C(sbi, info->gfi_low.fmr_physical),
541591
EXT4_B2C(sbi, info->gfi_high.fmr_physical),
592+
ext4_getfsmap_meta_helper,
542593
ext4_getfsmap_datadev_helper, info);
543594
if (error)
544595
goto err;
@@ -560,7 +611,8 @@ static int ext4_getfsmap_datadev(struct super_block *sb,
560611

561612
/* Report any gaps at the end of the bg */
562613
info->gfi_last = true;
563-
error = ext4_getfsmap_datadev_helper(sb, end_ag, last_cluster, 0, info);
614+
error = ext4_getfsmap_datadev_helper(sb, end_ag, last_cluster + 1,
615+
0, info);
564616
if (error)
565617
goto err;
566618

fs/ext4/mballoc.c

+14-4
Original file line numberDiff line numberDiff line change
@@ -6999,13 +6999,14 @@ int
69996999
ext4_mballoc_query_range(
70007000
struct super_block *sb,
70017001
ext4_group_t group,
7002-
ext4_grpblk_t start,
7002+
ext4_grpblk_t first,
70037003
ext4_grpblk_t end,
7004+
ext4_mballoc_query_range_fn meta_formatter,
70047005
ext4_mballoc_query_range_fn formatter,
70057006
void *priv)
70067007
{
70077008
void *bitmap;
7008-
ext4_grpblk_t next;
7009+
ext4_grpblk_t start, next;
70097010
struct ext4_buddy e4b;
70107011
int error;
70117012

@@ -7016,10 +7017,19 @@ ext4_mballoc_query_range(
70167017

70177018
ext4_lock_group(sb, group);
70187019

7019-
start = max(e4b.bd_info->bb_first_free, start);
7020+
start = max(e4b.bd_info->bb_first_free, first);
70207021
if (end >= EXT4_CLUSTERS_PER_GROUP(sb))
70217022
end = EXT4_CLUSTERS_PER_GROUP(sb) - 1;
7022-
7023+
if (meta_formatter && start != first) {
7024+
if (start > end)
7025+
start = end;
7026+
ext4_unlock_group(sb, group);
7027+
error = meta_formatter(sb, group, first, start - first,
7028+
priv);
7029+
if (error)
7030+
goto out_unload;
7031+
ext4_lock_group(sb, group);
7032+
}
70237033
while (start <= end) {
70247034
start = mb_find_next_zero_bit(bitmap, end + 1, start);
70257035
if (start > end)

fs/ext4/mballoc.h

+1
Original file line numberDiff line numberDiff line change
@@ -259,6 +259,7 @@ ext4_mballoc_query_range(
259259
ext4_group_t agno,
260260
ext4_grpblk_t start,
261261
ext4_grpblk_t end,
262+
ext4_mballoc_query_range_fn meta_formatter,
262263
ext4_mballoc_query_range_fn formatter,
263264
void *priv);
264265

0 commit comments

Comments
 (0)