[PATCH 12/16] irqchip/eip201-aic: Add support for Safexcel EIP-201 AIC
From: Miquel Raynal (Schneider Electric)
Date: Fri Mar 27 2026 - 16:20:01 EST
Describe the EIP-201 Advanced Interrupt Controller from Inside Secure,
typically found in a bigger block named EIP-150. This controller is
rather simple and is driven using the generic irqchip model. Its
own interrupt domain is limited to just a few interrupts connected to
other inner blocks, such as a Random Number Generator and a Public Key
Accelerator.
The one I used receives only rising edge interrupts and uses its own
logic to track them. It is theoretically possible to wire devices with
level interrupts, but not in the context of the EIP-150.
Signed-off-by: Miquel Raynal (Schneider Electric) <miquel.raynal@xxxxxxxxxxx>
---
drivers/irqchip/Kconfig | 8 ++
drivers/irqchip/Makefile | 1 +
drivers/irqchip/irq-eip201-aic.c | 221 +++++++++++++++++++++++++++++++++++++++
3 files changed, 230 insertions(+)
diff --git a/drivers/irqchip/Kconfig b/drivers/irqchip/Kconfig
index f07b00d7fef9..b098bb00a224 100644
--- a/drivers/irqchip/Kconfig
+++ b/drivers/irqchip/Kconfig
@@ -826,4 +826,12 @@ config SUNPLUS_SP7021_INTC
chained controller, routing all interrupt source in P-Chip to
the primary controller on C-Chip.
+config SAFEXCEL_EIP201_AIC
+ tristate "Safexcel EIP201 AIC"
+ select IRQ_DOMAIN
+ help
+ Support for the Advanced Interrupt Controller (AIC) typically
+ inside Safexcel EIP150 IPs, gathering Public Key Accelerator
+ and True Random Number Generator interrupts.
+
endmenu
diff --git a/drivers/irqchip/Makefile b/drivers/irqchip/Makefile
index 26aa3b6ec99f..80784a02f4a8 100644
--- a/drivers/irqchip/Makefile
+++ b/drivers/irqchip/Makefile
@@ -136,3 +136,4 @@ obj-$(CONFIG_APPLE_AIC) += irq-apple-aic.o
obj-$(CONFIG_MCHP_EIC) += irq-mchp-eic.o
obj-$(CONFIG_SOPHGO_SG2042_MSI) += irq-sg2042-msi.o
obj-$(CONFIG_SUNPLUS_SP7021_INTC) += irq-sp7021-intc.o
+obj-$(CONFIG_SAFEXCEL_EIP201_AIC) += irq-eip201-aic.o
diff --git a/drivers/irqchip/irq-eip201-aic.c b/drivers/irqchip/irq-eip201-aic.c
new file mode 100644
index 000000000000..514fd39e2777
--- /dev/null
+++ b/drivers/irqchip/irq-eip201-aic.c
@@ -0,0 +1,221 @@
+// SPDX-License-Identifier: GPL-2.0
+/*
+ * Copyright (C) 2026 Schneider Electric
+ * Authored by Miquel Raynal <miquel.raynal@xxxxxxxxxxx>
+ * Based on the work from Mathieu Hadjimegrian <mathieu.hadjimegrian@xxxxxxxxxx>
+ */
+
+#include "linux/irq.h"
+#include "linux/stddef.h"
+#include <linux/bitops.h>
+#include <linux/clk.h>
+#include <linux/init.h>
+#include <linux/interrupt.h>
+#include <linux/io.h>
+#include <linux/irqdomain.h>
+#include <linux/platform_device.h>
+
+#include <dt-bindings/interrupt-controller/inside-secure,safexcel-eip201.h>
+
+#define EIP201_AIC_POL_CTRL 0x0 /* RO */
+#define EIP201_AIC_POL_LOW_FALLING 0
+#define EIP201_AIC_POL_HIGH_RISING 1
+
+#define EIP201_AIC_TYP_CTRL 0x4 /* RO */
+#define EIP201_AIC_TYP_LEVEL 0
+#define EIP201_AIC_TYP_EDGE 1
+
+#define EIP201_AIC_ENABLE_SET 0xC /* WO */
+#define EIP201_AIC_ENABLED_STAT 0x10 /* RO */
+#define EIP201_AIC_ENABLE_CLR 0x14 /* WO */
+#define EIP201_AIC_ACK 0x10 /* WO */
+
+#define EIP201_AIC_REVISION 0x3FFC
+#define EIP201_AIC_REV_NUM(reg) FIELD_GET(GENMASK(7, 0), reg)
+#define EIP201_AIC_REV_COMP_NUM(reg) FIELD_GET(GENMASK(15, 8), reg)
+
+#define EIP201_AIC_INT(reg, int) field_get(BIT(int), reg)
+#define EIP201_AIC_NINT 7
+#define EIP201_AIC_INT_MASK (BIT(EIP201_AIC_NINT) - 1)
+
+struct eip201_aic {
+ struct device *dev;
+ void __iomem *regs;
+ struct irq_domain *domain;
+ struct irq_chip_generic *gc;
+ u32 type;
+ u32 pol;
+};
+
+static struct eip201_aic *irq_domain_to_aic(struct irq_domain *d)
+{
+ return d->host_data;
+}
+
+static int eip201_aic_irq_domain_xlate(struct irq_domain *d, struct device_node *ctrlr,
+ const u32 *intspec, unsigned int intsize,
+ irq_hw_number_t *out_hwirq, unsigned int *out_type)
+{
+ struct eip201_aic *aic = irq_domain_to_aic(d);
+ int ret;
+
+ ret = irq_domain_xlate_twocell(d, ctrlr, intspec, intsize, out_hwirq, out_type);
+ if (ret)
+ return ret;
+
+ /* One interrupt is reserved, two are for Inside Secure debugging purposes only */
+ switch (*out_hwirq) {
+ case AIC_PKA_INT0:
+ case AIC_PKA_INT1:
+ case AIC_RESERVED:
+ return -EINVAL;
+ default:
+ break;
+ }
+
+ /*
+ * Flow type is implementation specific and hardcoded, make sure it is corect,
+ * even though the documentation says EIP blocks generate edge interrupts.
+ */
+
+ /* Type register indicates:
+ * - '1' for edge interrupts
+ * - '0' for level interrupts
+ */
+ if (*out_type & IRQ_TYPE_LEVEL_MASK &&
+ EIP201_AIC_INT(aic->type, *out_hwirq))
+ return -EINVAL;
+
+ /*
+ * Polarity register indicates:
+ * - '1' for level high or rising edge
+ * - '0' for level low or falling edge
+ */
+ if (*out_type & (IRQ_TYPE_LEVEL_LOW | IRQ_TYPE_EDGE_FALLING) &&
+ EIP201_AIC_INT(aic->pol, *out_hwirq))
+ return -EINVAL;
+
+ return 0;
+}
+
+const struct irq_domain_ops eip201_aic_chip_ops = {
+ .map = irq_map_generic_chip,
+ .unmap = irq_unmap_generic_chip,
+ .xlate = eip201_aic_irq_domain_xlate,
+};
+
+static irqreturn_t eip201_aic_irq_handler(int irq, void *dev_id)
+{
+ struct eip201_aic *aic = dev_id;
+ unsigned long pending;
+ irq_hw_number_t hwirq;
+
+ pending = readl(aic->regs + EIP201_AIC_ENABLED_STAT);
+ if (!pending)
+ return IRQ_NONE;
+
+ /* Ack interrupts ASAP to decrease the likelyhood of missing an edge one */
+ writel(pending, aic->regs + EIP201_AIC_ACK);
+
+ for_each_set_bit(hwirq, &pending, EIP201_AIC_NINT)
+ generic_handle_domain_irq(aic->domain, hwirq);
+
+ return IRQ_HANDLED;
+}
+
+static int eip201_aic_probe(struct platform_device *pdev)
+{
+ struct eip201_aic *aic;
+ struct clk *clk;
+ u32 rev;
+ int irq;
+ int ret;
+
+ aic = devm_kzalloc(&pdev->dev, sizeof(*aic), GFP_KERNEL);
+ if (!aic)
+ return -ENOMEM;
+
+ aic->regs = devm_platform_ioremap_resource(pdev, 0);
+ if (!aic->regs)
+ return -EINVAL;
+
+ clk = devm_clk_get_enabled(&pdev->dev, NULL);
+ if (IS_ERR(clk))
+ return PTR_ERR(clk);
+
+ rev = readl(aic->regs + EIP201_AIC_REVISION);
+ if (!(EIP201_AIC_REV_NUM(rev) ^ EIP201_AIC_REV_COMP_NUM(rev)))
+ return -ENXIO;
+
+ platform_set_drvdata(pdev, aic);
+ aic->dev = &pdev->dev;
+
+ /* Cache the RO type and polarity of all interrupts */
+ aic->type = readl(aic->regs + EIP201_AIC_TYP_CTRL);
+ aic->pol = readl(aic->regs + EIP201_AIC_POL_CTRL);
+
+ /* Disable/clear all interrupts */
+ writel(EIP201_AIC_INT_MASK, aic->regs + EIP201_AIC_ENABLE_CLR);
+ writel(EIP201_AIC_INT_MASK, aic->regs + EIP201_AIC_ACK);
+
+ aic->domain = irq_domain_create_linear(dev_fwnode(&pdev->dev), EIP201_AIC_NINT,
+ &eip201_aic_chip_ops, aic);
+ if (!aic->domain)
+ return -ENXIO;
+
+ ret = irq_alloc_domain_generic_chips(aic->domain, EIP201_AIC_NINT, 1, "eip201-aic",
+ handle_edge_irq, 0, 0, 0);
+ if (ret)
+ goto remove_domain;
+
+ aic->gc = irq_get_domain_generic_chip(aic->domain, 0);
+ aic->gc->reg_base = aic->regs;
+ aic->gc->chip_types[0].regs.ack = EIP201_AIC_ACK;
+ aic->gc->chip_types[0].regs.enable = EIP201_AIC_ENABLE_SET;
+ aic->gc->chip_types[0].regs.disable = EIP201_AIC_ENABLE_CLR;
+ aic->gc->chip_types[0].chip.irq_ack = irq_gc_ack_set_bit;
+ aic->gc->chip_types[0].chip.irq_mask = irq_gc_mask_disable_reg;
+ aic->gc->chip_types[0].chip.irq_unmask = irq_gc_unmask_enable_reg;
+ aic->gc->chip_types[0].chip.flags = IRQCHIP_MASK_ON_SUSPEND;
+
+ irq = platform_get_irq(pdev, 0);
+ if (irq < 0)
+ return irq;
+
+ ret = devm_request_irq(&pdev->dev, irq, eip201_aic_irq_handler, 0,
+ dev_name(&pdev->dev), aic);
+ if (ret < 0)
+ goto remove_gc;
+
+ return 0;
+
+remove_gc:
+ irq_remove_generic_chip(aic->gc, EIP201_AIC_INT_MASK, 0, 0);
+remove_domain:
+ irq_domain_remove(aic->domain);
+
+ return ret;
+}
+
+static void eip201_aic_remove(struct platform_device *pdev)
+{
+ struct eip201_aic *aic = platform_get_drvdata(pdev);
+
+ irq_remove_generic_chip(aic->gc, EIP201_AIC_INT_MASK, 0, 0);
+ irq_domain_remove(aic->domain);
+}
+
+static const struct of_device_id eip201_aic_of_match[] = {
+ { .compatible = "inside-secure,safexcel-eip201", },
+ {},
+};
+
+static struct platform_driver eip201_aic_driver = {
+ .probe = eip201_aic_probe,
+ .remove = eip201_aic_remove,
+ .driver = {
+ .name = "safexcel-eip201-aic",
+ .of_match_table = eip201_aic_of_match,
+ },
+};
+module_platform_driver(eip201_aic_driver);
--
2.51.1