From: liujinbao1 <[email protected]>
During SPO tests, when mounting F2FS, an -EINVAL error was returned.
The issue originates from the f2fs_recover_inode_page function's
check, where old_ni.blk_addr != NULL_ADDR under the conditions of
IS_INODE(folio) && is_dent_dnode(folio).
Clear dent flag of the node block to fix this issue.
Test Step:
1.Modify files and induce SPO to generate fsync inode list.
blkaddr: 0x15828, ino: 1378, is_inode: 1, is_fsync: 2, is_dent: 0
2.Use inject.f2fs to set dent flag for a fsync inode.
inject.f2fs --cp 0 --mb flag --idx 2 --val 7 /dev/vda
Info: inject blkaddr[2] flag of cp 2: 0x3 -> 0x7
3.Boot verification, System fails to mount during recovery:
f2fs_recover_inode_page:ino: 1378, ni.blkaddr: 0x1581a,
old_ni.blk_addr != NULL_ADDR
F2FS-fs (vda): Cannot recover all fsync data errno=-22
4.Run fsck
fsck.f2fs -f /dev/vda
[ASSERT] (f2fs_find_fsync_inode:3924) --> Invalid dent flag:
blkaddr: 0x15828, ino: 1378, is_dent: 4, nat entry blkaddr: 0x1581a
[FIX] (f2fs_find_fsync_inode:3931) --> Clear dent flag: blkaddr: 0x15828,
ino: 1378
After this fix, the system boots normally.
Signed-off-by: Sheng Yong <[email protected]>
Signed-off-by: Yongpeng Yang <[email protected]>
Signed-off-by: Jinbao Liu <[email protected]>
---
v3:
- Add test steps to verify the effect.
---
v2:
- Clear the node_blk dent flag, proceed with recovery
of this and subsequent data.
---
v1:
- Clear the node_blk fsync flag, set next_blkaddr=NULL,
skip recovery of this and later data.
---
fsck/mount.c | 19 ++++++++++++++++++-
fsck/node.h | 13 +++++++++++++
2 files changed, 31 insertions(+), 1 deletion(-)
diff --git a/fsck/mount.c b/fsck/mount.c
index 6f640a0..3385489 100644
--- a/fsck/mount.c
+++ b/fsck/mount.c
@@ -3878,6 +3878,7 @@ int f2fs_find_fsync_inode(struct f2fs_sb_info *sbi,
struct list_head *head)
while (1) {
struct fsync_inode_entry *entry;
+ struct f2fs_nat_entry nat_entry;
if (!f2fs_is_valid_blkaddr(sbi, blkaddr, META_POR))
break;
@@ -3902,8 +3903,24 @@ int f2fs_find_fsync_inode(struct f2fs_sb_info *sbi,
struct list_head *head)
}
entry->blkaddr = blkaddr;
- if (IS_INODE(node_blk) && is_dent_dnode(node_blk))
+ if (IS_INODE(node_blk) && is_dent_dnode(node_blk)) {
+ get_nat_entry(sbi, ino_of_node(node_blk), &nat_entry);
+ if (is_valid_data_blkaddr(nat_entry.block_addr)) {
+ ASSERT_MSG("Invalid dent flag: blkaddr: 0x%x, "
+ "ino: %u, is_dent: %d, nat entry blkaddr:
0x%x\n",
+ blkaddr, ino_of_node(node_blk),
is_dent_dnode(node_blk),
+ nat_entry.block_addr);
+ if (c.fix_on && f2fs_dev_is_writable()) {
+ FIX_MSG("Clear dent flag: blkaddr: 0x%x,
ino: %u\n",
+ blkaddr, ino_of_node(node_blk));
+ set_dentry_mark(node_blk, 0);
+ err = update_inode(sbi, node_blk,
&blkaddr);
+ ASSERT(err >= 0);
+ goto next;
+ }
+ }
entry->last_dentry = blkaddr;
+ }
next:
blkaddr = next_blkaddr_of_node(node_blk);
diff --git a/fsck/node.h b/fsck/node.h
index 19f1e57..f3e2c46 100644
--- a/fsck/node.h
+++ b/fsck/node.h
@@ -173,6 +173,19 @@ static inline void set_cold_node(struct f2fs_node *rn,
bool is_dir)
F2FS_NODE_FOOTER(rn)->flag = cpu_to_le32(flag);
}
+static inline void set_mark(struct f2fs_node *rn, int mark, int type)
+{
+ unsigned int flag = le32_to_cpu(F2FS_NODE_FOOTER(rn)->flag);
+
+ if (mark)
+ flag |= (1 << type);
+ else
+ flag &= ~(1 << type);
+ F2FS_NODE_FOOTER(rn)->flag = cpu_to_le32(flag);
+}
+
+#define set_dentry_mark(page, mark) set_mark(page, mark, DENT_BIT_SHIFT)
+
#define is_fsync_dnode(node_blk) is_node(node_blk, FSYNC_BIT_SHIFT)
#define is_dent_dnode(node_blk) is_node(node_blk,
DENT_BIT_SHIFT)