aboutsummaryrefslogtreecommitdiffstats
path: root/fs/f2fs/super.c
diff options
context:
space:
mode:
authorJaegeuk Kim <jaegeuk.kim@samsung.com>2013-01-24 05:56:11 -0500
committerJaegeuk Kim <jaegeuk.kim@samsung.com>2013-02-11 17:15:00 -0500
commit577e349514452fa3fcd99fd06e587b02d3d1cf28 (patch)
treefe4e86d3f56b8050c85b1f5fb76117d6e99c5aa9 /fs/f2fs/super.c
parent7d79e75f6420fc13cfe72554b3ea5afad24c8625 (diff)
f2fs: prevent checkpoint once any IO failure is detected
This patch enhances the checkpoint routine to cope with IO errors. Basically f2fs detects IO errors from end_io_write, and the errors are able to be occurred during one of data, node, and meta page writes. In the previous code, when an IO error is occurred during writes, f2fs sets a flag, CP_ERROR_FLAG, in the raw ckeckpoint buffer which will be written to disk. Afterwards, write_checkpoint() will check the flag and remount f2fs as a read-only (ro) mode. However, even once f2fs is remounted as a ro mode, dirty checkpoint pages are freely able to be written to disk by flusher or kswapd in background. In such a case, after cold reboot, f2fs would restore the checkpoint data having CP_ERROR_FLAG, resulting in disabling write_checkpoint and remounting f2fs as a ro mode again. Therefore, let's prevent any checkpoint page (meta) writes once an IO error is occurred, and remount f2fs as a ro mode right away at that moment. Reported-by: Oliver Winker <oliver@oli1170.net> Signed-off-by: Jaegeuk Kim <jaegeuk.kim@samsung.com> Reviewed-by: Namjae Jeon <namjae.jeon@samsung.com>
Diffstat (limited to 'fs/f2fs/super.c')
-rw-r--r--fs/f2fs/super.c12
1 files changed, 9 insertions, 3 deletions
diff --git a/fs/f2fs/super.c b/fs/f2fs/super.c
index 37fad04c8669..117ca2a46e39 100644
--- a/fs/f2fs/super.c
+++ b/fs/f2fs/super.c
@@ -387,10 +387,11 @@ static int sanity_check_raw_super(struct super_block *sb,
387 return 0; 387 return 0;
388} 388}
389 389
390static int sanity_check_ckpt(struct f2fs_super_block *raw_super, 390static int sanity_check_ckpt(struct f2fs_sb_info *sbi)
391 struct f2fs_checkpoint *ckpt)
392{ 391{
393 unsigned int total, fsmeta; 392 unsigned int total, fsmeta;
393 struct f2fs_super_block *raw_super = F2FS_RAW_SUPER(sbi);
394 struct f2fs_checkpoint *ckpt = F2FS_CKPT(sbi);
394 395
395 total = le32_to_cpu(raw_super->segment_count); 396 total = le32_to_cpu(raw_super->segment_count);
396 fsmeta = le32_to_cpu(raw_super->segment_count_ckpt); 397 fsmeta = le32_to_cpu(raw_super->segment_count_ckpt);
@@ -401,6 +402,11 @@ static int sanity_check_ckpt(struct f2fs_super_block *raw_super,
401 402
402 if (fsmeta >= total) 403 if (fsmeta >= total)
403 return 1; 404 return 1;
405
406 if (is_set_ckpt_flags(ckpt, CP_ERROR_FLAG)) {
407 f2fs_msg(sbi->sb, KERN_ERR, "A bug case: need to run fsck");
408 return 1;
409 }
404 return 0; 410 return 0;
405} 411}
406 412
@@ -525,7 +531,7 @@ static int f2fs_fill_super(struct super_block *sb, void *data, int silent)
525 531
526 /* sanity checking of checkpoint */ 532 /* sanity checking of checkpoint */
527 err = -EINVAL; 533 err = -EINVAL;
528 if (sanity_check_ckpt(raw_super, sbi->ckpt)) { 534 if (sanity_check_ckpt(sbi)) {
529 f2fs_msg(sb, KERN_ERR, "Invalid F2FS checkpoint"); 535 f2fs_msg(sb, KERN_ERR, "Invalid F2FS checkpoint");
530 goto free_cp; 536 goto free_cp;
531 } 537 }