|
@@ -24,7 +24,7 @@
|
|
|
* This unit is responsible for scanning the flash media, checking UBI
|
|
|
* headers and providing complete information about the UBI flash image.
|
|
|
*
|
|
|
- * The scanning information is reoresented by a &struct ubi_scan_info' object.
|
|
|
+ * The scanning information is represented by a &struct ubi_scan_info' object.
|
|
|
* Information about found volumes is represented by &struct ubi_scan_volume
|
|
|
* objects which are kept in volume RB-tree with root at the @volumes field.
|
|
|
* The RB-tree is indexed by the volume ID.
|
|
@@ -55,8 +55,19 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
static struct ubi_ec_hdr *ech;
|
|
|
static struct ubi_vid_hdr *vidh;
|
|
|
|
|
|
-int ubi_scan_add_to_list(struct ubi_scan_info *si, int pnum, int ec,
|
|
|
- struct list_head *list)
|
|
|
+/**
|
|
|
+ * add_to_list - add physical eraseblock to a list.
|
|
|
+ * @si: scanning information
|
|
|
+ * @pnum: physical eraseblock number to add
|
|
|
+ * @ec: erase counter of the physical eraseblock
|
|
|
+ * @list: the list to add to
|
|
|
+ *
|
|
|
+ * This function adds physical eraseblock @pnum to free, erase, corrupted or
|
|
|
+ * alien lists. Returns zero in case of success and a negative error code in
|
|
|
+ * case of failure.
|
|
|
+ */
|
|
|
+static int add_to_list(struct ubi_scan_info *si, int pnum, int ec,
|
|
|
+ struct list_head *list)
|
|
|
{
|
|
|
struct ubi_scan_leb *seb;
|
|
|
|
|
@@ -121,9 +132,9 @@ static int validate_vid_hdr(const struct ubi_vid_hdr *vid_hdr,
|
|
|
const struct ubi_scan_volume *sv, int pnum)
|
|
|
{
|
|
|
int vol_type = vid_hdr->vol_type;
|
|
|
- int vol_id = ubi32_to_cpu(vid_hdr->vol_id);
|
|
|
- int used_ebs = ubi32_to_cpu(vid_hdr->used_ebs);
|
|
|
- int data_pad = ubi32_to_cpu(vid_hdr->data_pad);
|
|
|
+ int vol_id = be32_to_cpu(vid_hdr->vol_id);
|
|
|
+ int used_ebs = be32_to_cpu(vid_hdr->used_ebs);
|
|
|
+ int data_pad = be32_to_cpu(vid_hdr->data_pad);
|
|
|
|
|
|
if (sv->leb_count != 0) {
|
|
|
int sv_vol_type;
|
|
@@ -189,7 +200,7 @@ static struct ubi_scan_volume *add_volume(struct ubi_scan_info *si, int vol_id,
|
|
|
struct ubi_scan_volume *sv;
|
|
|
struct rb_node **p = &si->volumes.rb_node, *parent = NULL;
|
|
|
|
|
|
- ubi_assert(vol_id == ubi32_to_cpu(vid_hdr->vol_id));
|
|
|
+ ubi_assert(vol_id == be32_to_cpu(vid_hdr->vol_id));
|
|
|
|
|
|
/* Walk the volume RB-tree to look if this volume is already present */
|
|
|
while (*p) {
|
|
@@ -211,11 +222,10 @@ static struct ubi_scan_volume *add_volume(struct ubi_scan_info *si, int vol_id,
|
|
|
return ERR_PTR(-ENOMEM);
|
|
|
|
|
|
sv->highest_lnum = sv->leb_count = 0;
|
|
|
- si->max_sqnum = 0;
|
|
|
sv->vol_id = vol_id;
|
|
|
sv->root = RB_ROOT;
|
|
|
- sv->used_ebs = ubi32_to_cpu(vid_hdr->used_ebs);
|
|
|
- sv->data_pad = ubi32_to_cpu(vid_hdr->data_pad);
|
|
|
+ sv->used_ebs = be32_to_cpu(vid_hdr->used_ebs);
|
|
|
+ sv->data_pad = be32_to_cpu(vid_hdr->data_pad);
|
|
|
sv->compat = vid_hdr->compat;
|
|
|
sv->vol_type = vid_hdr->vol_type == UBI_VID_DYNAMIC ? UBI_DYNAMIC_VOLUME
|
|
|
: UBI_STATIC_VOLUME;
|
|
@@ -257,10 +267,10 @@ static int compare_lebs(const struct ubi_device *ubi,
|
|
|
int len, err, second_is_newer, bitflips = 0, corrupted = 0;
|
|
|
uint32_t data_crc, crc;
|
|
|
struct ubi_vid_hdr *vidh = NULL;
|
|
|
- unsigned long long sqnum2 = ubi64_to_cpu(vid_hdr->sqnum);
|
|
|
+ unsigned long long sqnum2 = be64_to_cpu(vid_hdr->sqnum);
|
|
|
|
|
|
if (seb->sqnum == 0 && sqnum2 == 0) {
|
|
|
- long long abs, v1 = seb->leb_ver, v2 = ubi32_to_cpu(vid_hdr->leb_ver);
|
|
|
+ long long abs, v1 = seb->leb_ver, v2 = be32_to_cpu(vid_hdr->leb_ver);
|
|
|
|
|
|
/*
|
|
|
* UBI constantly increases the logical eraseblock version
|
|
@@ -344,8 +354,8 @@ static int compare_lebs(const struct ubi_device *ubi,
|
|
|
|
|
|
/* Read the data of the copy and check the CRC */
|
|
|
|
|
|
- len = ubi32_to_cpu(vid_hdr->data_size);
|
|
|
- buf = kmalloc(len, GFP_KERNEL);
|
|
|
+ len = be32_to_cpu(vid_hdr->data_size);
|
|
|
+ buf = vmalloc(len);
|
|
|
if (!buf) {
|
|
|
err = -ENOMEM;
|
|
|
goto out_free_vidh;
|
|
@@ -355,7 +365,7 @@ static int compare_lebs(const struct ubi_device *ubi,
|
|
|
if (err && err != UBI_IO_BITFLIPS)
|
|
|
goto out_free_buf;
|
|
|
|
|
|
- data_crc = ubi32_to_cpu(vid_hdr->data_crc);
|
|
|
+ data_crc = be32_to_cpu(vid_hdr->data_crc);
|
|
|
crc = crc32(UBI_CRC32_INIT, buf, len);
|
|
|
if (crc != data_crc) {
|
|
|
dbg_bld("PEB %d CRC error: calculated %#08x, must be %#08x",
|
|
@@ -368,7 +378,7 @@ static int compare_lebs(const struct ubi_device *ubi,
|
|
|
bitflips = !!err;
|
|
|
}
|
|
|
|
|
|
- kfree(buf);
|
|
|
+ vfree(buf);
|
|
|
ubi_free_vid_hdr(ubi, vidh);
|
|
|
|
|
|
if (second_is_newer)
|
|
@@ -379,7 +389,7 @@ static int compare_lebs(const struct ubi_device *ubi,
|
|
|
return second_is_newer | (bitflips << 1) | (corrupted << 2);
|
|
|
|
|
|
out_free_buf:
|
|
|
- kfree(buf);
|
|
|
+ vfree(buf);
|
|
|
out_free_vidh:
|
|
|
ubi_free_vid_hdr(ubi, vidh);
|
|
|
ubi_assert(err < 0);
|
|
@@ -396,8 +406,12 @@ out_free_vidh:
|
|
|
* @vid_hdr: the volume identifier header
|
|
|
* @bitflips: if bit-flips were detected when this physical eraseblock was read
|
|
|
*
|
|
|
- * This function returns zero in case of success and a negative error code in
|
|
|
- * case of failure.
|
|
|
+ * This function adds information about a used physical eraseblock to the
|
|
|
+ * 'used' tree of the corresponding volume. The function is rather complex
|
|
|
+ * because it has to handle cases when this is not the first physical
|
|
|
+ * eraseblock belonging to the same logical eraseblock, and the newer one has
|
|
|
+ * to be picked, while the older one has to be dropped. This function returns
|
|
|
+ * zero in case of success and a negative error code in case of failure.
|
|
|
*/
|
|
|
int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
int pnum, int ec, const struct ubi_vid_hdr *vid_hdr,
|
|
@@ -410,10 +424,10 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
struct ubi_scan_leb *seb;
|
|
|
struct rb_node **p, *parent = NULL;
|
|
|
|
|
|
- vol_id = ubi32_to_cpu(vid_hdr->vol_id);
|
|
|
- lnum = ubi32_to_cpu(vid_hdr->lnum);
|
|
|
- sqnum = ubi64_to_cpu(vid_hdr->sqnum);
|
|
|
- leb_ver = ubi32_to_cpu(vid_hdr->leb_ver);
|
|
|
+ vol_id = be32_to_cpu(vid_hdr->vol_id);
|
|
|
+ lnum = be32_to_cpu(vid_hdr->lnum);
|
|
|
+ sqnum = be64_to_cpu(vid_hdr->sqnum);
|
|
|
+ leb_ver = be32_to_cpu(vid_hdr->leb_ver);
|
|
|
|
|
|
dbg_bld("PEB %d, LEB %d:%d, EC %d, sqnum %llu, ver %u, bitflips %d",
|
|
|
pnum, vol_id, lnum, ec, sqnum, leb_ver, bitflips);
|
|
@@ -422,6 +436,9 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
if (IS_ERR(sv) < 0)
|
|
|
return PTR_ERR(sv);
|
|
|
|
|
|
+ if (si->max_sqnum < sqnum)
|
|
|
+ si->max_sqnum = sqnum;
|
|
|
+
|
|
|
/*
|
|
|
* Walk the RB-tree of logical eraseblocks of volume @vol_id to look
|
|
|
* if this is the first instance of this logical eraseblock or not.
|
|
@@ -492,11 +509,11 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
return err;
|
|
|
|
|
|
if (cmp_res & 4)
|
|
|
- err = ubi_scan_add_to_list(si, seb->pnum,
|
|
|
- seb->ec, &si->corr);
|
|
|
+ err = add_to_list(si, seb->pnum, seb->ec,
|
|
|
+ &si->corr);
|
|
|
else
|
|
|
- err = ubi_scan_add_to_list(si, seb->pnum,
|
|
|
- seb->ec, &si->erase);
|
|
|
+ err = add_to_list(si, seb->pnum, seb->ec,
|
|
|
+ &si->erase);
|
|
|
if (err)
|
|
|
return err;
|
|
|
|
|
@@ -508,7 +525,7 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
|
|
|
if (sv->highest_lnum == lnum)
|
|
|
sv->last_data_size =
|
|
|
- ubi32_to_cpu(vid_hdr->data_size);
|
|
|
+ be32_to_cpu(vid_hdr->data_size);
|
|
|
|
|
|
return 0;
|
|
|
} else {
|
|
@@ -517,11 +534,9 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
* previously.
|
|
|
*/
|
|
|
if (cmp_res & 4)
|
|
|
- return ubi_scan_add_to_list(si, pnum, ec,
|
|
|
- &si->corr);
|
|
|
+ return add_to_list(si, pnum, ec, &si->corr);
|
|
|
else
|
|
|
- return ubi_scan_add_to_list(si, pnum, ec,
|
|
|
- &si->erase);
|
|
|
+ return add_to_list(si, pnum, ec, &si->erase);
|
|
|
}
|
|
|
}
|
|
|
|
|
@@ -547,12 +562,9 @@ int ubi_scan_add_used(const struct ubi_device *ubi, struct ubi_scan_info *si,
|
|
|
|
|
|
if (sv->highest_lnum <= lnum) {
|
|
|
sv->highest_lnum = lnum;
|
|
|
- sv->last_data_size = ubi32_to_cpu(vid_hdr->data_size);
|
|
|
+ sv->last_data_size = be32_to_cpu(vid_hdr->data_size);
|
|
|
}
|
|
|
|
|
|
- if (si->max_sqnum < sqnum)
|
|
|
- si->max_sqnum = sqnum;
|
|
|
-
|
|
|
sv->leb_count += 1;
|
|
|
rb_link_node(&seb->u.rb, parent, p);
|
|
|
rb_insert_color(&seb->u.rb, &sv->root);
|
|
@@ -674,7 +686,7 @@ int ubi_scan_erase_peb(const struct ubi_device *ubi,
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
- ec_hdr->ec = cpu_to_ubi64(ec);
|
|
|
+ ec_hdr->ec = cpu_to_be64(ec);
|
|
|
|
|
|
err = ubi_io_sync_erase(ubi, pnum, 0);
|
|
|
if (err < 0)
|
|
@@ -754,7 +766,7 @@ struct ubi_scan_leb *ubi_scan_get_free_peb(const struct ubi_device *ubi,
|
|
|
* @si: scanning information
|
|
|
* @pnum: the physical eraseblock number
|
|
|
*
|
|
|
- * This function returns a zero if the physical eraseblock was succesfully
|
|
|
+ * This function returns a zero if the physical eraseblock was successfully
|
|
|
* handled and a negative error code in case of failure.
|
|
|
*/
|
|
|
static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, int pnum)
|
|
@@ -783,8 +795,7 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, int pnum
|
|
|
else if (err == UBI_IO_BITFLIPS)
|
|
|
bitflips = 1;
|
|
|
else if (err == UBI_IO_PEB_EMPTY)
|
|
|
- return ubi_scan_add_to_list(si, pnum, UBI_SCAN_UNKNOWN_EC,
|
|
|
- &si->erase);
|
|
|
+ return add_to_list(si, pnum, UBI_SCAN_UNKNOWN_EC, &si->erase);
|
|
|
else if (err == UBI_IO_BAD_EC_HDR) {
|
|
|
/*
|
|
|
* We have to also look at the VID header, possibly it is not
|
|
@@ -806,7 +817,7 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, int pnum
|
|
|
return -EINVAL;
|
|
|
}
|
|
|
|
|
|
- ec = ubi64_to_cpu(ech->ec);
|
|
|
+ ec = be64_to_cpu(ech->ec);
|
|
|
if (ec > UBI_MAX_ERASECOUNTER) {
|
|
|
/*
|
|
|
* Erase counter overflow. The EC headers have 64 bits
|
|
@@ -832,28 +843,28 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, int pnum
|
|
|
else if (err == UBI_IO_BAD_VID_HDR ||
|
|
|
(err == UBI_IO_PEB_FREE && ec_corr)) {
|
|
|
/* VID header is corrupted */
|
|
|
- err = ubi_scan_add_to_list(si, pnum, ec, &si->corr);
|
|
|
+ err = add_to_list(si, pnum, ec, &si->corr);
|
|
|
if (err)
|
|
|
return err;
|
|
|
goto adjust_mean_ec;
|
|
|
} else if (err == UBI_IO_PEB_FREE) {
|
|
|
/* No VID header - the physical eraseblock is free */
|
|
|
- err = ubi_scan_add_to_list(si, pnum, ec, &si->free);
|
|
|
+ err = add_to_list(si, pnum, ec, &si->free);
|
|
|
if (err)
|
|
|
return err;
|
|
|
goto adjust_mean_ec;
|
|
|
}
|
|
|
|
|
|
- vol_id = ubi32_to_cpu(vidh->vol_id);
|
|
|
+ vol_id = be32_to_cpu(vidh->vol_id);
|
|
|
if (vol_id > UBI_MAX_VOLUMES && vol_id != UBI_LAYOUT_VOL_ID) {
|
|
|
- int lnum = ubi32_to_cpu(vidh->lnum);
|
|
|
+ int lnum = be32_to_cpu(vidh->lnum);
|
|
|
|
|
|
/* Unsupported internal volume */
|
|
|
switch (vidh->compat) {
|
|
|
case UBI_COMPAT_DELETE:
|
|
|
ubi_msg("\"delete\" compatible internal volume %d:%d"
|
|
|
" found, remove it", vol_id, lnum);
|
|
|
- err = ubi_scan_add_to_list(si, pnum, ec, &si->corr);
|
|
|
+ err = add_to_list(si, pnum, ec, &si->corr);
|
|
|
if (err)
|
|
|
return err;
|
|
|
break;
|
|
@@ -868,7 +879,7 @@ static int process_eb(struct ubi_device *ubi, struct ubi_scan_info *si, int pnum
|
|
|
case UBI_COMPAT_PRESERVE:
|
|
|
ubi_msg("\"preserve\" compatible internal volume %d:%d"
|
|
|
" found", vol_id, lnum);
|
|
|
- err = ubi_scan_add_to_list(si, pnum, ec, &si->alien);
|
|
|
+ err = add_to_list(si, pnum, ec, &si->alien);
|
|
|
if (err)
|
|
|
return err;
|
|
|
si->alien_peb_count += 1;
|
|
@@ -1109,7 +1120,7 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
uint8_t *buf;
|
|
|
|
|
|
/*
|
|
|
- * At first, check that scanning information is ok.
|
|
|
+ * At first, check that scanning information is OK.
|
|
|
*/
|
|
|
ubi_rb_for_each_entry(rb1, sv, &si->volumes, rb) {
|
|
|
int leb_count = 0;
|
|
@@ -1249,12 +1260,12 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (seb->sqnum != ubi64_to_cpu(vidh->sqnum)) {
|
|
|
+ if (seb->sqnum != be64_to_cpu(vidh->sqnum)) {
|
|
|
ubi_err("bad sqnum %llu", seb->sqnum);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (sv->vol_id != ubi32_to_cpu(vidh->vol_id)) {
|
|
|
+ if (sv->vol_id != be32_to_cpu(vidh->vol_id)) {
|
|
|
ubi_err("bad vol_id %d", sv->vol_id);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
@@ -1264,22 +1275,22 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (seb->lnum != ubi32_to_cpu(vidh->lnum)) {
|
|
|
+ if (seb->lnum != be32_to_cpu(vidh->lnum)) {
|
|
|
ubi_err("bad lnum %d", seb->lnum);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (sv->used_ebs != ubi32_to_cpu(vidh->used_ebs)) {
|
|
|
+ if (sv->used_ebs != be32_to_cpu(vidh->used_ebs)) {
|
|
|
ubi_err("bad used_ebs %d", sv->used_ebs);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (sv->data_pad != ubi32_to_cpu(vidh->data_pad)) {
|
|
|
+ if (sv->data_pad != be32_to_cpu(vidh->data_pad)) {
|
|
|
ubi_err("bad data_pad %d", sv->data_pad);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (seb->leb_ver != ubi32_to_cpu(vidh->leb_ver)) {
|
|
|
+ if (seb->leb_ver != be32_to_cpu(vidh->leb_ver)) {
|
|
|
ubi_err("bad leb_ver %u", seb->leb_ver);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
@@ -1288,12 +1299,12 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
if (!last_seb)
|
|
|
continue;
|
|
|
|
|
|
- if (sv->highest_lnum != ubi32_to_cpu(vidh->lnum)) {
|
|
|
+ if (sv->highest_lnum != be32_to_cpu(vidh->lnum)) {
|
|
|
ubi_err("bad highest_lnum %d", sv->highest_lnum);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
|
|
|
|
- if (sv->last_data_size != ubi32_to_cpu(vidh->data_size)) {
|
|
|
+ if (sv->last_data_size != be32_to_cpu(vidh->data_size)) {
|
|
|
ubi_err("bad last_data_size %d", sv->last_data_size);
|
|
|
goto bad_vid_hdr;
|
|
|
}
|
|
@@ -1310,8 +1321,10 @@ static int paranoid_check_si(const struct ubi_device *ubi,
|
|
|
memset(buf, 1, ubi->peb_count);
|
|
|
for (pnum = 0; pnum < ubi->peb_count; pnum++) {
|
|
|
err = ubi_io_is_bad(ubi, pnum);
|
|
|
- if (err < 0)
|
|
|
+ if (err < 0) {
|
|
|
+ kfree(buf);
|
|
|
return err;
|
|
|
+ }
|
|
|
else if (err)
|
|
|
buf[pnum] = 0;
|
|
|
}
|