2018-02-06 12:39:05 +08:00
|
|
|
From 5a5ff01c790d49c0f6fd247f68f2fd9a2128ea91 Mon Sep 17 00:00:00 2001
|
2017-09-27 15:31:31 +08:00
|
|
|
From: Yangbo Lu <yangbo.lu@nxp.com>
|
2018-02-06 12:39:05 +08:00
|
|
|
Date: Wed, 17 Jan 2018 15:36:28 +0800
|
|
|
|
Subject: [PATCH 23/30] irqchip: support layerscape
|
2017-09-27 15:31:31 +08:00
|
|
|
|
2018-02-06 12:39:05 +08:00
|
|
|
This is an integrated patch for layerscape gic support.
|
2017-09-27 15:31:31 +08:00
|
|
|
|
|
|
|
Signed-off-by: Eric Auger <eric.auger@redhat.com>
|
|
|
|
Signed-off-by: Zhao Qiang <qiang.zhao@nxp.com>
|
|
|
|
Signed-off-by: Yangbo Lu <yangbo.lu@nxp.com>
|
|
|
|
---
|
|
|
|
drivers/irqchip/Makefile | 1 +
|
|
|
|
drivers/irqchip/irq-gic-v3-its.c | 1 +
|
|
|
|
include/linux/irqdomain.h | 36 ++++++++++++++++++++++++++++++++++++
|
|
|
|
kernel/irq/irqdomain.c | 39 +++++++++++++++++++++++++++++++++++++++
|
|
|
|
kernel/irq/msi.c | 4 ++--
|
|
|
|
5 files changed, 79 insertions(+), 2 deletions(-)
|
|
|
|
|
|
|
|
--- a/drivers/irqchip/Makefile
|
|
|
|
+++ b/drivers/irqchip/Makefile
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -74,3 +74,4 @@ obj-$(CONFIG_LS_SCFG_MSI) += irq-ls-scf
|
2017-09-27 15:31:31 +08:00
|
|
|
obj-$(CONFIG_EZNPS_GIC) += irq-eznps.o
|
|
|
|
obj-$(CONFIG_ARCH_ASPEED) += irq-aspeed-vic.o
|
|
|
|
obj-$(CONFIG_STM32_EXTI) += irq-stm32-exti.o
|
|
|
|
+obj-$(CONFIG_QUICC_ENGINE) += irq-qeic.o
|
|
|
|
--- a/drivers/irqchip/irq-gic-v3-its.c
|
|
|
|
+++ b/drivers/irqchip/irq-gic-v3-its.c
|
2019-01-31 16:36:37 +01:00
|
|
|
@@ -1659,6 +1659,7 @@ static int its_init_domain(struct fwnode
|
2017-09-27 15:31:31 +08:00
|
|
|
|
|
|
|
inner_domain->parent = its_parent;
|
|
|
|
inner_domain->bus_token = DOMAIN_BUS_NEXUS;
|
|
|
|
+ inner_domain->flags |= IRQ_DOMAIN_FLAG_MSI_REMAP;
|
|
|
|
info->ops = &its_msi_domain_ops;
|
|
|
|
info->data = its;
|
|
|
|
inner_domain->host_data = info;
|
|
|
|
--- a/include/linux/irqdomain.h
|
|
|
|
+++ b/include/linux/irqdomain.h
|
|
|
|
@@ -183,6 +183,12 @@ enum {
|
|
|
|
/* Irq domain is an IPI domain with single virq */
|
|
|
|
IRQ_DOMAIN_FLAG_IPI_SINGLE = (1 << 3),
|
|
|
|
|
|
|
|
+ /* Irq domain implements MSIs */
|
|
|
|
+ IRQ_DOMAIN_FLAG_MSI = (1 << 4),
|
|
|
|
+
|
|
|
|
+ /* Irq domain implements MSI remapping */
|
|
|
|
+ IRQ_DOMAIN_FLAG_MSI_REMAP = (1 << 5),
|
|
|
|
+
|
|
|
|
/*
|
|
|
|
* Flags starting from IRQ_DOMAIN_FLAG_NONCORE are reserved
|
|
|
|
* for implementation specific purposes and ignored by the
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -216,6 +222,7 @@ struct irq_domain *irq_domain_add_legacy
|
2017-09-27 15:31:31 +08:00
|
|
|
void *host_data);
|
|
|
|
extern struct irq_domain *irq_find_matching_fwspec(struct irq_fwspec *fwspec,
|
|
|
|
enum irq_domain_bus_token bus_token);
|
|
|
|
+extern bool irq_domain_check_msi_remap(void);
|
|
|
|
extern void irq_set_default_host(struct irq_domain *host);
|
|
|
|
extern int irq_domain_alloc_descs(int virq, unsigned int nr_irqs,
|
|
|
|
irq_hw_number_t hwirq, int node,
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -446,6 +453,19 @@ static inline bool irq_domain_is_ipi_sin
|
2017-09-27 15:31:31 +08:00
|
|
|
{
|
|
|
|
return domain->flags & IRQ_DOMAIN_FLAG_IPI_SINGLE;
|
|
|
|
}
|
|
|
|
+
|
|
|
|
+static inline bool irq_domain_is_msi(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ return domain->flags & IRQ_DOMAIN_FLAG_MSI;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static inline bool irq_domain_is_msi_remap(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ return domain->flags & IRQ_DOMAIN_FLAG_MSI_REMAP;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+extern bool irq_domain_hierarchical_is_msi_remap(struct irq_domain *domain);
|
|
|
|
+
|
|
|
|
#else /* CONFIG_IRQ_DOMAIN_HIERARCHY */
|
|
|
|
static inline void irq_domain_activate_irq(struct irq_data *data) { }
|
|
|
|
static inline void irq_domain_deactivate_irq(struct irq_data *data) { }
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -477,6 +497,22 @@ static inline bool irq_domain_is_ipi_sin
|
2017-09-27 15:31:31 +08:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
+
|
|
|
|
+static inline bool irq_domain_is_msi(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static inline bool irq_domain_is_msi_remap(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
+
|
|
|
|
+static inline bool
|
|
|
|
+irq_domain_hierarchical_is_msi_remap(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
#endif /* CONFIG_IRQ_DOMAIN_HIERARCHY */
|
|
|
|
|
|
|
|
#else /* CONFIG_IRQ_DOMAIN */
|
|
|
|
--- a/kernel/irq/irqdomain.c
|
|
|
|
+++ b/kernel/irq/irqdomain.c
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -278,6 +278,31 @@ struct irq_domain *irq_find_matching_fws
|
2017-09-27 15:31:31 +08:00
|
|
|
EXPORT_SYMBOL_GPL(irq_find_matching_fwspec);
|
|
|
|
|
2017-10-08 16:34:37 +03:00
|
|
|
/**
|
2017-09-27 15:31:31 +08:00
|
|
|
+ * irq_domain_check_msi_remap - Check whether all MSI irq domains implement
|
|
|
|
+ * IRQ remapping
|
|
|
|
+ *
|
|
|
|
+ * Return: false if any MSI irq domain does not support IRQ remapping,
|
|
|
|
+ * true otherwise (including if there is no MSI irq domain)
|
|
|
|
+ */
|
|
|
|
+bool irq_domain_check_msi_remap(void)
|
|
|
|
+{
|
|
|
|
+ struct irq_domain *h;
|
|
|
|
+ bool ret = true;
|
|
|
|
+
|
|
|
|
+ mutex_lock(&irq_domain_mutex);
|
|
|
|
+ list_for_each_entry(h, &irq_domain_list, link) {
|
|
|
|
+ if (irq_domain_is_msi(h) &&
|
|
|
|
+ !irq_domain_hierarchical_is_msi_remap(h)) {
|
|
|
|
+ ret = false;
|
|
|
|
+ break;
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+ mutex_unlock(&irq_domain_mutex);
|
|
|
|
+ return ret;
|
|
|
|
+}
|
|
|
|
+EXPORT_SYMBOL_GPL(irq_domain_check_msi_remap);
|
|
|
|
+
|
2017-10-08 16:34:37 +03:00
|
|
|
+/**
|
2017-09-27 15:31:31 +08:00
|
|
|
* irq_set_default_host() - Set a "default" irq domain
|
|
|
|
* @domain: default domain pointer
|
2017-10-08 16:34:37 +03:00
|
|
|
*
|
|
|
|
@@ -1408,6 +1433,20 @@ static void irq_domain_check_hierarchy(s
|
2017-09-27 15:31:31 +08:00
|
|
|
if (domain->ops->alloc)
|
|
|
|
domain->flags |= IRQ_DOMAIN_FLAG_HIERARCHY;
|
|
|
|
}
|
|
|
|
+
|
|
|
|
+/**
|
|
|
|
+ * irq_domain_hierarchical_is_msi_remap - Check if the domain or any
|
|
|
|
+ * parent has MSI remapping support
|
|
|
|
+ * @domain: domain pointer
|
|
|
|
+ */
|
|
|
|
+bool irq_domain_hierarchical_is_msi_remap(struct irq_domain *domain)
|
|
|
|
+{
|
|
|
|
+ for (; domain; domain = domain->parent) {
|
|
|
|
+ if (irq_domain_is_msi_remap(domain))
|
|
|
|
+ return true;
|
|
|
|
+ }
|
|
|
|
+ return false;
|
|
|
|
+}
|
|
|
|
#else /* CONFIG_IRQ_DOMAIN_HIERARCHY */
|
|
|
|
/**
|
|
|
|
* irq_domain_get_irq_data - Get irq_data associated with @virq and @domain
|
|
|
|
--- a/kernel/irq/msi.c
|
|
|
|
+++ b/kernel/irq/msi.c
|
2017-10-08 16:34:37 +03:00
|
|
|
@@ -272,8 +272,8 @@ struct irq_domain *msi_create_irq_domain
|
2017-09-27 15:31:31 +08:00
|
|
|
if (info->flags & MSI_FLAG_USE_DEF_CHIP_OPS)
|
|
|
|
msi_domain_update_chip_ops(info);
|
|
|
|
|
|
|
|
- return irq_domain_create_hierarchy(parent, 0, 0, fwnode,
|
|
|
|
- &msi_domain_ops, info);
|
|
|
|
+ return irq_domain_create_hierarchy(parent, IRQ_DOMAIN_FLAG_MSI, 0,
|
|
|
|
+ fwnode, &msi_domain_ops, info);
|
|
|
|
}
|
|
|
|
|
|
|
|
int msi_domain_prepare_irqs(struct irq_domain *domain, struct device *dev,
|