aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorTomasz Nowicki <tomasz.nowicki@linaro.org>2015-03-24 10:02:49 -0400
committerWill Deacon <will.deacon@arm.com>2015-03-26 11:13:07 -0400
commitd60fc3892c4de4a25658786f941690462c5a5bab (patch)
tree79273968cdae4cc73bd97a89e2c646431224e542
parentfbe61ec71ac975279cd47b6c299d5e33f63aac4e (diff)
irqchip: Add GICv2 specific ACPI boot support
ACPI kernel uses MADT table for proper GIC initialization. It needs to parse GIC related subtables, collect CPU interface and distributor addresses and call driver initialization function (which is hardware abstraction agnostic). In a similar way, FDT initialize GICv1/2. NOTE: This commit allow to initialize GICv1/2 basic functionality. While now simple GICv2 init call is used, any further GIC features require generic infrastructure for proper ACPI irqchip initialization. That mechanism and stacked irqdomains to support GICv2 MSI/virtualization extension, GICv3/4 and its ITS are considered as next steps. CC: Jason Cooper <jason@lakedaemon.net> CC: Marc Zyngier <marc.zyngier@arm.com> CC: Thomas Gleixner <tglx@linutronix.de> Tested-by: Suravee Suthikulpanit <Suravee.Suthikulpanit@amd.com> Tested-by: Yijing Wang <wangyijing@huawei.com> Tested-by: Mark Langsdorf <mlangsdo@redhat.com> Tested-by: Jon Masters <jcm@redhat.com> Tested-by: Timur Tabi <timur@codeaurora.org> Tested-by: Robert Richter <rrichter@cavium.com> Acked-by: Robert Richter <rrichter@cavium.com> Acked-by: Marc Zyngier <marc.zyngier@arm.com> Acked-by: Jason Cooper <jason@lakedaemon.net> Reviewed-by: Grant Likely <grant.likely@linaro.org> Signed-off-by: Tomasz Nowicki <tomasz.nowicki@linaro.org> Signed-off-by: Hanjun Guo <hanjun.guo@linaro.org> Signed-off-by: Will Deacon <will.deacon@arm.com>
-rw-r--r--arch/arm64/include/asm/acpi.h2
-rw-r--r--arch/arm64/include/asm/irq.h13
-rw-r--r--arch/arm64/kernel/acpi.c25
-rw-r--r--drivers/irqchip/irq-gic.c102
-rw-r--r--drivers/irqchip/irqchip.c3
-rw-r--r--include/linux/acpi_irq.h10
-rw-r--r--include/linux/irqchip/arm-gic-acpi.h31
7 files changed, 186 insertions, 0 deletions
diff --git a/arch/arm64/include/asm/acpi.h b/arch/arm64/include/asm/acpi.h
index eea0bc3b09d5..59c05d8ea4a0 100644
--- a/arch/arm64/include/asm/acpi.h
+++ b/arch/arm64/include/asm/acpi.h
@@ -13,6 +13,8 @@
13#define _ASM_ACPI_H 13#define _ASM_ACPI_H
14 14
15#include <linux/mm.h> 15#include <linux/mm.h>
16#include <linux/irqchip/arm-gic-acpi.h>
17
16#include <asm/cputype.h> 18#include <asm/cputype.h>
17#include <asm/smp_plat.h> 19#include <asm/smp_plat.h>
18 20
diff --git a/arch/arm64/include/asm/irq.h b/arch/arm64/include/asm/irq.h
index 94c53674a31d..bbb251b14746 100644
--- a/arch/arm64/include/asm/irq.h
+++ b/arch/arm64/include/asm/irq.h
@@ -1,6 +1,8 @@
1#ifndef __ASM_IRQ_H 1#ifndef __ASM_IRQ_H
2#define __ASM_IRQ_H 2#define __ASM_IRQ_H
3 3
4#include <linux/irqchip/arm-gic-acpi.h>
5
4#include <asm-generic/irq.h> 6#include <asm-generic/irq.h>
5 7
6struct pt_regs; 8struct pt_regs;
@@ -8,4 +10,15 @@ struct pt_regs;
8extern void migrate_irqs(void); 10extern void migrate_irqs(void);
9extern void set_handle_irq(void (*handle_irq)(struct pt_regs *)); 11extern void set_handle_irq(void (*handle_irq)(struct pt_regs *));
10 12
13static inline void acpi_irq_init(void)
14{
15 /*
16 * Hardcode ACPI IRQ chip initialization to GICv2 for now.
17 * Proper irqchip infrastructure will be implemented along with
18 * incoming GICv2m|GICv3|ITS bits.
19 */
20 acpi_gic_init();
21}
22#define acpi_irq_init acpi_irq_init
23
11#endif 24#endif
diff --git a/arch/arm64/kernel/acpi.c b/arch/arm64/kernel/acpi.c
index dec6f8aed3dc..6468f8898530 100644
--- a/arch/arm64/kernel/acpi.c
+++ b/arch/arm64/kernel/acpi.c
@@ -359,3 +359,28 @@ void __init acpi_boot_table_init(void)
359 pr_err("Can't find FADT\n"); 359 pr_err("Can't find FADT\n");
360 } 360 }
361} 361}
362
363void __init acpi_gic_init(void)
364{
365 struct acpi_table_header *table;
366 acpi_status status;
367 acpi_size tbl_size;
368 int err;
369
370 if (acpi_disabled)
371 return;
372
373 status = acpi_get_table_with_size(ACPI_SIG_MADT, 0, &table, &tbl_size);
374 if (ACPI_FAILURE(status)) {
375 const char *msg = acpi_format_exception(status);
376
377 pr_err("Failed to get MADT table, %s\n", msg);
378 return;
379 }
380
381 err = gic_v2_acpi_init(table);
382 if (err)
383 pr_err("Failed to initialize GIC IRQ controller");
384
385 early_acpi_os_unmap_memory((char *)table, tbl_size);
386}
diff --git a/drivers/irqchip/irq-gic.c b/drivers/irqchip/irq-gic.c
index 471e1cdc1933..d15a36a93c52 100644
--- a/drivers/irqchip/irq-gic.c
+++ b/drivers/irqchip/irq-gic.c
@@ -33,12 +33,14 @@
33#include <linux/of.h> 33#include <linux/of.h>
34#include <linux/of_address.h> 34#include <linux/of_address.h>
35#include <linux/of_irq.h> 35#include <linux/of_irq.h>
36#include <linux/acpi.h>
36#include <linux/irqdomain.h> 37#include <linux/irqdomain.h>
37#include <linux/interrupt.h> 38#include <linux/interrupt.h>
38#include <linux/percpu.h> 39#include <linux/percpu.h>
39#include <linux/slab.h> 40#include <linux/slab.h>
40#include <linux/irqchip/chained_irq.h> 41#include <linux/irqchip/chained_irq.h>
41#include <linux/irqchip/arm-gic.h> 42#include <linux/irqchip/arm-gic.h>
43#include <linux/irqchip/arm-gic-acpi.h>
42 44
43#include <asm/cputype.h> 45#include <asm/cputype.h>
44#include <asm/irq.h> 46#include <asm/irq.h>
@@ -1090,3 +1092,103 @@ IRQCHIP_DECLARE(msm_8660_qgic, "qcom,msm-8660-qgic", gic_of_init);
1090IRQCHIP_DECLARE(msm_qgic2, "qcom,msm-qgic2", gic_of_init); 1092IRQCHIP_DECLARE(msm_qgic2, "qcom,msm-qgic2", gic_of_init);
1091 1093
1092#endif 1094#endif
1095
1096#ifdef CONFIG_ACPI
1097static phys_addr_t dist_phy_base, cpu_phy_base __initdata;
1098
1099static int __init
1100gic_acpi_parse_madt_cpu(struct acpi_subtable_header *header,
1101 const unsigned long end)
1102{
1103 struct acpi_madt_generic_interrupt *processor;
1104 phys_addr_t gic_cpu_base;
1105 static int cpu_base_assigned;
1106
1107 processor = (struct acpi_madt_generic_interrupt *)header;
1108
1109 if (BAD_MADT_ENTRY(processor, end))
1110 return -EINVAL;
1111
1112 /*
1113 * There is no support for non-banked GICv1/2 register in ACPI spec.
1114 * All CPU interface addresses have to be the same.
1115 */
1116 gic_cpu_base = processor->base_address;
1117 if (cpu_base_assigned && gic_cpu_base != cpu_phy_base)
1118 return -EINVAL;
1119
1120 cpu_phy_base = gic_cpu_base;
1121 cpu_base_assigned = 1;
1122 return 0;
1123}
1124
1125static int __init
1126gic_acpi_parse_madt_distributor(struct acpi_subtable_header *header,
1127 const unsigned long end)
1128{
1129 struct acpi_madt_generic_distributor *dist;
1130
1131 dist = (struct acpi_madt_generic_distributor *)header;
1132
1133 if (BAD_MADT_ENTRY(dist, end))
1134 return -EINVAL;
1135
1136 dist_phy_base = dist->base_address;
1137 return 0;
1138}
1139
1140int __init
1141gic_v2_acpi_init(struct acpi_table_header *table)
1142{
1143 void __iomem *cpu_base, *dist_base;
1144 int count;
1145
1146 /* Collect CPU base addresses */
1147 count = acpi_parse_entries(ACPI_SIG_MADT,
1148 sizeof(struct acpi_table_madt),
1149 gic_acpi_parse_madt_cpu, table,
1150 ACPI_MADT_TYPE_GENERIC_INTERRUPT, 0);
1151 if (count <= 0) {
1152 pr_err("No valid GICC entries exist\n");
1153 return -EINVAL;
1154 }
1155
1156 /*
1157 * Find distributor base address. We expect one distributor entry since
1158 * ACPI 5.1 spec neither support multi-GIC instances nor GIC cascade.
1159 */
1160 count = acpi_parse_entries(ACPI_SIG_MADT,
1161 sizeof(struct acpi_table_madt),
1162 gic_acpi_parse_madt_distributor, table,
1163 ACPI_MADT_TYPE_GENERIC_DISTRIBUTOR, 0);
1164 if (count <= 0) {
1165 pr_err("No valid GICD entries exist\n");
1166 return -EINVAL;
1167 } else if (count > 1) {
1168 pr_err("More than one GICD entry detected\n");
1169 return -EINVAL;
1170 }
1171
1172 cpu_base = ioremap(cpu_phy_base, ACPI_GIC_CPU_IF_MEM_SIZE);
1173 if (!cpu_base) {
1174 pr_err("Unable to map GICC registers\n");
1175 return -ENOMEM;
1176 }
1177
1178 dist_base = ioremap(dist_phy_base, ACPI_GICV2_DIST_MEM_SIZE);
1179 if (!dist_base) {
1180 pr_err("Unable to map GICD registers\n");
1181 iounmap(cpu_base);
1182 return -ENOMEM;
1183 }
1184
1185 /*
1186 * Initialize zero GIC instance (no multi-GIC support). Also, set GIC
1187 * as default IRQ domain to allow for GSI registration and GSI to IRQ
1188 * number translation (see acpi_register_gsi() and acpi_gsi_to_irq()).
1189 */
1190 gic_init_bases(0, -1, dist_base, cpu_base, 0, NULL);
1191 irq_set_default_host(gic_data[0].domain);
1192 return 0;
1193}
1194#endif
diff --git a/drivers/irqchip/irqchip.c b/drivers/irqchip/irqchip.c
index 0fe2f718d81c..afd1af3dfe5a 100644
--- a/drivers/irqchip/irqchip.c
+++ b/drivers/irqchip/irqchip.c
@@ -8,6 +8,7 @@
8 * warranty of any kind, whether express or implied. 8 * warranty of any kind, whether express or implied.
9 */ 9 */
10 10
11#include <linux/acpi_irq.h>
11#include <linux/init.h> 12#include <linux/init.h>
12#include <linux/of_irq.h> 13#include <linux/of_irq.h>
13#include <linux/irqchip.h> 14#include <linux/irqchip.h>
@@ -26,4 +27,6 @@ extern struct of_device_id __irqchip_of_table[];
26void __init irqchip_init(void) 27void __init irqchip_init(void)
27{ 28{
28 of_irq_init(__irqchip_of_table); 29 of_irq_init(__irqchip_of_table);
30
31 acpi_irq_init();
29} 32}
diff --git a/include/linux/acpi_irq.h b/include/linux/acpi_irq.h
new file mode 100644
index 000000000000..f10c87265855
--- /dev/null
+++ b/include/linux/acpi_irq.h
@@ -0,0 +1,10 @@
1#ifndef _LINUX_ACPI_IRQ_H
2#define _LINUX_ACPI_IRQ_H
3
4#include <linux/irq.h>
5
6#ifndef acpi_irq_init
7static inline void acpi_irq_init(void) { }
8#endif
9
10#endif /* _LINUX_ACPI_IRQ_H */
diff --git a/include/linux/irqchip/arm-gic-acpi.h b/include/linux/irqchip/arm-gic-acpi.h
new file mode 100644
index 000000000000..de3419ed3937
--- /dev/null
+++ b/include/linux/irqchip/arm-gic-acpi.h
@@ -0,0 +1,31 @@
1/*
2 * Copyright (C) 2014, Linaro Ltd.
3 * Author: Tomasz Nowicki <tomasz.nowicki@linaro.org>
4 *
5 * This program is free software; you can redistribute it and/or modify
6 * it under the terms of the GNU General Public License version 2 as
7 * published by the Free Software Foundation.
8 */
9
10#ifndef ARM_GIC_ACPI_H_
11#define ARM_GIC_ACPI_H_
12
13#ifdef CONFIG_ACPI
14
15/*
16 * Hard code here, we can not get memory size from MADT (but FDT does),
17 * Actually no need to do that, because this size can be inferred
18 * from GIC spec.
19 */
20#define ACPI_GICV2_DIST_MEM_SIZE (SZ_4K)
21#define ACPI_GIC_CPU_IF_MEM_SIZE (SZ_8K)
22
23struct acpi_table_header;
24
25int gic_v2_acpi_init(struct acpi_table_header *table);
26void acpi_gic_init(void);
27#else
28static inline void acpi_gic_init(void) { }
29#endif
30
31#endif /* ARM_GIC_ACPI_H_ */