|
@@ -47,6 +47,7 @@
|
|
|
#include <asm/irq.h>
|
|
|
#include <asm/exception.h>
|
|
|
#include <asm/smp_plat.h>
|
|
|
+#include <asm/virt.h>
|
|
|
|
|
|
#include "irq-gic-common.h"
|
|
|
|
|
@@ -82,6 +83,8 @@ static DEFINE_RAW_SPINLOCK(irq_controller_lock);
|
|
|
#define NR_GIC_CPU_IF 8
|
|
|
static u8 gic_cpu_map[NR_GIC_CPU_IF] __read_mostly;
|
|
|
|
|
|
+static struct static_key supports_deactivate = STATIC_KEY_INIT_TRUE;
|
|
|
+
|
|
|
#ifndef MAX_GIC_NR
|
|
|
#define MAX_GIC_NR 1
|
|
|
#endif
|
|
@@ -157,6 +160,11 @@ static void gic_mask_irq(struct irq_data *d)
|
|
|
gic_poke_irq(d, GIC_DIST_ENABLE_CLEAR);
|
|
|
}
|
|
|
|
|
|
+static void gic_eoimode1_mask_irq(struct irq_data *d)
|
|
|
+{
|
|
|
+ gic_mask_irq(d);
|
|
|
+}
|
|
|
+
|
|
|
static void gic_unmask_irq(struct irq_data *d)
|
|
|
{
|
|
|
gic_poke_irq(d, GIC_DIST_ENABLE_SET);
|
|
@@ -167,6 +175,11 @@ static void gic_eoi_irq(struct irq_data *d)
|
|
|
writel_relaxed(gic_irq(d), gic_cpu_base(d) + GIC_CPU_EOI);
|
|
|
}
|
|
|
|
|
|
+static void gic_eoimode1_eoi_irq(struct irq_data *d)
|
|
|
+{
|
|
|
+ writel_relaxed(gic_irq(d), gic_cpu_base(d) + GIC_CPU_DEACTIVATE);
|
|
|
+}
|
|
|
+
|
|
|
static int gic_irq_set_irqchip_state(struct irq_data *d,
|
|
|
enum irqchip_irq_state which, bool val)
|
|
|
{
|
|
@@ -272,11 +285,15 @@ static void __exception_irq_entry gic_handle_irq(struct pt_regs *regs)
|
|
|
irqnr = irqstat & GICC_IAR_INT_ID_MASK;
|
|
|
|
|
|
if (likely(irqnr > 15 && irqnr < 1021)) {
|
|
|
+ if (static_key_true(&supports_deactivate))
|
|
|
+ writel_relaxed(irqstat, cpu_base + GIC_CPU_EOI);
|
|
|
handle_domain_irq(gic->domain, irqnr, regs);
|
|
|
continue;
|
|
|
}
|
|
|
if (irqnr < 16) {
|
|
|
writel_relaxed(irqstat, cpu_base + GIC_CPU_EOI);
|
|
|
+ if (static_key_true(&supports_deactivate))
|
|
|
+ writel_relaxed(irqstat, cpu_base + GIC_CPU_DEACTIVATE);
|
|
|
#ifdef CONFIG_SMP
|
|
|
handle_IPI(irqnr, regs);
|
|
|
#endif
|
|
@@ -329,6 +346,22 @@ static struct irq_chip gic_chip = {
|
|
|
IRQCHIP_MASK_ON_SUSPEND,
|
|
|
};
|
|
|
|
|
|
+static struct irq_chip gic_eoimode1_chip = {
|
|
|
+ .name = "GICv2",
|
|
|
+ .irq_mask = gic_eoimode1_mask_irq,
|
|
|
+ .irq_unmask = gic_unmask_irq,
|
|
|
+ .irq_eoi = gic_eoimode1_eoi_irq,
|
|
|
+ .irq_set_type = gic_set_type,
|
|
|
+#ifdef CONFIG_SMP
|
|
|
+ .irq_set_affinity = gic_set_affinity,
|
|
|
+#endif
|
|
|
+ .irq_get_irqchip_state = gic_irq_get_irqchip_state,
|
|
|
+ .irq_set_irqchip_state = gic_irq_set_irqchip_state,
|
|
|
+ .flags = IRQCHIP_SET_TYPE_MASKED |
|
|
|
+ IRQCHIP_SKIP_SET_WAKE |
|
|
|
+ IRQCHIP_MASK_ON_SUSPEND,
|
|
|
+};
|
|
|
+
|
|
|
void __init gic_cascade_irq(unsigned int gic_nr, unsigned int irq)
|
|
|
{
|
|
|
if (gic_nr >= MAX_GIC_NR)
|
|
@@ -360,6 +393,10 @@ static void gic_cpu_if_up(struct gic_chip_data *gic)
|
|
|
{
|
|
|
void __iomem *cpu_base = gic_data_cpu_base(gic);
|
|
|
u32 bypass = 0;
|
|
|
+ u32 mode = 0;
|
|
|
+
|
|
|
+ if (static_key_true(&supports_deactivate))
|
|
|
+ mode = GIC_CPU_CTRL_EOImodeNS;
|
|
|
|
|
|
/*
|
|
|
* Preserve bypass disable bits to be written back later
|
|
@@ -367,7 +404,7 @@ static void gic_cpu_if_up(struct gic_chip_data *gic)
|
|
|
bypass = readl(cpu_base + GIC_CPU_CTRL);
|
|
|
bypass &= GICC_DIS_BYPASS_MASK;
|
|
|
|
|
|
- writel_relaxed(bypass | GICC_ENABLE, cpu_base + GIC_CPU_CTRL);
|
|
|
+ writel_relaxed(bypass | mode | GICC_ENABLE, cpu_base + GIC_CPU_CTRL);
|
|
|
}
|
|
|
|
|
|
|
|
@@ -803,13 +840,20 @@ void __init gic_init_physaddr(struct device_node *node)
|
|
|
static int gic_irq_domain_map(struct irq_domain *d, unsigned int irq,
|
|
|
irq_hw_number_t hw)
|
|
|
{
|
|
|
+ struct irq_chip *chip = &gic_chip;
|
|
|
+
|
|
|
+ if (static_key_true(&supports_deactivate)) {
|
|
|
+ if (d->host_data == (void *)&gic_data[0])
|
|
|
+ chip = &gic_eoimode1_chip;
|
|
|
+ }
|
|
|
+
|
|
|
if (hw < 32) {
|
|
|
irq_set_percpu_devid(irq);
|
|
|
- irq_domain_set_info(d, irq, hw, &gic_chip, d->host_data,
|
|
|
+ irq_domain_set_info(d, irq, hw, chip, d->host_data,
|
|
|
handle_percpu_devid_irq, NULL, NULL);
|
|
|
set_irq_flags(irq, IRQF_VALID | IRQF_NOAUTOEN);
|
|
|
} else {
|
|
|
- irq_domain_set_info(d, irq, hw, &gic_chip, d->host_data,
|
|
|
+ irq_domain_set_info(d, irq, hw, chip, d->host_data,
|
|
|
handle_fasteoi_irq, NULL, NULL);
|
|
|
set_irq_flags(irq, IRQF_VALID | IRQF_PROBE);
|
|
|
}
|
|
@@ -995,6 +1039,8 @@ void __init gic_init_bases(unsigned int gic_nr, int irq_start,
|
|
|
register_cpu_notifier(&gic_cpu_notifier);
|
|
|
#endif
|
|
|
set_handle_irq(gic_handle_irq);
|
|
|
+ if (static_key_true(&supports_deactivate))
|
|
|
+ pr_info("GIC: Using split EOI/Deactivate mode\n");
|
|
|
}
|
|
|
|
|
|
gic_dist_init(gic);
|
|
@@ -1010,6 +1056,7 @@ gic_of_init(struct device_node *node, struct device_node *parent)
|
|
|
{
|
|
|
void __iomem *cpu_base;
|
|
|
void __iomem *dist_base;
|
|
|
+ struct resource cpu_res;
|
|
|
u32 percpu_offset;
|
|
|
int irq;
|
|
|
|
|
@@ -1022,6 +1069,16 @@ gic_of_init(struct device_node *node, struct device_node *parent)
|
|
|
cpu_base = of_iomap(node, 1);
|
|
|
WARN(!cpu_base, "unable to map gic cpu registers\n");
|
|
|
|
|
|
+ of_address_to_resource(node, 1, &cpu_res);
|
|
|
+
|
|
|
+ /*
|
|
|
+ * Disable split EOI/Deactivate if either HYP is not available
|
|
|
+ * or the CPU interface is too small.
|
|
|
+ */
|
|
|
+ if (gic_cnt == 0 && (!is_hyp_mode_available() ||
|
|
|
+ resource_size(&cpu_res) < SZ_8K))
|
|
|
+ static_key_slow_dec(&supports_deactivate);
|
|
|
+
|
|
|
if (of_property_read_u32(node, "cpu-offset", &percpu_offset))
|
|
|
percpu_offset = 0;
|
|
|
|
|
@@ -1140,6 +1197,14 @@ gic_v2_acpi_init(struct acpi_table_header *table)
|
|
|
return -ENOMEM;
|
|
|
}
|
|
|
|
|
|
+ /*
|
|
|
+ * Disable split EOI/Deactivate if HYP is not available. ACPI
|
|
|
+ * guarantees that we'll always have a GICv2, so the CPU
|
|
|
+ * interface will always be the right size.
|
|
|
+ */
|
|
|
+ if (!is_hyp_mode_available())
|
|
|
+ static_key_slow_dec(&supports_deactivate);
|
|
|
+
|
|
|
/*
|
|
|
* Initialize zero GIC instance (no multi-GIC support). Also, set GIC
|
|
|
* as default IRQ domain to allow for GSI registration and GSI to IRQ
|