[PATCH v2 2/5] misc: fs_loader: reorganize and split to FS and FW loader

Francis, Neha n-francis at ti.com
Tue Mar 17 07:06:03 CET 2026


On 3/7/2026 12:35 AM, Christian Marangi wrote:
> In preparation to the introduction of variant of the FS loader,
> reorganize and split the driver to generic fw_loader function and
> specific fs_loader function. Create a dedicated directory for the loader
> and move the internal structs and functions to a dedicated header file.
> 
> This will permit to reuse all the property and logic of FS loader
> with container that are not exactly a readable filesystem.
> 
> Signed-off-by: Christian Marangi <ansuelsmth at gmail.com>
> ---
>  drivers/misc/Kconfig                     |   5 +
>  drivers/misc/Makefile                    |   2 +-
>  drivers/misc/fw_loader/Makefile          |   4 +
>  drivers/misc/{ => fw_loader}/fs_loader.c | 134 ++---------------------
>  drivers/misc/fw_loader/fw_loader.c       | 121 ++++++++++++++++++++
>  drivers/misc/fw_loader/internal.h        |  60 ++++++++++
>  include/fs_loader.h                      |  47 +-------
>  include/fw_loader.h                      |  19 ++++
>  8 files changed, 220 insertions(+), 172 deletions(-)
>  create mode 100644 drivers/misc/fw_loader/Makefile
>  rename drivers/misc/{ => fw_loader}/fs_loader.c (60%)
>  create mode 100644 drivers/misc/fw_loader/fw_loader.c
>  create mode 100644 drivers/misc/fw_loader/internal.h
> 
> diff --git a/drivers/misc/Kconfig b/drivers/misc/Kconfig
> index a0aa290480eb..9d332230b1f9 100644
> --- a/drivers/misc/Kconfig
> +++ b/drivers/misc/Kconfig
> @@ -610,8 +610,12 @@ config MPC83XX_SERDES
>  	help
>  	  Support for serdes found on MPC83xx SoCs.
>  
> +config FW_LOADER
> +	bool
> +
>  config FS_LOADER
>  	bool "Enable loader driver for file system"
> +	select FW_LOADER
>  	help
>  	  This is file system generic loader which can be used to load
>  	  the file image from the storage into target such as memory.
> @@ -621,6 +625,7 @@ config FS_LOADER
>  
>  config SPL_FS_LOADER
>  	bool "Enable loader driver for file system in SPL"
> +	select FW_LOADER
>  	depends on SPL
>  	help
>  	  This is file system generic loader which can be used to load
> diff --git a/drivers/misc/Makefile b/drivers/misc/Makefile
> index 1d950f7a0ab2..c1b83d576bbb 100644
> --- a/drivers/misc/Makefile
> +++ b/drivers/misc/Makefile
> @@ -36,7 +36,7 @@ obj-$(CONFIG_FSL_DEVICE_DISABLE) += fsl_devdis.o
>  obj-$(CONFIG_FSL_IFC) += fsl_ifc.o
>  obj-$(CONFIG_FSL_IIM) += fsl_iim.o
>  obj-$(CONFIG_FSL_SEC_MON) += fsl_sec_mon.o
> -obj-$(CONFIG_$(PHASE_)FS_LOADER) += fs_loader.o
> +obj-$(CONFIG_FW_LOADER) += fw_loader/
>  obj-$(CONFIG_GATEWORKS_SC) += gsc.o
>  obj-$(CONFIG_GDSYS_IOEP) += gdsys_ioep.o
>  obj-$(CONFIG_GDSYS_RXAUI_CTRL) += gdsys_rxaui_ctrl.o
> diff --git a/drivers/misc/fw_loader/Makefile b/drivers/misc/fw_loader/Makefile
> new file mode 100644
> index 000000000000..96baebede788
> --- /dev/null
> +++ b/drivers/misc/fw_loader/Makefile
> @@ -0,0 +1,4 @@
> +# SPDX-License-Identifier: GPL-2.0+
> +
> +obj-y += fw_loader.o
> +obj-$(CONFIG_$(PHASE_)FS_LOADER) += fs_loader.o
> diff --git a/drivers/misc/fs_loader.c b/drivers/misc/fw_loader/fs_loader.c
> similarity index 60%
> rename from drivers/misc/fs_loader.c
> rename to drivers/misc/fw_loader/fs_loader.c
> index 7e432a7ebd62..4ae01b9c8f0d 100644
> --- a/drivers/misc/fs_loader.c
> +++ b/drivers/misc/fw_loader/fs_loader.c
> @@ -25,25 +25,9 @@
>  #include <ubi_uboot.h>
>  #endif
>  
> -DECLARE_GLOBAL_DATA_PTR;
> +#include "internal.h"
>  
> -/**
> - * struct firmware - A place for storing firmware and its attribute data.
> - *
> - * This holds information about a firmware and its content.
> - *
> - * @size: Size of a file
> - * @data: Buffer for file
> - * @priv: Firmware loader private fields
> - * @name: Filename
> - * @offset: Offset of reading a file
> - */
> -struct firmware {
> -	size_t size;
> -	const u8 *data;
> -	const char *name;
> -	u32 offset;
> -};
> +DECLARE_GLOBAL_DATA_PTR;
>  
>  #ifdef CONFIG_CMD_UBIFS
>  static int mount_ubifs(char *mtdpart, char *ubivol)
> @@ -109,37 +93,6 @@ static int select_fs_dev(struct device_plat *plat)
>  	return ret;
>  }
>  
> -/**
> - * _request_firmware_prepare - Prepare firmware struct.
> - *
> - * @dev: An instance of a driver.
> - * @name: Name of firmware file.
> - * @dbuf: Address of buffer to load firmware into.
> - * @size: Size of buffer.
> - * @offset: Offset of a file for start reading into buffer.
> - *
> - * Return: Negative value if fail, 0 for successful.
> - */
> -static int _request_firmware_prepare(struct udevice *dev,
> -				    const char *name, void *dbuf,
> -				    size_t size, u32 offset)
> -{
> -	if (!name || name[0] == '\0')
> -		return -EINVAL;
> -
> -	struct firmware *firmwarep = dev_get_priv(dev);
> -
> -	if (!firmwarep)
> -		return -ENOMEM;
> -
> -	firmwarep->name = name;
> -	firmwarep->offset = offset;
> -	firmwarep->data = dbuf;
> -	firmwarep->size = size;
> -
> -	return 0;
> -}
> -
>  /**
>   * fw_get_filesystem_firmware - load firmware into an allocated buffer.
>   * @dev: An instance of a driver.
> @@ -200,87 +153,16 @@ out:
>  	return ret;
>  }
>  
> -/**
> - * request_firmware_into_buf - Load firmware into a previously allocated buffer.
> - * @dev: An instance of a driver.
> - * @name: Name of firmware file.
> - * @buf: Address of buffer to load firmware into.
> - * @size: Size of buffer.
> - * @offset: Offset of a file for start reading into buffer.
> - *
> - * The firmware is loaded directly into the buffer pointed to by @buf.
> - *
> - * Return: Size of total read, negative value when error.
> - */
> -int request_firmware_into_buf(struct udevice *dev,
> -			      const char *name,
> -			      void *buf, size_t size, u32 offset)
> -{
> -	int ret;
> -
> -	if (!dev)
> -		return -EINVAL;
> -
> -	ret = _request_firmware_prepare(dev, name, buf, size, offset);
> -	if (ret < 0) /* error */
> -		return ret;
> -
> -	ret = fw_get_filesystem_firmware(dev);
> -
> -	return ret;
> -}
> -
> -static int fs_loader_of_to_plat(struct udevice *dev)
> -{
> -	u32 phandlepart[2];
> -
> -	ofnode fs_loader_node = dev_ofnode(dev);
> -
> -	if (ofnode_valid(fs_loader_node)) {
> -		struct device_plat *plat;
> -
> -		plat = dev_get_plat(dev);
> -		if (!ofnode_read_u32_array(fs_loader_node,
> -					  "phandlepart",
> -					  phandlepart, 2)) {
> -			plat->phandlepart.phandle = phandlepart[0];
> -			plat->phandlepart.partition = phandlepart[1];
> -		}
> -
> -		plat->mtdpart = (char *)ofnode_read_string(
> -				 fs_loader_node, "mtdpart");
> -
> -		plat->ubivol = (char *)ofnode_read_string(
> -				 fs_loader_node, "ubivol");
> -	}
> -
> -	return 0;
> -}
> -
>  static int fs_loader_probe(struct udevice *dev)
>  {
> -#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(BLK)
> -	int ret;
>  	struct device_plat *plat = dev_get_plat(dev);
> +	int ret;
>  
> -	if (plat->phandlepart.phandle) {
> -		ofnode node = ofnode_get_by_phandle(plat->phandlepart.phandle);
> -		struct udevice *parent_dev = NULL;
> -
> -		ret = device_get_global_by_ofnode(node, &parent_dev);
> -		if (!ret) {
> -			struct udevice *dev;
> -
> -			ret = blk_get_from_parent(parent_dev, &dev);
> -			if (ret) {
> -				debug("fs_loader: No block device: %d\n",
> -					ret);
> +	ret = generic_fw_loader_probe(dev);
> +	if (ret)
> +		return ret;
>  
> -				return ret;
> -			}
> -		}
> -	}
> -#endif
> +	plat->get_firmware = fw_get_filesystem_firmware;
>  
>  	return 0;
>  };
> @@ -295,7 +177,7 @@ U_BOOT_DRIVER(fs_loader) = {
>  	.id			= UCLASS_FS_FIRMWARE_LOADER,
>  	.of_match		= fs_loader_ids,
>  	.probe			= fs_loader_probe,
> -	.of_to_plat	= fs_loader_of_to_plat,
> +	.of_to_plat	= generic_fw_loader_of_to_plat,
>  	.plat_auto	= sizeof(struct device_plat),
>  	.priv_auto	= sizeof(struct firmware),
>  };
> diff --git a/drivers/misc/fw_loader/fw_loader.c b/drivers/misc/fw_loader/fw_loader.c
> new file mode 100644
> index 000000000000..6459ba23a15b
> --- /dev/null
> +++ b/drivers/misc/fw_loader/fw_loader.c
> @@ -0,0 +1,121 @@
> +// SPDX-License-Identifier: GPL-2.0
> +/*
> + * Copyright (C) 2018-2019 Intel Corporation <www.intel.com>
> + *
> + */
> +
> +#include <errno.h>
> +#include <blk.h>
> +#include <linux/types.h>
> +#include <dm/device.h>
> +#include <fw_loader.h>
> +
> +#include "internal.h"
> +
> +int generic_fw_loader_of_to_plat(struct udevice *dev)
> +{
> +	u32 phandlepart[2];
> +
> +	ofnode fw_loader_node = dev_ofnode(dev);
> +
> +	if (ofnode_valid(fw_loader_node)) {
> +		struct device_plat *plat;
> +
> +		plat = dev_get_plat(dev);
> +		if (!ofnode_read_u32_array(fw_loader_node,
> +					   "phandlepart",
> +					   phandlepart, 2)) {
> +			plat->phandlepart.phandle = phandlepart[0];
> +			plat->phandlepart.partition = phandlepart[1];
> +		}
> +
> +		plat->mtdpart = (char *)ofnode_read_string(fw_loader_node,
> +							   "mtdpart");
> +
> +		plat->ubivol = (char *)ofnode_read_string(fw_loader_node,
> +							  "ubivol");
> +	}
> +
> +	return 0;
> +}
> +
> +int generic_fw_loader_probe(struct udevice *dev)
> +{
> +#if CONFIG_IS_ENABLED(DM) && CONFIG_IS_ENABLED(BLK)
> +	int ret;
> +	struct device_plat *plat = dev_get_plat(dev);
> +
> +	if (plat->phandlepart.phandle) {
> +		ofnode node = ofnode_get_by_phandle(plat->phandlepart.phandle);
> +		struct udevice *parent_dev = NULL;
> +
> +		ret = device_get_global_by_ofnode(node, &parent_dev);
> +		if (!ret) {
> +			struct udevice *dev;

Minor correction of readability since we are reorganizing anyway: s/dev/blk_dev?
instead of shadowing

> +
> +			ret = blk_get_from_parent(parent_dev, &dev);
> +			if (ret) {
> +				debug("fw_loader: No block device: %d\n",
> +				      ret);
> +
> +				return ret;
> +			}
> +		}
> +	}
> +#endif
> +
> +	return 0;
> +}
> +
> +/**
> + * _request_firmware_prepare - Prepare firmware struct.
> + *
> + * @dev: An instance of a driver.
> + * @name: Name of firmware file.
> + * @dbuf: Address of buffer to load firmware into.
> + * @size: Size of buffer.
> + * @offset: Offset of a file for start reading into buffer.
> + *
> + * Return: Negative value if fail, 0 for successful.
> + */
> +static int _request_firmware_prepare(struct udevice *dev,
> +				     const char *name, void *dbuf,
> +				     size_t size, u32 offset)
> +{
> +	if (!name || name[0] == '\0')
> +		return -EINVAL;
> +
> +	struct firmware *firmwarep = dev_get_priv(dev);
> +
> +	if (!firmwarep)
> +		return -ENOMEM;
> +
> +	firmwarep->name = name;
> +	firmwarep->offset = offset;
> +	firmwarep->data = dbuf;
> +	firmwarep->size = size;
> +
> +	return 0;
> +}
> +
> +int request_firmware_into_buf(struct udevice *dev,
> +			      const char *name,
> +			      void *buf, size_t size, u32 offset)
> +{
> +	struct device_plat *plat;
> +	int ret;
> +
> +	if (!dev)
> +		return -EINVAL;
> +
> +	ret = _request_firmware_prepare(dev, name, buf, size, offset);
> +	if (ret < 0) /* error */
> +		return ret;
> +
> +	plat = dev_get_plat(dev);
> +
> +	if (!plat->get_firmware)
> +		return -EOPNOTSUPP;
> +
> +	return plat->get_firmware(dev);
> +}
> diff --git a/drivers/misc/fw_loader/internal.h b/drivers/misc/fw_loader/internal.h
> new file mode 100644
> index 000000000000..fc78a4add59d
> --- /dev/null
> +++ b/drivers/misc/fw_loader/internal.h
> @@ -0,0 +1,60 @@
> +/* SPDX-License-Identifier: GPL-2.0 */
> +/*
> + * Copyright (C) 2018-2019 Intel Corporation <www.intel.com>
> + */
> +#ifndef _FW_LOADER_INTERNAL_H_
> +#define _FW_LOADER_INTERNAL_H_
> +
> +/**
> + * struct phandle_part - A place for storing phandle of node and its partition
> + *
> + * This holds information about a phandle of the block device, and its
> + * partition where the firmware would be loaded from.
> + *
> + * @phandle: Phandle of storage device node
> + * @partition: Partition of block device
> + */
> +struct phandle_part {
> +	u32 phandle;
> +	u32 partition;
> +};
> +
> +/**
> + * struct phandle_part - A place for storing all supported storage devices

Again nitpick s/phandle_part/device_plat

> + *
> + * This holds information about all supported storage devices for driver use.
> + *
> + * @phandlepart: Attribute data for block device.
> + * @mtdpart: MTD partition for ubi partition.
> + * @ubivol: UBI volume-name for ubifsmount.
> + */
> +struct device_plat {
> +	struct phandle_part phandlepart;
> +	char *mtdpart;
> +	char *ubivol;
> +
> +	int (*get_firmware)(struct udevice *dev);
> +};
> +
> +/**
> + * struct firmware - A place for storing firmware and its attribute data.
> + *
> + * This holds information about a firmware and its content.
> + *
> + * @size: Size of a file
> + * @data: Buffer for file
> + * @priv: Firmware loader private fields
> + * @name: Filename
> + * @offset: Offset of reading a file
> + */
> +struct firmware {
> +	size_t size;
> +	const u8 *data;
> +	const char *name;
> +	u32 offset;
> +};
> +
> +int generic_fw_loader_probe(struct udevice *dev);
> +int generic_fw_loader_of_to_plat(struct udevice *dev);
> +
> +#endif
> diff --git a/include/fs_loader.h b/include/fs_loader.h
> index 7e16e0f70309..3c64efe1b439 100644
> --- a/include/fs_loader.h
> +++ b/include/fs_loader.h
> @@ -6,52 +6,9 @@
>  #ifndef _FS_LOADER_H_
>  #define _FS_LOADER_H_
>  
> -struct udevice;
> -
> -/**
> - * struct phandle_part - A place for storing phandle of node and its partition
> - *
> - * This holds information about a phandle of the block device, and its
> - * partition where the firmware would be loaded from.
> - *
> - * @phandle: Phandle of storage device node
> - * @partition: Partition of block device
> - */
> -struct phandle_part {
> -	u32 phandle;
> -	u32 partition;
> -};
> -
> -/**
> - * struct phandle_part - A place for storing all supported storage devices
> - *
> - * This holds information about all supported storage devices for driver use.
> - *
> - * @phandlepart: Attribute data for block device.
> - * @mtdpart: MTD partition for ubi partition.
> - * @ubivol: UBI volume-name for ubifsmount.
> - */
> -struct device_plat {
> -	struct phandle_part phandlepart;
> -	char *mtdpart;
> -	char *ubivol;
> -};
> +#include <fw_loader.h>
>  
> -/**
> - * request_firmware_into_buf - Load firmware into a previously allocated buffer.
> - * @dev: An instance of a driver.
> - * @name: Name of firmware file.
> - * @buf: Address of buffer to load firmware into.
> - * @size: Size of buffer.
> - * @offset: Offset of a file for start reading into buffer.
> - *
> - * The firmware is loaded directly into the buffer pointed to by @buf.
> - *
> - * Return: Size of total read, negative value when error.
> - */
> -int request_firmware_into_buf(struct udevice *dev,
> -			      const char *name,
> -			      void *buf, size_t size, u32 offset);
> +struct udevice;
>  
>  /**
>   * get_fs_loader() - Get the chosen filesystem loader
> diff --git a/include/fw_loader.h b/include/fw_loader.h
> index 35574482b2b9..56f5e3be6195 100644
> --- a/include/fw_loader.h
> +++ b/include/fw_loader.h
> @@ -1,10 +1,29 @@
>  /* SPDX-License-Identifier: GPL-2.0 */
>  /*
> + * Copyright (C) 2018-2019 Intel Corporation <www.intel.com>
>   * Copyright (C) 2025 Lucien Jheng <lucienzx159 at gmail.com>
>   */
>  #ifndef _FW_LOADER_H_
>  #define _FW_LOADER_H_
>  
> +struct udevice;
> +
> +/**
> + * request_firmware_into_buf - Load firmware into a previously allocated buffer.
> + * @dev: An instance of a driver.
> + * @name: Name of firmware file.
> + * @buf: Address of buffer to load firmware into.
> + * @size: Size of buffer.
> + * @offset: Offset of a file for start reading into buffer.
> + *
> + * The firmware is loaded directly into the buffer pointed to by @buf.
> + *
> + * Return: Size of total read, negative value when error.
> + */
> +int request_firmware_into_buf(struct udevice *dev,
> +			      const char *name,
> +			      void *buf, size_t size, u32 offset);
> +
>  /**
>   * request_firmware_into_buf_via_script() -
>   * Load firmware using a U-Boot script and copy to buffer

With that:

Reviewed-by: Neha Malcom Francis <n-francis at ti.com>

-- 
Thanking You
Neha Malcom Francis



More information about the U-Boot mailing list