Add "ubi block" command to create a block device on top of a UBI volume. This block device can be used with some U-Boot commands, eg: fstype, ls and load.
Example use: => ubi part fs => ubi block root root dev=0 => fstype ubi 0 squashfs ubiblock_read() is based on drivers/mtd/ubi/block.c from Linux 4.19.75. Signed-off-by: Juha Sarlin <j...@sarlin.mobi> --- cmd/ubi.c | 36 ++++++++++ disk/part.c | 22 ------ drivers/block/blk-uclass.c | 4 +- drivers/mtd/ubi/Kconfig | 15 ++++ drivers/mtd/ubi/Makefile | 2 + drivers/mtd/ubi/block.c | 134 +++++++++++++++++++++++++++++++++++ drivers/mtd/ubi/build.c | 1 + drivers/mtd/ubi/ubi-uclass.c | 74 +++++++++++++++++++ drivers/mtd/ubi/ubi.h | 19 +++-- include/blk.h | 1 + include/dm/uclass-id.h | 1 + include/linux/mtd/ubi.h | 2 +- include/ubi_uboot.h | 2 - 13 files changed, 276 insertions(+), 37 deletions(-) create mode 100644 drivers/mtd/ubi/block.c create mode 100644 drivers/mtd/ubi/ubi-uclass.c diff --git a/cmd/ubi.c b/cmd/ubi.c index 22ba5b1a2c..ee5738a9f2 100644 --- a/cmd/ubi.c +++ b/cmd/ubi.c @@ -102,6 +102,34 @@ static int ubi_check(char *name) return 1; } +/** + * Create BLK device on UBI volume. + * + * @name: [[ubi]dev]:volume + * @return CMD_RET_SUCCESS or CMD_RET_FAILURE + */ +static int ubi_block(char *name) +{ + const char *device, *volume; + int ret, ubi_num = 0; + + volume = strchr(name, ':'); + if (volume) { + volume++; + device = strncmp(name, "ubi", 3) ? name : name + 3; + ubi_num = simple_strtoul(device, NULL, 0); + } else { + volume = name; + } + ret = ubiblock_create(ubi_num, volume); + if (ret < 0) { + pr_err("Can't create ubi block device: %s\n", errno_str(ret)); + return CMD_RET_FAILURE; + } + printf("%s dev=%d\n", name, ret); + return CMD_RET_SUCCESS; +} + static int verify_mkvol_req(const struct ubi_device *ubi, const struct ubi_mkvol_req *req) { @@ -541,6 +569,13 @@ static int do_ubi(cmd_tbl_t *cmdtp, int flag, int argc, char * const argv[]) return ubi_info(layout); } + if (strcmp(argv[1], "block") == 0) { + if (argc > 2) + return ubi_block(argv[2]); + printf("No volume name\n"); + return CMD_RET_FAILURE; + } + if (strcmp(argv[1], "check") == 0) { if (argc > 2) return ubi_check(argv[2]); @@ -688,6 +723,7 @@ U_BOOT_CMD( " - Write part of a volume from address\n" "ubi read[vol] address volume [size]" " - Read volume to address with size\n" + "ubi block volume - Create block device on UBI volume\n" "ubi remove[vol] volume" " - Remove volume\n" "ubi skipcheck volume on/off - Set or clear skip_check flag in volume header\n" diff --git a/disk/part.c b/disk/part.c index 8982ef3bae..f168311e9a 100644 --- a/disk/part.c +++ b/disk/part.c @@ -466,28 +466,6 @@ int blk_get_device_part_str(const char *ifname, const char *dev_part_str, } #endif -#ifdef CONFIG_CMD_UBIFS - /* - * Special-case ubi, ubi goes through a mtd, rather than through - * a regular block device. - */ - if (0 == strcmp(ifname, "ubi")) { - if (!ubifs_is_mounted()) { - printf("UBIFS not mounted, use ubifsmount to mount volume first!\n"); - return -1; - } - - *dev_desc = NULL; - memset(info, 0, sizeof(*info)); - strcpy((char *)info->type, BOOT_PART_TYPE); - strcpy((char *)info->name, "UBI"); -#if CONFIG_IS_ENABLED(PARTITION_UUIDS) - info->uuid[0] = 0; -#endif - return 0; - } -#endif - /* If no dev_part_str, use bootdevice environment variable */ if (!dev_part_str || !strlen(dev_part_str) || !strcmp(dev_part_str, "-")) diff --git a/drivers/block/blk-uclass.c b/drivers/block/blk-uclass.c index ca8978f0e1..433ea16db4 100644 --- a/drivers/block/blk-uclass.c +++ b/drivers/block/blk-uclass.c @@ -24,6 +24,7 @@ static const char *if_typename_str[IF_TYPE_COUNT] = { [IF_TYPE_NVME] = "nvme", [IF_TYPE_EFI] = "efi", [IF_TYPE_VIRTIO] = "virtio", + [IF_TYPE_UBI] = "ubi", }; static enum uclass_id if_type_uclass_id[IF_TYPE_COUNT] = { @@ -39,6 +40,7 @@ static enum uclass_id if_type_uclass_id[IF_TYPE_COUNT] = { [IF_TYPE_NVME] = UCLASS_NVME, [IF_TYPE_EFI] = UCLASS_EFI, [IF_TYPE_VIRTIO] = UCLASS_VIRTIO, + [IF_TYPE_UBI] = UCLASS_UBI, }; static enum if_type if_typename_to_iftype(const char *if_typename) @@ -112,7 +114,7 @@ struct blk_desc *blk_get_devnum_by_typename(const char *if_typename, int devnum) debug("%s: if_type=%d, devnum=%d: %s, %d, %d\n", __func__, if_type, devnum, dev->name, desc->if_type, desc->devnum); - if (desc->devnum != devnum) + if (desc->if_type != if_type || desc->devnum != devnum) continue; /* Find out the parent device uclass */ diff --git a/drivers/mtd/ubi/Kconfig b/drivers/mtd/ubi/Kconfig index a78fd51ba7..008494db1a 100644 --- a/drivers/mtd/ubi/Kconfig +++ b/drivers/mtd/ubi/Kconfig @@ -95,6 +95,21 @@ config MTD_UBI_FASTMAP_AUTOCONVERT Set this parameter to enable fastmap automatically on images without a fastmap. +config MTD_UBI_BLOCK + bool "Read-only block devices on top of UBI volumes" + default n + help + This option enables read-only UBI block devices support. UBI block + devices will be layered on top of UBI volumes, which means that the + UBI driver will transparently handle things like bad eraseblocks and + bit-flips. You can put any block-oriented file system on top of UBI + volumes in read-only mode (e.g., ext4), but it is probably most + practical for read-only file systems, like squashfs. + + When selected, this feature will be built in the UBI driver. + + If in doubt, say "N". + config MTD_UBI_FM_DEBUG int "Enable UBI fastmap debug" depends on MTD_UBI_FASTMAP diff --git a/drivers/mtd/ubi/Makefile b/drivers/mtd/ubi/Makefile index 30d00fbdfe..92ea1fa0d1 100644 --- a/drivers/mtd/ubi/Makefile +++ b/drivers/mtd/ubi/Makefile @@ -4,6 +4,8 @@ # Wolfgang Denk, DENX Software Engineering, w...@denx.de. obj-y += attach.o build.o vtbl.o vmt.o upd.o kapi.o eba.o io.o wl.o crc32.o +obj-$(CONFIG_MTD_UBI) += ubi-uclass.o +obj-$(CONFIG_MTD_UBI_BLOCK) += block.o obj-$(CONFIG_MTD_UBI_FASTMAP) += fastmap.o obj-y += misc.o obj-y += debug.o diff --git a/drivers/mtd/ubi/block.c b/drivers/mtd/ubi/block.c new file mode 100644 index 0000000000..b46cd32404 --- /dev/null +++ b/drivers/mtd/ubi/block.c @@ -0,0 +1,134 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (c) 2019 Juha Sarlin + * Copyright (c) 2014 Ezequiel Garcia + * Copyright (c) 2011 Free Electrons + */ + +/* + * Read-only block devices on top of UBI volumes + * + * A simple implementation to allow a block device to be layered on top of a + * UBI volume. The implementation is provided by creating a static 1-to-1 + * mapping between the block device and the UBI volume. + * + * The addressed byte is obtained from the addressed block sector, which is + * mapped linearly into the corresponding LEB: + * + * LEB number = addressed byte / LEB size + */ + +#include <div64.h> +#include <dm.h> +#include <dm/device-internal.h> +#include <ubi_uboot.h> +#include "ubi.h" + +static struct udevice *ubiblock_find(int ubi_num, const char *volume) +{ + struct udevice *dev; + char name[UBI_VOL_NAME_MAX + 8]; + + snprintf(name, sizeof(name), "ubi%d.%s", ubi_num, volume); + for (blk_first_device(IF_TYPE_UBI, &dev); dev; blk_next_device(&dev)) { + if (!strcmp(dev->name, name)) + return dev; + } + return NULL; +} + +static unsigned long ubiblock_read(struct udevice *dev, lbaint_t start, + lbaint_t blkcnt, void *buffer) +{ + struct ubi_volume_desc *desc = dev->priv; + struct ubi_volume *vol = desc->vol; + struct blk_desc *blk = dev_get_uclass_platdata(dev); + int ret, leb, offset, bytes_left, to_read; + int leb_size = vol->ubi->leb_size; + int blksz = blk->blksz; + u64 pos; + + to_read = blkcnt * blksz; + pos = start * blksz; + + /* Get LEB:offset address to read from */ + offset = do_div(pos, leb_size); + leb = pos; + bytes_left = to_read; + + while (bytes_left) { + /* + * We can only read one LEB at a time. Therefore if the read + * length is larger than one LEB size, we split the operation. + */ + if (offset + to_read > leb_size) + to_read = leb_size - offset; + + ret = ubi_read(desc, leb, buffer, offset, to_read); + if (ret < 0) { + pr_err("%s: ubi_read failed (%d)\n", __func__, ret); + return ret; + } + bytes_left -= to_read; + to_read = bytes_left; + leb += 1; + offset = 0; + } + return blkcnt; +} + +/** + * Create BLK device on UBI volume. + * + * @return devnum or -errno + */ +int ubiblock_create(int ubi_num, const char *volume) +{ + struct udevice *dev, *parent; + struct ubi_volume_desc *desc; + struct blk_desc *blk; + struct ubi_volume *vol; + int ret, leb_size, blksz, size; + + ret = ubi_get(ubi_num, &parent); + if (ret) + return ret; + desc = ubi_open_volume_nm(ubi_num, volume, UBI_READONLY); + if (IS_ERR(desc)) + return PTR_ERR(desc); + dev = ubiblock_find(ubi_num, volume); + if (!dev) { + vol = desc->vol; + leb_size = vol->ubi->leb_size; + /* Block size must be power of 2 */ + blksz = leb_size & ~(leb_size - 1); + size = vol->reserved_pebs * (leb_size / blksz); + ret = blk_create_devicef(parent, "ubiblock", volume, + IF_TYPE_UBI, -1, blksz, size, &dev); + if (ret) + return ret; + dev->priv = desc; + } + blk = dev_get_uclass_platdata(dev); + return blk->devnum; +} + +static int ubiblock_unbind(struct udevice *dev) +{ + if (dev->priv) { + ubi_close_volume(dev->priv); + dev->priv = NULL; + } + return 0; +} + +static const struct blk_ops ubiblock_ops = { + .read = ubiblock_read, +}; + +U_BOOT_DRIVER(ubiblock) = { + .name = "ubiblock", + .id = UCLASS_BLK, + .unbind = ubiblock_unbind, + .ops = &ubiblock_ops, +}; diff --git a/drivers/mtd/ubi/build.c b/drivers/mtd/ubi/build.c index 42c5270c7f..0c15057a8d 100644 --- a/drivers/mtd/ubi/build.c +++ b/drivers/mtd/ubi/build.c @@ -1112,6 +1112,7 @@ int ubi_detach_mtd_dev(int ubi_num, int anyway) if (!ubi) return -EINVAL; + ubi_put(ubi_num); spin_lock(&ubi_devices_lock); put_device(&ubi->dev); ubi->ref_count -= 1; diff --git a/drivers/mtd/ubi/ubi-uclass.c b/drivers/mtd/ubi/ubi-uclass.c new file mode 100644 index 0000000000..a13aa7512f --- /dev/null +++ b/drivers/mtd/ubi/ubi-uclass.c @@ -0,0 +1,74 @@ +// SPDX-License-Identifier: GPL-2.0+ +/* + * Copyright (c) 2019 Juha Sarlin + */ +#include <common.h> +#include <dm.h> +#include <dm/lists.h> +#include <dm/uclass-internal.h> +#include <dm/device-internal.h> +#include <dm/root.h> +#include "ubi.h" + +/** Get udevice for a UBI device. */ +int ubi_get(int ubi_num, struct udevice **devp) +{ + struct ubi_device *ubi; + struct udevice *dev; + int ret; + + ubi = ubi_get_device(ubi_num); + if (!ubi) + return -ENODEV; + ret = uclass_find_device_by_name(UCLASS_UBI, ubi->ubi_name, &dev); + if (ret) { + ret = device_bind_driver(dm_root(), "ubi", + ubi->ubi_name, &dev); + if (ret) { + ubi_put_device(ubi); + return ret; + } + dev->priv = ubi; + } else { + ubi_put_device(ubi); + } + *devp = dev; + return 0; +} + +/** Remove udevice for a UBI device. */ +int ubi_put(int ubi_num) +{ + struct ubi_device *ubi; + struct udevice *dev; + int ret; + + ubi = ubi_get_device(ubi_num); + if (!ubi) + return 0; + ubi_put_device(ubi); + ret = uclass_find_device_by_name(UCLASS_UBI, ubi->ubi_name, &dev); + if (!ret) + device_unbind(dev); + return 0; +} + +static int ubi_unbind(struct udevice *dev) +{ + if (dev->priv) { + ubi_put_device(dev->priv); + dev->priv = NULL; + } + return 0; +} + +U_BOOT_DRIVER(ubi) = { + .id = UCLASS_UBI, + .name = "ubi", + .unbind = ubi_unbind, +}; + +UCLASS_DRIVER(ubi) = { + .id = UCLASS_UBI, + .name = "ubi", +}; diff --git a/drivers/mtd/ubi/ubi.h b/drivers/mtd/ubi/ubi.h index f44960186b..c0acb67c23 100644 --- a/drivers/mtd/ubi/ubi.h +++ b/drivers/mtd/ubi/ubi.h @@ -922,6 +922,10 @@ void ubi_do_get_volume_info(struct ubi_device *ubi, struct ubi_volume *vol, int ubi_compare_lebs(struct ubi_device *ubi, const struct ubi_ainf_peb *aeb, int pnum, const struct ubi_vid_hdr *vid_hdr); +/* ubi-uclass.c */ +int ubi_get(int ubi_num, struct udevice **devp); +int ubi_put(int ubi_num); + /* fastmap.c */ #ifdef CONFIG_MTD_UBI_FASTMAP size_t ubi_calc_fm_size(struct ubi_device *ubi); @@ -933,19 +937,12 @@ static inline int ubi_update_fastmap(struct ubi_device *ubi) { return 0; } #endif /* block.c */ -#ifdef CONFIG_MTD_UBI_BLOCK -int ubiblock_init(void); -void ubiblock_exit(void); -int ubiblock_create(struct ubi_volume_info *vi); -int ubiblock_remove(struct ubi_volume_info *vi); -#else static inline int ubiblock_init(void) { return 0; } static inline void ubiblock_exit(void) {} -static inline int ubiblock_create(struct ubi_volume_info *vi) -{ - return -ENOSYS; -} -static inline int ubiblock_remove(struct ubi_volume_info *vi) +#ifdef CONFIG_MTD_UBI_BLOCK +int ubiblock_create(int ubi_num, const char *volume); +#else +static inline int ubiblock_create(int ubi_num, const char *volume) { return -ENOSYS; } diff --git a/include/blk.h b/include/blk.h index d0c033aece..b4c5c35248 100644 --- a/include/blk.h +++ b/include/blk.h @@ -34,6 +34,7 @@ enum if_type { IF_TYPE_NVME, IF_TYPE_EFI, IF_TYPE_VIRTIO, + IF_TYPE_UBI, IF_TYPE_COUNT, /* Number of interface types */ }; diff --git a/include/dm/uclass-id.h b/include/dm/uclass-id.h index 0c563d898b..b8bc400661 100644 --- a/include/dm/uclass-id.h +++ b/include/dm/uclass-id.h @@ -102,6 +102,7 @@ enum uclass_id { UCLASS_THERMAL, /* Thermal sensor */ UCLASS_TIMER, /* Timer device */ UCLASS_TPM, /* Trusted Platform Module TIS interface */ + UCLASS_UBI, /* Unsorted Block Images */ UCLASS_UFS, /* Universal Flash Storage */ UCLASS_USB, /* USB bus */ UCLASS_USB_DEV_GENERIC, /* USB generic device */ diff --git a/include/linux/mtd/ubi.h b/include/linux/mtd/ubi.h index badf6a0c6c..6c7f22a42b 100644 --- a/include/linux/mtd/ubi.h +++ b/include/linux/mtd/ubi.h @@ -271,7 +271,7 @@ int ubi_flush(int ubi_num, int vol_id, int lnum); * This function is the same as the 'ubi_leb_read()' function, but it does not * provide the checking capability. */ -static inline int ubi_read(struct ubi_volume_desc *desc, int lnum, char *buf, +static inline int ubi_read(struct ubi_volume_desc *desc, int lnum, void *buf, int offset, int len) { return ubi_leb_read(desc, lnum, buf, offset, len, 0); diff --git a/include/ubi_uboot.h b/include/ubi_uboot.h index 0770228cd8..7a602c766c 100644 --- a/include/ubi_uboot.h +++ b/include/ubi_uboot.h @@ -46,8 +46,6 @@ #undef CONFIG_MTD_UBI_DEBUG_MSG_IO #undef CONFIG_MTD_UBI_DEBUG_MSG_BLD -#undef CONFIG_MTD_UBI_BLOCK - /* ubi_init() disables returning error codes when built into the Linux * kernel so that it doesn't hang the Linux kernel boot process. Since * the U-Boot driver code depends on getting valid error codes from this -- 2.20.1.98.gecbdaf0899 _______________________________________________ U-Boot mailing list U-Boot@lists.denx.de https://lists.denx.de/listinfo/u-boot