aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorArtem Bityutskiy <Artem.Bityutskiy@nokia.com>2007-12-06 11:47:30 -0500
committerArtem Bityutskiy <Artem.Bityutskiy@nokia.com>2007-12-26 12:15:14 -0500
commit393852ecfeec575ac78216b0eb58e4fd92f0816c (patch)
tree99edbcd7fee9cdaa69f51187834f93ada73feada
parent94780d4de2e9339ab93df63420db70f11882634d (diff)
UBI: add ubi_leb_map interface
The idea of this interface belongs to Adrian Hunter. The interface is extremely useful when one has to have a guarantee that an LEB will contain all 0xFFs even in case of an unclean reboot. UBI does have an 'ubi_leb_erase()' call which may do this, but it is stupid and ineffecient, because it flushes whole queue. I should be re-worked to just be a pair of unmap, map calls. The user of the interfaci is UBIFS at the moment. Signed-off-by: Artem Bityutskiy <Artem.Bityutskiy@nokia.com>
-rw-r--r--drivers/mtd/ubi/eba.c13
-rw-r--r--drivers/mtd/ubi/kapi.c45
-rw-r--r--include/linux/mtd/ubi.h1
3 files changed, 54 insertions, 5 deletions
diff --git a/drivers/mtd/ubi/eba.c b/drivers/mtd/ubi/eba.c
index 85f50c83cf42..c87db07bcd0c 100644
--- a/drivers/mtd/ubi/eba.c
+++ b/drivers/mtd/ubi/eba.c
@@ -656,11 +656,14 @@ retry:
656 goto write_error; 656 goto write_error;
657 } 657 }
658 658
659 err = ubi_io_write_data(ubi, buf, pnum, offset, len); 659 if (len) {
660 if (err) { 660 err = ubi_io_write_data(ubi, buf, pnum, offset, len);
661 ubi_warn("failed to write %d bytes at offset %d of LEB %d:%d, " 661 if (err) {
662 "PEB %d", len, offset, vol_id, lnum, pnum); 662 ubi_warn("failed to write %d bytes at offset %d of "
663 goto write_error; 663 "LEB %d:%d, PEB %d", len, offset, vol_id,
664 lnum, pnum);
665 goto write_error;
666 }
664 } 667 }
665 668
666 vol->eba_tbl[lnum] = pnum; 669 vol->eba_tbl[lnum] = pnum;
diff --git a/drivers/mtd/ubi/kapi.c b/drivers/mtd/ubi/kapi.c
index 03c774f41549..e1ef802a03a7 100644
--- a/drivers/mtd/ubi/kapi.c
+++ b/drivers/mtd/ubi/kapi.c
@@ -547,6 +547,51 @@ int ubi_leb_unmap(struct ubi_volume_desc *desc, int lnum)
547EXPORT_SYMBOL_GPL(ubi_leb_unmap); 547EXPORT_SYMBOL_GPL(ubi_leb_unmap);
548 548
549/** 549/**
550 * ubi_leb_map - map logical erasblock to a physical eraseblock.
551 * @desc: volume descriptor
552 * @lnum: logical eraseblock number
553 * @dtype: expected data type
554 *
555 * This function maps an un-mapped logical eraseblock @lnum to a physical
556 * eraseblock. This means, that after a successfull invocation of this
557 * function the logical eraseblock @lnum will be empty (contain only %0xFF
558 * bytes) and be mapped to a physical eraseblock, even if an unclean reboot
559 * happens.
560 *
561 * This function returns zero in case of success, %-EBADF if the volume is
562 * damaged because of an interrupted update, %-EBADMSG if the logical
563 * eraseblock is already mapped, and other negative error codes in case of
564 * other failures.
565 */
566int ubi_leb_map(struct ubi_volume_desc *desc, int lnum, int dtype)
567{
568 struct ubi_volume *vol = desc->vol;
569 struct ubi_device *ubi = vol->ubi;
570 int vol_id = vol->vol_id;
571
572 dbg_msg("unmap LEB %d:%d", vol_id, lnum);
573
574 if (desc->mode == UBI_READONLY || vol->vol_type == UBI_STATIC_VOLUME)
575 return -EROFS;
576
577 if (lnum < 0 || lnum >= vol->reserved_pebs)
578 return -EINVAL;
579
580 if (dtype != UBI_LONGTERM && dtype != UBI_SHORTTERM &&
581 dtype != UBI_UNKNOWN)
582 return -EINVAL;
583
584 if (vol->upd_marker)
585 return -EBADF;
586
587 if (vol->eba_tbl[lnum] >= 0)
588 return -EBADMSG;
589
590 return ubi_eba_write_leb(ubi, vol_id, lnum, NULL, 0, 0, dtype);
591}
592EXPORT_SYMBOL_GPL(ubi_leb_map);
593
594/**
550 * ubi_is_mapped - check if logical eraseblock is mapped. 595 * ubi_is_mapped - check if logical eraseblock is mapped.
551 * @desc: volume descriptor 596 * @desc: volume descriptor
552 * @lnum: logical eraseblock number 597 * @lnum: logical eraseblock number
diff --git a/include/linux/mtd/ubi.h b/include/linux/mtd/ubi.h
index 3d967b6b120a..c4abe0351225 100644
--- a/include/linux/mtd/ubi.h
+++ b/include/linux/mtd/ubi.h
@@ -167,6 +167,7 @@ int ubi_leb_change(struct ubi_volume_desc *desc, int lnum, const void *buf,
167 int len, int dtype); 167 int len, int dtype);
168int ubi_leb_erase(struct ubi_volume_desc *desc, int lnum); 168int ubi_leb_erase(struct ubi_volume_desc *desc, int lnum);
169int ubi_leb_unmap(struct ubi_volume_desc *desc, int lnum); 169int ubi_leb_unmap(struct ubi_volume_desc *desc, int lnum);
170int ubi_leb_map(struct ubi_volume_desc *desc, int lnum, int dtype);
170int ubi_is_mapped(struct ubi_volume_desc *desc, int lnum); 171int ubi_is_mapped(struct ubi_volume_desc *desc, int lnum);
171 172
172/* 173/*