/** * main.c * * Copyright (c) 2013 Samsung Electronics Co., Ltd. * http://www.samsung.com/ * Copyright (c) 2015 Jaegeuk Kim * : implement defrag.f2fs * Copyright (C) 2015 Huawei Ltd. * Hou Pengyang * Liu Shuoran * Jaegeuk Kim * : add sload.f2fs * * This program is free software; you can redistribute it and/or modify * it under the terms of the GNU General Public License version 2 as * published by the Free Software Foundation. */ #include "fsck.h" #include struct f2fs_fsck gfsck; void fsck_usage() { MSG(0, "\nUsage: fsck.f2fs [options] device\n"); MSG(0, "[options]:\n"); MSG(0, " -a check/fix potential corruption, reported by f2fs\n"); MSG(0, " -d debug level [default:0]\n"); MSG(0, " -f check/fix entire partition\n"); MSG(0, " -p preen mode [default:0 the same as -a [0|1]]\n"); MSG(0, " -t show directory tree [-d -1]\n"); exit(1); } void dump_usage() { MSG(0, "\nUsage: dump.f2fs [options] device\n"); MSG(0, "[options]:\n"); MSG(0, " -d debug level [default:0]\n"); MSG(0, " -i inode no (hex)\n"); MSG(0, " -n [NAT dump segno from #1~#2 (decimal), for all 0~-1]\n"); MSG(0, " -s [SIT dump segno from #1~#2 (decimal), for all 0~-1]\n"); MSG(0, " -a [SSA dump segno from #1~#2 (decimal), for all 0~-1]\n"); MSG(0, " -b blk_addr (in 4KB)\n"); exit(1); } void defrag_usage() { MSG(0, "\nUsage: defrag.f2fs [options] device\n"); MSG(0, "[options]:\n"); MSG(0, " -d debug level [default:0]\n"); MSG(0, " -s start block address [default: main_blkaddr]\n"); MSG(0, " -l length [default:512 (2MB)]\n"); MSG(0, " -t target block address [default: main_blkaddr + 2MB]\n"); MSG(0, " -i set direction as shrink [default: expand]\n"); exit(1); } void resize_usage() { MSG(0, "\nUsage: resize.f2fs [options] device\n"); MSG(0, "[options]:\n"); MSG(0, " -d debug level [default:0]\n"); MSG(0, " -t target sectors [default: device size]\n"); exit(1); } void sload_usage() { MSG(0, "\nUsage: sload.f2fs [options] device\n"); MSG(0, "[options]:\n"); MSG(0, " -f source directory [path of the source directory]\n"); MSG(0, " -t mount point [prefix of target fs path, default:/]\n"); MSG(0, " -d debug level [default:0]\n"); exit(1); } void f2fs_parse_options(int argc, char *argv[]) { int option = 0; char *prog = basename(argv[0]); if (!strcmp("fsck.f2fs", prog)) { const char *option_string = "ad:fp:t"; c.func = FSCK; while ((option = getopt(argc, argv, option_string)) != EOF) { switch (option) { case 'a': c.auto_fix = 1; MSG(0, "Info: Fix the reported corruption.\n"); break; case 'p': /* preen mode has different levels: * 0: default level, the same as -a * 1: check meta */ c.preen_mode = atoi(optarg); if (c.preen_mode < 0) c.preen_mode = PREEN_MODE_0; else if (c.preen_mode >= PREEN_MODE_MAX) c.preen_mode = PREEN_MODE_MAX - 1; if (c.preen_mode == PREEN_MODE_0) c.auto_fix = 1; MSG(0, "Info: Fix the reported corruption in " "preen mode %d\n", c.preen_mode); break; case 'd': c.dbg_lv = atoi(optarg); MSG(0, "Info: Debug level = %d\n", c.dbg_lv); break; case 'f': c.fix_on = 1; MSG(0, "Info: Force to fix corruption\n"); break; case 't': c.dbg_lv = -1; break; default: MSG(0, "\tError: Unknown option %c\n", option); fsck_usage(); break; } } } else if (!strcmp("dump.f2fs", prog)) { const char *option_string = "d:i:n:s:a:b:"; static struct dump_option dump_opt = { .nid = 0, /* default root ino */ .start_nat = -1, .end_nat = -1, .start_sit = -1, .end_sit = -1, .start_ssa = -1, .end_ssa = -1, .blk_addr = -1, }; c.func = DUMP; while ((option = getopt(argc, argv, option_string)) != EOF) { int ret = 0; switch (option) { case 'd': c.dbg_lv = atoi(optarg); MSG(0, "Info: Debug level = %d\n", c.dbg_lv); break; case 'i': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%d", &dump_opt.nid); else ret = sscanf(optarg, "%x", &dump_opt.nid); break; case 'n': ret = sscanf(optarg, "%d~%d", &dump_opt.start_nat, &dump_opt.end_nat); break; case 's': ret = sscanf(optarg, "%d~%d", &dump_opt.start_sit, &dump_opt.end_sit); break; case 'a': ret = sscanf(optarg, "%d~%d", &dump_opt.start_ssa, &dump_opt.end_ssa); break; case 'b': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%d", &dump_opt.blk_addr); else ret = sscanf(optarg, "%x", &dump_opt.blk_addr); break; default: MSG(0, "\tError: Unknown option %c\n", option); dump_usage(); break; } ASSERT(ret >= 0); } c.private = &dump_opt; } else if (!strcmp("defrag.f2fs", prog)) { const char *option_string = "d:s:l:t:i"; c.func = DEFRAG; while ((option = getopt(argc, argv, option_string)) != EOF) { int ret = 0; switch (option) { case 'd': c.dbg_lv = atoi(optarg); MSG(0, "Info: Debug level = %d\n", c.dbg_lv); break; case 's': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%"PRIu64"", &c.defrag_start); else ret = sscanf(optarg, "%"PRIx64"", &c.defrag_start); break; case 'l': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%"PRIu64"", &c.defrag_len); else ret = sscanf(optarg, "%"PRIx64"", &c.defrag_len); break; case 't': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%"PRIu64"", &c.defrag_target); else ret = sscanf(optarg, "%"PRIx64"", &c.defrag_target); break; case 'i': c.defrag_shrink = 1; break; default: MSG(0, "\tError: Unknown option %c\n", option); defrag_usage(); break; } ASSERT(ret >= 0); } } else if (!strcmp("resize.f2fs", prog)) { const char *option_string = "d:t:"; c.func = RESIZE; while ((option = getopt(argc, argv, option_string)) != EOF) { int ret = 0; switch (option) { case 'd': c.dbg_lv = atoi(optarg); MSG(0, "Info: Debug level = %d\n", c.dbg_lv); break; case 't': if (strncmp(optarg, "0x", 2)) ret = sscanf(optarg, "%"PRIu64"", &c.target_sectors); else ret = sscanf(optarg, "%"PRIx64"", &c.target_sectors); break; default: MSG(0, "\tError: Unknown option %c\n", option); resize_usage(); break; } ASSERT(ret >= 0); } } else if (!strcmp("sload.f2fs", prog)) { const char *option_string = "d:f:t:"; c.func = SLOAD; while ((option = getopt(argc, argv, option_string)) != EOF) { switch (option) { case 'd': c.dbg_lv = atoi(optarg); MSG(0, "Info: Debug level = %d\n", c.dbg_lv); break; case 'f': c.from_dir = (char *)optarg; break; case 't': c.mount_point = (char *)optarg; break; default: MSG(0, "\tError: Unknown option %c\n", option); sload_usage(); break; } } } if ((optind + 1) != argc) { MSG(0, "\tError: Device not specified\n"); if (c.func == FSCK) fsck_usage(); else if (c.func == DUMP) dump_usage(); else if (c.func == DEFRAG) defrag_usage(); else if (c.func == RESIZE) resize_usage(); else if (c.func == SLOAD) sload_usage(); } c.device_name = argv[optind]; } static void do_fsck(struct f2fs_sb_info *sbi) { struct f2fs_checkpoint *ckpt = F2FS_CKPT(sbi); u32 flag = le32_to_cpu(ckpt->ckpt_flags); u32 blk_cnt; fsck_init(sbi); print_cp_state(flag); if (!c.fix_on && !c.bug_on) { switch (c.preen_mode) { case PREEN_MODE_1: if (fsck_chk_meta(sbi)) { MSG(0, "[FSCK] F2FS metadata [Fail]"); MSG(0, "\tError: meta does not match, " "force check all\n"); } else { MSG(0, "[FSCK] F2FS metadata [Ok..]"); fsck_free(sbi); return; } if (!c.ro) c.fix_on = 1; break; } } else { /* * we can hit this in 3 situations: * 1. fsck -f, fix_on has already been set to 1 when * parsing options; * 2. fsck -a && CP_FSCK_FLAG is set, fix_on has already * been set to 1 when checking CP_FSCK_FLAG; * 3. fsck -p 1 && error is detected, then bug_on is set, * we set fix_on = 1 here, so that fsck can fix errors * automatically */ c.fix_on = 1; } fsck_chk_orphan_node(sbi); /* Traverse all block recursively from root inode */ blk_cnt = 1; fsck_chk_node_blk(sbi, NULL, sbi->root_ino_num, (u8 *)"/", F2FS_FT_DIR, TYPE_INODE, &blk_cnt, NULL); fsck_verify(sbi); fsck_free(sbi); } static void do_dump(struct f2fs_sb_info *sbi) { struct dump_option *opt = (struct dump_option *)c.private; struct f2fs_checkpoint *ckpt = F2FS_CKPT(sbi); u32 flag = le32_to_cpu(ckpt->ckpt_flags); if (opt->end_nat == -1) opt->end_nat = NM_I(sbi)->max_nid; if (opt->end_sit == -1) opt->end_sit = SM_I(sbi)->main_segments; if (opt->end_ssa == -1) opt->end_ssa = SM_I(sbi)->main_segments; if (opt->start_nat != -1) nat_dump(sbi, opt->start_nat, opt->end_nat); if (opt->start_sit != -1) sit_dump(sbi, opt->start_sit, opt->end_sit); if (opt->start_ssa != -1) ssa_dump(sbi, opt->start_ssa, opt->end_ssa); if (opt->blk_addr != -1) dump_info_from_blkaddr(sbi, opt->blk_addr); if (opt->nid) dump_node(sbi, opt->nid); print_cp_state(flag); } static int do_defrag(struct f2fs_sb_info *sbi) { struct f2fs_super_block *sb = F2FS_RAW_SUPER(sbi); if (c.defrag_start > get_sb(block_count)) goto out_range; if (c.defrag_start < SM_I(sbi)->main_blkaddr) c.defrag_start = SM_I(sbi)->main_blkaddr; if (c.defrag_len == 0) c.defrag_len = sbi->blocks_per_seg; if (c.defrag_start + c.defrag_len > get_sb(block_count)) c.defrag_len = get_sb(block_count) - c.defrag_start; if (c.defrag_target == 0) { c.defrag_target = c.defrag_start - 1; if (!c.defrag_shrink) c.defrag_target += c.defrag_len + 1; } if (c.defrag_target < SM_I(sbi)->main_blkaddr || c.defrag_target > get_sb(block_count)) goto out_range; if (c.defrag_target >= c.defrag_start && c.defrag_target < c.defrag_start + c.defrag_len) goto out_range; if (c.defrag_start > c.defrag_target) MSG(0, "Info: Move 0x%"PRIx64" <- [0x%"PRIx64"-0x%"PRIx64"]\n", c.defrag_target, c.defrag_start, c.defrag_start + c.defrag_len - 1); else MSG(0, "Info: Move [0x%"PRIx64"-0x%"PRIx64"] -> 0x%"PRIx64"\n", c.defrag_start, c.defrag_start + c.defrag_len - 1, c.defrag_target); return f2fs_defragment(sbi, c.defrag_start, c.defrag_len, c.defrag_target, c.defrag_shrink); out_range: ASSERT_MSG("Out-of-range [0x%"PRIx64" ~ 0x%"PRIx64"] to 0x%"PRIx64"", c.defrag_start, c.defrag_start + c.defrag_len - 1, c.defrag_target); return -1; } static int do_resize(struct f2fs_sb_info *sbi) { struct f2fs_super_block *sb = F2FS_RAW_SUPER(sbi); if (!c.target_sectors) c.target_sectors = c.total_sectors; if (c.target_sectors > c.total_sectors) { ASSERT_MSG("Out-of-range Target=0x%"PRIx64" / 0x%"PRIx64"", c.target_sectors, c.total_sectors); return -1; } if (c.target_sectors == (get_sb(block_count) << get_sb(log_sectors_per_block))) { ASSERT_MSG("Nothing to resize; it's same"); return -1; } return f2fs_resize(sbi); } static int do_sload(struct f2fs_sb_info *sbi) { if (!c.from_dir) { MSG(0, "\tError: Need source directory\n"); sload_usage(); return -1; } if (!c.mount_point) c.mount_point = "/"; return f2fs_sload(sbi, c.from_dir, c.mount_point, NULL, NULL); } int main(int argc, char **argv) { struct f2fs_sb_info *sbi; int ret = 0; f2fs_init_configuration(); f2fs_parse_options(argc, argv); if (f2fs_dev_is_umounted() < 0) { if (!c.ro || c.func == DEFRAG) { MSG(0, "\tError: Not available on mounted device!\n"); return -1; } /* allow ro-mounted partition */ MSG(0, "Info: Check FS only due to RO\n"); c.fix_on = 0; c.auto_fix = 0; } /* Get device */ if (f2fs_get_device_info() < 0) return -1; fsck_again: memset(&gfsck, 0, sizeof(gfsck)); gfsck.sbi.fsck = &gfsck; sbi = &gfsck.sbi; ret = f2fs_do_mount(sbi); if (ret != 0) { if (ret == 1) { MSG(0, "Info: No error was reported\n"); ret = 0; } goto out_err; } switch (c.func) { case FSCK: do_fsck(sbi); break; case DUMP: do_dump(sbi); break; case DEFRAG: ret = do_defrag(sbi); if (ret) goto out_err; break; case RESIZE: if (do_resize(sbi)) goto out_err; break; case SLOAD: do_sload(sbi); break; } f2fs_do_umount(sbi); if (c.func == FSCK && c.bug_on) { if (!c.ro && c.fix_on == 0 && c.auto_fix == 0) { char ans[255] = {0}; retry: printf("Do you want to fix this partition? [Y/N] "); ret = scanf("%s", ans); ASSERT(ret >= 0); if (!strcasecmp(ans, "y")) c.fix_on = 1; else if (!strcasecmp(ans, "n")) c.fix_on = 0; else goto retry; if (c.fix_on) goto fsck_again; } } f2fs_finalize_device(); printf("\nDone.\n"); return 0; out_err: if (sbi->ckpt) free(sbi->ckpt); if (sbi->raw_super) free(sbi->raw_super); return ret; }