Interrupt controller eiointc routes irq to cpu interface IP0 - IP7,
now it is hard-coded that eiointc routes irq to CPU started from IP1,
however with function irq_create_mapping() parameter parent hwirq
uses irq parsed from ACPI or DTS table.
Routed interrupt pin need be the consistent with parent hwirq.
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
---
drivers/irqchip/irq-loongson-eiointc.c | 19 ++++++++++++++++---
1 file changed, 16 insertions(+), 3 deletions(-)
diff --git a/drivers/irqchip/irq-loongson-eiointc.c b/drivers/irqchip/irq-loongson-eiointc.c
index b2860eb2d32c..3e987d1232d2 100644
--- a/drivers/irqchip/irq-loongson-eiointc.c
+++ b/drivers/irqchip/irq-loongson-eiointc.c
@@ -68,6 +68,7 @@ struct eiointc_priv {
struct fwnode_handle *domain_handle;
struct irq_domain *eiointc_domain;
int flags;
+ irq_hw_number_t parent_hwirq;
};
static struct eiointc_priv *eiointc_priv[MAX_IO_PICS];
@@ -211,7 +212,12 @@ static int eiointc_router_init(unsigned int cpu)
}
for (i = 0; i < eiointc_priv[0]->vec_count / 32 / 4; i++) {
- bit = BIT(1 + index); /* Route to IP[1 + index] */
+ /*
+ * Route to interrupt pin, relative offset used here
+ * Offset 0 means routing to IP0 and so on
+ * Every 32 vector routing to one interrupt pin
+ */
+ bit = BIT(eiointc_priv[index]->parent_hwirq - INT_HWI0);
data = bit | (bit << 8) | (bit << 16) | (bit << 24);
iocsr_write32(data, EIOINTC_REG_IPMAP + i * 4);
}
@@ -495,7 +501,7 @@ int __init eiointc_acpi_init(struct irq_domain *parent,
priv->vec_count = VEC_COUNT;
priv->node = acpi_eiointc->node;
-
+ priv->parent_hwirq = acpi_eiointc->cascade;
parent_irq = irq_create_mapping(parent, acpi_eiointc->cascade);
ret = eiointc_init(priv, parent_irq, acpi_eiointc->node_map);
@@ -529,6 +535,7 @@ static int __init eiointc_of_init(struct device_node *of_node,
{
int parent_irq, ret;
struct eiointc_priv *priv;
+ struct irq_data *irq_data;
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv)
@@ -544,6 +551,12 @@ static int __init eiointc_of_init(struct device_node *of_node,
if (ret < 0)
goto out_free_priv;
+ irq_data = irq_get_irq_data(parent_irq);
+ if (!irq_data) {
+ ret = -ENODEV;
+ goto out_free_priv;
+ }
+
/*
* In particular, the number of devices supported by the LS2K0500
* extended I/O interrupt vector is 128.
@@ -552,7 +565,7 @@ static int __init eiointc_of_init(struct device_node *of_node,
priv->vec_count = 128;
else
priv->vec_count = VEC_COUNT;
-
+ priv->parent_hwirq = irqd_to_hwirq(irq_data);
priv->node = 0;
priv->domain_handle = of_fwnode_handle(of_node);
--
2.39.3
Hi, Mingcong, On Mon, Aug 4, 2025 at 4:19 PM Bibo Mao <maobibo@loongson.cn> wrote: > > Interrupt controller eiointc routes irq to cpu interface IP0 - IP7, > now it is hard-coded that eiointc routes irq to CPU started from IP1, > however with function irq_create_mapping() parameter parent hwirq > uses irq parsed from ACPI or DTS table. > > Routed interrupt pin need be the consistent with parent hwirq. I remember that you want to test as many machine types as possible, have you completed it? Huacai > > Signed-off-by: Bibo Mao <maobibo@loongson.cn> > --- > drivers/irqchip/irq-loongson-eiointc.c | 19 ++++++++++++++++--- > 1 file changed, 16 insertions(+), 3 deletions(-) > > diff --git a/drivers/irqchip/irq-loongson-eiointc.c b/drivers/irqchip/irq-loongson-eiointc.c > index b2860eb2d32c..3e987d1232d2 100644 > --- a/drivers/irqchip/irq-loongson-eiointc.c > +++ b/drivers/irqchip/irq-loongson-eiointc.c > @@ -68,6 +68,7 @@ struct eiointc_priv { > struct fwnode_handle *domain_handle; > struct irq_domain *eiointc_domain; > int flags; > + irq_hw_number_t parent_hwirq; > }; > > static struct eiointc_priv *eiointc_priv[MAX_IO_PICS]; > @@ -211,7 +212,12 @@ static int eiointc_router_init(unsigned int cpu) > } > > for (i = 0; i < eiointc_priv[0]->vec_count / 32 / 4; i++) { > - bit = BIT(1 + index); /* Route to IP[1 + index] */ > + /* > + * Route to interrupt pin, relative offset used here > + * Offset 0 means routing to IP0 and so on > + * Every 32 vector routing to one interrupt pin > + */ > + bit = BIT(eiointc_priv[index]->parent_hwirq - INT_HWI0); > data = bit | (bit << 8) | (bit << 16) | (bit << 24); > iocsr_write32(data, EIOINTC_REG_IPMAP + i * 4); > } > @@ -495,7 +501,7 @@ int __init eiointc_acpi_init(struct irq_domain *parent, > > priv->vec_count = VEC_COUNT; > priv->node = acpi_eiointc->node; > - > + priv->parent_hwirq = acpi_eiointc->cascade; > parent_irq = irq_create_mapping(parent, acpi_eiointc->cascade); > > ret = eiointc_init(priv, parent_irq, acpi_eiointc->node_map); > @@ -529,6 +535,7 @@ static int __init eiointc_of_init(struct device_node *of_node, > { > int parent_irq, ret; > struct eiointc_priv *priv; > + struct irq_data *irq_data; > > priv = kzalloc(sizeof(*priv), GFP_KERNEL); > if (!priv) > @@ -544,6 +551,12 @@ static int __init eiointc_of_init(struct device_node *of_node, > if (ret < 0) > goto out_free_priv; > > + irq_data = irq_get_irq_data(parent_irq); > + if (!irq_data) { > + ret = -ENODEV; > + goto out_free_priv; > + } > + > /* > * In particular, the number of devices supported by the LS2K0500 > * extended I/O interrupt vector is 128. > @@ -552,7 +565,7 @@ static int __init eiointc_of_init(struct device_node *of_node, > priv->vec_count = 128; > else > priv->vec_count = VEC_COUNT; > - > + priv->parent_hwirq = irqd_to_hwirq(irq_data); > priv->node = 0; > priv->domain_handle = of_fwnode_handle(of_node); > > -- > 2.39.3 >
The following commit has been merged into the irq/drivers branch of tip:
Commit-ID: 7fb83eb664e9b3a0438dd28859e9f0fd49d4c165
Gitweb: https://git.kernel.org/tip/7fb83eb664e9b3a0438dd28859e9f0fd49d4c165
Author: Bibo Mao <maobibo@loongson.cn>
AuthorDate: Mon, 04 Aug 2025 16:19:45 +08:00
Committer: Thomas Gleixner <tglx@linutronix.de>
CommitterDate: Sun, 24 Aug 2025 12:51:04 +02:00
irqchip/loongson-eiointc: Route interrupt parsed from bios table
Interrupt controller eiointc routes interrupts to CPU interface IP0 - IP7.
It is currently hard-coded that eiointc routes interrupts to the CPU
starting from IP1, but it should base that decision on the parent
interrupt, which is provided by ACPI or DTS.
Retrieve the parent's hardware interrupt number and store it in the
descriptor of the eointc instance, so that the routing function can utilize
it for the correct route settings.
[ tglx: Massaged change log ]
Signed-off-by: Bibo Mao <maobibo@loongson.cn>
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Link: https://lore.kernel.org/all/20250804081946.1456573-2-maobibo@loongson.cn
---
drivers/irqchip/irq-loongson-eiointc.c | 21 +++++++++++++++++----
1 file changed, 17 insertions(+), 4 deletions(-)
diff --git a/drivers/irqchip/irq-loongson-eiointc.c b/drivers/irqchip/irq-loongson-eiointc.c
index b2860eb..baa4069 100644
--- a/drivers/irqchip/irq-loongson-eiointc.c
+++ b/drivers/irqchip/irq-loongson-eiointc.c
@@ -68,6 +68,7 @@ struct eiointc_priv {
struct fwnode_handle *domain_handle;
struct irq_domain *eiointc_domain;
int flags;
+ irq_hw_number_t parent_hwirq;
};
static struct eiointc_priv *eiointc_priv[MAX_IO_PICS];
@@ -211,7 +212,12 @@ static int eiointc_router_init(unsigned int cpu)
}
for (i = 0; i < eiointc_priv[0]->vec_count / 32 / 4; i++) {
- bit = BIT(1 + index); /* Route to IP[1 + index] */
+ /*
+ * Route to interrupt pin, relative offset used here
+ * Offset 0 means routing to IP0 and so on
+ * Every 32 vector routing to one interrupt pin
+ */
+ bit = BIT(eiointc_priv[index]->parent_hwirq - INT_HWI0);
data = bit | (bit << 8) | (bit << 16) | (bit << 24);
iocsr_write32(data, EIOINTC_REG_IPMAP + i * 4);
}
@@ -495,7 +501,7 @@ int __init eiointc_acpi_init(struct irq_domain *parent,
priv->vec_count = VEC_COUNT;
priv->node = acpi_eiointc->node;
-
+ priv->parent_hwirq = acpi_eiointc->cascade;
parent_irq = irq_create_mapping(parent, acpi_eiointc->cascade);
ret = eiointc_init(priv, parent_irq, acpi_eiointc->node_map);
@@ -527,8 +533,9 @@ out_free_priv:
static int __init eiointc_of_init(struct device_node *of_node,
struct device_node *parent)
{
- int parent_irq, ret;
struct eiointc_priv *priv;
+ struct irq_data *irq_data;
+ int parent_irq, ret;
priv = kzalloc(sizeof(*priv), GFP_KERNEL);
if (!priv)
@@ -544,6 +551,12 @@ static int __init eiointc_of_init(struct device_node *of_node,
if (ret < 0)
goto out_free_priv;
+ irq_data = irq_get_irq_data(parent_irq);
+ if (!irq_data) {
+ ret = -ENODEV;
+ goto out_free_priv;
+ }
+
/*
* In particular, the number of devices supported by the LS2K0500
* extended I/O interrupt vector is 128.
@@ -552,7 +565,7 @@ static int __init eiointc_of_init(struct device_node *of_node,
priv->vec_count = 128;
else
priv->vec_count = VEC_COUNT;
-
+ priv->parent_hwirq = irqd_to_hwirq(irq_data);
priv->node = 0;
priv->domain_handle = of_fwnode_handle(of_node);
© 2016 - 2025 Red Hat, Inc.