xref: /optee_os/core/drivers/gic.c (revision 9e935234082a30d5a5f8d8dd68dc4bdd82b02174)
11bb92983SJerome Forissier // SPDX-License-Identifier: BSD-2-Clause
2b0104773SPascal Brand /*
3*9e935234SJens Wiklander  * Copyright (c) 2016-2017, 2023-2024 Linaro Limited
4b0104773SPascal Brand  * Copyright (c) 2014, STMicroelectronics International N.V.
5b0104773SPascal Brand  */
6b0104773SPascal Brand 
718901324SDavid Wang #include <arm.h>
88ddf5a4eSEtienne Carriere #include <assert.h>
967e55c51SEtienne Carriere #include <compiler.h>
1005089e5fSJens Wiklander #include <config.h>
11b0104773SPascal Brand #include <drivers/gic.h>
12*9e935234SJens Wiklander #include <dt-bindings/interrupt-controller/arm-gic.h>
13*9e935234SJens Wiklander #include <initcall.h>
1405089e5fSJens Wiklander #include <io.h>
150f93de74SEtienne Carriere #include <keep.h>
1667729d8dSLudovic Barre #include <kernel/dt.h>
1714885eb1SEtienne Carriere #include <kernel/dt_driver.h>
187315b7b4SJens Wiklander #include <kernel/interrupt.h>
1905089e5fSJens Wiklander #include <kernel/misc.h>
20d13278b8SEtienne Carriere #include <kernel/panic.h>
2105089e5fSJens Wiklander #include <libfdt.h>
2260801696SVolodymyr Babchuk #include <mm/core_memprot.h>
2360801696SVolodymyr Babchuk #include <mm/core_mmu.h>
244de4bebcSJens Wiklander #include <trace.h>
2505089e5fSJens Wiklander #include <util.h>
26b0104773SPascal Brand 
27b0104773SPascal Brand /* Offsets from gic.gicc_base */
28b0104773SPascal Brand #define GICC_CTLR		(0x000)
2930a673e3SPeter Maydell #define GICC_PMR		(0x004)
30b0104773SPascal Brand #define GICC_IAR		(0x00C)
31b0104773SPascal Brand #define GICC_EOIR		(0x010)
32b0104773SPascal Brand 
33b0104773SPascal Brand #define GICC_CTLR_ENABLEGRP0	(1 << 0)
34b0104773SPascal Brand #define GICC_CTLR_ENABLEGRP1	(1 << 1)
35b0104773SPascal Brand #define GICC_CTLR_FIQEN		(1 << 3)
36b0104773SPascal Brand 
37b0104773SPascal Brand /* Offsets from gic.gicd_base */
38b0104773SPascal Brand #define GICD_CTLR		(0x000)
39b0104773SPascal Brand #define GICD_TYPER		(0x004)
40b0104773SPascal Brand #define GICD_IGROUPR(n)		(0x080 + (n) * 4)
41b0104773SPascal Brand #define GICD_ISENABLER(n)	(0x100 + (n) * 4)
42b0104773SPascal Brand #define GICD_ICENABLER(n)	(0x180 + (n) * 4)
4326ed70ecSGuanchao Liang #define GICD_ISPENDR(n)		(0x200 + (n) * 4)
44b0104773SPascal Brand #define GICD_ICPENDR(n)		(0x280 + (n) * 4)
45b0104773SPascal Brand #define GICD_IPRIORITYR(n)	(0x400 + (n) * 4)
46b0104773SPascal Brand #define GICD_ITARGETSR(n)	(0x800 + (n) * 4)
471fcac774SSandeep Tripathy #define GICD_IGROUPMODR(n)	(0xd00 + (n) * 4)
4826ed70ecSGuanchao Liang #define GICD_SGIR		(0xF00)
49b0104773SPascal Brand 
5069171becSJens Wiklander #ifdef CFG_ARM_GICV3
5169171becSJens Wiklander #define GICD_PIDR2		(0xFFE8)
5269171becSJens Wiklander #else
5369171becSJens Wiklander /* Called ICPIDR2 in GICv2 specification */
5469171becSJens Wiklander #define GICD_PIDR2		(0xFE8)
5569171becSJens Wiklander #endif
5669171becSJens Wiklander 
5705089e5fSJens Wiklander #define GICD_CTLR_ENABLEGRP0	BIT32(0)
5805089e5fSJens Wiklander #define GICD_CTLR_ENABLEGRP1NS	BIT32(1)
5905089e5fSJens Wiklander #define GICD_CTLR_ENABLEGRP1S	BIT32(2)
6005089e5fSJens Wiklander #define GICD_CTLR_ARE_S		BIT32(4)
6105089e5fSJens Wiklander #define GICD_CTLR_ARE_NS	BIT32(5)
6205089e5fSJens Wiklander 
6305089e5fSJens Wiklander /* Offsets from gic.gicr_base[core_pos] */
6405089e5fSJens Wiklander #define GICR_V3_PCPUBASE_SIZE	(2 * 64 * 1024)
6505089e5fSJens Wiklander #define GICR_SGI_BASE_OFFSET	(64 * 1024)
6605089e5fSJens Wiklander #define GICR_CTLR		(0x00)
6705089e5fSJens Wiklander #define GICR_TYPER		(0x08)
6805089e5fSJens Wiklander 
6905089e5fSJens Wiklander #define GICR_IGROUPR0		(GICR_SGI_BASE_OFFSET + 0x080)
7005089e5fSJens Wiklander #define GICR_IGRPMODR0		(GICR_SGI_BASE_OFFSET + 0xD00)
7184603456SJens Wiklander #define GICR_ICENABLER0		(GICR_SGI_BASE_OFFSET + 0x180)
7284603456SJens Wiklander #define GICR_ICPENDR0		(GICR_SGI_BASE_OFFSET + 0x280)
73*9e935234SJens Wiklander #define GICR_ISENABLER0		(GICR_SGI_BASE_OFFSET + 0x100)
74*9e935234SJens Wiklander #define GICR_ICFGR0		(GICR_SGI_BASE_OFFSET + 0xC00)
75*9e935234SJens Wiklander #define GICR_ICFGR1		(GICR_SGI_BASE_OFFSET + 0xC04)
76*9e935234SJens Wiklander #define GICR_IPRIORITYR(n)	(GICR_SGI_BASE_OFFSET + 0x400 + (n) * 4)
77*9e935234SJens Wiklander 
7805089e5fSJens Wiklander 
7905089e5fSJens Wiklander #define GICR_TYPER_LAST		BIT64(4)
8005089e5fSJens Wiklander #define GICR_TYPER_AFF3_SHIFT	56
8105089e5fSJens Wiklander #define GICR_TYPER_AFF2_SHIFT	48
8205089e5fSJens Wiklander #define GICR_TYPER_AFF1_SHIFT	40
8305089e5fSJens Wiklander #define GICR_TYPER_AFF0_SHIFT	32
84b0104773SPascal Brand 
8569171becSJens Wiklander /* GICD IDR2 name differs on GICv3 and GICv2 but uses same bit map */
8669171becSJens Wiklander #define GICD_PIDR2_ARCHREV_SHIFT	4
8769171becSJens Wiklander #define GICD_PIDR2_ARCHREV_MASK		0xF
8869171becSJens Wiklander 
8953bd332aSSY Chiu /* Number of Private Peripheral Interrupt */
9053bd332aSSY Chiu #define NUM_PPI	32
9153bd332aSSY Chiu 
9226ed70ecSGuanchao Liang /* Number of Software Generated Interrupt */
9326ed70ecSGuanchao Liang #define NUM_SGI			16
9426ed70ecSGuanchao Liang 
9526ed70ecSGuanchao Liang /* Number of Non-secure Software Generated Interrupt */
9626ed70ecSGuanchao Liang #define NUM_NS_SGI		8
9726ed70ecSGuanchao Liang 
9853bd332aSSY Chiu /* Number of interrupts in one register */
9953bd332aSSY Chiu #define NUM_INTS_PER_REG	32
10053bd332aSSY Chiu 
10153bd332aSSY Chiu /* Number of targets in one register */
10253bd332aSSY Chiu #define NUM_TARGETS_PER_REG	4
10353bd332aSSY Chiu 
10453bd332aSSY Chiu /* Accessors to access ITARGETSRn */
10553bd332aSSY Chiu #define ITARGETSR_FIELD_BITS	8
10653bd332aSSY Chiu #define ITARGETSR_FIELD_MASK	0xff
10753bd332aSSY Chiu 
1081b4c5002SIzhar Nevo #define GICD_TYPER_IT_LINES_NUM_MASK	0x1f
1097315b7b4SJens Wiklander #define GICC_IAR_IT_ID_MASK	0x3ff
1107315b7b4SJens Wiklander #define GICC_IAR_CPU_ID_MASK	0x7
1117315b7b4SJens Wiklander #define GICC_IAR_CPU_ID_SHIFT	10
112b0104773SPascal Brand 
113ec740b9fSJens Wiklander #define GICC_SGI_IRM_BIT	40
114ec740b9fSJens Wiklander #define GICC_SGI_AFF1_SHIFT	16
115ec740b9fSJens Wiklander #define GICC_SGI_AFF2_SHIFT	32
116ec740b9fSJens Wiklander #define GICC_SGI_AFF3_SHIFT	48
117ec740b9fSJens Wiklander 
118ec740b9fSJens Wiklander #define GICD_SGIR_SIGINTID_MASK			0xf
119ec740b9fSJens Wiklander #define GICD_SGIR_TO_OTHER_CPUS			0x1
120ec740b9fSJens Wiklander #define GICD_SGIR_TO_THIS_CPU			0x2
121ec740b9fSJens Wiklander #define GICD_SGIR_TARGET_LIST_FILTER_SHIFT	24
122ec740b9fSJens Wiklander #define GICD_SGIR_NSATT_SHIFT			15
123ec740b9fSJens Wiklander #define GICD_SGIR_CPU_TARGET_LIST_SHIFT		16
124ec740b9fSJens Wiklander 
12567e55c51SEtienne Carriere struct gic_data {
12667e55c51SEtienne Carriere 	vaddr_t gicc_base;
12767e55c51SEtienne Carriere 	vaddr_t gicd_base;
12805089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
12905089e5fSJens Wiklander 	vaddr_t gicr_base[CFG_TEE_CORE_NB_CORE];
13005089e5fSJens Wiklander #endif
13167e55c51SEtienne Carriere 	size_t max_it;
13205089e5fSJens Wiklander 	uint32_t per_cpu_group_status;
13305089e5fSJens Wiklander 	uint32_t per_cpu_group_modifier;
134*9e935234SJens Wiklander 	uint32_t per_cpu_enable;
13567e55c51SEtienne Carriere 	struct itr_chip chip;
13667e55c51SEtienne Carriere };
13767e55c51SEtienne Carriere 
138*9e935234SJens Wiklander static bool gic_primary_done __nex_bss;
13967e55c51SEtienne Carriere static struct gic_data gic_data __nex_bss;
14067e55c51SEtienne Carriere 
141702fe5a7SClément Léger static void gic_op_add(struct itr_chip *chip, size_t it, uint32_t type,
142702fe5a7SClément Léger 		       uint32_t prio);
1437315b7b4SJens Wiklander static void gic_op_enable(struct itr_chip *chip, size_t it);
1447315b7b4SJens Wiklander static void gic_op_disable(struct itr_chip *chip, size_t it);
14526ed70ecSGuanchao Liang static void gic_op_raise_pi(struct itr_chip *chip, size_t it);
14626ed70ecSGuanchao Liang static void gic_op_raise_sgi(struct itr_chip *chip, size_t it,
147ec740b9fSJens Wiklander 			     uint32_t cpu_mask);
14826ed70ecSGuanchao Liang static void gic_op_set_affinity(struct itr_chip *chip, size_t it,
14926ed70ecSGuanchao Liang 			uint8_t cpu_mask);
1507315b7b4SJens Wiklander 
1517315b7b4SJens Wiklander static const struct itr_ops gic_ops = {
1527315b7b4SJens Wiklander 	.add = gic_op_add,
15308ded0e1SEtienne Carriere 	.mask = gic_op_disable,
15408ded0e1SEtienne Carriere 	.unmask = gic_op_enable,
1557315b7b4SJens Wiklander 	.enable = gic_op_enable,
1567315b7b4SJens Wiklander 	.disable = gic_op_disable,
15726ed70ecSGuanchao Liang 	.raise_pi = gic_op_raise_pi,
15826ed70ecSGuanchao Liang 	.raise_sgi = gic_op_raise_sgi,
15926ed70ecSGuanchao Liang 	.set_affinity = gic_op_set_affinity,
1607315b7b4SJens Wiklander };
1613639b55fSJerome Forissier DECLARE_KEEP_PAGER(gic_ops);
1627315b7b4SJens Wiklander 
16305089e5fSJens Wiklander static vaddr_t __maybe_unused get_gicr_base(struct gic_data *gd __maybe_unused)
16405089e5fSJens Wiklander {
16505089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
16605089e5fSJens Wiklander 	return gd->gicr_base[get_core_pos()];
16705089e5fSJens Wiklander #else
16805089e5fSJens Wiklander 	return 0;
16905089e5fSJens Wiklander #endif
17005089e5fSJens Wiklander }
17105089e5fSJens Wiklander 
1727c4883aeSJens Wiklander static bool affinity_routing_is_enabled(struct gic_data *gd)
1737c4883aeSJens Wiklander {
1747c4883aeSJens Wiklander 	return IS_ENABLED(CFG_ARM_GICV3) &&
1757c4883aeSJens Wiklander 	       io_read32(gd->gicd_base + GICD_CTLR) & GICD_CTLR_ARE_S;
1767c4883aeSJens Wiklander }
1777c4883aeSJens Wiklander 
17818901324SDavid Wang static size_t probe_max_it(vaddr_t gicc_base __maybe_unused, vaddr_t gicd_base)
179b0104773SPascal Brand {
180b0104773SPascal Brand 	int i;
181b0104773SPascal Brand 	uint32_t old_ctlr;
182b0104773SPascal Brand 	size_t ret = 0;
1831b4c5002SIzhar Nevo 	size_t max_regs = io_read32(gicd_base + GICD_TYPER) &
1841b4c5002SIzhar Nevo 			  GICD_TYPER_IT_LINES_NUM_MASK;
185b0104773SPascal Brand 
186b0104773SPascal Brand 	/*
187b0104773SPascal Brand 	 * Probe which interrupt number is the largest.
188b0104773SPascal Brand 	 */
18918901324SDavid Wang #if defined(CFG_ARM_GICV3)
19018901324SDavid Wang 	old_ctlr = read_icc_ctlr();
19118901324SDavid Wang 	write_icc_ctlr(0);
19218901324SDavid Wang #else
193918bb3a5SEtienne Carriere 	old_ctlr = io_read32(gicc_base + GICC_CTLR);
194918bb3a5SEtienne Carriere 	io_write32(gicc_base + GICC_CTLR, 0);
19518901324SDavid Wang #endif
19679f008d3SJens Wiklander 	for (i = max_regs; i >= 0; i--) {
197b0104773SPascal Brand 		uint32_t old_reg;
198b0104773SPascal Brand 		uint32_t reg;
199b0104773SPascal Brand 		int b;
200b0104773SPascal Brand 
201918bb3a5SEtienne Carriere 		old_reg = io_read32(gicd_base + GICD_ISENABLER(i));
202918bb3a5SEtienne Carriere 		io_write32(gicd_base + GICD_ISENABLER(i), 0xffffffff);
203918bb3a5SEtienne Carriere 		reg = io_read32(gicd_base + GICD_ISENABLER(i));
204918bb3a5SEtienne Carriere 		io_write32(gicd_base + GICD_ICENABLER(i), ~old_reg);
20579f008d3SJens Wiklander 		for (b = NUM_INTS_PER_REG - 1; b >= 0; b--) {
206007a97a2SJens Wiklander 			if (BIT32(b) & reg) {
20753bd332aSSY Chiu 				ret = i * NUM_INTS_PER_REG + b;
208b0104773SPascal Brand 				goto out;
209b0104773SPascal Brand 			}
210b0104773SPascal Brand 		}
211b0104773SPascal Brand 	}
212b0104773SPascal Brand out:
21318901324SDavid Wang #if defined(CFG_ARM_GICV3)
21418901324SDavid Wang 	write_icc_ctlr(old_ctlr);
21518901324SDavid Wang #else
216918bb3a5SEtienne Carriere 	io_write32(gicc_base + GICC_CTLR, old_ctlr);
21718901324SDavid Wang #endif
218b0104773SPascal Brand 	return ret;
219b0104773SPascal Brand }
220b0104773SPascal Brand 
221*9e935234SJens Wiklander static void gicv3_sync_redist_config(struct gic_data *gd)
22284603456SJens Wiklander {
22384603456SJens Wiklander 	vaddr_t gicr_base = get_gicr_base(gd);
22484603456SJens Wiklander 	bool need_sync = false;
22584603456SJens Wiklander 	uint32_t gmod0 = 0;
22684603456SJens Wiklander 	uint32_t grp0 = 0;
22784603456SJens Wiklander 	size_t n = 0;
22884603456SJens Wiklander 
2297c4883aeSJens Wiklander 	/*
2307c4883aeSJens Wiklander 	 * If gicr_base isn't available there's no need to synchronize SGI
2317c4883aeSJens Wiklander 	 * configuration since gic_init_donate_sgi_to_ns() would panic.
2327c4883aeSJens Wiklander 	 */
23384603456SJens Wiklander 	if (!gicr_base)
2347c4883aeSJens Wiklander 		return;
23584603456SJens Wiklander 
23684603456SJens Wiklander 	grp0 = io_read32(gicr_base + GICR_IGROUPR0);
23784603456SJens Wiklander 	gmod0 = io_read32(gicr_base + GICR_IGRPMODR0);
238*9e935234SJens Wiklander 	for (n = GIC_SGI_SEC_BASE; n < GIC_SPI_BASE; n++) {
23984603456SJens Wiklander 		/* Ignore matching bits */
24084603456SJens Wiklander 		if (!(BIT32(n) & (grp0 ^ gd->per_cpu_group_status)) &&
24184603456SJens Wiklander 		    !(BIT32(n) & (gmod0 ^ gd->per_cpu_group_modifier)))
24284603456SJens Wiklander 			continue;
24384603456SJens Wiklander 		/*
244*9e935234SJens Wiklander 		 * SGI/PPI-n differs from primary CPU configuration,
24584603456SJens Wiklander 		 * let's sync up.
24684603456SJens Wiklander 		 */
24784603456SJens Wiklander 		need_sync = true;
24884603456SJens Wiklander 
24984603456SJens Wiklander 		/* Disable interrupt */
25084603456SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(n));
25184603456SJens Wiklander 
25284603456SJens Wiklander 		/* Make interrupt non-pending */
25384603456SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(n));
25484603456SJens Wiklander 
25584603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_status)
25684603456SJens Wiklander 			grp0 |= BIT32(n);
25784603456SJens Wiklander 		else
25884603456SJens Wiklander 			grp0 &= ~BIT32(n);
25984603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_modifier)
26084603456SJens Wiklander 			gmod0 |= BIT32(n);
26184603456SJens Wiklander 		else
26284603456SJens Wiklander 			gmod0 &= ~BIT32(n);
26384603456SJens Wiklander 	}
26484603456SJens Wiklander 
26584603456SJens Wiklander 	if (need_sync) {
26684603456SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, grp0);
26784603456SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0, gmod0);
268*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_ISENABLER0, gd->per_cpu_enable);
26984603456SJens Wiklander 	}
27084603456SJens Wiklander }
27184603456SJens Wiklander 
272*9e935234SJens Wiklander static void gic_legacy_sync_dist_config(struct gic_data *gd)
27384603456SJens Wiklander {
27484603456SJens Wiklander 	bool need_sync = false;
27584603456SJens Wiklander 	uint32_t grp0 = 0;
27684603456SJens Wiklander 	size_t n = 0;
27784603456SJens Wiklander 
27884603456SJens Wiklander 	grp0 = io_read32(gd->gicd_base + GICD_IGROUPR(0));
279*9e935234SJens Wiklander 	for (n = GIC_SGI_SEC_BASE; n < GIC_SPI_BASE; n++) {
28084603456SJens Wiklander 		/* Ignore matching bits */
28184603456SJens Wiklander 		if (!(BIT32(n) & (grp0 ^ gd->per_cpu_group_status)))
28284603456SJens Wiklander 			continue;
28384603456SJens Wiklander 		/*
284*9e935234SJens Wiklander 		 * SGI/PPI-n differs from primary CPU configuration,
28584603456SJens Wiklander 		 * let's sync up.
28684603456SJens Wiklander 		 */
28784603456SJens Wiklander 		need_sync = true;
28884603456SJens Wiklander 
28984603456SJens Wiklander 		/* Disable interrupt */
29084603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICENABLER(0), BIT(n));
29184603456SJens Wiklander 
29284603456SJens Wiklander 		/* Make interrupt non-pending */
29384603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICPENDR(0), BIT(n));
29484603456SJens Wiklander 
29584603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_status)
29684603456SJens Wiklander 			grp0 |= BIT32(n);
29784603456SJens Wiklander 		else
29884603456SJens Wiklander 			grp0 &= ~BIT32(n);
29984603456SJens Wiklander 	}
30084603456SJens Wiklander 
301*9e935234SJens Wiklander 	if (need_sync) {
30284603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_IGROUPR(0), grp0);
303*9e935234SJens Wiklander 		io_write32(gd->gicd_base + GICD_ISENABLER(0),
304*9e935234SJens Wiklander 			   gd->per_cpu_enable);
305*9e935234SJens Wiklander 	}
30684603456SJens Wiklander }
30784603456SJens Wiklander 
3085da157f5SJens Wiklander static void init_gic_per_cpu(struct gic_data *gd)
309bedc2b9fSsunny {
31005089e5fSJens Wiklander 	io_write32(gd->gicd_base + GICD_IGROUPR(0), gd->per_cpu_group_status);
311bedc2b9fSsunny 
31205089e5fSJens Wiklander 	/*
31305089e5fSJens Wiklander 	 * Set the priority mask to permit Non-secure interrupts, and to
31430a673e3SPeter Maydell 	 * allow the Non-secure world to adjust the priority mask itself
31530a673e3SPeter Maydell 	 */
31618901324SDavid Wang #if defined(CFG_ARM_GICV3)
31718901324SDavid Wang 	write_icc_pmr(0x80);
3181fcac774SSandeep Tripathy 	write_icc_igrpen1(1);
31918901324SDavid Wang #else
320918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_PMR, 0x80);
32130a673e3SPeter Maydell 
322bedc2b9fSsunny 	/* Enable GIC */
323918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_CTLR,
324918bb3a5SEtienne Carriere 		   GICC_CTLR_ENABLEGRP0 | GICC_CTLR_ENABLEGRP1 |
325918bb3a5SEtienne Carriere 		   GICC_CTLR_FIQEN);
32618901324SDavid Wang #endif
327bedc2b9fSsunny }
328bedc2b9fSsunny 
3295da157f5SJens Wiklander void gic_init_per_cpu(void)
3305da157f5SJens Wiklander {
3315da157f5SJens Wiklander 	struct gic_data *gd = &gic_data;
3325da157f5SJens Wiklander 
3335da157f5SJens Wiklander #if defined(CFG_ARM_GICV3)
3345da157f5SJens Wiklander 	assert(gd->gicd_base);
3355da157f5SJens Wiklander #else
3365da157f5SJens Wiklander 	assert(gd->gicd_base && gd->gicc_base);
3375da157f5SJens Wiklander #endif
3385da157f5SJens Wiklander 
33984603456SJens Wiklander 	if (IS_ENABLED(CFG_WITH_ARM_TRUSTED_FW)) {
34084603456SJens Wiklander 		/*
34184603456SJens Wiklander 		 * GIC is already initialized by TF-A, we only need to
342*9e935234SJens Wiklander 		 * handle eventual SGI or PPI configuration changes.
34384603456SJens Wiklander 		 */
3447c4883aeSJens Wiklander 		if (affinity_routing_is_enabled(gd))
345*9e935234SJens Wiklander 			gicv3_sync_redist_config(gd);
34684603456SJens Wiklander 		else
347*9e935234SJens Wiklander 			gic_legacy_sync_dist_config(gd);
34884603456SJens Wiklander 	} else {
34984603456SJens Wiklander 		/*
35084603456SJens Wiklander 		 * Non-TF-A case where all CPU specific configuration
35184603456SJens Wiklander 		 * of GIC must be done here.
35284603456SJens Wiklander 		 */
3535da157f5SJens Wiklander 		init_gic_per_cpu(gd);
3545da157f5SJens Wiklander 	}
35584603456SJens Wiklander }
3565da157f5SJens Wiklander 
35784603456SJens Wiklander void gic_init_donate_sgi_to_ns(size_t it)
35884603456SJens Wiklander {
35984603456SJens Wiklander 	struct gic_data *gd = &gic_data;
36084603456SJens Wiklander 
36184603456SJens Wiklander 	assert(it >= GIC_SGI_SEC_BASE && it <= GIC_SGI_SEC_MAX);
36284603456SJens Wiklander 
36384603456SJens Wiklander 	/* Assert it's secure to start with. */
36484603456SJens Wiklander 	assert(!(gd->per_cpu_group_status & BIT32(it)) &&
36584603456SJens Wiklander 	       (gd->per_cpu_group_modifier & BIT32(it)));
36684603456SJens Wiklander 
36784603456SJens Wiklander 	gd->per_cpu_group_modifier &= ~BIT32(it);
36884603456SJens Wiklander 	gd->per_cpu_group_status |= BIT32(it);
36984603456SJens Wiklander 
3707c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd)) {
37184603456SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
37284603456SJens Wiklander 
3737c4883aeSJens Wiklander 		if (!gicr_base)
3747c4883aeSJens Wiklander 			panic("GICR_BASE missing");
3757c4883aeSJens Wiklander 
37684603456SJens Wiklander 		/* Disable interrupt */
37784603456SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(it));
37884603456SJens Wiklander 
37984603456SJens Wiklander 		/* Make interrupt non-pending */
38084603456SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(it));
38184603456SJens Wiklander 
38284603456SJens Wiklander 		/* Make it to non-secure */
38384603456SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, gd->per_cpu_group_status);
38484603456SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0,
38584603456SJens Wiklander 			   gd->per_cpu_group_modifier);
38684603456SJens Wiklander 	} else {
38784603456SJens Wiklander 		/* Disable interrupt */
38884603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICENABLER(0), BIT(it));
38984603456SJens Wiklander 
39084603456SJens Wiklander 		/* Make interrupt non-pending */
39184603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICPENDR(0), BIT(it));
39284603456SJens Wiklander 
39384603456SJens Wiklander 		/* Make it to non-secure */
39484603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_IGROUPR(0),
39584603456SJens Wiklander 			   gd->per_cpu_group_status);
39684603456SJens Wiklander 	}
39784603456SJens Wiklander }
39884603456SJens Wiklander 
3990ee3f52eSEtienne Carriere static int gic_dt_get_irq(const uint32_t *properties, int count, uint32_t *type,
4000ee3f52eSEtienne Carriere 			  uint32_t *prio)
4010ee3f52eSEtienne Carriere {
4020ee3f52eSEtienne Carriere 	int it_num = DT_INFO_INVALID_INTERRUPT;
4030ee3f52eSEtienne Carriere 
4040ee3f52eSEtienne Carriere 	if (type)
4050ee3f52eSEtienne Carriere 		*type = IRQ_TYPE_NONE;
4060ee3f52eSEtienne Carriere 
4070ee3f52eSEtienne Carriere 	if (prio)
4080ee3f52eSEtienne Carriere 		*prio = 0;
4090ee3f52eSEtienne Carriere 
4100ee3f52eSEtienne Carriere 	if (!properties || count < 2)
4110ee3f52eSEtienne Carriere 		return DT_INFO_INVALID_INTERRUPT;
4120ee3f52eSEtienne Carriere 
4130ee3f52eSEtienne Carriere 	it_num = fdt32_to_cpu(properties[1]);
4140ee3f52eSEtienne Carriere 
4150ee3f52eSEtienne Carriere 	switch (fdt32_to_cpu(properties[0])) {
4168c7282beSEtienne Carriere 	case GIC_PPI:
4170ee3f52eSEtienne Carriere 		it_num += 16;
4180ee3f52eSEtienne Carriere 		break;
4198c7282beSEtienne Carriere 	case GIC_SPI:
4200ee3f52eSEtienne Carriere 		it_num += 32;
4210ee3f52eSEtienne Carriere 		break;
4220ee3f52eSEtienne Carriere 	default:
4230ee3f52eSEtienne Carriere 		it_num = DT_INFO_INVALID_INTERRUPT;
4240ee3f52eSEtienne Carriere 	}
4250ee3f52eSEtienne Carriere 
4260ee3f52eSEtienne Carriere 	return it_num;
4270ee3f52eSEtienne Carriere }
4280ee3f52eSEtienne Carriere 
42905089e5fSJens Wiklander static void __maybe_unused probe_redist_base_addrs(vaddr_t *gicr_base_addrs,
43005089e5fSJens Wiklander 						   paddr_t gicr_base_pa)
43105089e5fSJens Wiklander {
43205089e5fSJens Wiklander 	size_t sz = GICR_V3_PCPUBASE_SIZE;
43305089e5fSJens Wiklander 	paddr_t pa = gicr_base_pa;
43405089e5fSJens Wiklander 	size_t core_pos = 0;
43505089e5fSJens Wiklander 	uint64_t mt_bit = 0;
43605089e5fSJens Wiklander 	uint64_t mpidr = 0;
43705089e5fSJens Wiklander 	uint64_t tv = 0;
43805089e5fSJens Wiklander 	vaddr_t va = 0;
43905089e5fSJens Wiklander 
44005089e5fSJens Wiklander #ifdef ARM64
44105089e5fSJens Wiklander 	mt_bit = read_mpidr_el1() & MPIDR_MT_MASK;
44205089e5fSJens Wiklander #endif
44305089e5fSJens Wiklander 	do {
44405089e5fSJens Wiklander 		va = core_mmu_get_va(pa, MEM_AREA_IO_SEC, sz);
44505089e5fSJens Wiklander 		if (!va)
44605089e5fSJens Wiklander 			panic();
44705089e5fSJens Wiklander 		tv = io_read64(va + GICR_TYPER);
44805089e5fSJens Wiklander 
44905089e5fSJens Wiklander 		/*
45005089e5fSJens Wiklander 		 * Extract an mpidr from the Type register to calculate the
45105089e5fSJens Wiklander 		 * core position of this redistributer instance.
45205089e5fSJens Wiklander 		 */
45305089e5fSJens Wiklander 		mpidr = mt_bit;
45405089e5fSJens Wiklander 		mpidr |= SHIFT_U64((tv >> GICR_TYPER_AFF3_SHIFT) &
45505089e5fSJens Wiklander 				   MPIDR_AFFLVL_MASK, MPIDR_AFF3_SHIFT);
45605089e5fSJens Wiklander 		mpidr |= (tv >> GICR_TYPER_AFF0_SHIFT) &
45705089e5fSJens Wiklander 			 (MPIDR_AFF0_MASK | MPIDR_AFF1_MASK | MPIDR_AFF2_MASK);
45805089e5fSJens Wiklander 		core_pos = get_core_pos_mpidr(mpidr);
45905089e5fSJens Wiklander 		if (core_pos < CFG_TEE_CORE_NB_CORE) {
46005089e5fSJens Wiklander 			DMSG("GICR_BASE[%zu] at %#"PRIxVA, core_pos, va);
46105089e5fSJens Wiklander 			gicr_base_addrs[core_pos] = va;
46205089e5fSJens Wiklander 		} else {
46305089e5fSJens Wiklander 			EMSG("Skipping too large core_pos %zu from GICR_TYPER",
46405089e5fSJens Wiklander 			     core_pos);
46505089e5fSJens Wiklander 		}
46605089e5fSJens Wiklander 		pa += sz;
46705089e5fSJens Wiklander 	} while (!(tv & GICR_TYPER_LAST));
46805089e5fSJens Wiklander }
46905089e5fSJens Wiklander 
47005089e5fSJens Wiklander static void gic_init_base_addr(paddr_t gicc_base_pa, paddr_t gicd_base_pa,
47105089e5fSJens Wiklander 			       paddr_t gicr_base_pa __maybe_unused)
472b0104773SPascal Brand {
47367e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
4740ee3f52eSEtienne Carriere 	vaddr_t gicc_base = 0;
4750ee3f52eSEtienne Carriere 	vaddr_t gicd_base = 0;
47669171becSJens Wiklander 	uint32_t vers __maybe_unused = 0;
4770ee3f52eSEtienne Carriere 
4780ee3f52eSEtienne Carriere 	assert(cpu_mmu_enabled());
4790ee3f52eSEtienne Carriere 
4800ee3f52eSEtienne Carriere 	gicd_base = core_mmu_get_va(gicd_base_pa, MEM_AREA_IO_SEC,
4810ee3f52eSEtienne Carriere 				    GIC_DIST_REG_SIZE);
4820ee3f52eSEtienne Carriere 	if (!gicd_base)
4830ee3f52eSEtienne Carriere 		panic();
4840ee3f52eSEtienne Carriere 
48569171becSJens Wiklander 	vers = io_read32(gicd_base + GICD_PIDR2);
48669171becSJens Wiklander 	vers >>= GICD_PIDR2_ARCHREV_SHIFT;
48769171becSJens Wiklander 	vers &= GICD_PIDR2_ARCHREV_MASK;
48869171becSJens Wiklander 
48969171becSJens Wiklander 	if (IS_ENABLED(CFG_ARM_GICV3)) {
49069171becSJens Wiklander 		assert(vers == 3);
49169171becSJens Wiklander 	} else {
492d3f6526eSJens Wiklander 		assert(vers == 2 || vers == 1);
4930ee3f52eSEtienne Carriere 		gicc_base = core_mmu_get_va(gicc_base_pa, MEM_AREA_IO_SEC,
4940ee3f52eSEtienne Carriere 					    GIC_CPU_REG_SIZE);
4950ee3f52eSEtienne Carriere 		if (!gicc_base)
4960ee3f52eSEtienne Carriere 			panic();
4970ee3f52eSEtienne Carriere 	}
4980ee3f52eSEtienne Carriere 
4990ee3f52eSEtienne Carriere 	gd->gicc_base = gicc_base;
5000ee3f52eSEtienne Carriere 	gd->gicd_base = gicd_base;
5010ee3f52eSEtienne Carriere 	gd->max_it = probe_max_it(gicc_base, gicd_base);
50205089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
5037c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd) && gicr_base_pa)
50405089e5fSJens Wiklander 		probe_redist_base_addrs(gd->gicr_base, gicr_base_pa);
50505089e5fSJens Wiklander #endif
5060ee3f52eSEtienne Carriere 	gd->chip.ops = &gic_ops;
5070ee3f52eSEtienne Carriere 
5080ee3f52eSEtienne Carriere 	if (IS_ENABLED(CFG_DT))
5090ee3f52eSEtienne Carriere 		gd->chip.dt_get_irq = gic_dt_get_irq;
5100ee3f52eSEtienne Carriere }
5110ee3f52eSEtienne Carriere 
51205089e5fSJens Wiklander void gic_init_v3(paddr_t gicc_base_pa, paddr_t gicd_base_pa,
51305089e5fSJens Wiklander 		 paddr_t gicr_base_pa)
5140ee3f52eSEtienne Carriere {
5150ee3f52eSEtienne Carriere 	struct gic_data __maybe_unused *gd = &gic_data;
5160ee3f52eSEtienne Carriere 	size_t __maybe_unused n = 0;
517b0104773SPascal Brand 
51805089e5fSJens Wiklander 	gic_init_base_addr(gicc_base_pa, gicd_base_pa, gicr_base_pa);
519b0104773SPascal Brand 
52005089e5fSJens Wiklander #if defined(CFG_WITH_ARM_TRUSTED_FW)
5210ee3f52eSEtienne Carriere 	/* GIC configuration is initialized from TF-A when embedded */
5227c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd)) {
5237c4883aeSJens Wiklander 		/* Secure affinity routing enabled */
52405089e5fSJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
52505089e5fSJens Wiklander 
5267c4883aeSJens Wiklander 		if (gicr_base) {
5277c4883aeSJens Wiklander 			gd->per_cpu_group_status = io_read32(gicr_base +
5287c4883aeSJens Wiklander 							     GICR_IGROUPR0);
52905089e5fSJens Wiklander 			gd->per_cpu_group_modifier = io_read32(gicr_base +
53005089e5fSJens Wiklander 							       GICR_IGRPMODR0);
53105089e5fSJens Wiklander 		} else {
5327c4883aeSJens Wiklander 			IMSG("GIC redistributor base address not provided");
5337c4883aeSJens Wiklander 			IMSG("Assuming default GIC group status and modifier");
5347c4883aeSJens Wiklander 			gd->per_cpu_group_status = 0xffff00ff;
5357c4883aeSJens Wiklander 			gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
5367c4883aeSJens Wiklander 		}
5377c4883aeSJens Wiklander 	} else {
53805089e5fSJens Wiklander 		/* Legacy operation with secure affinity routing disabled */
53905089e5fSJens Wiklander 		gd->per_cpu_group_status = io_read32(gd->gicd_base +
54005089e5fSJens Wiklander 						     GICD_IGROUPR(0));
54105089e5fSJens Wiklander 		gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
54205089e5fSJens Wiklander 	}
54305089e5fSJens Wiklander #else /*!CFG_WITH_ARM_TRUSTED_FW*/
54405089e5fSJens Wiklander 	/*
54505089e5fSJens Wiklander 	 * Without TF-A, GIC is always configured in for legacy operation
54605089e5fSJens Wiklander 	 * with secure affinity routing disabled.
54705089e5fSJens Wiklander 	 */
5487315b7b4SJens Wiklander 	for (n = 0; n <= gd->max_it / NUM_INTS_PER_REG; n++) {
549b0104773SPascal Brand 		/* Disable interrupts */
550918bb3a5SEtienne Carriere 		io_write32(gd->gicd_base + GICD_ICENABLER(n), 0xffffffff);
551b0104773SPascal Brand 
552b0104773SPascal Brand 		/* Make interrupts non-pending */
553918bb3a5SEtienne Carriere 		io_write32(gd->gicd_base + GICD_ICPENDR(n), 0xffffffff);
554b0104773SPascal Brand 
555b0104773SPascal Brand 		/* Mark interrupts non-secure */
556bedc2b9fSsunny 		if (n == 0) {
557bedc2b9fSsunny 			/* per-CPU inerrupts config:
558bedc2b9fSsunny 			 * ID0-ID7(SGI)	  for Non-secure interrupts
559bedc2b9fSsunny 			 * ID8-ID15(SGI)  for Secure interrupts.
560bedc2b9fSsunny 			 * All PPI config as Non-secure interrupts.
561bedc2b9fSsunny 			 */
56205089e5fSJens Wiklander 			gd->per_cpu_group_status = 0xffff00ff;
56305089e5fSJens Wiklander 			gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
56405089e5fSJens Wiklander 			io_write32(gd->gicd_base + GICD_IGROUPR(n),
56505089e5fSJens Wiklander 				   gd->per_cpu_group_status);
566bedc2b9fSsunny 		} else {
567918bb3a5SEtienne Carriere 			io_write32(gd->gicd_base + GICD_IGROUPR(n), 0xffffffff);
568b0104773SPascal Brand 		}
569bedc2b9fSsunny 	}
570b0104773SPascal Brand 
57130a673e3SPeter Maydell 	/* Set the priority mask to permit Non-secure interrupts, and to
57230a673e3SPeter Maydell 	 * allow the Non-secure world to adjust the priority mask itself
57330a673e3SPeter Maydell 	 */
57418901324SDavid Wang #if defined(CFG_ARM_GICV3)
57518901324SDavid Wang 	write_icc_pmr(0x80);
5761fcac774SSandeep Tripathy 	write_icc_igrpen1(1);
5771fcac774SSandeep Tripathy 	io_setbits32(gd->gicd_base + GICD_CTLR, GICD_CTLR_ENABLEGRP1S);
57818901324SDavid Wang #else
579918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_PMR, 0x80);
58030a673e3SPeter Maydell 
581b0104773SPascal Brand 	/* Enable GIC */
582918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_CTLR, GICC_CTLR_FIQEN |
583918bb3a5SEtienne Carriere 		   GICC_CTLR_ENABLEGRP0 | GICC_CTLR_ENABLEGRP1);
584918bb3a5SEtienne Carriere 	io_setbits32(gd->gicd_base + GICD_CTLR,
58505089e5fSJens Wiklander 		     GICD_CTLR_ENABLEGRP0 | GICD_CTLR_ENABLEGRP1NS);
5861fcac774SSandeep Tripathy #endif
58705089e5fSJens Wiklander #endif /*!CFG_WITH_ARM_TRUSTED_FW*/
58867e55c51SEtienne Carriere 
58901980f3fSEtienne Carriere 	interrupt_main_init(&gic_data.chip);
590b0104773SPascal Brand }
591b0104773SPascal Brand 
5927315b7b4SJens Wiklander static void gic_it_add(struct gic_data *gd, size_t it)
593b0104773SPascal Brand {
59453bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
59553bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
596b0104773SPascal Brand 
59767e55c51SEtienne Carriere 	assert(gd == &gic_data);
59867e55c51SEtienne Carriere 
599b0104773SPascal Brand 	/* Disable the interrupt */
600918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICENABLER(idx), mask);
601b0104773SPascal Brand 	/* Make it non-pending */
602918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICPENDR(idx), mask);
603b0104773SPascal Brand 	/* Assign it to group0 */
604918bb3a5SEtienne Carriere 	io_clrbits32(gd->gicd_base + GICD_IGROUPR(idx), mask);
6051fcac774SSandeep Tripathy #if defined(CFG_ARM_GICV3)
6061fcac774SSandeep Tripathy 	/* Assign it to group1S */
6071fcac774SSandeep Tripathy 	io_setbits32(gd->gicd_base + GICD_IGROUPMODR(idx), mask);
6081fcac774SSandeep Tripathy #endif
609b0104773SPascal Brand }
610b0104773SPascal Brand 
6117315b7b4SJens Wiklander static void gic_it_set_cpu_mask(struct gic_data *gd, size_t it,
6127315b7b4SJens Wiklander 				uint8_t cpu_mask)
613b0104773SPascal Brand {
6148ddf5a4eSEtienne Carriere 	size_t idx __maybe_unused = it / NUM_INTS_PER_REG;
6158ddf5a4eSEtienne Carriere 	uint32_t mask __maybe_unused = 1 << (it % NUM_INTS_PER_REG);
61653bd332aSSY Chiu 	uint32_t target, target_shift;
617918bb3a5SEtienne Carriere 	vaddr_t itargetsr = gd->gicd_base +
618918bb3a5SEtienne Carriere 			    GICD_ITARGETSR(it / NUM_TARGETS_PER_REG);
619b0104773SPascal Brand 
62067e55c51SEtienne Carriere 	assert(gd == &gic_data);
62167e55c51SEtienne Carriere 
622b0104773SPascal Brand 	/* Assigned to group0 */
623918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
624b0104773SPascal Brand 
625b0104773SPascal Brand 	/* Route it to selected CPUs */
626918bb3a5SEtienne Carriere 	target = io_read32(itargetsr);
62753bd332aSSY Chiu 	target_shift = (it % NUM_TARGETS_PER_REG) * ITARGETSR_FIELD_BITS;
62853bd332aSSY Chiu 	target &= ~(ITARGETSR_FIELD_MASK << target_shift);
62953bd332aSSY Chiu 	target |= cpu_mask << target_shift;
630918bb3a5SEtienne Carriere 	DMSG("cpu_mask: writing 0x%x to 0x%" PRIxVA, target, itargetsr);
631918bb3a5SEtienne Carriere 	io_write32(itargetsr, target);
632918bb3a5SEtienne Carriere 	DMSG("cpu_mask: 0x%x", io_read32(itargetsr));
633b0104773SPascal Brand }
634b0104773SPascal Brand 
6357315b7b4SJens Wiklander static void gic_it_set_prio(struct gic_data *gd, size_t it, uint8_t prio)
636b0104773SPascal Brand {
6378ddf5a4eSEtienne Carriere 	size_t idx __maybe_unused = it / NUM_INTS_PER_REG;
6388ddf5a4eSEtienne Carriere 	uint32_t mask __maybe_unused = 1 << (it % NUM_INTS_PER_REG);
639b0104773SPascal Brand 
64067e55c51SEtienne Carriere 	assert(gd == &gic_data);
64167e55c51SEtienne Carriere 
642b0104773SPascal Brand 	/* Assigned to group0 */
643918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
644b0104773SPascal Brand 
645b0104773SPascal Brand 	/* Set prio it to selected CPUs */
6461f60363aSJens Wiklander 	DMSG("prio: writing 0x%x to 0x%" PRIxVA,
6477315b7b4SJens Wiklander 		prio, gd->gicd_base + GICD_IPRIORITYR(0) + it);
648918bb3a5SEtienne Carriere 	io_write8(gd->gicd_base + GICD_IPRIORITYR(0) + it, prio);
649b0104773SPascal Brand }
650b0104773SPascal Brand 
6517315b7b4SJens Wiklander static void gic_it_enable(struct gic_data *gd, size_t it)
652b0104773SPascal Brand {
65353bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
65453bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
655918bb3a5SEtienne Carriere 	vaddr_t base = gd->gicd_base;
656b0104773SPascal Brand 
65767e55c51SEtienne Carriere 	assert(gd == &gic_data);
65867e55c51SEtienne Carriere 
659b0104773SPascal Brand 	/* Assigned to group0 */
660918bb3a5SEtienne Carriere 	assert(!(io_read32(base + GICD_IGROUPR(idx)) & mask));
661b0104773SPascal Brand 
662b0104773SPascal Brand 	/* Enable the interrupt */
663918bb3a5SEtienne Carriere 	io_write32(base + GICD_ISENABLER(idx), mask);
664b0104773SPascal Brand }
665b0104773SPascal Brand 
6667315b7b4SJens Wiklander static void gic_it_disable(struct gic_data *gd, size_t it)
667b0104773SPascal Brand {
66853bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
66953bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
670b0104773SPascal Brand 
67167e55c51SEtienne Carriere 	assert(gd == &gic_data);
67267e55c51SEtienne Carriere 
673b0104773SPascal Brand 	/* Assigned to group0 */
674918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
675b0104773SPascal Brand 
676b0104773SPascal Brand 	/* Disable the interrupt */
677918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICENABLER(idx), mask);
678b0104773SPascal Brand }
679b0104773SPascal Brand 
68026ed70ecSGuanchao Liang static void gic_it_set_pending(struct gic_data *gd, size_t it)
68126ed70ecSGuanchao Liang {
68226ed70ecSGuanchao Liang 	size_t idx = it / NUM_INTS_PER_REG;
68326ed70ecSGuanchao Liang 	uint32_t mask = BIT32(it % NUM_INTS_PER_REG);
68426ed70ecSGuanchao Liang 
68567e55c51SEtienne Carriere 	assert(gd == &gic_data);
68667e55c51SEtienne Carriere 
68726ed70ecSGuanchao Liang 	/* Should be Peripheral Interrupt */
68826ed70ecSGuanchao Liang 	assert(it >= NUM_SGI);
68926ed70ecSGuanchao Liang 
69026ed70ecSGuanchao Liang 	/* Raise the interrupt */
691918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ISPENDR(idx), mask);
69226ed70ecSGuanchao Liang }
69326ed70ecSGuanchao Liang 
694ec740b9fSJens Wiklander static void assert_cpu_mask_is_valid(uint32_t cpu_mask)
695ec740b9fSJens Wiklander {
696ec740b9fSJens Wiklander 	bool __maybe_unused to_others = cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS;
697ec740b9fSJens Wiklander 	bool __maybe_unused to_current = cpu_mask & ITR_CPU_MASK_TO_THIS_CPU;
698ec740b9fSJens Wiklander 	bool __maybe_unused to_list = cpu_mask & 0xff;
699ec740b9fSJens Wiklander 
700ec740b9fSJens Wiklander 	/* One and only one of the bit fields shall be non-zero */
701ec740b9fSJens Wiklander 	assert(to_others + to_current + to_list == 1);
702ec740b9fSJens Wiklander }
703ec740b9fSJens Wiklander 
70454739cb4SMark-PK Tsai static void gic_it_raise_sgi(struct gic_data *gd __maybe_unused, size_t it,
70584603456SJens Wiklander 			     uint32_t cpu_mask, bool ns)
70626ed70ecSGuanchao Liang {
70754739cb4SMark-PK Tsai #if defined(CFG_ARM_GICV3)
70854739cb4SMark-PK Tsai 	uint32_t mask_id = it & 0xf;
709ec740b9fSJens Wiklander 	uint64_t mask = SHIFT_U64(mask_id, 24);
710ec740b9fSJens Wiklander 
711ec740b9fSJens Wiklander 	assert_cpu_mask_is_valid(cpu_mask);
712ec740b9fSJens Wiklander 
713ec740b9fSJens Wiklander 	if (cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS) {
714ec740b9fSJens Wiklander 		mask |= BIT64(GICC_SGI_IRM_BIT);
715ec740b9fSJens Wiklander 	} else {
71654739cb4SMark-PK Tsai 		uint64_t mpidr = read_mpidr();
717ec740b9fSJens Wiklander 		uint64_t mask_aff1 = (mpidr & MPIDR_AFF1_MASK) >>
718ec740b9fSJens Wiklander 				     MPIDR_AFF1_SHIFT;
719ec740b9fSJens Wiklander 		uint64_t mask_aff2 = (mpidr & MPIDR_AFF2_MASK) >>
720ec740b9fSJens Wiklander 				     MPIDR_AFF2_SHIFT;
721ec740b9fSJens Wiklander 		uint64_t mask_aff3 = (mpidr & MPIDR_AFF3_MASK) >>
722ec740b9fSJens Wiklander 				     MPIDR_AFF3_SHIFT;
723ec740b9fSJens Wiklander 
724ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff1, GICC_SGI_AFF1_SHIFT);
725ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff2, GICC_SGI_AFF2_SHIFT);
726ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff3, GICC_SGI_AFF3_SHIFT);
727ec740b9fSJens Wiklander 
728ec740b9fSJens Wiklander 		if (cpu_mask & ITR_CPU_MASK_TO_THIS_CPU) {
729ec740b9fSJens Wiklander 			mask |= BIT32(mpidr & 0xf);
730ec740b9fSJens Wiklander 		} else {
731ec740b9fSJens Wiklander 			/*
732ec740b9fSJens Wiklander 			 * Only support sending SGI to the cores in the
733ec740b9fSJens Wiklander 			 * same cluster now.
734ec740b9fSJens Wiklander 			 */
735ec740b9fSJens Wiklander 			mask |= cpu_mask & 0xff;
736ec740b9fSJens Wiklander 		}
737ec740b9fSJens Wiklander 	}
73854739cb4SMark-PK Tsai 
73954739cb4SMark-PK Tsai 	/* Raise the interrupt */
74084603456SJens Wiklander 	if (ns)
74154739cb4SMark-PK Tsai 		write_icc_asgi1r(mask);
74254739cb4SMark-PK Tsai 	else
74354739cb4SMark-PK Tsai 		write_icc_sgi1r(mask);
74454739cb4SMark-PK Tsai #else
745ec740b9fSJens Wiklander 	uint32_t mask_id = it & GICD_SGIR_SIGINTID_MASK;
74684603456SJens Wiklander 	uint32_t mask_group = ns;
747ec740b9fSJens Wiklander 	uint32_t mask = mask_id;
748ec740b9fSJens Wiklander 
749ec740b9fSJens Wiklander 	assert_cpu_mask_is_valid(cpu_mask);
750ec740b9fSJens Wiklander 
751ec740b9fSJens Wiklander 	mask |= SHIFT_U32(mask_group, GICD_SGIR_NSATT_SHIFT);
752ec740b9fSJens Wiklander 	if (cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS) {
753ec740b9fSJens Wiklander 		mask |= SHIFT_U32(GICD_SGIR_TO_OTHER_CPUS,
754ec740b9fSJens Wiklander 				  GICD_SGIR_TARGET_LIST_FILTER_SHIFT);
755ec740b9fSJens Wiklander 	} else if (cpu_mask & ITR_CPU_MASK_TO_THIS_CPU) {
756ec740b9fSJens Wiklander 		mask |= SHIFT_U32(GICD_SGIR_TO_THIS_CPU,
757ec740b9fSJens Wiklander 				  GICD_SGIR_TARGET_LIST_FILTER_SHIFT);
758ec740b9fSJens Wiklander 	} else {
759ec740b9fSJens Wiklander 		mask |= SHIFT_U32(cpu_mask & 0xff,
760ec740b9fSJens Wiklander 				  GICD_SGIR_CPU_TARGET_LIST_SHIFT);
761ec740b9fSJens Wiklander 	}
76226ed70ecSGuanchao Liang 
76326ed70ecSGuanchao Liang 	/* Raise the interrupt */
764918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_SGIR, mask);
76554739cb4SMark-PK Tsai #endif
76626ed70ecSGuanchao Liang }
76726ed70ecSGuanchao Liang 
76818901324SDavid Wang static uint32_t gic_read_iar(struct gic_data *gd __maybe_unused)
769b0104773SPascal Brand {
77067e55c51SEtienne Carriere 	assert(gd == &gic_data);
77167e55c51SEtienne Carriere 
77218901324SDavid Wang #if defined(CFG_ARM_GICV3)
7731de462e1SSumit Garg 	return read_icc_iar1();
77418901324SDavid Wang #else
775918bb3a5SEtienne Carriere 	return io_read32(gd->gicc_base + GICC_IAR);
77618901324SDavid Wang #endif
777b0104773SPascal Brand }
778b0104773SPascal Brand 
77918901324SDavid Wang static void gic_write_eoir(struct gic_data *gd __maybe_unused, uint32_t eoir)
780b0104773SPascal Brand {
78167e55c51SEtienne Carriere 	assert(gd == &gic_data);
78267e55c51SEtienne Carriere 
78318901324SDavid Wang #if defined(CFG_ARM_GICV3)
7841de462e1SSumit Garg 	write_icc_eoir1(eoir);
78518901324SDavid Wang #else
786918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_EOIR, eoir);
78718901324SDavid Wang #endif
788b0104773SPascal Brand }
789b0104773SPascal Brand 
7907315b7b4SJens Wiklander static bool gic_it_is_enabled(struct gic_data *gd, size_t it)
7917315b7b4SJens Wiklander {
79253bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
79353bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
79467e55c51SEtienne Carriere 
79567e55c51SEtienne Carriere 	assert(gd == &gic_data);
796918bb3a5SEtienne Carriere 	return !!(io_read32(gd->gicd_base + GICD_ISENABLER(idx)) & mask);
79753bd332aSSY Chiu }
79853bd332aSSY Chiu 
7997315b7b4SJens Wiklander static bool __maybe_unused gic_it_get_group(struct gic_data *gd, size_t it)
8007315b7b4SJens Wiklander {
80153bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
80253bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
80367e55c51SEtienne Carriere 
80467e55c51SEtienne Carriere 	assert(gd == &gic_data);
805918bb3a5SEtienne Carriere 	return !!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask);
80653bd332aSSY Chiu }
80753bd332aSSY Chiu 
8087315b7b4SJens Wiklander static uint32_t __maybe_unused gic_it_get_target(struct gic_data *gd, size_t it)
8097315b7b4SJens Wiklander {
81053bd332aSSY Chiu 	size_t reg_idx = it / NUM_TARGETS_PER_REG;
8117315b7b4SJens Wiklander 	uint32_t target_shift = (it % NUM_TARGETS_PER_REG) *
8127315b7b4SJens Wiklander 				ITARGETSR_FIELD_BITS;
81353bd332aSSY Chiu 	uint32_t target_mask = ITARGETSR_FIELD_MASK << target_shift;
814918bb3a5SEtienne Carriere 	uint32_t target = io_read32(gd->gicd_base + GICD_ITARGETSR(reg_idx));
8157315b7b4SJens Wiklander 
81667e55c51SEtienne Carriere 	assert(gd == &gic_data);
817918bb3a5SEtienne Carriere 	return (target & target_mask) >> target_shift;
81853bd332aSSY Chiu }
81953bd332aSSY Chiu 
82067e55c51SEtienne Carriere void gic_dump_state(void)
82153bd332aSSY Chiu {
82267e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
82367e55c51SEtienne Carriere 	int i = 0;
82453bd332aSSY Chiu 
82518901324SDavid Wang #if defined(CFG_ARM_GICV3)
82618901324SDavid Wang 	DMSG("GICC_CTLR: 0x%x", read_icc_ctlr());
82718901324SDavid Wang #else
828918bb3a5SEtienne Carriere 	DMSG("GICC_CTLR: 0x%x", io_read32(gd->gicc_base + GICC_CTLR));
82918901324SDavid Wang #endif
830918bb3a5SEtienne Carriere 	DMSG("GICD_CTLR: 0x%x", io_read32(gd->gicd_base + GICD_CTLR));
8317315b7b4SJens Wiklander 
8324a9ea08cSFangsuo Wu 	for (i = 0; i <= (int)gd->max_it; i++) {
8337315b7b4SJens Wiklander 		if (gic_it_is_enabled(gd, i)) {
83453bd332aSSY Chiu 			DMSG("irq%d: enabled, group:%d, target:%x", i,
8357315b7b4SJens Wiklander 			     gic_it_get_group(gd, i), gic_it_get_target(gd, i));
83653bd332aSSY Chiu 		}
83753bd332aSSY Chiu 	}
83853bd332aSSY Chiu }
8397315b7b4SJens Wiklander 
84067e55c51SEtienne Carriere static void __maybe_unused gic_native_itr_handler(void)
8417315b7b4SJens Wiklander {
84267e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
84367e55c51SEtienne Carriere 	uint32_t iar = 0;
84467e55c51SEtienne Carriere 	uint32_t id = 0;
8457315b7b4SJens Wiklander 
8467315b7b4SJens Wiklander 	iar = gic_read_iar(gd);
8477315b7b4SJens Wiklander 	id = iar & GICC_IAR_IT_ID_MASK;
8487315b7b4SJens Wiklander 
8494a9ea08cSFangsuo Wu 	if (id <= gd->max_it)
85099e2612cSEtienne Carriere 		interrupt_call_handlers(&gd->chip, id);
8513b3a4611SMathieu Briand 	else
8523b3a4611SMathieu Briand 		DMSG("ignoring interrupt %" PRIu32, id);
8537315b7b4SJens Wiklander 
8547315b7b4SJens Wiklander 	gic_write_eoir(gd, iar);
8557315b7b4SJens Wiklander }
8567315b7b4SJens Wiklander 
85767e55c51SEtienne Carriere #ifndef CFG_CORE_WORKAROUND_ARM_NMFI
858358bf47cSEtienne Carriere /* Override interrupt_main_handler() with driver implementation */
859358bf47cSEtienne Carriere void interrupt_main_handler(void)
86067e55c51SEtienne Carriere {
86167e55c51SEtienne Carriere 	gic_native_itr_handler();
86267e55c51SEtienne Carriere }
86367e55c51SEtienne Carriere #endif /*CFG_CORE_WORKAROUND_ARM_NMFI*/
86467e55c51SEtienne Carriere 
8657315b7b4SJens Wiklander static void gic_op_add(struct itr_chip *chip, size_t it,
866702fe5a7SClément Léger 		       uint32_t type __unused,
867702fe5a7SClément Léger 		       uint32_t prio __unused)
8687315b7b4SJens Wiklander {
8697315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
8707315b7b4SJens Wiklander 
87167e55c51SEtienne Carriere 	assert(gd == &gic_data);
87267e55c51SEtienne Carriere 
8734a9ea08cSFangsuo Wu 	if (it > gd->max_it)
874d13278b8SEtienne Carriere 		panic();
875d13278b8SEtienne Carriere 
876*9e935234SJens Wiklander 	if (it < GIC_SPI_BASE) {
877*9e935234SJens Wiklander 		if (gic_primary_done)
878*9e935234SJens Wiklander 			panic("Cannot add SGI or PPI after boot");
879*9e935234SJens Wiklander 
880*9e935234SJens Wiklander 		/* Assign it to Secure Group 1, G1S */
881*9e935234SJens Wiklander 		gd->per_cpu_group_modifier |= BIT32(it);
882*9e935234SJens Wiklander 		gd->per_cpu_group_status &= ~BIT32(it);
883*9e935234SJens Wiklander 	}
884*9e935234SJens Wiklander 
885*9e935234SJens Wiklander 	if (it < GIC_SPI_BASE && affinity_routing_is_enabled(gd)) {
886*9e935234SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
887*9e935234SJens Wiklander 
888*9e935234SJens Wiklander 		if (!gicr_base)
889*9e935234SJens Wiklander 			panic("GICR_BASE missing");
890*9e935234SJens Wiklander 
891*9e935234SJens Wiklander 		/* Disable interrupt */
892*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(it));
893*9e935234SJens Wiklander 
894*9e935234SJens Wiklander 		/* Make interrupt non-pending */
895*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(it));
896*9e935234SJens Wiklander 
897*9e935234SJens Wiklander 		/* Make it to Secure */
898*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, gd->per_cpu_group_status);
899*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0,
900*9e935234SJens Wiklander 			   gd->per_cpu_group_modifier);
901*9e935234SJens Wiklander 	} else {
9027315b7b4SJens Wiklander 		gic_it_add(gd, it);
9037315b7b4SJens Wiklander 		/* Set the CPU mask to deliver interrupts to any online core */
9047315b7b4SJens Wiklander 		gic_it_set_cpu_mask(gd, it, 0xff);
9057315b7b4SJens Wiklander 		gic_it_set_prio(gd, it, 0x1);
9067315b7b4SJens Wiklander 	}
907*9e935234SJens Wiklander }
9087315b7b4SJens Wiklander 
9097315b7b4SJens Wiklander static void gic_op_enable(struct itr_chip *chip, size_t it)
9107315b7b4SJens Wiklander {
9117315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
9127315b7b4SJens Wiklander 
91367e55c51SEtienne Carriere 	assert(gd == &gic_data);
91467e55c51SEtienne Carriere 
9154a9ea08cSFangsuo Wu 	if (it > gd->max_it)
916d13278b8SEtienne Carriere 		panic();
917d13278b8SEtienne Carriere 
918*9e935234SJens Wiklander 	if (it < GIC_SPI_BASE)
919*9e935234SJens Wiklander 		gd->per_cpu_enable |= BIT(it);
920*9e935234SJens Wiklander 
921*9e935234SJens Wiklander 	if (it < GIC_SPI_BASE && affinity_routing_is_enabled(gd)) {
922*9e935234SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
923*9e935234SJens Wiklander 
924*9e935234SJens Wiklander 		if (!gicr_base)
925*9e935234SJens Wiklander 			panic("GICR_BASE missing");
926*9e935234SJens Wiklander 
927*9e935234SJens Wiklander 		/* Assigned to G1S */
928*9e935234SJens Wiklander 		assert(gd->per_cpu_group_modifier & BIT(it) &&
929*9e935234SJens Wiklander 		       !(gd->per_cpu_group_status & BIT(it)));
930*9e935234SJens Wiklander 		io_write32(gicr_base + GICR_ISENABLER0, gd->per_cpu_enable);
931*9e935234SJens Wiklander 	} else {
9327315b7b4SJens Wiklander 		gic_it_enable(gd, it);
9337315b7b4SJens Wiklander 	}
934*9e935234SJens Wiklander }
9357315b7b4SJens Wiklander 
9367315b7b4SJens Wiklander static void gic_op_disable(struct itr_chip *chip, size_t it)
9377315b7b4SJens Wiklander {
9387315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
9397315b7b4SJens Wiklander 
94067e55c51SEtienne Carriere 	assert(gd == &gic_data);
94167e55c51SEtienne Carriere 
9424a9ea08cSFangsuo Wu 	if (it > gd->max_it)
943d13278b8SEtienne Carriere 		panic();
944d13278b8SEtienne Carriere 
9457315b7b4SJens Wiklander 	gic_it_disable(gd, it);
9467315b7b4SJens Wiklander }
94726ed70ecSGuanchao Liang 
94826ed70ecSGuanchao Liang static void gic_op_raise_pi(struct itr_chip *chip, size_t it)
94926ed70ecSGuanchao Liang {
95026ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
95126ed70ecSGuanchao Liang 
95267e55c51SEtienne Carriere 	assert(gd == &gic_data);
95367e55c51SEtienne Carriere 
9544a9ea08cSFangsuo Wu 	if (it > gd->max_it)
95526ed70ecSGuanchao Liang 		panic();
95626ed70ecSGuanchao Liang 
95726ed70ecSGuanchao Liang 	gic_it_set_pending(gd, it);
95826ed70ecSGuanchao Liang }
95926ed70ecSGuanchao Liang 
96026ed70ecSGuanchao Liang static void gic_op_raise_sgi(struct itr_chip *chip, size_t it,
961ec740b9fSJens Wiklander 			     uint32_t cpu_mask)
96226ed70ecSGuanchao Liang {
96326ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
96484603456SJens Wiklander 	bool ns = false;
96526ed70ecSGuanchao Liang 
96667e55c51SEtienne Carriere 	assert(gd == &gic_data);
96767e55c51SEtienne Carriere 
96854739cb4SMark-PK Tsai 	/* Should be Software Generated Interrupt */
96954739cb4SMark-PK Tsai 	assert(it < NUM_SGI);
97054739cb4SMark-PK Tsai 
97184603456SJens Wiklander 	ns = BIT32(it) & gd->per_cpu_group_status;
97284603456SJens Wiklander 	gic_it_raise_sgi(gd, it, cpu_mask, ns);
97326ed70ecSGuanchao Liang }
97467e55c51SEtienne Carriere 
97526ed70ecSGuanchao Liang static void gic_op_set_affinity(struct itr_chip *chip, size_t it,
97626ed70ecSGuanchao Liang 			uint8_t cpu_mask)
97726ed70ecSGuanchao Liang {
97826ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
97926ed70ecSGuanchao Liang 
98067e55c51SEtienne Carriere 	assert(gd == &gic_data);
98167e55c51SEtienne Carriere 
9824a9ea08cSFangsuo Wu 	if (it > gd->max_it)
98326ed70ecSGuanchao Liang 		panic();
98426ed70ecSGuanchao Liang 
98526ed70ecSGuanchao Liang 	gic_it_set_cpu_mask(gd, it, cpu_mask);
98626ed70ecSGuanchao Liang }
98714885eb1SEtienne Carriere 
98814885eb1SEtienne Carriere #ifdef CFG_DT
98914885eb1SEtienne Carriere /* Callback for "interrupts" and "interrupts-extended" DT node properties */
99014885eb1SEtienne Carriere static TEE_Result dt_get_gic_chip_cb(struct dt_pargs *arg, void *priv_data,
99114885eb1SEtienne Carriere 				     struct itr_desc *itr_desc)
99214885eb1SEtienne Carriere {
99314885eb1SEtienne Carriere 	int itr_num = DT_INFO_INVALID_INTERRUPT;
99414885eb1SEtienne Carriere 	struct itr_chip *chip = priv_data;
99514885eb1SEtienne Carriere 	uint32_t phandle_args[2] = { };
99614885eb1SEtienne Carriere 	uint32_t type = 0;
99714885eb1SEtienne Carriere 	uint32_t prio = 0;
99814885eb1SEtienne Carriere 
99914885eb1SEtienne Carriere 	assert(arg && itr_desc);
100014885eb1SEtienne Carriere 
100114885eb1SEtienne Carriere 	/*
100214885eb1SEtienne Carriere 	 * gic_dt_get_irq() expects phandle arguments passed are still in DT
100314885eb1SEtienne Carriere 	 * format (big-endian) whereas struct dt_pargs carries converted
100414885eb1SEtienne Carriere 	 * formats. Therefore swap again phandle arguments. gic_dt_get_irq()
100514885eb1SEtienne Carriere 	 * consumes only the 2 first arguments.
100614885eb1SEtienne Carriere 	 */
100714885eb1SEtienne Carriere 	if (arg->args_count < 2)
100814885eb1SEtienne Carriere 		return TEE_ERROR_GENERIC;
100914885eb1SEtienne Carriere 	phandle_args[0] = cpu_to_fdt32(arg->args[0]);
101014885eb1SEtienne Carriere 	phandle_args[1] = cpu_to_fdt32(arg->args[1]);
101114885eb1SEtienne Carriere 
101214885eb1SEtienne Carriere 	itr_num = gic_dt_get_irq((const void *)phandle_args, 2, &type, &prio);
101314885eb1SEtienne Carriere 	if (itr_num == DT_INFO_INVALID_INTERRUPT)
101414885eb1SEtienne Carriere 		return TEE_ERROR_GENERIC;
101514885eb1SEtienne Carriere 
101614885eb1SEtienne Carriere 	gic_op_add(chip, itr_num, type, prio);
101714885eb1SEtienne Carriere 
101814885eb1SEtienne Carriere 	itr_desc->chip = chip;
101914885eb1SEtienne Carriere 	itr_desc->itr_num = itr_num;
102014885eb1SEtienne Carriere 
102114885eb1SEtienne Carriere 	return TEE_SUCCESS;
102214885eb1SEtienne Carriere }
102314885eb1SEtienne Carriere 
102414885eb1SEtienne Carriere static TEE_Result gic_probe(const void *fdt, int offs, const void *cd __unused)
102514885eb1SEtienne Carriere {
102614885eb1SEtienne Carriere 	if (interrupt_register_provider(fdt, offs, dt_get_gic_chip_cb,
102714885eb1SEtienne Carriere 					&gic_data.chip))
102814885eb1SEtienne Carriere 		panic();
102914885eb1SEtienne Carriere 
103014885eb1SEtienne Carriere 	return TEE_SUCCESS;
103114885eb1SEtienne Carriere }
103214885eb1SEtienne Carriere 
103314885eb1SEtienne Carriere static const struct dt_device_match gic_match_table[] = {
103414885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a15-gic" },
103514885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a7-gic" },
103614885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a5-gic" },
103714885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a9-gic" },
103814885eb1SEtienne Carriere 	{ .compatible = "arm,gic-400" },
103914885eb1SEtienne Carriere 	{ }
104014885eb1SEtienne Carriere };
104114885eb1SEtienne Carriere 
104214885eb1SEtienne Carriere DEFINE_DT_DRIVER(gic_dt_driver) = {
104314885eb1SEtienne Carriere 	.name = "gic",
104414885eb1SEtienne Carriere 	.match_table = gic_match_table,
104514885eb1SEtienne Carriere 	.probe = gic_probe,
104614885eb1SEtienne Carriere };
104714885eb1SEtienne Carriere #endif /*CFG_DT*/
1048*9e935234SJens Wiklander 
1049*9e935234SJens Wiklander static TEE_Result gic_set_primary_done(void)
1050*9e935234SJens Wiklander {
1051*9e935234SJens Wiklander 	gic_primary_done = true;
1052*9e935234SJens Wiklander 	return TEE_SUCCESS;
1053*9e935234SJens Wiklander }
1054*9e935234SJens Wiklander 
1055*9e935234SJens Wiklander nex_release_init_resource(gic_set_primary_done);
1056