mirror of
https://github.com/Ysurac/openmptcprouter.git
synced 2025-02-12 11:21:55 +00:00
Remove patch for FIT support
This commit is contained in:
parent
bf9718efe5
commit
e595565c9c
2 changed files with 0 additions and 1015 deletions
|
@ -1,260 +0,0 @@
|
|||
From 9703951cdfe868b130e64d6122420396c2807be8 Mon Sep 17 00:00:00 2001
|
||||
From: Daniel Golle <daniel@makrotopia.org>
|
||||
Date: Thu, 30 May 2024 03:15:02 +0100
|
||||
Subject: [PATCH 5/9] nvmem: implement block NVMEM provider
|
||||
|
||||
On embedded devices using an eMMC it is common that one or more partitions
|
||||
on the eMMC are used to store MAC addresses and Wi-Fi calibration EEPROM
|
||||
data. Allow referencing any block device or partition in Device Tree to
|
||||
allow e.g. Ethernet and Wi-Fi drivers accessing them via the NVMEM layer.
|
||||
|
||||
Signed-off-by: Daniel Golle <daniel@makrotopia.org>
|
||||
---
|
||||
drivers/nvmem/Kconfig | 11 +++
|
||||
drivers/nvmem/Makefile | 2 +
|
||||
drivers/nvmem/block.c | 197 +++++++++++++++++++++++++++++++++++++++++
|
||||
3 files changed, 210 insertions(+)
|
||||
create mode 100644 drivers/nvmem/block.c
|
||||
|
||||
--- a/drivers/nvmem/Kconfig
|
||||
+++ b/drivers/nvmem/Kconfig
|
||||
@@ -40,6 +40,17 @@ config NVMEM_APPLE_EFUSES
|
||||
This driver can also be built as a module. If so, the module will
|
||||
be called nvmem-apple-efuses.
|
||||
|
||||
+config NVMEM_BLOCK
|
||||
+ tristate "Block device NVMEM provider"
|
||||
+ depends on BLOCK
|
||||
+ depends on OF
|
||||
+ depends on NVMEM
|
||||
+ select BLOCK_NOTIFIERS
|
||||
+ help
|
||||
+ Allow block devices (or partitions) to act as NVMEM prodivers,
|
||||
+ typically used with eMMC to store MAC addresses or Wi-Fi
|
||||
+ calibration data on embedded devices.
|
||||
+
|
||||
config NVMEM_BCM_OCOTP
|
||||
tristate "Broadcom On-Chip OTP Controller support"
|
||||
depends on ARCH_BCM_IPROC || COMPILE_TEST
|
||||
--- a/drivers/nvmem/Makefile
|
||||
+++ b/drivers/nvmem/Makefile
|
||||
@@ -14,6 +14,8 @@ obj-$(CONFIG_NVMEM_APPLE_EFUSES) += nvme
|
||||
nvmem-apple-efuses-y := apple-efuses.o
|
||||
obj-$(CONFIG_NVMEM_BCM_OCOTP) += nvmem-bcm-ocotp.o
|
||||
nvmem-bcm-ocotp-y := bcm-ocotp.o
|
||||
+obj-$(CONFIG_NVMEM_BLOCK) += nvmem-block.o
|
||||
+nvmem-block-y := block.o
|
||||
obj-$(CONFIG_NVMEM_BRCM_NVRAM) += nvmem_brcm_nvram.o
|
||||
nvmem_brcm_nvram-y := brcm_nvram.o
|
||||
obj-$(CONFIG_NVMEM_IMX_IIM) += nvmem-imx-iim.o
|
||||
--- /dev/null
|
||||
+++ b/drivers/nvmem/block.c
|
||||
@@ -0,0 +1,208 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0-or-later
|
||||
+/*
|
||||
+ * block device NVMEM provider
|
||||
+ *
|
||||
+ * Copyright (c) 2024 Daniel Golle <daniel@makrotopia.org>
|
||||
+ *
|
||||
+ * Useful on devices using a partition on an eMMC for MAC addresses or
|
||||
+ * Wi-Fi calibration EEPROM data.
|
||||
+ */
|
||||
+
|
||||
+#include <linux/blkdev.h>
|
||||
+#include <linux/nvmem-provider.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/pagemap.h>
|
||||
+#include <linux/property.h>
|
||||
+
|
||||
+/* List of all NVMEM devices */
|
||||
+static LIST_HEAD(nvmem_devices);
|
||||
+static DEFINE_MUTEX(devices_mutex);
|
||||
+
|
||||
+struct blk_nvmem {
|
||||
+ struct nvmem_device *nvmem;
|
||||
+ struct block_device *bdev;
|
||||
+ struct list_head list;
|
||||
+};
|
||||
+
|
||||
+static int blk_nvmem_reg_read(void *priv, unsigned int from,
|
||||
+ void *val, size_t bytes)
|
||||
+{
|
||||
+ unsigned long offs = from & ~PAGE_MASK, to_read;
|
||||
+ pgoff_t f_index = from >> PAGE_SHIFT;
|
||||
+ struct address_space *mapping;
|
||||
+ struct blk_nvmem *bnv = priv;
|
||||
+ size_t bytes_left = bytes;
|
||||
+ struct folio *folio;
|
||||
+ void *p;
|
||||
+ int ret;
|
||||
+
|
||||
+ if (!bnv->bdev)
|
||||
+ return -ENODEV;
|
||||
+
|
||||
+ if (!bnv->bdev->bd_disk)
|
||||
+ return -EINVAL;
|
||||
+
|
||||
+ if (!bnv->bdev->bd_disk->fops)
|
||||
+ return -EIO;
|
||||
+
|
||||
+ if (!bnv->bdev->bd_disk->fops->open)
|
||||
+ return -EIO;
|
||||
+
|
||||
+ ret = bnv->bdev->bd_disk->fops->open(bnv->bdev->bd_disk, BLK_OPEN_READ);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ mapping = bnv->bdev->bd_inode->i_mapping;
|
||||
+
|
||||
+ while (bytes_left) {
|
||||
+ folio = read_mapping_folio(mapping, f_index++, NULL);
|
||||
+ if (IS_ERR(folio)) {
|
||||
+ ret = PTR_ERR(folio);
|
||||
+ goto err_release_bdev;
|
||||
+ }
|
||||
+ to_read = min_t(unsigned long, bytes_left, PAGE_SIZE - offs);
|
||||
+ p = folio_address(folio) + offset_in_folio(folio, offs);
|
||||
+ memcpy(val, p, to_read);
|
||||
+ offs = 0;
|
||||
+ bytes_left -= to_read;
|
||||
+ val += to_read;
|
||||
+ folio_put(folio);
|
||||
+ }
|
||||
+
|
||||
+err_release_bdev:
|
||||
+ bnv->bdev->bd_disk->fops->release(bnv->bdev->bd_disk);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static int blk_nvmem_register(struct device *dev)
|
||||
+{
|
||||
+ struct device_node *np = dev_of_node(dev);
|
||||
+ struct block_device *bdev = dev_to_bdev(dev);
|
||||
+ struct nvmem_config config = {};
|
||||
+ struct blk_nvmem *bnv;
|
||||
+
|
||||
+ /* skip devices which do not have a device tree node */
|
||||
+ if (!np)
|
||||
+ return 0;
|
||||
+
|
||||
+ /* skip devices without an nvmem layout defined */
|
||||
+ if (!of_get_child_by_name(np, "nvmem-layout"))
|
||||
+ return 0;
|
||||
+
|
||||
+ /*
|
||||
+ * skip devices which don't have GENHD_FL_NVMEM set
|
||||
+ *
|
||||
+ * This flag is used for mtdblock and ubiblock devices because
|
||||
+ * both, MTD and UBI already implement their own NVMEM provider.
|
||||
+ * To avoid registering multiple NVMEM providers for the same
|
||||
+ * device node, don't register the block NVMEM provider for them.
|
||||
+ */
|
||||
+ if (!(bdev->bd_disk->flags & GENHD_FL_NVMEM))
|
||||
+ return 0;
|
||||
+
|
||||
+ /*
|
||||
+ * skip block device too large to be represented as NVMEM devices
|
||||
+ * which are using an 'int' as address
|
||||
+ */
|
||||
+ if (bdev_nr_bytes(bdev) > INT_MAX)
|
||||
+ return -EFBIG;
|
||||
+
|
||||
+ bnv = kzalloc(sizeof(struct blk_nvmem), GFP_KERNEL);
|
||||
+ if (!bnv)
|
||||
+ return -ENOMEM;
|
||||
+
|
||||
+ config.id = NVMEM_DEVID_NONE;
|
||||
+ config.dev = &bdev->bd_device;
|
||||
+ config.name = dev_name(&bdev->bd_device);
|
||||
+ config.owner = THIS_MODULE;
|
||||
+ config.priv = bnv;
|
||||
+ config.reg_read = blk_nvmem_reg_read;
|
||||
+ config.size = bdev_nr_bytes(bdev);
|
||||
+ config.word_size = 1;
|
||||
+ config.stride = 1;
|
||||
+ config.read_only = true;
|
||||
+ config.root_only = true;
|
||||
+ config.ignore_wp = true;
|
||||
+ config.of_node = to_of_node(dev->fwnode);
|
||||
+
|
||||
+ bnv->bdev = bdev;
|
||||
+ bnv->nvmem = nvmem_register(&config);
|
||||
+ if (IS_ERR(bnv->nvmem)) {
|
||||
+ dev_err_probe(&bdev->bd_device, PTR_ERR(bnv->nvmem),
|
||||
+ "Failed to register NVMEM device\n");
|
||||
+
|
||||
+ kfree(bnv);
|
||||
+ return PTR_ERR(bnv->nvmem);
|
||||
+ }
|
||||
+
|
||||
+ mutex_lock(&devices_mutex);
|
||||
+ list_add_tail(&bnv->list, &nvmem_devices);
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static void blk_nvmem_unregister(struct device *dev)
|
||||
+{
|
||||
+ struct block_device *bdev = dev_to_bdev(dev);
|
||||
+ struct blk_nvmem *bnv_c, *bnv = NULL;
|
||||
+
|
||||
+ mutex_lock(&devices_mutex);
|
||||
+ list_for_each_entry(bnv_c, &nvmem_devices, list) {
|
||||
+ if (bnv_c->bdev == bdev) {
|
||||
+ bnv = bnv_c;
|
||||
+ break;
|
||||
+ }
|
||||
+ }
|
||||
+
|
||||
+ if (!bnv) {
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+ return;
|
||||
+ }
|
||||
+
|
||||
+ list_del(&bnv->list);
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+ nvmem_unregister(bnv->nvmem);
|
||||
+ kfree(bnv);
|
||||
+}
|
||||
+
|
||||
+static int blk_nvmem_handler(struct notifier_block *this, unsigned long code, void *obj)
|
||||
+{
|
||||
+ struct device *dev = (struct device *)obj;
|
||||
+
|
||||
+ switch (code) {
|
||||
+ case BLK_DEVICE_ADD:
|
||||
+ return blk_nvmem_register(dev);
|
||||
+ case BLK_DEVICE_REMOVE:
|
||||
+ blk_nvmem_unregister(dev);
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static struct notifier_block blk_nvmem_notifier = {
|
||||
+ .notifier_call = blk_nvmem_handler,
|
||||
+};
|
||||
+
|
||||
+static int __init blk_nvmem_init(void)
|
||||
+{
|
||||
+ blk_register_notify(&blk_nvmem_notifier);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static void __exit blk_nvmem_exit(void)
|
||||
+{
|
||||
+ blk_unregister_notify(&blk_nvmem_notifier);
|
||||
+}
|
||||
+
|
||||
+module_init(blk_nvmem_init);
|
||||
+module_exit(blk_nvmem_exit);
|
||||
+
|
||||
+MODULE_LICENSE("GPL");
|
||||
+MODULE_AUTHOR("Daniel Golle <daniel@makrotopia.org>");
|
||||
+MODULE_DESCRIPTION("block device NVMEM provider");
|
|
@ -1,755 +0,0 @@
|
|||
From 6173a065cb395d4a9528c4e49810af127db68141 Mon Sep 17 00:00:00 2001
|
||||
From: Daniel Golle <daniel@makrotopia.org>
|
||||
Date: Wed, 16 Nov 2022 12:49:52 +0000
|
||||
Subject: [PATCH 1/2] block: add uImage.FIT subimage block driver
|
||||
|
||||
Add a small block driver which exposes filesystem sub-images contained
|
||||
in U-Boot uImage.FIT images as block devices.
|
||||
|
||||
The uImage.FIT image has to be stored directly on a block device or
|
||||
partition, MTD device or partition, or UBI volume.
|
||||
|
||||
The driver is intended for systems using the U-Boot bootloader and
|
||||
uses the root device hint left by the bootloader (or the user) in
|
||||
the 'chosen' section of the device-tree.
|
||||
|
||||
Example:
|
||||
/dts-v1/;
|
||||
/ {
|
||||
chosen {
|
||||
rootdisk = <&mmc0_part3>;
|
||||
};
|
||||
};
|
||||
|
||||
Signed-off-by: Daniel Golle <daniel@makrotopia.org>
|
||||
---
|
||||
MAINTAINERS | 6 +
|
||||
drivers/block/Kconfig | 12 +
|
||||
drivers/block/Makefile | 2 +
|
||||
drivers/block/fitblk.c | 658 ++++++++++++++++++++++++++++++++++++
|
||||
drivers/block/open | 4 +
|
||||
include/uapi/linux/fitblk.h | 10 +
|
||||
6 files changed, 692 insertions(+)
|
||||
create mode 100644 drivers/block/fitblk.c
|
||||
create mode 100644 drivers/block/open
|
||||
create mode 100644 include/uapi/linux/fitblk.h
|
||||
|
||||
--- a/MAINTAINERS
|
||||
+++ b/MAINTAINERS
|
||||
@@ -23658,6 +23658,12 @@ F: Documentation/filesystems/ubifs-authe
|
||||
F: Documentation/filesystems/ubifs.rst
|
||||
F: fs/ubifs/
|
||||
|
||||
+U-BOOT UIMAGE.FIT PARSER
|
||||
+M: Daniel Golle <daniel@makrotopia.org>
|
||||
+L: linux-block@vger.kernel.org
|
||||
+S: Maintained
|
||||
+F: drivers/block/fitblk.c
|
||||
+
|
||||
UBLK USERSPACE BLOCK DRIVER
|
||||
M: Ming Lei <ming.lei@redhat.com>
|
||||
L: linux-block@vger.kernel.org
|
||||
--- a/drivers/block/Kconfig
|
||||
+++ b/drivers/block/Kconfig
|
||||
@@ -363,6 +363,18 @@ config BLK_DEV_RUST_NULL
|
||||
|
||||
If unsure, say N.
|
||||
|
||||
+config UIMAGE_FIT_BLK
|
||||
+ bool "uImage.FIT block driver"
|
||||
+ help
|
||||
+ This driver allows using filesystems contained in uImage.FIT images
|
||||
+ by mapping them as block devices.
|
||||
+
|
||||
+ It can currently not be built as a module due to libfdt symbols not
|
||||
+ being exported.
|
||||
+
|
||||
+ Say Y if you want to mount filesystems sub-images of a uImage.FIT
|
||||
+ stored in a block device partition, mtdblock or ubiblock device.
|
||||
+
|
||||
config BLK_DEV_RBD
|
||||
tristate "Rados block device (RBD)"
|
||||
depends on INET && BLOCK
|
||||
--- a/drivers/block/Makefile
|
||||
+++ b/drivers/block/Makefile
|
||||
@@ -42,4 +42,6 @@ obj-$(CONFIG_BLK_DEV_NULL_BLK) += null_b
|
||||
|
||||
obj-$(CONFIG_BLK_DEV_UBLK) += ublk_drv.o
|
||||
|
||||
+obj-$(CONFIG_UIMAGE_FIT_BLK) += fitblk.o
|
||||
+
|
||||
swim_mod-y := swim.o swim_asm.o
|
||||
--- /dev/null
|
||||
+++ b/drivers/block/fitblk.c
|
||||
@@ -0,0 +1,658 @@
|
||||
+// SPDX-License-Identifier: GPL-2.0-only
|
||||
+/*
|
||||
+ * uImage.FIT virtual block device driver.
|
||||
+ *
|
||||
+ * Copyright (C) 2023 Daniel Golle
|
||||
+ * Copyright (C) 2007 Nick Piggin
|
||||
+ * Copyright (C) 2007 Novell Inc.
|
||||
+ *
|
||||
+ * Initially derived from drivers/block/brd.c which is in parts derived from
|
||||
+ * drivers/block/rd.c, and drivers/block/loop.c, copyright of their respective
|
||||
+ * owners.
|
||||
+ *
|
||||
+ * uImage.FIT headers extracted from Das U-Boot
|
||||
+ * (C) Copyright 2008 Semihalf
|
||||
+ * (C) Copyright 2000-2005
|
||||
+ * Wolfgang Denk, DENX Software Engineering, wd@denx.de.
|
||||
+ */
|
||||
+
|
||||
+#include <linux/init.h>
|
||||
+#include <linux/initrd.h>
|
||||
+#include <linux/module.h>
|
||||
+#include <linux/moduleparam.h>
|
||||
+#include <linux/major.h>
|
||||
+#include <linux/blkdev.h>
|
||||
+#include <linux/blkpg.h>
|
||||
+#include <linux/blk-mq.h>
|
||||
+#include <linux/ctype.h>
|
||||
+#include <linux/hdreg.h>
|
||||
+#include <linux/list.h>
|
||||
+#include <linux/mutex.h>
|
||||
+#include <linux/of.h>
|
||||
+#include <linux/of_device.h>
|
||||
+#include <linux/of_fdt.h>
|
||||
+#include <linux/pagemap.h>
|
||||
+#include <linux/platform_device.h>
|
||||
+#include <linux/property.h>
|
||||
+#include <linux/refcount.h>
|
||||
+#include <linux/task_work.h>
|
||||
+#include <linux/types.h>
|
||||
+#include <linux/libfdt.h>
|
||||
+#include <linux/mtd/mtd.h>
|
||||
+#include <linux/root_dev.h>
|
||||
+#include <uapi/linux/fitblk.h>
|
||||
+
|
||||
+#define FIT_DEVICE_PREFIX "fit"
|
||||
+
|
||||
+/* maximum number of pages used for the uImage.FIT index structure */
|
||||
+#define FIT_MAX_PAGES 1024
|
||||
+
|
||||
+/* minimum free sectors to map as read-write "remainder" volume */
|
||||
+#define MIN_FREE_SECT 16
|
||||
+
|
||||
+/* maximum number of mapped loadables */
|
||||
+#define MAX_FIT_LOADABLES 16
|
||||
+
|
||||
+/* constants for uImage.FIT structrure traversal */
|
||||
+#define FIT_IMAGES_PATH "/images"
|
||||
+#define FIT_CONFS_PATH "/configurations"
|
||||
+
|
||||
+/* hash/signature/key node */
|
||||
+#define FIT_HASH_NODENAME "hash"
|
||||
+#define FIT_ALGO_PROP "algo"
|
||||
+#define FIT_VALUE_PROP "value"
|
||||
+#define FIT_IGNORE_PROP "uboot-ignore"
|
||||
+#define FIT_SIG_NODENAME "signature"
|
||||
+#define FIT_KEY_REQUIRED "required"
|
||||
+#define FIT_KEY_HINT "key-name-hint"
|
||||
+
|
||||
+/* cipher node */
|
||||
+#define FIT_CIPHER_NODENAME "cipher"
|
||||
+#define FIT_ALGO_PROP "algo"
|
||||
+
|
||||
+/* image node */
|
||||
+#define FIT_DATA_PROP "data"
|
||||
+#define FIT_DATA_POSITION_PROP "data-position"
|
||||
+#define FIT_DATA_OFFSET_PROP "data-offset"
|
||||
+#define FIT_DATA_SIZE_PROP "data-size"
|
||||
+#define FIT_TIMESTAMP_PROP "timestamp"
|
||||
+#define FIT_DESC_PROP "description"
|
||||
+#define FIT_ARCH_PROP "arch"
|
||||
+#define FIT_TYPE_PROP "type"
|
||||
+#define FIT_OS_PROP "os"
|
||||
+#define FIT_COMP_PROP "compression"
|
||||
+#define FIT_ENTRY_PROP "entry"
|
||||
+#define FIT_LOAD_PROP "load"
|
||||
+
|
||||
+/* configuration node */
|
||||
+#define FIT_KERNEL_PROP "kernel"
|
||||
+#define FIT_FILESYSTEM_PROP "filesystem"
|
||||
+#define FIT_RAMDISK_PROP "ramdisk"
|
||||
+#define FIT_FDT_PROP "fdt"
|
||||
+#define FIT_LOADABLE_PROP "loadables"
|
||||
+#define FIT_DEFAULT_PROP "default"
|
||||
+#define FIT_SETUP_PROP "setup"
|
||||
+#define FIT_FPGA_PROP "fpga"
|
||||
+#define FIT_FIRMWARE_PROP "firmware"
|
||||
+#define FIT_STANDALONE_PROP "standalone"
|
||||
+
|
||||
+/* fitblk driver data */
|
||||
+static const char *_fitblk_claim_ptr = "I belong to fitblk";
|
||||
+static const char *ubootver;
|
||||
+struct device_node *rootdisk;
|
||||
+static struct platform_device *pdev;
|
||||
+static LIST_HEAD(fitblk_devices);
|
||||
+static DEFINE_MUTEX(devices_mutex);
|
||||
+refcount_t num_devs;
|
||||
+
|
||||
+struct fitblk {
|
||||
+ struct platform_device *pdev;
|
||||
+ struct block_device *lower_bdev;
|
||||
+ sector_t start_sect;
|
||||
+ struct gendisk *disk;
|
||||
+ struct work_struct remove_work;
|
||||
+ struct list_head list;
|
||||
+ bool dead;
|
||||
+};
|
||||
+
|
||||
+static int fitblk_open(struct gendisk *disk, fmode_t mode)
|
||||
+{
|
||||
+ struct fitblk *fitblk = disk->private_data;
|
||||
+
|
||||
+ if (fitblk->dead)
|
||||
+ return -ENOENT;
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static void fitblk_release(struct gendisk *disk)
|
||||
+{
|
||||
+ return;
|
||||
+}
|
||||
+
|
||||
+static void fitblk_submit_bio(struct bio *orig_bio)
|
||||
+{
|
||||
+ struct bio *bio = orig_bio;
|
||||
+ struct fitblk *fitblk = bio->bi_bdev->bd_disk->private_data;
|
||||
+
|
||||
+ if (fitblk->dead)
|
||||
+ return;
|
||||
+
|
||||
+ /* mangle bio and re-submit */
|
||||
+ while (bio) {
|
||||
+ bio->bi_iter.bi_sector += fitblk->start_sect;
|
||||
+ bio->bi_bdev = fitblk->lower_bdev;
|
||||
+ bio = bio->bi_next;
|
||||
+ }
|
||||
+ submit_bio(orig_bio);
|
||||
+}
|
||||
+
|
||||
+static void fitblk_remove(struct fitblk *fitblk)
|
||||
+{
|
||||
+ blk_mark_disk_dead(fitblk->disk);
|
||||
+ mutex_lock(&devices_mutex);
|
||||
+ fitblk->dead = true;
|
||||
+ list_del(&fitblk->list);
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+
|
||||
+ schedule_work(&fitblk->remove_work);
|
||||
+}
|
||||
+
|
||||
+static int fitblk_ioctl(struct block_device *bdev, fmode_t mode,
|
||||
+ unsigned int cmd, unsigned long arg)
|
||||
+{
|
||||
+ struct fitblk *fitblk = bdev->bd_disk->private_data;
|
||||
+
|
||||
+ if (!capable(CAP_SYS_ADMIN))
|
||||
+ return -EACCES;
|
||||
+
|
||||
+ if (fitblk->dead)
|
||||
+ return -ENOENT;
|
||||
+
|
||||
+ switch (cmd) {
|
||||
+ case FITBLK_RELEASE:
|
||||
+ fitblk_remove(fitblk);
|
||||
+ break;
|
||||
+ default:
|
||||
+ return -EINVAL;
|
||||
+ }
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+
|
||||
+static const struct block_device_operations fitblk_fops = {
|
||||
+ .owner = THIS_MODULE,
|
||||
+ .ioctl = fitblk_ioctl,
|
||||
+ .open = fitblk_open,
|
||||
+ .release = fitblk_release,
|
||||
+ .submit_bio = fitblk_submit_bio,
|
||||
+};
|
||||
+
|
||||
+static void fitblk_purge(struct work_struct *work)
|
||||
+{
|
||||
+ struct fitblk *fitblk = container_of(work, struct fitblk, remove_work);
|
||||
+
|
||||
+ //del_gendisk(fitblk->disk); // causes crash, not doing it doesn't matter
|
||||
+ refcount_dec(&num_devs);
|
||||
+ platform_device_del(fitblk->pdev);
|
||||
+ platform_device_put(fitblk->pdev);
|
||||
+
|
||||
+ if (refcount_dec_if_one(&num_devs)) {
|
||||
+ sysfs_remove_link(&pdev->dev.kobj, "lower_dev");
|
||||
+ blkdev_put(fitblk->lower_bdev, &_fitblk_claim_ptr);
|
||||
+ }
|
||||
+
|
||||
+ kfree(fitblk);
|
||||
+}
|
||||
+
|
||||
+static int add_fit_subimage_device(struct block_device *lower_bdev,
|
||||
+ unsigned int slot, sector_t start_sect,
|
||||
+ sector_t nr_sect, bool readonly)
|
||||
+{
|
||||
+ struct fitblk *fitblk;
|
||||
+ struct gendisk *disk;
|
||||
+ int err;
|
||||
+
|
||||
+ mutex_lock(&devices_mutex);
|
||||
+ if (!refcount_inc_not_zero(&num_devs))
|
||||
+ return -EBADF;
|
||||
+
|
||||
+ fitblk = kzalloc(sizeof(struct fitblk), GFP_KERNEL);
|
||||
+ if (!fitblk) {
|
||||
+ err = -ENOMEM;
|
||||
+ goto out_unlock;
|
||||
+ }
|
||||
+
|
||||
+ fitblk->lower_bdev = lower_bdev;
|
||||
+ fitblk->start_sect = start_sect;
|
||||
+ INIT_WORK(&fitblk->remove_work, fitblk_purge);
|
||||
+
|
||||
+ disk = blk_alloc_disk(NUMA_NO_NODE);
|
||||
+ if (!disk) {
|
||||
+ err = -ENOMEM;
|
||||
+ goto out_free_fitblk;
|
||||
+ }
|
||||
+
|
||||
+ disk->first_minor = 0;
|
||||
+ disk->flags = lower_bdev->bd_disk->flags | GENHD_FL_NO_PART;
|
||||
+ disk->fops = &fitblk_fops;
|
||||
+ disk->private_data = fitblk;
|
||||
+ if (readonly) {
|
||||
+ set_disk_ro(disk, 1);
|
||||
+ snprintf(disk->disk_name, sizeof(disk->disk_name), FIT_DEVICE_PREFIX "%u", slot);
|
||||
+ } else {
|
||||
+ strcpy(disk->disk_name, FIT_DEVICE_PREFIX "rw");
|
||||
+ }
|
||||
+
|
||||
+ set_capacity(disk, nr_sect);
|
||||
+
|
||||
+ disk->queue->queue_flags = lower_bdev->bd_disk->queue->queue_flags;
|
||||
+ memcpy(&disk->queue->limits, &lower_bdev->bd_disk->queue->limits,
|
||||
+ sizeof(struct queue_limits));
|
||||
+
|
||||
+ fitblk->disk = disk;
|
||||
+ fitblk->pdev = platform_device_alloc(disk->disk_name, PLATFORM_DEVID_NONE);
|
||||
+ if (!fitblk->pdev) {
|
||||
+ err = -ENOMEM;
|
||||
+ goto out_cleanup_disk;
|
||||
+ }
|
||||
+
|
||||
+ fitblk->pdev->dev.parent = &pdev->dev;
|
||||
+ err = platform_device_add(fitblk->pdev);
|
||||
+ if (err)
|
||||
+ goto out_put_pdev;
|
||||
+
|
||||
+ err = device_add_disk(&fitblk->pdev->dev, disk, NULL);
|
||||
+ if (err)
|
||||
+ goto out_del_pdev;
|
||||
+
|
||||
+ if (!ROOT_DEV)
|
||||
+ ROOT_DEV = disk->part0->bd_dev;
|
||||
+
|
||||
+ list_add_tail(&fitblk->list, &fitblk_devices);
|
||||
+
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+
|
||||
+ return 0;
|
||||
+
|
||||
+out_del_pdev:
|
||||
+ platform_device_del(fitblk->pdev);
|
||||
+out_put_pdev:
|
||||
+ platform_device_put(fitblk->pdev);
|
||||
+out_cleanup_disk:
|
||||
+ put_disk(disk);
|
||||
+out_free_fitblk:
|
||||
+ kfree(fitblk);
|
||||
+out_unlock:
|
||||
+ refcount_dec(&num_devs);
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+ return err;
|
||||
+}
|
||||
+
|
||||
+static void fitblk_mark_dead(struct block_device *bdev, bool surprise)
|
||||
+{
|
||||
+ struct list_head *n, *tmp;
|
||||
+ struct fitblk *fitblk;
|
||||
+
|
||||
+ mutex_lock(&devices_mutex);
|
||||
+ list_for_each_safe(n, tmp, &fitblk_devices) {
|
||||
+ fitblk = list_entry(n, struct fitblk, list);
|
||||
+ if (fitblk->lower_bdev != bdev)
|
||||
+ continue;
|
||||
+
|
||||
+ fitblk->dead = true;
|
||||
+ list_del(&fitblk->list);
|
||||
+ /* removal needs to be deferred to avoid deadlock */
|
||||
+ schedule_work(&fitblk->remove_work);
|
||||
+ }
|
||||
+ mutex_unlock(&devices_mutex);
|
||||
+}
|
||||
+
|
||||
+static const struct blk_holder_ops fitblk_hops = {
|
||||
+ .mark_dead = fitblk_mark_dead,
|
||||
+};
|
||||
+
|
||||
+static int parse_fit_on_dev(struct device *dev)
|
||||
+{
|
||||
+ struct block_device *bdev;
|
||||
+ struct address_space *mapping;
|
||||
+ struct folio *folio;
|
||||
+ pgoff_t f_index = 0;
|
||||
+ size_t bytes_left, bytes_to_copy;
|
||||
+ void *pre_fit, *fit, *fit_c;
|
||||
+ u64 dsize, dsectors, imgmaxsect = 0;
|
||||
+ u32 size, image_pos, image_len;
|
||||
+ const __be32 *image_offset_be, *image_len_be, *image_pos_be;
|
||||
+ int ret = 0, node, images, config;
|
||||
+ const char *image_name, *image_type, *image_description,
|
||||
+ *config_default, *config_description, *config_loadables;
|
||||
+ u32 image_name_len, image_type_len, image_description_len,
|
||||
+ bootconf_len, config_default_len, config_description_len,
|
||||
+ config_loadables_len;
|
||||
+ sector_t start_sect, nr_sects;
|
||||
+ struct device_node *np = NULL;
|
||||
+ const char *bootconf_c;
|
||||
+ const char *loadable;
|
||||
+ char *bootconf = NULL, *bootconf_term;
|
||||
+ bool found;
|
||||
+ int loadables_rem_len, loadable_len;
|
||||
+ u16 loadcnt;
|
||||
+ unsigned int slot = 0;
|
||||
+
|
||||
+ /* Exclusive open the block device to receive holder notifications */
|
||||
+ bdev = blkdev_get_by_dev(dev->devt, BLK_OPEN_READ, &_fitblk_claim_ptr, &fitblk_hops);
|
||||
+ if (!bdev)
|
||||
+ return -ENODEV;
|
||||
+
|
||||
+ if (IS_ERR(bdev))
|
||||
+ return PTR_ERR(bdev);
|
||||
+
|
||||
+ mapping = bdev->bd_inode->i_mapping;
|
||||
+
|
||||
+ /* map first page */
|
||||
+ folio = read_mapping_folio(mapping, f_index++, NULL);
|
||||
+ if (IS_ERR(folio)) {
|
||||
+ ret = PTR_ERR(folio);
|
||||
+ goto out_blkdev;
|
||||
+ }
|
||||
+ pre_fit = folio_address(folio) + offset_in_folio(folio, 0);
|
||||
+
|
||||
+ /* uImage.FIT is based on flattened device tree structure */
|
||||
+ if (fdt_check_header(pre_fit)) {
|
||||
+ ret = -EINVAL;
|
||||
+ folio_put(folio);
|
||||
+ goto out_blkdev;
|
||||
+ }
|
||||
+
|
||||
+ size = fdt_totalsize(pre_fit);
|
||||
+
|
||||
+ if (size > PAGE_SIZE * FIT_MAX_PAGES) {
|
||||
+ ret = -EOPNOTSUPP;
|
||||
+ folio_put(folio);
|
||||
+ goto out_blkdev;
|
||||
+ }
|
||||
+
|
||||
+ /* acquire disk size */
|
||||
+ dsectors = bdev_nr_sectors(bdev);
|
||||
+ dsize = dsectors << SECTOR_SHIFT;
|
||||
+
|
||||
+ /* abort if FIT structure is larger than disk or partition size */
|
||||
+ if (size >= dsize) {
|
||||
+ ret = -EFBIG;
|
||||
+ folio_put(folio);
|
||||
+ goto out_blkdev;
|
||||
+ }
|
||||
+
|
||||
+ fit = kmalloc(size, GFP_KERNEL);
|
||||
+ if (!fit) {
|
||||
+ ret = -ENOMEM;
|
||||
+ folio_put(folio);
|
||||
+ goto out_blkdev;
|
||||
+ }
|
||||
+
|
||||
+ bytes_left = size;
|
||||
+ fit_c = fit;
|
||||
+ while (bytes_left > 0) {
|
||||
+ bytes_to_copy = min_t(size_t, bytes_left,
|
||||
+ folio_size(folio) - offset_in_folio(folio, 0));
|
||||
+ memcpy(fit_c, pre_fit, bytes_to_copy);
|
||||
+ fit_c += bytes_to_copy;
|
||||
+ bytes_left -= bytes_to_copy;
|
||||
+ if (bytes_left) {
|
||||
+ folio_put(folio);
|
||||
+ folio = read_mapping_folio(mapping, f_index++, NULL);
|
||||
+ if (IS_ERR(folio)) {
|
||||
+ ret = PTR_ERR(folio);
|
||||
+ goto out_blkdev;
|
||||
+ };
|
||||
+ pre_fit = folio_address(folio) + offset_in_folio(folio, 0);
|
||||
+ }
|
||||
+ }
|
||||
+ folio_put(folio);
|
||||
+
|
||||
+ /* set boot config node name U-Boot may have added to the device tree */
|
||||
+ np = of_find_node_by_path("/chosen");
|
||||
+ if (np) {
|
||||
+ bootconf_c = of_get_property(np, "u-boot,bootconf", &bootconf_len);
|
||||
+ if (bootconf_c && bootconf_len)
|
||||
+ bootconf = kmemdup_nul(bootconf_c, bootconf_len, GFP_KERNEL);
|
||||
+ }
|
||||
+
|
||||
+ if (bootconf) {
|
||||
+ bootconf_term = strchr(bootconf, '#');
|
||||
+ if (bootconf_term)
|
||||
+ *bootconf_term = '\0';
|
||||
+ }
|
||||
+
|
||||
+ /* find configuration path in uImage.FIT */
|
||||
+ config = fdt_path_offset(fit, FIT_CONFS_PATH);
|
||||
+ if (config < 0) {
|
||||
+ pr_err("FIT: Cannot find %s node: %d\n",
|
||||
+ FIT_CONFS_PATH, config);
|
||||
+ ret = -ENOENT;
|
||||
+ goto out_bootconf;
|
||||
+ }
|
||||
+
|
||||
+ /* get default configuration node name */
|
||||
+ config_default =
|
||||
+ fdt_getprop(fit, config, FIT_DEFAULT_PROP, &config_default_len);
|
||||
+
|
||||
+ /* make sure we got either default or selected boot config node name */
|
||||
+ if (!config_default && !bootconf) {
|
||||
+ pr_err("FIT: Cannot find default configuration\n");
|
||||
+ ret = -ENOENT;
|
||||
+ goto out_bootconf;
|
||||
+ }
|
||||
+
|
||||
+ /* find selected boot config node, fallback on default config node */
|
||||
+ node = fdt_subnode_offset(fit, config, bootconf ?: config_default);
|
||||
+ if (node < 0) {
|
||||
+ pr_err("FIT: Cannot find %s node: %d\n",
|
||||
+ bootconf ?: config_default, node);
|
||||
+ ret = -ENOENT;
|
||||
+ goto out_bootconf;
|
||||
+ }
|
||||
+
|
||||
+ pr_info("FIT: Detected U-Boot %s\n", ubootver);
|
||||
+
|
||||
+ /* get selected configuration data */
|
||||
+ config_description =
|
||||
+ fdt_getprop(fit, node, FIT_DESC_PROP, &config_description_len);
|
||||
+ config_loadables = fdt_getprop(fit, node, FIT_LOADABLE_PROP,
|
||||
+ &config_loadables_len);
|
||||
+
|
||||
+ pr_info("FIT: %s configuration: \"%.*s\"%s%.*s%s\n",
|
||||
+ bootconf ? "Selected" : "Default",
|
||||
+ bootconf ? bootconf_len : config_default_len,
|
||||
+ bootconf ?: config_default,
|
||||
+ config_description ? " (" : "",
|
||||
+ config_description ? config_description_len : 0,
|
||||
+ config_description ?: "",
|
||||
+ config_description ? ")" : "");
|
||||
+
|
||||
+ if (!config_loadables || !config_loadables_len) {
|
||||
+ pr_err("FIT: No loadables configured in \"%s\"\n",
|
||||
+ bootconf ?: config_default);
|
||||
+ ret = -ENOENT;
|
||||
+ goto out_bootconf;
|
||||
+ }
|
||||
+
|
||||
+ /* get images path in uImage.FIT */
|
||||
+ images = fdt_path_offset(fit, FIT_IMAGES_PATH);
|
||||
+ if (images < 0) {
|
||||
+ pr_err("FIT: Cannot find %s node: %d\n", FIT_IMAGES_PATH, images);
|
||||
+ ret = -EINVAL;
|
||||
+ goto out_bootconf;
|
||||
+ }
|
||||
+
|
||||
+ /* iterate over images in uImage.FIT */
|
||||
+ fdt_for_each_subnode(node, fit, images) {
|
||||
+ image_name = fdt_get_name(fit, node, &image_name_len);
|
||||
+ image_type = fdt_getprop(fit, node, FIT_TYPE_PROP, &image_type_len);
|
||||
+ image_offset_be = fdt_getprop(fit, node, FIT_DATA_OFFSET_PROP, NULL);
|
||||
+ image_pos_be = fdt_getprop(fit, node, FIT_DATA_POSITION_PROP, NULL);
|
||||
+ image_len_be = fdt_getprop(fit, node, FIT_DATA_SIZE_PROP, NULL);
|
||||
+
|
||||
+ if (!image_name || !image_type || !image_len_be ||
|
||||
+ !image_name_len || !image_type_len)
|
||||
+ continue;
|
||||
+
|
||||
+ image_len = be32_to_cpu(*image_len_be);
|
||||
+ if (!image_len)
|
||||
+ continue;
|
||||
+
|
||||
+ if (image_offset_be)
|
||||
+ image_pos = be32_to_cpu(*image_offset_be) + size;
|
||||
+ else if (image_pos_be)
|
||||
+ image_pos = be32_to_cpu(*image_pos_be);
|
||||
+ else
|
||||
+ continue;
|
||||
+
|
||||
+ image_description = fdt_getprop(fit, node, FIT_DESC_PROP,
|
||||
+ &image_description_len);
|
||||
+
|
||||
+ pr_info("FIT: %16s sub-image 0x%08x..0x%08x \"%.*s\"%s%.*s%s\n",
|
||||
+ image_type, image_pos, image_pos + image_len - 1,
|
||||
+ image_name_len, image_name, image_description ? " (" : "",
|
||||
+ image_description ? image_description_len : 0,
|
||||
+ image_description ?: "", image_description ? ") " : "");
|
||||
+
|
||||
+ /* only 'filesystem' images should be mapped as partitions */
|
||||
+ if (strncmp(image_type, FIT_FILESYSTEM_PROP, image_type_len))
|
||||
+ continue;
|
||||
+
|
||||
+ /* check if sub-image is part of configured loadables */
|
||||
+ found = false;
|
||||
+ loadable = config_loadables;
|
||||
+ loadables_rem_len = config_loadables_len;
|
||||
+ for (loadcnt = 0; loadables_rem_len > 1 &&
|
||||
+ loadcnt < MAX_FIT_LOADABLES; ++loadcnt) {
|
||||
+ loadable_len =
|
||||
+ strnlen(loadable, loadables_rem_len - 1) + 1;
|
||||
+ loadables_rem_len -= loadable_len;
|
||||
+ if (!strncmp(image_name, loadable, loadable_len)) {
|
||||
+ found = true;
|
||||
+ break;
|
||||
+ }
|
||||
+ loadable += loadable_len;
|
||||
+ }
|
||||
+ if (!found)
|
||||
+ continue;
|
||||
+
|
||||
+ if (image_pos % (1 << PAGE_SHIFT)) {
|
||||
+ dev_err(dev, "FIT: image %.*s start not aligned to page boundaries, skipping\n",
|
||||
+ image_name_len, image_name);
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ if (image_len % (1 << PAGE_SHIFT)) {
|
||||
+ dev_err(dev, "FIT: sub-image %.*s end not aligned to page boundaries, skipping\n",
|
||||
+ image_name_len, image_name);
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ start_sect = image_pos >> SECTOR_SHIFT;
|
||||
+ nr_sects = image_len >> SECTOR_SHIFT;
|
||||
+ imgmaxsect = max_t(sector_t, imgmaxsect, start_sect + nr_sects);
|
||||
+
|
||||
+ if (start_sect + nr_sects > dsectors) {
|
||||
+ dev_err(dev, "FIT: sub-image %.*s disk access beyond EOD\n",
|
||||
+ image_name_len, image_name);
|
||||
+ continue;
|
||||
+ }
|
||||
+
|
||||
+ if (!slot) {
|
||||
+ ret = sysfs_create_link_nowarn(&pdev->dev.kobj, bdev_kobj(bdev), "lower_dev");
|
||||
+ if (ret && ret != -EEXIST)
|
||||
+ goto out_bootconf;
|
||||
+
|
||||
+ ret = 0;
|
||||
+ }
|
||||
+
|
||||
+ add_fit_subimage_device(bdev, slot++, start_sect, nr_sects, true);
|
||||
+ }
|
||||
+
|
||||
+ if (!found || !slot)
|
||||
+ goto out_bootconf;
|
||||
+
|
||||
+ dev_info(dev, "mapped %u uImage.FIT filesystem sub-image%s as /dev/fit%s%u%s\n",
|
||||
+ slot, (slot > 1)?"s":"", (slot > 1)?"[0...":"", slot - 1,
|
||||
+ (slot > 1)?"]":"");
|
||||
+
|
||||
+ /* in case uImage.FIT is stored in a partition, map the remaining space */
|
||||
+ if (!bdev->bd_read_only && bdev_is_partition(bdev) &&
|
||||
+ (imgmaxsect + MIN_FREE_SECT) < dsectors) {
|
||||
+ add_fit_subimage_device(bdev, slot++, imgmaxsect,
|
||||
+ dsectors - imgmaxsect, false);
|
||||
+ dev_info(dev, "mapped remaining space as /dev/fitrw\n");
|
||||
+ }
|
||||
+
|
||||
+out_bootconf:
|
||||
+ kfree(bootconf);
|
||||
+ kfree(fit);
|
||||
+out_blkdev:
|
||||
+ if (!found || ret)
|
||||
+ blkdev_put(bdev, &_fitblk_claim_ptr);
|
||||
+
|
||||
+ return ret;
|
||||
+}
|
||||
+
|
||||
+static int fitblk_match_of_node(struct device *dev, const void *np)
|
||||
+{
|
||||
+ int ret;
|
||||
+
|
||||
+ ret = device_match_of_node(dev, np);
|
||||
+ if (ret)
|
||||
+ return ret;
|
||||
+
|
||||
+ /*
|
||||
+ * To match ubiblock and mtdblock devices by their parent ubi
|
||||
+ * or mtd device, also consider block device parent
|
||||
+ */
|
||||
+ if (!dev->parent)
|
||||
+ return 0;
|
||||
+
|
||||
+ return device_match_of_node(dev->parent, np);
|
||||
+}
|
||||
+
|
||||
+static int fitblk_probe(struct platform_device *pdev)
|
||||
+{
|
||||
+ struct device *dev;
|
||||
+
|
||||
+ dev = class_find_device(&block_class, NULL, rootdisk, fitblk_match_of_node);
|
||||
+ if (!dev)
|
||||
+ return -EPROBE_DEFER;
|
||||
+
|
||||
+ return parse_fit_on_dev(dev);
|
||||
+}
|
||||
+
|
||||
+static struct platform_driver fitblk_driver = {
|
||||
+ .probe = fitblk_probe,
|
||||
+ .driver = {
|
||||
+ .name = "fitblk",
|
||||
+ },
|
||||
+};
|
||||
+
|
||||
+static int __init fitblk_init(void)
|
||||
+{
|
||||
+ /* detect U-Boot firmware */
|
||||
+ ubootver = of_get_property(of_chosen, "u-boot,version", NULL);
|
||||
+ if (!ubootver)
|
||||
+ return 0;
|
||||
+
|
||||
+ /* parse 'rootdisk' property phandle */
|
||||
+ rootdisk = of_parse_phandle(of_chosen, "rootdisk", 0);
|
||||
+ if (!rootdisk)
|
||||
+ return 0;
|
||||
+
|
||||
+ if (platform_driver_register(&fitblk_driver))
|
||||
+ return -ENODEV;
|
||||
+
|
||||
+ refcount_set(&num_devs, 1);
|
||||
+ pdev = platform_device_register_simple("fitblk", -1, NULL, 0);
|
||||
+ if (IS_ERR(pdev))
|
||||
+ return PTR_ERR(pdev);
|
||||
+
|
||||
+ return 0;
|
||||
+}
|
||||
+device_initcall(fitblk_init);
|
||||
--- /dev/null
|
||||
+++ b/include/uapi/linux/fitblk.h
|
||||
@@ -0,0 +1,10 @@
|
||||
+/* SPDX-License-Identifier: GPL-2.0+ WITH Linux-syscall-note */
|
||||
+#ifndef _UAPI_LINUX_FITBLK_H
|
||||
+#define _UAPI_LINUX_FITBLK_H
|
||||
+
|
||||
+/*
|
||||
+ * IOCTL commands --- we will commandeer 0x46 ('F')
|
||||
+ */
|
||||
+#define FITBLK_RELEASE 0x4600
|
||||
+
|
||||
+#endif /* _UAPI_LINUX_FITBLK_H */
|
Loading…
Reference in a new issue