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

Reply via email to