Skip to content

Commit d625a2b

Browse files
chaseyuJaegeuk Kim
authored andcommitted
f2fs: fix to avoid migrating empty section
It reports a bug from device w/ zufs: F2FS-fs (dm-64): Inconsistent segment (173822) type [1, 0] in SSA and SIT F2FS-fs (dm-64): Stopped filesystem due to reason: 4 Thread A Thread B - f2fs_expand_inode_data - f2fs_allocate_pinning_section - f2fs_gc_range - do_garbage_collect w/ segno #x - writepage - f2fs_allocate_data_block - new_curseg - allocate segno #x The root cause is: fallocate on pinning file may race w/ block allocation as above, result in do_garbage_collect() from fallocate() may migrate segment which is just allocated by a log, the log will update segment type in its in-memory structure, however GC will get segment type from on-disk SSA block, once segment type changes by log, we can detect such inconsistency, then shutdown filesystem. In this case, on-disk SSA shows type of segno #173822 is 1 (SUM_TYPE_NODE), however segno #173822 was just allocated as data type segment, so in-memory SIT shows type of segno #173822 is 0 (SUM_TYPE_DATA). Change as below to fix this issue: - check whether current section is empty before gc - add sanity checks on do_garbage_collect() to avoid any race case, result in migrating segment used by log. - btw, it fixes misc issue in printed logs: "SSA and SIT" -> "SIT and SSA". Fixes: 9703d69 ("f2fs: support file pinning for zoned devices") Cc: Daeho Jeong <daehojeong@google.com> Signed-off-by: Chao Yu <chao@kernel.org> Signed-off-by: Jaegeuk Kim <jaegeuk@kernel.org>
1 parent 9251a9e commit d625a2b

1 file changed

Lines changed: 15 additions & 1 deletion

File tree

fs/f2fs/gc.c

Lines changed: 15 additions & 1 deletion
Original file line numberDiff line numberDiff line change
@@ -1794,6 +1794,13 @@ static int do_garbage_collect(struct f2fs_sb_info *sbi,
17941794
struct folio *sum_folio = filemap_get_folio(META_MAPPING(sbi),
17951795
GET_SUM_BLOCK(sbi, segno));
17961796

1797+
if (is_cursec(sbi, GET_SEC_FROM_SEG(sbi, segno))) {
1798+
f2fs_err(sbi, "%s: segment %u is used by log",
1799+
__func__, segno);
1800+
f2fs_bug_on(sbi, 1);
1801+
goto skip;
1802+
}
1803+
17971804
if (get_valid_blocks(sbi, segno, false) == 0)
17981805
goto freed;
17991806
if (gc_type == BG_GC && __is_large_section(sbi) &&
@@ -1805,7 +1812,7 @@ static int do_garbage_collect(struct f2fs_sb_info *sbi,
18051812

18061813
sum = folio_address(sum_folio);
18071814
if (type != GET_SUM_TYPE((&sum->footer))) {
1808-
f2fs_err(sbi, "Inconsistent segment (%u) type [%d, %d] in SSA and SIT",
1815+
f2fs_err(sbi, "Inconsistent segment (%u) type [%d, %d] in SIT and SSA",
18091816
segno, type, GET_SUM_TYPE((&sum->footer)));
18101817
f2fs_stop_checkpoint(sbi, false,
18111818
STOP_CP_REASON_CORRUPTED_SUMMARY);
@@ -2068,6 +2075,13 @@ int f2fs_gc_range(struct f2fs_sb_info *sbi,
20682075
.iroot = RADIX_TREE_INIT(gc_list.iroot, GFP_NOFS),
20692076
};
20702077

2078+
/*
2079+
* avoid migrating empty section, as it can be allocated by
2080+
* log in parallel.
2081+
*/
2082+
if (!get_valid_blocks(sbi, segno, true))
2083+
continue;
2084+
20712085
if (is_cursec(sbi, GET_SEC_FROM_SEG(sbi, segno)))
20722086
continue;
20732087

0 commit comments

Comments
 (0)