diff options
| -rw-r--r-- | drivers/mtd/ubi/io.c | 54 | ||||
| -rw-r--r-- | drivers/mtd/ubi/scan.c | 4 | ||||
| -rw-r--r-- | drivers/mtd/ubi/ubi.h | 10 | ||||
| -rw-r--r-- | drivers/mtd/ubi/wl.c | 10 |
4 files changed, 51 insertions, 27 deletions
diff --git a/drivers/mtd/ubi/io.c b/drivers/mtd/ubi/io.c index 1677a215af64..b76252465c87 100644 --- a/drivers/mtd/ubi/io.c +++ b/drivers/mtd/ubi/io.c | |||
| @@ -720,16 +720,16 @@ bad: | |||
| 720 | int ubi_io_read_ec_hdr(struct ubi_device *ubi, int pnum, | 720 | int ubi_io_read_ec_hdr(struct ubi_device *ubi, int pnum, |
| 721 | struct ubi_ec_hdr *ec_hdr, int verbose) | 721 | struct ubi_ec_hdr *ec_hdr, int verbose) |
| 722 | { | 722 | { |
| 723 | int err, read_err = 0; | 723 | int err, read_err; |
| 724 | uint32_t crc, magic, hdr_crc; | 724 | uint32_t crc, magic, hdr_crc; |
| 725 | 725 | ||
| 726 | dbg_io("read EC header from PEB %d", pnum); | 726 | dbg_io("read EC header from PEB %d", pnum); |
| 727 | ubi_assert(pnum >= 0 && pnum < ubi->peb_count); | 727 | ubi_assert(pnum >= 0 && pnum < ubi->peb_count); |
| 728 | 728 | ||
| 729 | err = ubi_io_read(ubi, ec_hdr, pnum, 0, UBI_EC_HDR_SIZE); | 729 | read_err = ubi_io_read(ubi, ec_hdr, pnum, 0, UBI_EC_HDR_SIZE); |
| 730 | if (err) { | 730 | if (read_err) { |
| 731 | if (err != UBI_IO_BITFLIPS && err != -EBADMSG) | 731 | if (read_err != UBI_IO_BITFLIPS && read_err != -EBADMSG) |
| 732 | return err; | 732 | return read_err; |
| 733 | 733 | ||
| 734 | /* | 734 | /* |
| 735 | * We read all the data, but either a correctable bit-flip | 735 | * We read all the data, but either a correctable bit-flip |
| @@ -740,14 +740,12 @@ int ubi_io_read_ec_hdr(struct ubi_device *ubi, int pnum, | |||
| 740 | * this. If the EC header is still OK, we just report this as | 740 | * this. If the EC header is still OK, we just report this as |
| 741 | * there was a bit-flip, to force scrubbing. | 741 | * there was a bit-flip, to force scrubbing. |
| 742 | */ | 742 | */ |
| 743 | if (err == -EBADMSG) | ||
| 744 | read_err = UBI_IO_BAD_HDR_EBADMSG; | ||
| 745 | } | 743 | } |
| 746 | 744 | ||
| 747 | magic = be32_to_cpu(ec_hdr->magic); | 745 | magic = be32_to_cpu(ec_hdr->magic); |
| 748 | if (magic != UBI_EC_HDR_MAGIC) { | 746 | if (magic != UBI_EC_HDR_MAGIC) { |
| 749 | if (read_err) | 747 | if (read_err == -EBADMSG) |
| 750 | return read_err; | 748 | return UBI_IO_BAD_HDR_EBADMSG; |
| 751 | 749 | ||
| 752 | /* | 750 | /* |
| 753 | * The magic field is wrong. Let's check if we have read all | 751 | * The magic field is wrong. Let's check if we have read all |
| @@ -762,7 +760,10 @@ int ubi_io_read_ec_hdr(struct ubi_device *ubi, int pnum, | |||
| 762 | else if (UBI_IO_DEBUG) | 760 | else if (UBI_IO_DEBUG) |
| 763 | dbg_msg("no EC header found at PEB %d, " | 761 | dbg_msg("no EC header found at PEB %d, " |
| 764 | "only 0xFF bytes", pnum); | 762 | "only 0xFF bytes", pnum); |
| 765 | return UBI_IO_FF; | 763 | if (!read_err) |
| 764 | return UBI_IO_FF; | ||
| 765 | else | ||
| 766 | return UBI_IO_FF_BITFLIPS; | ||
| 766 | } | 767 | } |
| 767 | 768 | ||
| 768 | /* | 769 | /* |
| @@ -790,7 +791,11 @@ int ubi_io_read_ec_hdr(struct ubi_device *ubi, int pnum, | |||
| 790 | } else if (UBI_IO_DEBUG) | 791 | } else if (UBI_IO_DEBUG) |
| 791 | dbg_msg("bad EC header CRC at PEB %d, calculated " | 792 | dbg_msg("bad EC header CRC at PEB %d, calculated " |
| 792 | "%#08x, read %#08x", pnum, crc, hdr_crc); | 793 | "%#08x, read %#08x", pnum, crc, hdr_crc); |
| 793 | return read_err ?: UBI_IO_BAD_HDR; | 794 | |
| 795 | if (!read_err) | ||
| 796 | return UBI_IO_BAD_HDR; | ||
| 797 | else | ||
| 798 | return UBI_IO_BAD_HDR_EBADMSG; | ||
| 794 | } | 799 | } |
| 795 | 800 | ||
| 796 | /* And of course validate what has just been read from the media */ | 801 | /* And of course validate what has just been read from the media */ |
| @@ -986,7 +991,7 @@ bad: | |||
| 986 | int ubi_io_read_vid_hdr(struct ubi_device *ubi, int pnum, | 991 | int ubi_io_read_vid_hdr(struct ubi_device *ubi, int pnum, |
| 987 | struct ubi_vid_hdr *vid_hdr, int verbose) | 992 | struct ubi_vid_hdr *vid_hdr, int verbose) |
| 988 | { | 993 | { |
| 989 | int err, read_err = 0; | 994 | int err, read_err; |
| 990 | uint32_t crc, magic, hdr_crc; | 995 | uint32_t crc, magic, hdr_crc; |
| 991 | void *p; | 996 | void *p; |
| 992 | 997 | ||
| @@ -994,20 +999,15 @@ int ubi_io_read_vid_hdr(struct ubi_device *ubi, int pnum, | |||
| 994 | ubi_assert(pnum >= 0 && pnum < ubi->peb_count); | 999 | ubi_assert(pnum >= 0 && pnum < ubi->peb_count); |
| 995 | 1000 | ||
| 996 | p = (char *)vid_hdr - ubi->vid_hdr_shift; | 1001 | p = (char *)vid_hdr - ubi->vid_hdr_shift; |
| 997 | err = ubi_io_read(ubi, p, pnum, ubi->vid_hdr_aloffset, | 1002 | read_err = ubi_io_read(ubi, p, pnum, ubi->vid_hdr_aloffset, |
| 998 | ubi->vid_hdr_alsize); | 1003 | ubi->vid_hdr_alsize); |
| 999 | if (err) { | 1004 | if (read_err && read_err != UBI_IO_BITFLIPS && read_err != -EBADMSG) |
| 1000 | if (err != UBI_IO_BITFLIPS && err != -EBADMSG) | 1005 | return read_err; |
| 1001 | return err; | ||
| 1002 | |||
| 1003 | if (err == -EBADMSG) | ||
| 1004 | read_err = UBI_IO_BAD_HDR_EBADMSG; | ||
| 1005 | } | ||
| 1006 | 1006 | ||
| 1007 | magic = be32_to_cpu(vid_hdr->magic); | 1007 | magic = be32_to_cpu(vid_hdr->magic); |
| 1008 | if (magic != UBI_VID_HDR_MAGIC) { | 1008 | if (magic != UBI_VID_HDR_MAGIC) { |
| 1009 | if (read_err) | 1009 | if (read_err == -EBADMSG) |
| 1010 | return read_err; | 1010 | return UBI_IO_BAD_HDR_EBADMSG; |
| 1011 | 1011 | ||
| 1012 | if (check_pattern(vid_hdr, 0xFF, UBI_VID_HDR_SIZE)) { | 1012 | if (check_pattern(vid_hdr, 0xFF, UBI_VID_HDR_SIZE)) { |
| 1013 | if (verbose) | 1013 | if (verbose) |
| @@ -1016,7 +1016,10 @@ int ubi_io_read_vid_hdr(struct ubi_device *ubi, int pnum, | |||
| 1016 | else if (UBI_IO_DEBUG) | 1016 | else if (UBI_IO_DEBUG) |
| 1017 | dbg_msg("no VID header found at PEB %d, " | 1017 | dbg_msg("no VID header found at PEB %d, " |
| 1018 | "only 0xFF bytes", pnum); | 1018 | "only 0xFF bytes", pnum); |
| 1019 | return UBI_IO_FF; | 1019 | if (!read_err) |
| 1020 | return UBI_IO_FF; | ||
| 1021 | else | ||
| 1022 | return UBI_IO_FF_BITFLIPS; | ||
| 1020 | } | 1023 | } |
| 1021 | 1024 | ||
| 1022 | if (verbose) { | 1025 | if (verbose) { |
| @@ -1040,7 +1043,10 @@ int ubi_io_read_vid_hdr(struct ubi_device *ubi, int pnum, | |||
| 1040 | } else if (UBI_IO_DEBUG) | 1043 | } else if (UBI_IO_DEBUG) |
| 1041 | dbg_msg("bad CRC at PEB %d, calculated %#08x, " | 1044 | dbg_msg("bad CRC at PEB %d, calculated %#08x, " |
| 1042 | "read %#08x", pnum, crc, hdr_crc); | 1045 | "read %#08x", pnum, crc, hdr_crc); |
| 1043 | return read_err ?: UBI_IO_BAD_HDR; | 1046 | if (!read_err) |
| 1047 | return UBI_IO_BAD_HDR; | ||
| 1048 | else | ||
| 1049 | return UBI_IO_BAD_HDR_EBADMSG; | ||
| 1044 | } | 1050 | } |
| 1045 | 1051 | ||
| 1046 | err = validate_vid_hdr(ubi, vid_hdr); | 1052 | err = validate_vid_hdr(ubi, vid_hdr); |
diff --git a/drivers/mtd/ubi/scan.c b/drivers/mtd/ubi/scan.c index 37cb18ff10c6..6f9080767e3f 100644 --- a/drivers/mtd/ubi/scan.c +++ b/drivers/mtd/ubi/scan.c | |||
| @@ -748,7 +748,7 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, | |||
| 748 | return err; | 748 | return err; |
| 749 | else if (err == UBI_IO_BITFLIPS) | 749 | else if (err == UBI_IO_BITFLIPS) |
| 750 | bitflips = 1; | 750 | bitflips = 1; |
| 751 | else if (err == UBI_IO_FF) | 751 | else if (err == UBI_IO_FF || err == UBI_IO_FF_BITFLIPS) |
| 752 | return add_to_list(si, pnum, UBI_SCAN_UNKNOWN_EC, &si->erase); | 752 | return add_to_list(si, pnum, UBI_SCAN_UNKNOWN_EC, &si->erase); |
| 753 | else if (err == UBI_IO_BAD_HDR_EBADMSG || err == UBI_IO_BAD_HDR) { | 753 | else if (err == UBI_IO_BAD_HDR_EBADMSG || err == UBI_IO_BAD_HDR) { |
| 754 | /* | 754 | /* |
| @@ -817,7 +817,7 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, | |||
| 817 | else if (err == UBI_IO_BITFLIPS) | 817 | else if (err == UBI_IO_BITFLIPS) |
| 818 | bitflips = 1; | 818 | bitflips = 1; |
| 819 | else if (err == UBI_IO_BAD_HDR_EBADMSG || err == UBI_IO_BAD_HDR || | 819 | else if (err == UBI_IO_BAD_HDR_EBADMSG || err == UBI_IO_BAD_HDR || |
| 820 | (err == UBI_IO_FF && ec_corr)) { | 820 | (err == UBI_IO_FF && ec_corr) || err == UBI_IO_FF_BITFLIPS) { |
| 821 | /* VID header is corrupted */ | 821 | /* VID header is corrupted */ |
| 822 | if (err == UBI_IO_BAD_HDR_EBADMSG || | 822 | if (err == UBI_IO_BAD_HDR_EBADMSG || |
| 823 | ec_corr == UBI_IO_BAD_HDR_EBADMSG) | 823 | ec_corr == UBI_IO_BAD_HDR_EBADMSG) |
diff --git a/drivers/mtd/ubi/ubi.h b/drivers/mtd/ubi/ubi.h index 774bdcad6a07..10990770bc9e 100644 --- a/drivers/mtd/ubi/ubi.h +++ b/drivers/mtd/ubi/ubi.h | |||
| @@ -86,17 +86,25 @@ | |||
| 86 | * Error codes returned by the I/O sub-system. | 86 | * Error codes returned by the I/O sub-system. |
| 87 | * | 87 | * |
| 88 | * UBI_IO_FF: the read region of flash contains only 0xFFs | 88 | * UBI_IO_FF: the read region of flash contains only 0xFFs |
| 89 | * UBI_IO_FF_BITFLIPS: the same as %UBI_IO_FF, but also also there was a data | ||
| 90 | * integrity error reported by the MTD driver | ||
| 91 | * (uncorrectable ECC error in case of NAND) | ||
| 89 | * UBI_IO_BAD_HDR: the EC or VID header is corrupted (bad magic or CRC) | 92 | * UBI_IO_BAD_HDR: the EC or VID header is corrupted (bad magic or CRC) |
| 90 | * UBI_IO_BAD_HDR_EBADMSG: the same as %UBI_IO_BAD_HDR, but also there was a | 93 | * UBI_IO_BAD_HDR_EBADMSG: the same as %UBI_IO_BAD_HDR, but also there was a |
| 91 | * data integrity error reported by the MTD driver | 94 | * data integrity error reported by the MTD driver |
| 92 | * (uncorrectable ECC error in case of NAND) | 95 | * (uncorrectable ECC error in case of NAND) |
| 93 | * UBI_IO_BITFLIPS: bit-flips were detected and corrected | 96 | * UBI_IO_BITFLIPS: bit-flips were detected and corrected |
| 97 | * | ||
| 98 | * Note, it is probably better to have bit-flip and ebadmsg as flags which can | ||
| 99 | * be or'ed with other error code. But this is a big change because there are | ||
| 100 | * may callers, so it does not worth the risk of introducing a bug | ||
| 94 | */ | 101 | */ |
| 95 | enum { | 102 | enum { |
| 96 | UBI_IO_FF = 1, | 103 | UBI_IO_FF = 1, |
| 104 | UBI_IO_FF_BITFLIPS, | ||
| 97 | UBI_IO_BAD_HDR, | 105 | UBI_IO_BAD_HDR, |
| 98 | UBI_IO_BAD_HDR_EBADMSG, | 106 | UBI_IO_BAD_HDR_EBADMSG, |
| 99 | UBI_IO_BITFLIPS | 107 | UBI_IO_BITFLIPS, |
| 100 | }; | 108 | }; |
| 101 | 109 | ||
| 102 | /* | 110 | /* |
diff --git a/drivers/mtd/ubi/wl.c b/drivers/mtd/ubi/wl.c index a9e7c9eed703..605ecb1e22bb 100644 --- a/drivers/mtd/ubi/wl.c +++ b/drivers/mtd/ubi/wl.c | |||
| @@ -759,6 +759,16 @@ static int wear_leveling_worker(struct ubi_device *ubi, struct ubi_work *wrk, | |||
| 759 | dbg_wl("PEB %d has no VID header", e1->pnum); | 759 | dbg_wl("PEB %d has no VID header", e1->pnum); |
| 760 | protect = 1; | 760 | protect = 1; |
| 761 | goto out_not_moved; | 761 | goto out_not_moved; |
| 762 | } else if (err == UBI_IO_FF_BITFLIPS) { | ||
| 763 | /* | ||
| 764 | * The same situation as %UBI_IO_FF, but bit-flips were | ||
| 765 | * detected. It is better to schedule this PEB for | ||
| 766 | * scrubbing. | ||
| 767 | */ | ||
| 768 | dbg_wl("PEB %d has no VID header but has bit-flips", | ||
| 769 | e1->pnum); | ||
| 770 | scrubbing = 1; | ||
| 771 | goto out_not_moved; | ||
| 762 | } | 772 | } |
| 763 | 773 | ||
| 764 | ubi_err("error %d while reading VID header from PEB %d", | 774 | ubi_err("error %d while reading VID header from PEB %d", |
