ubiformat.c
来自「mtd-utils 是一套更改linux mtd設備的工具」· C语言 代码 · 共 773 行 · 第 1/2 页
C
773 行
if (si->ec[eb] == EB_BAD) { divisor += 1; continue; } if (args.verbose) { normsg_cont("eraseblock %d: erase", eb); fflush(stdout); } err = mtd_erase(mtd, eb); if (err) { sys_errmsg("failed to erase eraseblock %d", eb); goto out_close; } err = read_all(fd, buf, mtd->eb_size); if (err) { sys_errmsg("failed to read eraseblock %d from \"%s\"", written_ebs, args.image); goto out_close; } if (si->ec[eb] <= EC_MAX) ec = si->ec[eb] + 1; else if (!args.override_ec) ec = si->mean_ec; else ec = args.ec; if (args.verbose) { printf(", change EC to %lld", ec); fflush(stdout); } err = change_ec((struct ubi_ec_hdr *)buf, ec); if (err) { errmsg("bad EC header at eraseblock %d of \"%s\"", written_ebs, args.image); goto out_close; } if (args.verbose) { printf(", write data\n"); fflush(stdout); } new_len = drop_ffs(mtd, buf, mtd->eb_size); err = mtd_write(mtd, eb, 0, buf, new_len); if (err) { sys_errmsg("cannot write eraseblock %d", eb); goto out_close; } if (++written_ebs >= img_ebs) break; } if (!args.quiet && !args.verbose) printf("\n"); close(fd); return eb + 1;out_close: close(fd); return -1;}static int format(const struct mtd_info *mtd, const struct ubigen_info *ui, const struct ubi_scan_info *si, int start_eb, int novtbl){ int eb, err, write_size; struct ubi_ec_hdr *hdr; struct ubi_vtbl_record *vtbl; int eb1 = -1, eb2 = -1; long long ec1 = -1, ec2 = -1; write_size = UBI_EC_HDR_SIZE + mtd->subpage_size - 1; write_size /= mtd->subpage_size; write_size *= mtd->subpage_size; hdr = malloc(write_size); if (!hdr) return sys_errmsg("cannot allocate %d bytes of memory", write_size); memset(hdr, 0xFF, write_size); for (eb = start_eb; eb < mtd->eb_cnt; eb++) { long long ec; if (!args.quiet && !args.verbose) { printf("\r" PROGRAM_NAME ": formatting eraseblock %d -- %2lld %% complete ", eb, (long long)(eb + 1 - start_eb) * 100 / (mtd->eb_cnt - start_eb)); fflush(stdout); } if (si->ec[eb] == EB_BAD) continue; if (si->ec[eb] <= EC_MAX) ec = si->ec[eb] + 1; else if (!args.override_ec) ec = si->mean_ec; else ec = args.ec; ubigen_init_ec_hdr(ui, hdr, ec); if (args.verbose) { normsg_cont("eraseblock %d: erase", eb); fflush(stdout); } err = mtd_erase(mtd, eb); if (err) { sys_errmsg("failed to erase eraseblock %d", eb); goto out_free; } if ((eb1 == -1 || eb2 == -1) && !novtbl) { if (eb1 == -1) { eb1 = eb; ec1 = ec; } else if (eb2 == -1) { eb2 = eb; ec2 = ec; } if (args.verbose) printf(", do not write EC, leave for vtbl\n"); continue; } if (args.verbose) { printf(", write EC %lld\n", ec); fflush(stdout); } err = mtd_write(mtd, eb, 0, hdr, write_size); if (err) { sys_errmsg("cannot write EC header (%d bytes buffer) to eraseblock %d", write_size, eb); if (args.subpage_size != mtd->min_io_size) normsg("may be %d is incorrect?", args.subpage_size); goto out_free; } } if (!args.quiet && !args.verbose) printf("\n"); if (!novtbl) { if (eb1 == -1 || eb2 == -1) { errmsg("no eraseblocks for volume table"); goto out_free; } verbose(args.verbose, "write volume table to eraseblocks %d and %d", eb1, eb2); vtbl = ubigen_create_empty_vtbl(ui); if (!vtbl) goto out_free; err = ubigen_write_layout_vol(ui, eb1, eb2, ec1, ec2, vtbl, mtd->fd); free(vtbl); if (err) { errmsg("cannot write layout volume"); goto out_free; } } free(hdr); return 0;out_free: free(hdr); return -1;}int main(int argc, char * const argv[]){ int err, verbose; struct mtd_info mtd; libubi_t libubi; struct ubigen_info ui; struct ubi_scan_info *si; err = parse_opt(argc, argv); if (err) return -1; err = mtd_get_info(args.node, &mtd); if (err) return errmsg("cannot get information about \"%s\"", args.node); if (args.subpage_size == 0) args.subpage_size = mtd.min_io_size; else { if (args.subpage_size > mtd.min_io_size) { errmsg("sub-page cannot be larger than min. I/O unit"); goto out_close; } if (mtd.min_io_size % args.subpage_size) { errmsg("min. I/O unit size should be multiple of sub-page size"); goto out_close; } } /* Validate VID header offset if it was specified */ if (args.vid_hdr_offs != 0) { if (args.vid_hdr_offs % 8) { errmsg("VID header offset has to be multiple of min. I/O unit size"); goto out_close; } if (args.vid_hdr_offs + UBI_VID_HDR_SIZE > mtd.eb_size) { errmsg("bad VID header offset"); goto out_close; } } /* * Because of MTD interface limitations 'mtd_get_info()' cannot get * sub-page so we force the user to pass it via the command line. Let's * hope the user passed us something sane. */ mtd.subpage_size = args.subpage_size; if (mtd.rdonly) { errmsg("mtd%d (%s) is a read-only device", mtd.num, args.node); goto out_close; } /* Make sure this MTD device is not attached to UBI */ libubi = libubi_open(0); if (libubi) { int ubi_dev_num; err = mtd_num2ubi_dev(libubi, mtd.num, &ubi_dev_num); libubi_close(libubi); if (!err) { errmsg("please, first detach mtd%d (%s) from ubi%d", mtd.num, args.node, ubi_dev_num); goto out_close; } } if (!args.quiet) { normsg_cont("mtd%d (%s), size ", mtd.num, mtd.type_str); ubiutils_print_bytes(mtd.size, 1); printf(", %d eraseblocks of ", mtd.eb_size); ubiutils_print_bytes(mtd.eb_size, 1); printf(", min. I/O size %d bytes\n", mtd.min_io_size); } if (args.quiet) verbose = 0; else if (args.verbose) verbose = 2; else verbose = 1; err = ubi_scan(&mtd, &si, verbose); if (err) { errmsg("failed to scan mtd%d (%s)", mtd.num, args.node); goto out_close; } if (si->good_cnt == 0) { errmsg("all %d eraseblocks are bad", si->bad_cnt); goto out_free; } if (si->good_cnt < 2 && (!args.novtbl || args.image)) { errmsg("too few non-bad eraseblocks (%d) on mtd%d", si->good_cnt, mtd.num); goto out_free; } if (!args.quiet) { if (si->ok_cnt) normsg("%d eraseblocks have valid erase counter, mean value is %lld", si->ok_cnt, si->mean_ec); if (si->empty_cnt) normsg("%d eraseblocks are supposedly empty", si->empty_cnt); if (si->corrupted_cnt) normsg("%d corrupted erase counters", si->corrupted_cnt); print_bad_eraseblocks(&mtd, si); } if (si->alien_cnt) { if (!args.yes || !args.quiet) warnmsg("%d of %d eraseblocks contain non-ubifs data", si->alien_cnt, si->good_cnt); if (!args.yes && want_exit()) { if (args.yes && !args.quiet) printf("yes\n"); goto out_free; } } if (!args.override_ec && si->empty_cnt < si->good_cnt) { int percent = ((double)si->ok_cnt)/si->good_cnt * 100; /* * Make sure the majority of eraseblocks have valid * erase counters. */ if (percent < 50) { if (!args.yes || !args.quiet) warnmsg("only %d of %d eraseblocks have valid erase counter", si->ok_cnt, si->good_cnt); normsg("erase counter 0 will be used for all eraseblocks"); normsg("note, arbitrary erase counter value may be specified using -e option"); if (!args.yes && want_exit()) { if (args.yes && !args.quiet) printf("yes\n"); goto out_free; } args.ec = 0; args.override_ec = 1; } else if (percent < 95) { if (!args.yes || !args.quiet) warnmsg("only %d of %d eraseblocks have valid erase counter", si->ok_cnt, si->good_cnt); normsg("mean erase counter %lld will be used for the rest of eraseblock", si->mean_ec); if (!args.yes && want_exit()) { if (args.yes && !args.quiet) printf("yes\n"); goto out_free; } args.ec = si->mean_ec; args.override_ec = 1; } } if (!args.quiet && args.override_ec) normsg("use erase counter %lld for all eraseblocks", args.ec); ubigen_info_init(&ui, mtd.eb_size, mtd.min_io_size, args.subpage_size, args.vid_hdr_offs, args.ubi_ver); if (si->vid_hdr_offs != -1 && ui.vid_hdr_offs != si->vid_hdr_offs) { /* * Hmm, what we read from flash and what we calculated using * min. I/O unit size and sub-page size differs. */ if (!args.yes || !args.quiet) { warnmsg("VID header and data offsets on flash are %d and %d, " "which is different to calculated offsets %d and %d", si->vid_hdr_offs, si->data_offs, ui.vid_hdr_offs, ui.data_offs); normsg_cont("use new offsets %d and %d? (yes/no) ", si->vid_hdr_offs, si->data_offs); } if (args.yes || answer_is_yes()) { if (args.yes && !args.quiet) printf("yes\n"); } else { ui.vid_hdr_offs = si->vid_hdr_offs; ui.data_offs = si->data_offs; } } if (args.image) { err = flash_image(&mtd, &ui, si); if (err < 0) goto out_free; err = format(&mtd, &ui, si, err, 1); if (err) goto out_free; } else { err = format(&mtd, &ui, si, 0, args.novtbl); if (err) goto out_free; } ubi_scan_free(si); close(mtd.fd); return 0;out_free: ubi_scan_free(si);out_close: close(mtd.fd); return -1;}
⌨️ 快捷键说明
复制代码Ctrl + C
搜索代码Ctrl + F
全屏模式F11
增大字号Ctrl + =
减小字号Ctrl + -
显示快捷键?