On 4/22/25 15:12, Patrice Chotard wrote:
> From: Lionel Debieve <lionel.debi...@foss.st.com>
>
> This driver is checking the access rights of the different
> peripherals connected to the ETZPC bus. If access is denied,
> the associated device is not bound.
>
> Signed-off-by: Lionel Debieve <lionel.debi...@foss.st.com>
> Signed-off-by: Gatien Chevallier <gatien.chevall...@foss.st.com>
> Signed-off-by: Patrick Delaunay <patrick.delau...@foss.st.com>
> Signed-off-by: Patrice Chotard <patrice.chot...@foss.st.com>
> Reviewed-by: Patrick Delaunay <patrick.delau...@foss.st.com>
> ---
>
> (no changes since v1)
>
> arch/arm/mach-stm32mp/include/mach/etzpc.h | 32 ++++
> arch/arm/mach-stm32mp/stm32mp1/Makefile | 1 +
> arch/arm/mach-stm32mp/stm32mp1/etzpc.c | 194 +++++++++++++++++++++
> 3 files changed, 227 insertions(+)
> create mode 100644 arch/arm/mach-stm32mp/include/mach/etzpc.h
> create mode 100644 arch/arm/mach-stm32mp/stm32mp1/etzpc.c
>
> diff --git a/arch/arm/mach-stm32mp/include/mach/etzpc.h
> b/arch/arm/mach-stm32mp/include/mach/etzpc.h
> new file mode 100644
> index 00000000000..fd697c3e2ac
> --- /dev/null
> +++ b/arch/arm/mach-stm32mp/include/mach/etzpc.h
> @@ -0,0 +1,32 @@
> +/* SPDX-License-Identifier: GPL-2.0-or-later OR BSD-3-Clause */
> +/*
> + * Copyright (C) 2023, STMicroelectronics - All Rights Reserved
> + */
> +
> +#ifndef MACH_ETZPC_H
> +#define MACH_ETZPC_H
> +
> +#include <linux/types.h>
> +
> +/**
> + * stm32_etzpc_check_access - Check ETZPC accesses for given device node
> + *
> + * @device_node Node of the device for which the accesses are
> checked
> + *
> + * @returns 0 on success (if access is granted), -EINVAL if access is denied.
> + * Else, returns an appropriate negative ERRNO value
> + */
> +int stm32_etzpc_check_access(ofnode device_node);
> +
> +/**
> + * stm32_etzpc_check_access_by_id - Check ETZPC accesses for given id
> + *
> + * @device_node Node of the device to get a reference on ETZPC
> + * @id ID of the resource to check
> + *
> + * @returns 0 on success (if access is granted), -EINVAL if access is denied.
> + * Else, returns an appropriate negative ERRNO value
> + */
> +int stm32_etzpc_check_access_by_id(ofnode device_node, u32 id);
> +
> +#endif /* MACH_ETZPC_H*/
> diff --git a/arch/arm/mach-stm32mp/stm32mp1/Makefile
> b/arch/arm/mach-stm32mp/stm32mp1/Makefile
> index 0df6dabaaab..1f4ada3ac70 100644
> --- a/arch/arm/mach-stm32mp/stm32mp1/Makefile
> +++ b/arch/arm/mach-stm32mp/stm32mp1/Makefile
> @@ -4,6 +4,7 @@
> #
>
> obj-y += cpu.o
> +obj-y += etzpc.o
>
> obj-$(CONFIG_STM32MP13X) += stm32mp13x.o
> obj-$(CONFIG_STM32MP15X) += stm32mp15x.o
> diff --git a/arch/arm/mach-stm32mp/stm32mp1/etzpc.c
> b/arch/arm/mach-stm32mp/stm32mp1/etzpc.c
> new file mode 100644
> index 00000000000..7013bf97167
> --- /dev/null
> +++ b/arch/arm/mach-stm32mp/stm32mp1/etzpc.c
> @@ -0,0 +1,194 @@
> +// SPDX-License-Identifier: GPL-2.0-or-later OR BSD-3-Clause
> +/*
> + * Copyright (C) 2023, STMicroelectronics - All Rights Reserved
> + */
> +
> +#define LOG_CATEGORY UCLASS_NOP
> +
> +#include <dm.h>
> +#include <asm/io.h>
> +#include <dm/device.h>
> +#include <dm/device_compat.h>
> +#include <dm/lists.h>
> +#include <linux/bitfield.h>
> +#include <mach/etzpc.h>
> +
> +/* ETZPC peripheral as firewall bus */
> +/* ETZPC registers */
> +#define ETZPC_DECPROT 0x10
> +#define ETZPC_HWCFGR 0x3F0
> +
> +/* ETZPC miscellaneous */
> +#define ETZPC_PROT_MASK GENMASK(1, 0)
> +#define ETZPC_PROT_A7NS 0x3
> +#define ETZPC_DECPROT_SHIFT 1
> +
> +#define IDS_PER_DECPROT_REGS 16
> +
> +#define ETZPC_HWCFGR_NUM_PER_SEC GENMASK(15, 8)
> +#define ETZPC_HWCFGR_NUM_AHB_SEC GENMASK(23, 16)
> +
> +/*
> + * struct stm32_etzpc_plat: Information about ETZPC device
> + *
> + * @base: Base address of ETZPC
> + * @max_entries: Number of securable peripherals in ETZPC
> + */
> +struct stm32_etzpc_plat {
> + void *base;
> + unsigned int max_entries;
> +};
> +
> +static int etzpc_parse_feature_domain(ofnode node, struct
> ofnode_phandle_args *args)
> +{
> + int ret;
> +
> + ret = ofnode_parse_phandle_with_args(node, "access-controllers",
> + "#access-controller-cells", 0,
> + 0, args);
> + if (ret) {
> + log_debug("failed to parse access-controller (%d)\n", ret);
> + return ret;
> + }
> +
> + if (args->args_count != 1) {
> + log_debug("invalid domain args_count: %d\n", args->args_count);
> + return -EINVAL;
> + }
> +
> + return 0;
> +}
> +
> +static int etzpc_check_access(void *base, u32 id)
> +{
> + u32 reg_offset, offset, sec_val;
> +
> + /* Check access configuration, 16 peripherals per register */
> + reg_offset = ETZPC_DECPROT + 0x4 * (id / IDS_PER_DECPROT_REGS);
> + offset = (id % IDS_PER_DECPROT_REGS) << ETZPC_DECPROT_SHIFT;
> +
> + /* Verify peripheral is non-secure and attributed to cortex A7 */
> + sec_val = (readl(base + reg_offset) >> offset) & ETZPC_PROT_MASK;
> + if (sec_val != ETZPC_PROT_A7NS) {
> + log_debug("Invalid bus configuration: reg_offset %#x, value
> %d\n",
> + reg_offset, sec_val);
> + return -EACCES;
> + }
> +
> + return 0;
> +}
> +
> +int stm32_etzpc_check_access_by_id(ofnode device_node, u32 id)
> +{
> + struct stm32_etzpc_plat *plat;
> + struct ofnode_phandle_args args;
> + struct udevice *dev;
> + int err;
> +
> + err = etzpc_parse_feature_domain(device_node, &args);
> + if (err)
> + return err;
> +
> + if (id == -1U)
> + id = args.args[0];
> +
> + err = uclass_get_device_by_ofnode(UCLASS_NOP, args.node, &dev);
> + if (err || dev->driver != DM_DRIVER_GET(stm32_etzpc)) {
> + log_err("No device found\n");
> + return -EINVAL;
> + }
> +
> + plat = dev_get_plat(dev);
> +
> + if (id >= plat->max_entries) {
> + dev_err(dev, "Invalid sys bus ID for %s\n",
> ofnode_get_name(device_node));
> + return -EINVAL;
> + }
> +
> + return etzpc_check_access(plat->base, id);
> +}
> +
> +int stm32_etzpc_check_access(ofnode device_node)
> +{
> + return stm32_etzpc_check_access_by_id(device_node, -1U);
> +}
> +
> +static int stm32_etzpc_bind(struct udevice *dev)
> +{
> + struct stm32_etzpc_plat *plat = dev_get_plat(dev);
> + struct ofnode_phandle_args args;
> + u32 nb_per, nb_master;
> + int ret = 0, err = 0;
> + ofnode node, parent;
> +
> + plat->base = dev_read_addr_ptr(dev);
> + if (!plat->base) {
> + dev_err(dev, "can't get registers base address\n");
> + return -ENOENT;
> + }
> +
> + /* Get number of etzpc entries*/
> + nb_per = FIELD_GET(ETZPC_HWCFGR_NUM_PER_SEC,
> + readl(plat->base + ETZPC_HWCFGR));
> + nb_master = FIELD_GET(ETZPC_HWCFGR_NUM_AHB_SEC,
> + readl(plat->base + ETZPC_HWCFGR));
> + plat->max_entries = nb_per + nb_master;
> +
> + parent = dev_ofnode(dev);
> + for (node = ofnode_first_subnode(parent);
> + ofnode_valid(node);
> + node = ofnode_next_subnode(node)) {
> + const char *node_name = ofnode_get_name(node);
> +
> + if (!ofnode_is_enabled(node))
> + continue;
> +
> + err = etzpc_parse_feature_domain(node, &args);
> + if (err) {
> + dev_err(dev, "%s failed to parse child on bus (%d)\n",
> node_name, err);
> + continue;
> + }
> +
> + if (!ofnode_equal(args.node, parent)) {
> + dev_err(dev, "%s phandle to %s\n",
> + node_name, ofnode_get_name(args.node));
> + continue;
> + }
> +
> + if (args.args[0] >= plat->max_entries) {
> + dev_err(dev, "Invalid sys bus ID for %s\n", node_name);
> + return -EINVAL;
> + }
> +
> + err = etzpc_check_access(plat->base, args.args[0]);
> + if (err) {
> + dev_info(dev, "%s not allowed on bus (%d)\n",
> node_name, err);
> + continue;
> + }
> +
> + err = lists_bind_fdt(dev, node, NULL, NULL,
> + gd->flags & GD_FLG_RELOC ? false : true);
> + if (err) {
> + ret = err;
> + dev_err(dev, "%s failed to bind on bus (%d)\n",
> node_name, ret);
> + }
> + }
> +
> + if (ret)
> + dev_err(dev, "Some child failed to bind (%d)\n", ret);
> +
> + return ret;
> +}
> +
> +static const struct udevice_id stm32_etzpc_ids[] = {
> + { .compatible = "st,stm32-etzpc" },
> + {},
> +};
> +
> +U_BOOT_DRIVER(stm32_etzpc) = {
> + .name = "stm32_etzpc",
> + .id = UCLASS_NOP,
> + .of_match = stm32_etzpc_ids,
> + .bind = stm32_etzpc_bind,
> + .plat_auto = sizeof(struct stm32_etzpc_plat),
> +};
Applied to u-boot-stm32/master
Thanks
Patrice