xref: /optee_os/core/drivers/gic.c (revision dd18bd87893402b3372b469f95442a747d776ecd)
11bb92983SJerome Forissier // SPDX-License-Identifier: BSD-2-Clause
2b0104773SPascal Brand /*
39e935234SJens 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>
129e935234SJens Wiklander #include <dt-bindings/interrupt-controller/arm-gic.h>
139e935234SJens 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)
739e935234SJens Wiklander #define GICR_ISENABLER0		(GICR_SGI_BASE_OFFSET + 0x100)
749e935234SJens Wiklander #define GICR_ICFGR0		(GICR_SGI_BASE_OFFSET + 0xC00)
759e935234SJens Wiklander #define GICR_ICFGR1		(GICR_SGI_BASE_OFFSET + 0xC04)
769e935234SJens Wiklander #define GICR_IPRIORITYR(n)	(GICR_SGI_BASE_OFFSET + 0x400 + (n) * 4)
779e935234SJens Wiklander 
786c2d2e8aSJens Wiklander #define GICR_CTLR_RWP		BIT32(3)
7905089e5fSJens Wiklander 
8005089e5fSJens Wiklander #define GICR_TYPER_LAST		BIT64(4)
8105089e5fSJens Wiklander #define GICR_TYPER_AFF3_SHIFT	56
8205089e5fSJens Wiklander #define GICR_TYPER_AFF2_SHIFT	48
8305089e5fSJens Wiklander #define GICR_TYPER_AFF1_SHIFT	40
8405089e5fSJens Wiklander #define GICR_TYPER_AFF0_SHIFT	32
85b0104773SPascal Brand 
8669171becSJens Wiklander /* GICD IDR2 name differs on GICv3 and GICv2 but uses same bit map */
8769171becSJens Wiklander #define GICD_PIDR2_ARCHREV_SHIFT	4
8869171becSJens Wiklander #define GICD_PIDR2_ARCHREV_MASK		0xF
8969171becSJens Wiklander 
9053bd332aSSY Chiu /* Number of Private Peripheral Interrupt */
9153bd332aSSY Chiu #define NUM_PPI	32
9253bd332aSSY Chiu 
9326ed70ecSGuanchao Liang /* Number of Software Generated Interrupt */
9426ed70ecSGuanchao Liang #define NUM_SGI			16
9526ed70ecSGuanchao Liang 
9626ed70ecSGuanchao Liang /* Number of Non-secure Software Generated Interrupt */
9726ed70ecSGuanchao Liang #define NUM_NS_SGI		8
9826ed70ecSGuanchao Liang 
9953bd332aSSY Chiu /* Number of interrupts in one register */
10053bd332aSSY Chiu #define NUM_INTS_PER_REG	32
10153bd332aSSY Chiu 
10253bd332aSSY Chiu /* Number of targets in one register */
10353bd332aSSY Chiu #define NUM_TARGETS_PER_REG	4
10453bd332aSSY Chiu 
10553bd332aSSY Chiu /* Accessors to access ITARGETSRn */
10653bd332aSSY Chiu #define ITARGETSR_FIELD_BITS	8
10753bd332aSSY Chiu #define ITARGETSR_FIELD_MASK	0xff
10853bd332aSSY Chiu 
1091b4c5002SIzhar Nevo #define GICD_TYPER_IT_LINES_NUM_MASK	0x1f
1107315b7b4SJens Wiklander #define GICC_IAR_IT_ID_MASK	0x3ff
1117315b7b4SJens Wiklander #define GICC_IAR_CPU_ID_MASK	0x7
1127315b7b4SJens Wiklander #define GICC_IAR_CPU_ID_SHIFT	10
113b0104773SPascal Brand 
114ec740b9fSJens Wiklander #define GICC_SGI_IRM_BIT	40
115ec740b9fSJens Wiklander #define GICC_SGI_AFF1_SHIFT	16
116ec740b9fSJens Wiklander #define GICC_SGI_AFF2_SHIFT	32
117ec740b9fSJens Wiklander #define GICC_SGI_AFF3_SHIFT	48
118ec740b9fSJens Wiklander 
119ec740b9fSJens Wiklander #define GICD_SGIR_SIGINTID_MASK			0xf
120ec740b9fSJens Wiklander #define GICD_SGIR_TO_OTHER_CPUS			0x1
121ec740b9fSJens Wiklander #define GICD_SGIR_TO_THIS_CPU			0x2
122ec740b9fSJens Wiklander #define GICD_SGIR_TARGET_LIST_FILTER_SHIFT	24
123ec740b9fSJens Wiklander #define GICD_SGIR_NSATT_SHIFT			15
124ec740b9fSJens Wiklander #define GICD_SGIR_CPU_TARGET_LIST_SHIFT		16
125ec740b9fSJens Wiklander 
12667e55c51SEtienne Carriere struct gic_data {
12767e55c51SEtienne Carriere 	vaddr_t gicc_base;
12867e55c51SEtienne Carriere 	vaddr_t gicd_base;
12905089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
13005089e5fSJens Wiklander 	vaddr_t gicr_base[CFG_TEE_CORE_NB_CORE];
13105089e5fSJens Wiklander #endif
13267e55c51SEtienne Carriere 	size_t max_it;
13305089e5fSJens Wiklander 	uint32_t per_cpu_group_status;
13405089e5fSJens Wiklander 	uint32_t per_cpu_group_modifier;
1359e935234SJens Wiklander 	uint32_t per_cpu_enable;
13667e55c51SEtienne Carriere 	struct itr_chip chip;
13767e55c51SEtienne Carriere };
13867e55c51SEtienne Carriere 
1399e935234SJens Wiklander static bool gic_primary_done __nex_bss;
14067e55c51SEtienne Carriere static struct gic_data gic_data __nex_bss;
14167e55c51SEtienne Carriere 
142702fe5a7SClément Léger static void gic_op_add(struct itr_chip *chip, size_t it, uint32_t type,
143702fe5a7SClément Léger 		       uint32_t prio);
1447315b7b4SJens Wiklander static void gic_op_enable(struct itr_chip *chip, size_t it);
1457315b7b4SJens Wiklander static void gic_op_disable(struct itr_chip *chip, size_t it);
14626ed70ecSGuanchao Liang static void gic_op_raise_pi(struct itr_chip *chip, size_t it);
14726ed70ecSGuanchao Liang static void gic_op_raise_sgi(struct itr_chip *chip, size_t it,
148ec740b9fSJens Wiklander 			     uint32_t cpu_mask);
14926ed70ecSGuanchao Liang static void gic_op_set_affinity(struct itr_chip *chip, size_t it,
15026ed70ecSGuanchao Liang 			uint8_t cpu_mask);
1517315b7b4SJens Wiklander 
1527315b7b4SJens Wiklander static const struct itr_ops gic_ops = {
1537315b7b4SJens Wiklander 	.add = gic_op_add,
15408ded0e1SEtienne Carriere 	.mask = gic_op_disable,
15508ded0e1SEtienne Carriere 	.unmask = gic_op_enable,
1567315b7b4SJens Wiklander 	.enable = gic_op_enable,
1577315b7b4SJens Wiklander 	.disable = gic_op_disable,
15826ed70ecSGuanchao Liang 	.raise_pi = gic_op_raise_pi,
15926ed70ecSGuanchao Liang 	.raise_sgi = gic_op_raise_sgi,
16026ed70ecSGuanchao Liang 	.set_affinity = gic_op_set_affinity,
1617315b7b4SJens Wiklander };
1623639b55fSJerome Forissier DECLARE_KEEP_PAGER(gic_ops);
1637315b7b4SJens Wiklander 
16405089e5fSJens Wiklander static vaddr_t __maybe_unused get_gicr_base(struct gic_data *gd __maybe_unused)
16505089e5fSJens Wiklander {
16605089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
16705089e5fSJens Wiklander 	return gd->gicr_base[get_core_pos()];
16805089e5fSJens Wiklander #else
16905089e5fSJens Wiklander 	return 0;
17005089e5fSJens Wiklander #endif
17105089e5fSJens Wiklander }
17205089e5fSJens Wiklander 
1737c4883aeSJens Wiklander static bool affinity_routing_is_enabled(struct gic_data *gd)
1747c4883aeSJens Wiklander {
1757c4883aeSJens Wiklander 	return IS_ENABLED(CFG_ARM_GICV3) &&
1767c4883aeSJens Wiklander 	       io_read32(gd->gicd_base + GICD_CTLR) & GICD_CTLR_ARE_S;
1777c4883aeSJens Wiklander }
1787c4883aeSJens Wiklander 
17918901324SDavid Wang static size_t probe_max_it(vaddr_t gicc_base __maybe_unused, vaddr_t gicd_base)
180b0104773SPascal Brand {
181b0104773SPascal Brand 	int i;
182b0104773SPascal Brand 	uint32_t old_ctlr;
183b0104773SPascal Brand 	size_t ret = 0;
1841b4c5002SIzhar Nevo 	size_t max_regs = io_read32(gicd_base + GICD_TYPER) &
1851b4c5002SIzhar Nevo 			  GICD_TYPER_IT_LINES_NUM_MASK;
186b0104773SPascal Brand 
187b0104773SPascal Brand 	/*
188b0104773SPascal Brand 	 * Probe which interrupt number is the largest.
189b0104773SPascal Brand 	 */
19018901324SDavid Wang #if defined(CFG_ARM_GICV3)
19118901324SDavid Wang 	old_ctlr = read_icc_ctlr();
19218901324SDavid Wang 	write_icc_ctlr(0);
19318901324SDavid Wang #else
194918bb3a5SEtienne Carriere 	old_ctlr = io_read32(gicc_base + GICC_CTLR);
195918bb3a5SEtienne Carriere 	io_write32(gicc_base + GICC_CTLR, 0);
19618901324SDavid Wang #endif
19779f008d3SJens Wiklander 	for (i = max_regs; i >= 0; i--) {
198b0104773SPascal Brand 		uint32_t old_reg;
199b0104773SPascal Brand 		uint32_t reg;
200b0104773SPascal Brand 		int b;
201b0104773SPascal Brand 
202918bb3a5SEtienne Carriere 		old_reg = io_read32(gicd_base + GICD_ISENABLER(i));
203918bb3a5SEtienne Carriere 		io_write32(gicd_base + GICD_ISENABLER(i), 0xffffffff);
204918bb3a5SEtienne Carriere 		reg = io_read32(gicd_base + GICD_ISENABLER(i));
205918bb3a5SEtienne Carriere 		io_write32(gicd_base + GICD_ICENABLER(i), ~old_reg);
20679f008d3SJens Wiklander 		for (b = NUM_INTS_PER_REG - 1; b >= 0; b--) {
207007a97a2SJens Wiklander 			if (BIT32(b) & reg) {
20853bd332aSSY Chiu 				ret = i * NUM_INTS_PER_REG + b;
209b0104773SPascal Brand 				goto out;
210b0104773SPascal Brand 			}
211b0104773SPascal Brand 		}
212b0104773SPascal Brand 	}
213b0104773SPascal Brand out:
21418901324SDavid Wang #if defined(CFG_ARM_GICV3)
21518901324SDavid Wang 	write_icc_ctlr(old_ctlr);
21618901324SDavid Wang #else
217918bb3a5SEtienne Carriere 	io_write32(gicc_base + GICC_CTLR, old_ctlr);
21818901324SDavid Wang #endif
219b0104773SPascal Brand 	return ret;
220b0104773SPascal Brand }
221b0104773SPascal Brand 
2226c2d2e8aSJens Wiklander static void gicr_wait_for_pending_write(vaddr_t gicr_base)
2236c2d2e8aSJens Wiklander {
2246c2d2e8aSJens Wiklander 	/*
2256c2d2e8aSJens Wiklander 	 * Wait for changes to
2266c2d2e8aSJens Wiklander 	 * - GICR_ICENABLER0
2276c2d2e8aSJens Wiklander 	 * - GICR_CTLR.DPG1S
2286c2d2e8aSJens Wiklander 	 * - GICR_CTLR.DPG1NS
2296c2d2e8aSJens Wiklander 	 * - GICR_CTLR.DPG0
2306c2d2e8aSJens Wiklander 	 * to be visible to all agents in the system.
2316c2d2e8aSJens Wiklander 	 */
2326c2d2e8aSJens Wiklander 	while (io_read32(gicr_base + GICR_CTLR) & GICR_CTLR_RWP)
2336c2d2e8aSJens Wiklander 		;
2346c2d2e8aSJens Wiklander }
2356c2d2e8aSJens Wiklander 
2369e935234SJens Wiklander static void gicv3_sync_redist_config(struct gic_data *gd)
23784603456SJens Wiklander {
23884603456SJens Wiklander 	vaddr_t gicr_base = get_gicr_base(gd);
23984603456SJens Wiklander 	bool need_sync = false;
24084603456SJens Wiklander 	uint32_t gmod0 = 0;
24184603456SJens Wiklander 	uint32_t grp0 = 0;
24284603456SJens Wiklander 	size_t n = 0;
24384603456SJens Wiklander 
2447c4883aeSJens Wiklander 	/*
2457c4883aeSJens Wiklander 	 * If gicr_base isn't available there's no need to synchronize SGI
2467c4883aeSJens Wiklander 	 * configuration since gic_init_donate_sgi_to_ns() would panic.
2477c4883aeSJens Wiklander 	 */
24884603456SJens Wiklander 	if (!gicr_base)
2497c4883aeSJens Wiklander 		return;
25084603456SJens Wiklander 
25184603456SJens Wiklander 	grp0 = io_read32(gicr_base + GICR_IGROUPR0);
25284603456SJens Wiklander 	gmod0 = io_read32(gicr_base + GICR_IGRPMODR0);
2539e935234SJens Wiklander 	for (n = GIC_SGI_SEC_BASE; n < GIC_SPI_BASE; n++) {
25484603456SJens Wiklander 		/* Ignore matching bits */
25584603456SJens Wiklander 		if (!(BIT32(n) & (grp0 ^ gd->per_cpu_group_status)) &&
25684603456SJens Wiklander 		    !(BIT32(n) & (gmod0 ^ gd->per_cpu_group_modifier)))
25784603456SJens Wiklander 			continue;
25884603456SJens Wiklander 		/*
2599e935234SJens Wiklander 		 * SGI/PPI-n differs from primary CPU configuration,
26084603456SJens Wiklander 		 * let's sync up.
26184603456SJens Wiklander 		 */
26284603456SJens Wiklander 		need_sync = true;
26384603456SJens Wiklander 
26484603456SJens Wiklander 		/* Disable interrupt */
26584603456SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(n));
26684603456SJens Wiklander 
2676c2d2e8aSJens Wiklander 		/* Wait for the write to GICR_ICENABLER0 to propagate */
2686c2d2e8aSJens Wiklander 		gicr_wait_for_pending_write(gicr_base);
2696c2d2e8aSJens Wiklander 
27084603456SJens Wiklander 		/* Make interrupt non-pending */
27184603456SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(n));
27284603456SJens Wiklander 
27384603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_status)
27484603456SJens Wiklander 			grp0 |= BIT32(n);
27584603456SJens Wiklander 		else
27684603456SJens Wiklander 			grp0 &= ~BIT32(n);
27784603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_modifier)
27884603456SJens Wiklander 			gmod0 |= BIT32(n);
27984603456SJens Wiklander 		else
28084603456SJens Wiklander 			gmod0 &= ~BIT32(n);
28184603456SJens Wiklander 	}
28284603456SJens Wiklander 
28384603456SJens Wiklander 	if (need_sync) {
28484603456SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, grp0);
28584603456SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0, gmod0);
2869e935234SJens Wiklander 		io_write32(gicr_base + GICR_ISENABLER0, gd->per_cpu_enable);
28784603456SJens Wiklander 	}
28884603456SJens Wiklander }
28984603456SJens Wiklander 
2909e935234SJens Wiklander static void gic_legacy_sync_dist_config(struct gic_data *gd)
29184603456SJens Wiklander {
29284603456SJens Wiklander 	bool need_sync = false;
29384603456SJens Wiklander 	uint32_t grp0 = 0;
29484603456SJens Wiklander 	size_t n = 0;
29584603456SJens Wiklander 
29684603456SJens Wiklander 	grp0 = io_read32(gd->gicd_base + GICD_IGROUPR(0));
2979e935234SJens Wiklander 	for (n = GIC_SGI_SEC_BASE; n < GIC_SPI_BASE; n++) {
29884603456SJens Wiklander 		/* Ignore matching bits */
29984603456SJens Wiklander 		if (!(BIT32(n) & (grp0 ^ gd->per_cpu_group_status)))
30084603456SJens Wiklander 			continue;
30184603456SJens Wiklander 		/*
3029e935234SJens Wiklander 		 * SGI/PPI-n differs from primary CPU configuration,
30384603456SJens Wiklander 		 * let's sync up.
30484603456SJens Wiklander 		 */
30584603456SJens Wiklander 		need_sync = true;
30684603456SJens Wiklander 
30784603456SJens Wiklander 		/* Disable interrupt */
30884603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICENABLER(0), BIT(n));
30984603456SJens Wiklander 
31084603456SJens Wiklander 		/* Make interrupt non-pending */
31184603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICPENDR(0), BIT(n));
31284603456SJens Wiklander 
31384603456SJens Wiklander 		if (BIT32(n) & gd->per_cpu_group_status)
31484603456SJens Wiklander 			grp0 |= BIT32(n);
31584603456SJens Wiklander 		else
31684603456SJens Wiklander 			grp0 &= ~BIT32(n);
31784603456SJens Wiklander 	}
31884603456SJens Wiklander 
3199e935234SJens Wiklander 	if (need_sync) {
32084603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_IGROUPR(0), grp0);
3219e935234SJens Wiklander 		io_write32(gd->gicd_base + GICD_ISENABLER(0),
3229e935234SJens Wiklander 			   gd->per_cpu_enable);
3239e935234SJens Wiklander 	}
32484603456SJens Wiklander }
32584603456SJens Wiklander 
3265da157f5SJens Wiklander static void init_gic_per_cpu(struct gic_data *gd)
327bedc2b9fSsunny {
32805089e5fSJens Wiklander 	io_write32(gd->gicd_base + GICD_IGROUPR(0), gd->per_cpu_group_status);
329bedc2b9fSsunny 
33005089e5fSJens Wiklander 	/*
33105089e5fSJens Wiklander 	 * Set the priority mask to permit Non-secure interrupts, and to
33230a673e3SPeter Maydell 	 * allow the Non-secure world to adjust the priority mask itself
33330a673e3SPeter Maydell 	 */
33418901324SDavid Wang #if defined(CFG_ARM_GICV3)
33518901324SDavid Wang 	write_icc_pmr(0x80);
3361fcac774SSandeep Tripathy 	write_icc_igrpen1(1);
33718901324SDavid Wang #else
338918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_PMR, 0x80);
33930a673e3SPeter Maydell 
340bedc2b9fSsunny 	/* Enable GIC */
341918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_CTLR,
342918bb3a5SEtienne Carriere 		   GICC_CTLR_ENABLEGRP0 | GICC_CTLR_ENABLEGRP1 |
343918bb3a5SEtienne Carriere 		   GICC_CTLR_FIQEN);
34418901324SDavid Wang #endif
345bedc2b9fSsunny }
346bedc2b9fSsunny 
3475da157f5SJens Wiklander void gic_init_per_cpu(void)
3485da157f5SJens Wiklander {
3495da157f5SJens Wiklander 	struct gic_data *gd = &gic_data;
3505da157f5SJens Wiklander 
3515da157f5SJens Wiklander #if defined(CFG_ARM_GICV3)
3525da157f5SJens Wiklander 	assert(gd->gicd_base);
3535da157f5SJens Wiklander #else
3545da157f5SJens Wiklander 	assert(gd->gicd_base && gd->gicc_base);
3555da157f5SJens Wiklander #endif
3565da157f5SJens Wiklander 
35784603456SJens Wiklander 	if (IS_ENABLED(CFG_WITH_ARM_TRUSTED_FW)) {
35884603456SJens Wiklander 		/*
35984603456SJens Wiklander 		 * GIC is already initialized by TF-A, we only need to
3609e935234SJens Wiklander 		 * handle eventual SGI or PPI configuration changes.
36184603456SJens Wiklander 		 */
3627c4883aeSJens Wiklander 		if (affinity_routing_is_enabled(gd))
3639e935234SJens Wiklander 			gicv3_sync_redist_config(gd);
36484603456SJens Wiklander 		else
3659e935234SJens Wiklander 			gic_legacy_sync_dist_config(gd);
36684603456SJens Wiklander 	} else {
36784603456SJens Wiklander 		/*
36884603456SJens Wiklander 		 * Non-TF-A case where all CPU specific configuration
36984603456SJens Wiklander 		 * of GIC must be done here.
37084603456SJens Wiklander 		 */
3715da157f5SJens Wiklander 		init_gic_per_cpu(gd);
3725da157f5SJens Wiklander 	}
37384603456SJens Wiklander }
3745da157f5SJens Wiklander 
37584603456SJens Wiklander void gic_init_donate_sgi_to_ns(size_t it)
37684603456SJens Wiklander {
37784603456SJens Wiklander 	struct gic_data *gd = &gic_data;
37884603456SJens Wiklander 
37984603456SJens Wiklander 	assert(it >= GIC_SGI_SEC_BASE && it <= GIC_SGI_SEC_MAX);
38084603456SJens Wiklander 
38184603456SJens Wiklander 	/* Assert it's secure to start with. */
38284603456SJens Wiklander 	assert(!(gd->per_cpu_group_status & BIT32(it)) &&
38384603456SJens Wiklander 	       (gd->per_cpu_group_modifier & BIT32(it)));
38484603456SJens Wiklander 
38584603456SJens Wiklander 	gd->per_cpu_group_modifier &= ~BIT32(it);
38684603456SJens Wiklander 	gd->per_cpu_group_status |= BIT32(it);
38784603456SJens Wiklander 
3887c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd)) {
38984603456SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
39084603456SJens Wiklander 
3917c4883aeSJens Wiklander 		if (!gicr_base)
3927c4883aeSJens Wiklander 			panic("GICR_BASE missing");
3937c4883aeSJens Wiklander 
39484603456SJens Wiklander 		/* Disable interrupt */
39584603456SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(it));
39684603456SJens Wiklander 
3976c2d2e8aSJens Wiklander 		/* Wait for the write to GICR_ICENABLER0 to propagate */
3986c2d2e8aSJens Wiklander 		gicr_wait_for_pending_write(gicr_base);
3996c2d2e8aSJens Wiklander 
40084603456SJens Wiklander 		/* Make interrupt non-pending */
40184603456SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(it));
40284603456SJens Wiklander 
40384603456SJens Wiklander 		/* Make it to non-secure */
40484603456SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, gd->per_cpu_group_status);
40584603456SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0,
40684603456SJens Wiklander 			   gd->per_cpu_group_modifier);
40784603456SJens Wiklander 	} else {
40884603456SJens Wiklander 		/* Disable interrupt */
40984603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICENABLER(0), BIT(it));
41084603456SJens Wiklander 
41184603456SJens Wiklander 		/* Make interrupt non-pending */
41284603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_ICPENDR(0), BIT(it));
41384603456SJens Wiklander 
41484603456SJens Wiklander 		/* Make it to non-secure */
41584603456SJens Wiklander 		io_write32(gd->gicd_base + GICD_IGROUPR(0),
41684603456SJens Wiklander 			   gd->per_cpu_group_status);
41784603456SJens Wiklander 	}
41884603456SJens Wiklander }
41984603456SJens Wiklander 
4200ee3f52eSEtienne Carriere static int gic_dt_get_irq(const uint32_t *properties, int count, uint32_t *type,
4210ee3f52eSEtienne Carriere 			  uint32_t *prio)
4220ee3f52eSEtienne Carriere {
4230ee3f52eSEtienne Carriere 	int it_num = DT_INFO_INVALID_INTERRUPT;
4240ee3f52eSEtienne Carriere 
4250ee3f52eSEtienne Carriere 	if (type)
4260ee3f52eSEtienne Carriere 		*type = IRQ_TYPE_NONE;
4270ee3f52eSEtienne Carriere 
4280ee3f52eSEtienne Carriere 	if (prio)
4290ee3f52eSEtienne Carriere 		*prio = 0;
4300ee3f52eSEtienne Carriere 
4310ee3f52eSEtienne Carriere 	if (!properties || count < 2)
4320ee3f52eSEtienne Carriere 		return DT_INFO_INVALID_INTERRUPT;
4330ee3f52eSEtienne Carriere 
4340ee3f52eSEtienne Carriere 	it_num = fdt32_to_cpu(properties[1]);
4350ee3f52eSEtienne Carriere 
4360ee3f52eSEtienne Carriere 	switch (fdt32_to_cpu(properties[0])) {
4378c7282beSEtienne Carriere 	case GIC_PPI:
4380ee3f52eSEtienne Carriere 		it_num += 16;
4390ee3f52eSEtienne Carriere 		break;
4408c7282beSEtienne Carriere 	case GIC_SPI:
4410ee3f52eSEtienne Carriere 		it_num += 32;
4420ee3f52eSEtienne Carriere 		break;
4430ee3f52eSEtienne Carriere 	default:
4440ee3f52eSEtienne Carriere 		it_num = DT_INFO_INVALID_INTERRUPT;
4450ee3f52eSEtienne Carriere 	}
4460ee3f52eSEtienne Carriere 
4470ee3f52eSEtienne Carriere 	return it_num;
4480ee3f52eSEtienne Carriere }
4490ee3f52eSEtienne Carriere 
45005089e5fSJens Wiklander static void __maybe_unused probe_redist_base_addrs(vaddr_t *gicr_base_addrs,
45105089e5fSJens Wiklander 						   paddr_t gicr_base_pa)
45205089e5fSJens Wiklander {
45305089e5fSJens Wiklander 	size_t sz = GICR_V3_PCPUBASE_SIZE;
45405089e5fSJens Wiklander 	paddr_t pa = gicr_base_pa;
45505089e5fSJens Wiklander 	size_t core_pos = 0;
45605089e5fSJens Wiklander 	uint64_t mt_bit = 0;
45705089e5fSJens Wiklander 	uint64_t mpidr = 0;
45805089e5fSJens Wiklander 	uint64_t tv = 0;
45905089e5fSJens Wiklander 	vaddr_t va = 0;
46005089e5fSJens Wiklander 
46105089e5fSJens Wiklander #ifdef ARM64
46205089e5fSJens Wiklander 	mt_bit = read_mpidr_el1() & MPIDR_MT_MASK;
46305089e5fSJens Wiklander #endif
46405089e5fSJens Wiklander 	do {
46505089e5fSJens Wiklander 		va = core_mmu_get_va(pa, MEM_AREA_IO_SEC, sz);
46605089e5fSJens Wiklander 		if (!va)
46705089e5fSJens Wiklander 			panic();
46805089e5fSJens Wiklander 		tv = io_read64(va + GICR_TYPER);
46905089e5fSJens Wiklander 
47005089e5fSJens Wiklander 		/*
47105089e5fSJens Wiklander 		 * Extract an mpidr from the Type register to calculate the
47205089e5fSJens Wiklander 		 * core position of this redistributer instance.
47305089e5fSJens Wiklander 		 */
47405089e5fSJens Wiklander 		mpidr = mt_bit;
47505089e5fSJens Wiklander 		mpidr |= SHIFT_U64((tv >> GICR_TYPER_AFF3_SHIFT) &
47605089e5fSJens Wiklander 				   MPIDR_AFFLVL_MASK, MPIDR_AFF3_SHIFT);
47705089e5fSJens Wiklander 		mpidr |= (tv >> GICR_TYPER_AFF0_SHIFT) &
47805089e5fSJens Wiklander 			 (MPIDR_AFF0_MASK | MPIDR_AFF1_MASK | MPIDR_AFF2_MASK);
47905089e5fSJens Wiklander 		core_pos = get_core_pos_mpidr(mpidr);
48005089e5fSJens Wiklander 		if (core_pos < CFG_TEE_CORE_NB_CORE) {
48105089e5fSJens Wiklander 			DMSG("GICR_BASE[%zu] at %#"PRIxVA, core_pos, va);
48205089e5fSJens Wiklander 			gicr_base_addrs[core_pos] = va;
48305089e5fSJens Wiklander 		} else {
48405089e5fSJens Wiklander 			EMSG("Skipping too large core_pos %zu from GICR_TYPER",
48505089e5fSJens Wiklander 			     core_pos);
48605089e5fSJens Wiklander 		}
48705089e5fSJens Wiklander 		pa += sz;
48805089e5fSJens Wiklander 	} while (!(tv & GICR_TYPER_LAST));
48905089e5fSJens Wiklander }
49005089e5fSJens Wiklander 
49105089e5fSJens Wiklander static void gic_init_base_addr(paddr_t gicc_base_pa, paddr_t gicd_base_pa,
49205089e5fSJens Wiklander 			       paddr_t gicr_base_pa __maybe_unused)
493b0104773SPascal Brand {
49467e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
4950ee3f52eSEtienne Carriere 	vaddr_t gicc_base = 0;
4960ee3f52eSEtienne Carriere 	vaddr_t gicd_base = 0;
49769171becSJens Wiklander 	uint32_t vers __maybe_unused = 0;
4980ee3f52eSEtienne Carriere 
4990ee3f52eSEtienne Carriere 	assert(cpu_mmu_enabled());
5000ee3f52eSEtienne Carriere 
5010ee3f52eSEtienne Carriere 	gicd_base = core_mmu_get_va(gicd_base_pa, MEM_AREA_IO_SEC,
5020ee3f52eSEtienne Carriere 				    GIC_DIST_REG_SIZE);
5030ee3f52eSEtienne Carriere 	if (!gicd_base)
5040ee3f52eSEtienne Carriere 		panic();
5050ee3f52eSEtienne Carriere 
50669171becSJens Wiklander 	vers = io_read32(gicd_base + GICD_PIDR2);
50769171becSJens Wiklander 	vers >>= GICD_PIDR2_ARCHREV_SHIFT;
50869171becSJens Wiklander 	vers &= GICD_PIDR2_ARCHREV_MASK;
50969171becSJens Wiklander 
51069171becSJens Wiklander 	if (IS_ENABLED(CFG_ARM_GICV3)) {
511*dd18bd87SZiad Elhanafy 		assert(vers == 4 || vers == 3);
51269171becSJens Wiklander 	} else {
513d3f6526eSJens Wiklander 		assert(vers == 2 || vers == 1);
5140ee3f52eSEtienne Carriere 		gicc_base = core_mmu_get_va(gicc_base_pa, MEM_AREA_IO_SEC,
5150ee3f52eSEtienne Carriere 					    GIC_CPU_REG_SIZE);
5160ee3f52eSEtienne Carriere 		if (!gicc_base)
5170ee3f52eSEtienne Carriere 			panic();
5180ee3f52eSEtienne Carriere 	}
5190ee3f52eSEtienne Carriere 
5200ee3f52eSEtienne Carriere 	gd->gicc_base = gicc_base;
5210ee3f52eSEtienne Carriere 	gd->gicd_base = gicd_base;
5220ee3f52eSEtienne Carriere 	gd->max_it = probe_max_it(gicc_base, gicd_base);
52305089e5fSJens Wiklander #if defined(CFG_ARM_GICV3)
5247c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd) && gicr_base_pa)
52505089e5fSJens Wiklander 		probe_redist_base_addrs(gd->gicr_base, gicr_base_pa);
52605089e5fSJens Wiklander #endif
5270ee3f52eSEtienne Carriere 	gd->chip.ops = &gic_ops;
5280ee3f52eSEtienne Carriere 
5290ee3f52eSEtienne Carriere 	if (IS_ENABLED(CFG_DT))
5300ee3f52eSEtienne Carriere 		gd->chip.dt_get_irq = gic_dt_get_irq;
5310ee3f52eSEtienne Carriere }
5320ee3f52eSEtienne Carriere 
53305089e5fSJens Wiklander void gic_init_v3(paddr_t gicc_base_pa, paddr_t gicd_base_pa,
53405089e5fSJens Wiklander 		 paddr_t gicr_base_pa)
5350ee3f52eSEtienne Carriere {
5360ee3f52eSEtienne Carriere 	struct gic_data __maybe_unused *gd = &gic_data;
5370ee3f52eSEtienne Carriere 	size_t __maybe_unused n = 0;
538b0104773SPascal Brand 
53905089e5fSJens Wiklander 	gic_init_base_addr(gicc_base_pa, gicd_base_pa, gicr_base_pa);
540b0104773SPascal Brand 
54105089e5fSJens Wiklander #if defined(CFG_WITH_ARM_TRUSTED_FW)
5420ee3f52eSEtienne Carriere 	/* GIC configuration is initialized from TF-A when embedded */
5437c4883aeSJens Wiklander 	if (affinity_routing_is_enabled(gd)) {
5447c4883aeSJens Wiklander 		/* Secure affinity routing enabled */
54505089e5fSJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
54605089e5fSJens Wiklander 
5477c4883aeSJens Wiklander 		if (gicr_base) {
5487c4883aeSJens Wiklander 			gd->per_cpu_group_status = io_read32(gicr_base +
5497c4883aeSJens Wiklander 							     GICR_IGROUPR0);
55005089e5fSJens Wiklander 			gd->per_cpu_group_modifier = io_read32(gicr_base +
55105089e5fSJens Wiklander 							       GICR_IGRPMODR0);
55205089e5fSJens Wiklander 		} else {
5537c4883aeSJens Wiklander 			IMSG("GIC redistributor base address not provided");
5547c4883aeSJens Wiklander 			IMSG("Assuming default GIC group status and modifier");
5557c4883aeSJens Wiklander 			gd->per_cpu_group_status = 0xffff00ff;
5567c4883aeSJens Wiklander 			gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
5577c4883aeSJens Wiklander 		}
5587c4883aeSJens Wiklander 	} else {
55905089e5fSJens Wiklander 		/* Legacy operation with secure affinity routing disabled */
56005089e5fSJens Wiklander 		gd->per_cpu_group_status = io_read32(gd->gicd_base +
56105089e5fSJens Wiklander 						     GICD_IGROUPR(0));
56205089e5fSJens Wiklander 		gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
56305089e5fSJens Wiklander 	}
56405089e5fSJens Wiklander #else /*!CFG_WITH_ARM_TRUSTED_FW*/
56505089e5fSJens Wiklander 	/*
56605089e5fSJens Wiklander 	 * Without TF-A, GIC is always configured in for legacy operation
56705089e5fSJens Wiklander 	 * with secure affinity routing disabled.
56805089e5fSJens Wiklander 	 */
5697315b7b4SJens Wiklander 	for (n = 0; n <= gd->max_it / NUM_INTS_PER_REG; n++) {
570b0104773SPascal Brand 		/* Disable interrupts */
571918bb3a5SEtienne Carriere 		io_write32(gd->gicd_base + GICD_ICENABLER(n), 0xffffffff);
572b0104773SPascal Brand 
573b0104773SPascal Brand 		/* Make interrupts non-pending */
574918bb3a5SEtienne Carriere 		io_write32(gd->gicd_base + GICD_ICPENDR(n), 0xffffffff);
575b0104773SPascal Brand 
576b0104773SPascal Brand 		/* Mark interrupts non-secure */
577bedc2b9fSsunny 		if (n == 0) {
578bedc2b9fSsunny 			/* per-CPU inerrupts config:
579bedc2b9fSsunny 			 * ID0-ID7(SGI)	  for Non-secure interrupts
580bedc2b9fSsunny 			 * ID8-ID15(SGI)  for Secure interrupts.
581bedc2b9fSsunny 			 * All PPI config as Non-secure interrupts.
582bedc2b9fSsunny 			 */
58305089e5fSJens Wiklander 			gd->per_cpu_group_status = 0xffff00ff;
58405089e5fSJens Wiklander 			gd->per_cpu_group_modifier = ~gd->per_cpu_group_status;
58505089e5fSJens Wiklander 			io_write32(gd->gicd_base + GICD_IGROUPR(n),
58605089e5fSJens Wiklander 				   gd->per_cpu_group_status);
587bedc2b9fSsunny 		} else {
588918bb3a5SEtienne Carriere 			io_write32(gd->gicd_base + GICD_IGROUPR(n), 0xffffffff);
589b0104773SPascal Brand 		}
590bedc2b9fSsunny 	}
591b0104773SPascal Brand 
59230a673e3SPeter Maydell 	/* Set the priority mask to permit Non-secure interrupts, and to
59330a673e3SPeter Maydell 	 * allow the Non-secure world to adjust the priority mask itself
59430a673e3SPeter Maydell 	 */
59518901324SDavid Wang #if defined(CFG_ARM_GICV3)
59618901324SDavid Wang 	write_icc_pmr(0x80);
5971fcac774SSandeep Tripathy 	write_icc_igrpen1(1);
5981fcac774SSandeep Tripathy 	io_setbits32(gd->gicd_base + GICD_CTLR, GICD_CTLR_ENABLEGRP1S);
59918901324SDavid Wang #else
600918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_PMR, 0x80);
60130a673e3SPeter Maydell 
602b0104773SPascal Brand 	/* Enable GIC */
603918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_CTLR, GICC_CTLR_FIQEN |
604918bb3a5SEtienne Carriere 		   GICC_CTLR_ENABLEGRP0 | GICC_CTLR_ENABLEGRP1);
605918bb3a5SEtienne Carriere 	io_setbits32(gd->gicd_base + GICD_CTLR,
60605089e5fSJens Wiklander 		     GICD_CTLR_ENABLEGRP0 | GICD_CTLR_ENABLEGRP1NS);
6071fcac774SSandeep Tripathy #endif
60805089e5fSJens Wiklander #endif /*!CFG_WITH_ARM_TRUSTED_FW*/
60967e55c51SEtienne Carriere 
61001980f3fSEtienne Carriere 	interrupt_main_init(&gic_data.chip);
611b0104773SPascal Brand }
612b0104773SPascal Brand 
6137315b7b4SJens Wiklander static void gic_it_add(struct gic_data *gd, size_t it)
614b0104773SPascal Brand {
61553bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
61653bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
617b0104773SPascal Brand 
61867e55c51SEtienne Carriere 	assert(gd == &gic_data);
61967e55c51SEtienne Carriere 
620b0104773SPascal Brand 	/* Disable the interrupt */
621918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICENABLER(idx), mask);
622b0104773SPascal Brand 	/* Make it non-pending */
623918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICPENDR(idx), mask);
624b0104773SPascal Brand 	/* Assign it to group0 */
625918bb3a5SEtienne Carriere 	io_clrbits32(gd->gicd_base + GICD_IGROUPR(idx), mask);
6261fcac774SSandeep Tripathy #if defined(CFG_ARM_GICV3)
6271fcac774SSandeep Tripathy 	/* Assign it to group1S */
6281fcac774SSandeep Tripathy 	io_setbits32(gd->gicd_base + GICD_IGROUPMODR(idx), mask);
6291fcac774SSandeep Tripathy #endif
630b0104773SPascal Brand }
631b0104773SPascal Brand 
6327315b7b4SJens Wiklander static void gic_it_set_cpu_mask(struct gic_data *gd, size_t it,
6337315b7b4SJens Wiklander 				uint8_t cpu_mask)
634b0104773SPascal Brand {
6358ddf5a4eSEtienne Carriere 	size_t idx __maybe_unused = it / NUM_INTS_PER_REG;
6368ddf5a4eSEtienne Carriere 	uint32_t mask __maybe_unused = 1 << (it % NUM_INTS_PER_REG);
63753bd332aSSY Chiu 	uint32_t target, target_shift;
638918bb3a5SEtienne Carriere 	vaddr_t itargetsr = gd->gicd_base +
639918bb3a5SEtienne Carriere 			    GICD_ITARGETSR(it / NUM_TARGETS_PER_REG);
640b0104773SPascal Brand 
64167e55c51SEtienne Carriere 	assert(gd == &gic_data);
64267e55c51SEtienne Carriere 
643b0104773SPascal Brand 	/* Assigned to group0 */
644918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
645b0104773SPascal Brand 
646b0104773SPascal Brand 	/* Route it to selected CPUs */
647918bb3a5SEtienne Carriere 	target = io_read32(itargetsr);
64853bd332aSSY Chiu 	target_shift = (it % NUM_TARGETS_PER_REG) * ITARGETSR_FIELD_BITS;
64953bd332aSSY Chiu 	target &= ~(ITARGETSR_FIELD_MASK << target_shift);
65053bd332aSSY Chiu 	target |= cpu_mask << target_shift;
651918bb3a5SEtienne Carriere 	DMSG("cpu_mask: writing 0x%x to 0x%" PRIxVA, target, itargetsr);
652918bb3a5SEtienne Carriere 	io_write32(itargetsr, target);
653918bb3a5SEtienne Carriere 	DMSG("cpu_mask: 0x%x", io_read32(itargetsr));
654b0104773SPascal Brand }
655b0104773SPascal Brand 
6567315b7b4SJens Wiklander static void gic_it_set_prio(struct gic_data *gd, size_t it, uint8_t prio)
657b0104773SPascal Brand {
6588ddf5a4eSEtienne Carriere 	size_t idx __maybe_unused = it / NUM_INTS_PER_REG;
6598ddf5a4eSEtienne Carriere 	uint32_t mask __maybe_unused = 1 << (it % NUM_INTS_PER_REG);
660b0104773SPascal Brand 
66167e55c51SEtienne Carriere 	assert(gd == &gic_data);
66267e55c51SEtienne Carriere 
663b0104773SPascal Brand 	/* Assigned to group0 */
664918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
665b0104773SPascal Brand 
666b0104773SPascal Brand 	/* Set prio it to selected CPUs */
6671f60363aSJens Wiklander 	DMSG("prio: writing 0x%x to 0x%" PRIxVA,
6687315b7b4SJens Wiklander 		prio, gd->gicd_base + GICD_IPRIORITYR(0) + it);
669918bb3a5SEtienne Carriere 	io_write8(gd->gicd_base + GICD_IPRIORITYR(0) + it, prio);
670b0104773SPascal Brand }
671b0104773SPascal Brand 
6727315b7b4SJens Wiklander static void gic_it_enable(struct gic_data *gd, size_t it)
673b0104773SPascal Brand {
67453bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
67553bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
676918bb3a5SEtienne Carriere 	vaddr_t base = gd->gicd_base;
677b0104773SPascal Brand 
67867e55c51SEtienne Carriere 	assert(gd == &gic_data);
67967e55c51SEtienne Carriere 
680b0104773SPascal Brand 	/* Assigned to group0 */
681918bb3a5SEtienne Carriere 	assert(!(io_read32(base + GICD_IGROUPR(idx)) & mask));
682b0104773SPascal Brand 
683b0104773SPascal Brand 	/* Enable the interrupt */
684918bb3a5SEtienne Carriere 	io_write32(base + GICD_ISENABLER(idx), mask);
685b0104773SPascal Brand }
686b0104773SPascal Brand 
6877315b7b4SJens Wiklander static void gic_it_disable(struct gic_data *gd, size_t it)
688b0104773SPascal Brand {
68953bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
69053bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
691b0104773SPascal Brand 
69267e55c51SEtienne Carriere 	assert(gd == &gic_data);
69367e55c51SEtienne Carriere 
694b0104773SPascal Brand 	/* Assigned to group0 */
695918bb3a5SEtienne Carriere 	assert(!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask));
696b0104773SPascal Brand 
697b0104773SPascal Brand 	/* Disable the interrupt */
698918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ICENABLER(idx), mask);
699b0104773SPascal Brand }
700b0104773SPascal Brand 
70126ed70ecSGuanchao Liang static void gic_it_set_pending(struct gic_data *gd, size_t it)
70226ed70ecSGuanchao Liang {
70326ed70ecSGuanchao Liang 	size_t idx = it / NUM_INTS_PER_REG;
70426ed70ecSGuanchao Liang 	uint32_t mask = BIT32(it % NUM_INTS_PER_REG);
70526ed70ecSGuanchao Liang 
70667e55c51SEtienne Carriere 	assert(gd == &gic_data);
70767e55c51SEtienne Carriere 
70826ed70ecSGuanchao Liang 	/* Should be Peripheral Interrupt */
70926ed70ecSGuanchao Liang 	assert(it >= NUM_SGI);
71026ed70ecSGuanchao Liang 
71126ed70ecSGuanchao Liang 	/* Raise the interrupt */
712918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_ISPENDR(idx), mask);
71326ed70ecSGuanchao Liang }
71426ed70ecSGuanchao Liang 
715ec740b9fSJens Wiklander static void assert_cpu_mask_is_valid(uint32_t cpu_mask)
716ec740b9fSJens Wiklander {
717ec740b9fSJens Wiklander 	bool __maybe_unused to_others = cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS;
718ec740b9fSJens Wiklander 	bool __maybe_unused to_current = cpu_mask & ITR_CPU_MASK_TO_THIS_CPU;
719ec740b9fSJens Wiklander 	bool __maybe_unused to_list = cpu_mask & 0xff;
720ec740b9fSJens Wiklander 
721ec740b9fSJens Wiklander 	/* One and only one of the bit fields shall be non-zero */
722ec740b9fSJens Wiklander 	assert(to_others + to_current + to_list == 1);
723ec740b9fSJens Wiklander }
724ec740b9fSJens Wiklander 
72554739cb4SMark-PK Tsai static void gic_it_raise_sgi(struct gic_data *gd __maybe_unused, size_t it,
72684603456SJens Wiklander 			     uint32_t cpu_mask, bool ns)
72726ed70ecSGuanchao Liang {
72854739cb4SMark-PK Tsai #if defined(CFG_ARM_GICV3)
72954739cb4SMark-PK Tsai 	uint32_t mask_id = it & 0xf;
730ec740b9fSJens Wiklander 	uint64_t mask = SHIFT_U64(mask_id, 24);
731ec740b9fSJens Wiklander 
732ec740b9fSJens Wiklander 	assert_cpu_mask_is_valid(cpu_mask);
733ec740b9fSJens Wiklander 
734ec740b9fSJens Wiklander 	if (cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS) {
735ec740b9fSJens Wiklander 		mask |= BIT64(GICC_SGI_IRM_BIT);
736ec740b9fSJens Wiklander 	} else {
73754739cb4SMark-PK Tsai 		uint64_t mpidr = read_mpidr();
738ec740b9fSJens Wiklander 		uint64_t mask_aff1 = (mpidr & MPIDR_AFF1_MASK) >>
739ec740b9fSJens Wiklander 				     MPIDR_AFF1_SHIFT;
740ec740b9fSJens Wiklander 		uint64_t mask_aff2 = (mpidr & MPIDR_AFF2_MASK) >>
741ec740b9fSJens Wiklander 				     MPIDR_AFF2_SHIFT;
742ec740b9fSJens Wiklander 		uint64_t mask_aff3 = (mpidr & MPIDR_AFF3_MASK) >>
743ec740b9fSJens Wiklander 				     MPIDR_AFF3_SHIFT;
744ec740b9fSJens Wiklander 
745ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff1, GICC_SGI_AFF1_SHIFT);
746ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff2, GICC_SGI_AFF2_SHIFT);
747ec740b9fSJens Wiklander 		mask |= SHIFT_U64(mask_aff3, GICC_SGI_AFF3_SHIFT);
748ec740b9fSJens Wiklander 
749ec740b9fSJens Wiklander 		if (cpu_mask & ITR_CPU_MASK_TO_THIS_CPU) {
750ec740b9fSJens Wiklander 			mask |= BIT32(mpidr & 0xf);
751ec740b9fSJens Wiklander 		} else {
752ec740b9fSJens Wiklander 			/*
753ec740b9fSJens Wiklander 			 * Only support sending SGI to the cores in the
754ec740b9fSJens Wiklander 			 * same cluster now.
755ec740b9fSJens Wiklander 			 */
756ec740b9fSJens Wiklander 			mask |= cpu_mask & 0xff;
757ec740b9fSJens Wiklander 		}
758ec740b9fSJens Wiklander 	}
75954739cb4SMark-PK Tsai 
76054739cb4SMark-PK Tsai 	/* Raise the interrupt */
76184603456SJens Wiklander 	if (ns)
76254739cb4SMark-PK Tsai 		write_icc_asgi1r(mask);
76354739cb4SMark-PK Tsai 	else
76454739cb4SMark-PK Tsai 		write_icc_sgi1r(mask);
76554739cb4SMark-PK Tsai #else
766ec740b9fSJens Wiklander 	uint32_t mask_id = it & GICD_SGIR_SIGINTID_MASK;
76784603456SJens Wiklander 	uint32_t mask_group = ns;
768ec740b9fSJens Wiklander 	uint32_t mask = mask_id;
769ec740b9fSJens Wiklander 
770ec740b9fSJens Wiklander 	assert_cpu_mask_is_valid(cpu_mask);
771ec740b9fSJens Wiklander 
772ec740b9fSJens Wiklander 	mask |= SHIFT_U32(mask_group, GICD_SGIR_NSATT_SHIFT);
773ec740b9fSJens Wiklander 	if (cpu_mask & ITR_CPU_MASK_TO_OTHER_CPUS) {
774ec740b9fSJens Wiklander 		mask |= SHIFT_U32(GICD_SGIR_TO_OTHER_CPUS,
775ec740b9fSJens Wiklander 				  GICD_SGIR_TARGET_LIST_FILTER_SHIFT);
776ec740b9fSJens Wiklander 	} else if (cpu_mask & ITR_CPU_MASK_TO_THIS_CPU) {
777ec740b9fSJens Wiklander 		mask |= SHIFT_U32(GICD_SGIR_TO_THIS_CPU,
778ec740b9fSJens Wiklander 				  GICD_SGIR_TARGET_LIST_FILTER_SHIFT);
779ec740b9fSJens Wiklander 	} else {
780ec740b9fSJens Wiklander 		mask |= SHIFT_U32(cpu_mask & 0xff,
781ec740b9fSJens Wiklander 				  GICD_SGIR_CPU_TARGET_LIST_SHIFT);
782ec740b9fSJens Wiklander 	}
78326ed70ecSGuanchao Liang 
78426ed70ecSGuanchao Liang 	/* Raise the interrupt */
785918bb3a5SEtienne Carriere 	io_write32(gd->gicd_base + GICD_SGIR, mask);
78654739cb4SMark-PK Tsai #endif
78726ed70ecSGuanchao Liang }
78826ed70ecSGuanchao Liang 
78918901324SDavid Wang static uint32_t gic_read_iar(struct gic_data *gd __maybe_unused)
790b0104773SPascal Brand {
79167e55c51SEtienne Carriere 	assert(gd == &gic_data);
79267e55c51SEtienne Carriere 
79318901324SDavid Wang #if defined(CFG_ARM_GICV3)
7941de462e1SSumit Garg 	return read_icc_iar1();
79518901324SDavid Wang #else
796918bb3a5SEtienne Carriere 	return io_read32(gd->gicc_base + GICC_IAR);
79718901324SDavid Wang #endif
798b0104773SPascal Brand }
799b0104773SPascal Brand 
80018901324SDavid Wang static void gic_write_eoir(struct gic_data *gd __maybe_unused, uint32_t eoir)
801b0104773SPascal Brand {
80267e55c51SEtienne Carriere 	assert(gd == &gic_data);
80367e55c51SEtienne Carriere 
80418901324SDavid Wang #if defined(CFG_ARM_GICV3)
8051de462e1SSumit Garg 	write_icc_eoir1(eoir);
80618901324SDavid Wang #else
807918bb3a5SEtienne Carriere 	io_write32(gd->gicc_base + GICC_EOIR, eoir);
80818901324SDavid Wang #endif
809b0104773SPascal Brand }
810b0104773SPascal Brand 
8117315b7b4SJens Wiklander static bool gic_it_is_enabled(struct gic_data *gd, size_t it)
8127315b7b4SJens Wiklander {
81353bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
81453bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
81567e55c51SEtienne Carriere 
81667e55c51SEtienne Carriere 	assert(gd == &gic_data);
817918bb3a5SEtienne Carriere 	return !!(io_read32(gd->gicd_base + GICD_ISENABLER(idx)) & mask);
81853bd332aSSY Chiu }
81953bd332aSSY Chiu 
8207315b7b4SJens Wiklander static bool __maybe_unused gic_it_get_group(struct gic_data *gd, size_t it)
8217315b7b4SJens Wiklander {
82253bd332aSSY Chiu 	size_t idx = it / NUM_INTS_PER_REG;
82353bd332aSSY Chiu 	uint32_t mask = 1 << (it % NUM_INTS_PER_REG);
82467e55c51SEtienne Carriere 
82567e55c51SEtienne Carriere 	assert(gd == &gic_data);
826918bb3a5SEtienne Carriere 	return !!(io_read32(gd->gicd_base + GICD_IGROUPR(idx)) & mask);
82753bd332aSSY Chiu }
82853bd332aSSY Chiu 
8297315b7b4SJens Wiklander static uint32_t __maybe_unused gic_it_get_target(struct gic_data *gd, size_t it)
8307315b7b4SJens Wiklander {
83153bd332aSSY Chiu 	size_t reg_idx = it / NUM_TARGETS_PER_REG;
8327315b7b4SJens Wiklander 	uint32_t target_shift = (it % NUM_TARGETS_PER_REG) *
8337315b7b4SJens Wiklander 				ITARGETSR_FIELD_BITS;
83453bd332aSSY Chiu 	uint32_t target_mask = ITARGETSR_FIELD_MASK << target_shift;
835918bb3a5SEtienne Carriere 	uint32_t target = io_read32(gd->gicd_base + GICD_ITARGETSR(reg_idx));
8367315b7b4SJens Wiklander 
83767e55c51SEtienne Carriere 	assert(gd == &gic_data);
838918bb3a5SEtienne Carriere 	return (target & target_mask) >> target_shift;
83953bd332aSSY Chiu }
84053bd332aSSY Chiu 
84167e55c51SEtienne Carriere void gic_dump_state(void)
84253bd332aSSY Chiu {
84367e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
84467e55c51SEtienne Carriere 	int i = 0;
84553bd332aSSY Chiu 
84618901324SDavid Wang #if defined(CFG_ARM_GICV3)
84718901324SDavid Wang 	DMSG("GICC_CTLR: 0x%x", read_icc_ctlr());
84818901324SDavid Wang #else
849918bb3a5SEtienne Carriere 	DMSG("GICC_CTLR: 0x%x", io_read32(gd->gicc_base + GICC_CTLR));
85018901324SDavid Wang #endif
851918bb3a5SEtienne Carriere 	DMSG("GICD_CTLR: 0x%x", io_read32(gd->gicd_base + GICD_CTLR));
8527315b7b4SJens Wiklander 
8534a9ea08cSFangsuo Wu 	for (i = 0; i <= (int)gd->max_it; i++) {
8547315b7b4SJens Wiklander 		if (gic_it_is_enabled(gd, i)) {
85553bd332aSSY Chiu 			DMSG("irq%d: enabled, group:%d, target:%x", i,
8567315b7b4SJens Wiklander 			     gic_it_get_group(gd, i), gic_it_get_target(gd, i));
85753bd332aSSY Chiu 		}
85853bd332aSSY Chiu 	}
85953bd332aSSY Chiu }
8607315b7b4SJens Wiklander 
86167e55c51SEtienne Carriere static void __maybe_unused gic_native_itr_handler(void)
8627315b7b4SJens Wiklander {
86367e55c51SEtienne Carriere 	struct gic_data *gd = &gic_data;
86467e55c51SEtienne Carriere 	uint32_t iar = 0;
86567e55c51SEtienne Carriere 	uint32_t id = 0;
8667315b7b4SJens Wiklander 
8677315b7b4SJens Wiklander 	iar = gic_read_iar(gd);
8687315b7b4SJens Wiklander 	id = iar & GICC_IAR_IT_ID_MASK;
8697315b7b4SJens Wiklander 
8704a9ea08cSFangsuo Wu 	if (id <= gd->max_it)
87199e2612cSEtienne Carriere 		interrupt_call_handlers(&gd->chip, id);
8723b3a4611SMathieu Briand 	else
8733b3a4611SMathieu Briand 		DMSG("ignoring interrupt %" PRIu32, id);
8747315b7b4SJens Wiklander 
8757315b7b4SJens Wiklander 	gic_write_eoir(gd, iar);
8767315b7b4SJens Wiklander }
8777315b7b4SJens Wiklander 
87867e55c51SEtienne Carriere #ifndef CFG_CORE_WORKAROUND_ARM_NMFI
879358bf47cSEtienne Carriere /* Override interrupt_main_handler() with driver implementation */
880358bf47cSEtienne Carriere void interrupt_main_handler(void)
88167e55c51SEtienne Carriere {
88267e55c51SEtienne Carriere 	gic_native_itr_handler();
88367e55c51SEtienne Carriere }
88467e55c51SEtienne Carriere #endif /*CFG_CORE_WORKAROUND_ARM_NMFI*/
88567e55c51SEtienne Carriere 
8867315b7b4SJens Wiklander static void gic_op_add(struct itr_chip *chip, size_t it,
887702fe5a7SClément Léger 		       uint32_t type __unused,
888702fe5a7SClément Léger 		       uint32_t prio __unused)
8897315b7b4SJens Wiklander {
8907315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
8917315b7b4SJens Wiklander 
89267e55c51SEtienne Carriere 	assert(gd == &gic_data);
89367e55c51SEtienne Carriere 
8944a9ea08cSFangsuo Wu 	if (it > gd->max_it)
895d13278b8SEtienne Carriere 		panic();
896d13278b8SEtienne Carriere 
8979e935234SJens Wiklander 	if (it < GIC_SPI_BASE) {
8989e935234SJens Wiklander 		if (gic_primary_done)
8999e935234SJens Wiklander 			panic("Cannot add SGI or PPI after boot");
9009e935234SJens Wiklander 
9019e935234SJens Wiklander 		/* Assign it to Secure Group 1, G1S */
9029e935234SJens Wiklander 		gd->per_cpu_group_modifier |= BIT32(it);
9039e935234SJens Wiklander 		gd->per_cpu_group_status &= ~BIT32(it);
9049e935234SJens Wiklander 	}
9059e935234SJens Wiklander 
9069e935234SJens Wiklander 	if (it < GIC_SPI_BASE && affinity_routing_is_enabled(gd)) {
9079e935234SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
9089e935234SJens Wiklander 
9099e935234SJens Wiklander 		if (!gicr_base)
9109e935234SJens Wiklander 			panic("GICR_BASE missing");
9119e935234SJens Wiklander 
9129e935234SJens Wiklander 		/* Disable interrupt */
9139e935234SJens Wiklander 		io_write32(gicr_base + GICR_ICENABLER0, BIT32(it));
9149e935234SJens Wiklander 
9156c2d2e8aSJens Wiklander 		/* Wait for the write to GICR_ICENABLER0 to propagate */
9166c2d2e8aSJens Wiklander 		gicr_wait_for_pending_write(gicr_base);
9176c2d2e8aSJens Wiklander 
9189e935234SJens Wiklander 		/* Make interrupt non-pending */
9199e935234SJens Wiklander 		io_write32(gicr_base + GICR_ICPENDR0, BIT32(it));
9209e935234SJens Wiklander 
9219e935234SJens Wiklander 		/* Make it to Secure */
9229e935234SJens Wiklander 		io_write32(gicr_base + GICR_IGROUPR0, gd->per_cpu_group_status);
9239e935234SJens Wiklander 		io_write32(gicr_base + GICR_IGRPMODR0,
9249e935234SJens Wiklander 			   gd->per_cpu_group_modifier);
9259e935234SJens Wiklander 	} else {
9267315b7b4SJens Wiklander 		gic_it_add(gd, it);
9277315b7b4SJens Wiklander 		/* Set the CPU mask to deliver interrupts to any online core */
9287315b7b4SJens Wiklander 		gic_it_set_cpu_mask(gd, it, 0xff);
9297315b7b4SJens Wiklander 		gic_it_set_prio(gd, it, 0x1);
9307315b7b4SJens Wiklander 	}
9319e935234SJens Wiklander }
9327315b7b4SJens Wiklander 
9337315b7b4SJens Wiklander static void gic_op_enable(struct itr_chip *chip, size_t it)
9347315b7b4SJens Wiklander {
9357315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
9367315b7b4SJens Wiklander 
93767e55c51SEtienne Carriere 	assert(gd == &gic_data);
93867e55c51SEtienne Carriere 
9394a9ea08cSFangsuo Wu 	if (it > gd->max_it)
940d13278b8SEtienne Carriere 		panic();
941d13278b8SEtienne Carriere 
9429e935234SJens Wiklander 	if (it < GIC_SPI_BASE)
9439e935234SJens Wiklander 		gd->per_cpu_enable |= BIT(it);
9449e935234SJens Wiklander 
9459e935234SJens Wiklander 	if (it < GIC_SPI_BASE && affinity_routing_is_enabled(gd)) {
9469e935234SJens Wiklander 		vaddr_t gicr_base = get_gicr_base(gd);
9479e935234SJens Wiklander 
9489e935234SJens Wiklander 		if (!gicr_base)
9499e935234SJens Wiklander 			panic("GICR_BASE missing");
9509e935234SJens Wiklander 
9519e935234SJens Wiklander 		/* Assigned to G1S */
9529e935234SJens Wiklander 		assert(gd->per_cpu_group_modifier & BIT(it) &&
9539e935234SJens Wiklander 		       !(gd->per_cpu_group_status & BIT(it)));
9549e935234SJens Wiklander 		io_write32(gicr_base + GICR_ISENABLER0, gd->per_cpu_enable);
9559e935234SJens Wiklander 	} else {
9567315b7b4SJens Wiklander 		gic_it_enable(gd, it);
9577315b7b4SJens Wiklander 	}
9589e935234SJens Wiklander }
9597315b7b4SJens Wiklander 
9607315b7b4SJens Wiklander static void gic_op_disable(struct itr_chip *chip, size_t it)
9617315b7b4SJens Wiklander {
9627315b7b4SJens Wiklander 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
9637315b7b4SJens Wiklander 
96467e55c51SEtienne Carriere 	assert(gd == &gic_data);
96567e55c51SEtienne Carriere 
9664a9ea08cSFangsuo Wu 	if (it > gd->max_it)
967d13278b8SEtienne Carriere 		panic();
968d13278b8SEtienne Carriere 
9697315b7b4SJens Wiklander 	gic_it_disable(gd, it);
9707315b7b4SJens Wiklander }
97126ed70ecSGuanchao Liang 
97226ed70ecSGuanchao Liang static void gic_op_raise_pi(struct itr_chip *chip, size_t it)
97326ed70ecSGuanchao Liang {
97426ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
97526ed70ecSGuanchao Liang 
97667e55c51SEtienne Carriere 	assert(gd == &gic_data);
97767e55c51SEtienne Carriere 
9784a9ea08cSFangsuo Wu 	if (it > gd->max_it)
97926ed70ecSGuanchao Liang 		panic();
98026ed70ecSGuanchao Liang 
98126ed70ecSGuanchao Liang 	gic_it_set_pending(gd, it);
98226ed70ecSGuanchao Liang }
98326ed70ecSGuanchao Liang 
98426ed70ecSGuanchao Liang static void gic_op_raise_sgi(struct itr_chip *chip, size_t it,
985ec740b9fSJens Wiklander 			     uint32_t cpu_mask)
98626ed70ecSGuanchao Liang {
98726ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
98884603456SJens Wiklander 	bool ns = false;
98926ed70ecSGuanchao Liang 
99067e55c51SEtienne Carriere 	assert(gd == &gic_data);
99167e55c51SEtienne Carriere 
99254739cb4SMark-PK Tsai 	/* Should be Software Generated Interrupt */
99354739cb4SMark-PK Tsai 	assert(it < NUM_SGI);
99454739cb4SMark-PK Tsai 
99584603456SJens Wiklander 	ns = BIT32(it) & gd->per_cpu_group_status;
99684603456SJens Wiklander 	gic_it_raise_sgi(gd, it, cpu_mask, ns);
99726ed70ecSGuanchao Liang }
99867e55c51SEtienne Carriere 
99926ed70ecSGuanchao Liang static void gic_op_set_affinity(struct itr_chip *chip, size_t it,
100026ed70ecSGuanchao Liang 			uint8_t cpu_mask)
100126ed70ecSGuanchao Liang {
100226ed70ecSGuanchao Liang 	struct gic_data *gd = container_of(chip, struct gic_data, chip);
100326ed70ecSGuanchao Liang 
100467e55c51SEtienne Carriere 	assert(gd == &gic_data);
100567e55c51SEtienne Carriere 
10064a9ea08cSFangsuo Wu 	if (it > gd->max_it)
100726ed70ecSGuanchao Liang 		panic();
100826ed70ecSGuanchao Liang 
100926ed70ecSGuanchao Liang 	gic_it_set_cpu_mask(gd, it, cpu_mask);
101026ed70ecSGuanchao Liang }
101114885eb1SEtienne Carriere 
101214885eb1SEtienne Carriere #ifdef CFG_DT
101314885eb1SEtienne Carriere /* Callback for "interrupts" and "interrupts-extended" DT node properties */
101414885eb1SEtienne Carriere static TEE_Result dt_get_gic_chip_cb(struct dt_pargs *arg, void *priv_data,
101514885eb1SEtienne Carriere 				     struct itr_desc *itr_desc)
101614885eb1SEtienne Carriere {
101714885eb1SEtienne Carriere 	int itr_num = DT_INFO_INVALID_INTERRUPT;
101814885eb1SEtienne Carriere 	struct itr_chip *chip = priv_data;
101914885eb1SEtienne Carriere 	uint32_t phandle_args[2] = { };
102014885eb1SEtienne Carriere 	uint32_t type = 0;
102114885eb1SEtienne Carriere 	uint32_t prio = 0;
102214885eb1SEtienne Carriere 
102314885eb1SEtienne Carriere 	assert(arg && itr_desc);
102414885eb1SEtienne Carriere 
102514885eb1SEtienne Carriere 	/*
102614885eb1SEtienne Carriere 	 * gic_dt_get_irq() expects phandle arguments passed are still in DT
102714885eb1SEtienne Carriere 	 * format (big-endian) whereas struct dt_pargs carries converted
102814885eb1SEtienne Carriere 	 * formats. Therefore swap again phandle arguments. gic_dt_get_irq()
102914885eb1SEtienne Carriere 	 * consumes only the 2 first arguments.
103014885eb1SEtienne Carriere 	 */
103114885eb1SEtienne Carriere 	if (arg->args_count < 2)
103214885eb1SEtienne Carriere 		return TEE_ERROR_GENERIC;
103314885eb1SEtienne Carriere 	phandle_args[0] = cpu_to_fdt32(arg->args[0]);
103414885eb1SEtienne Carriere 	phandle_args[1] = cpu_to_fdt32(arg->args[1]);
103514885eb1SEtienne Carriere 
103614885eb1SEtienne Carriere 	itr_num = gic_dt_get_irq((const void *)phandle_args, 2, &type, &prio);
103714885eb1SEtienne Carriere 	if (itr_num == DT_INFO_INVALID_INTERRUPT)
103814885eb1SEtienne Carriere 		return TEE_ERROR_GENERIC;
103914885eb1SEtienne Carriere 
104014885eb1SEtienne Carriere 	gic_op_add(chip, itr_num, type, prio);
104114885eb1SEtienne Carriere 
104214885eb1SEtienne Carriere 	itr_desc->chip = chip;
104314885eb1SEtienne Carriere 	itr_desc->itr_num = itr_num;
104414885eb1SEtienne Carriere 
104514885eb1SEtienne Carriere 	return TEE_SUCCESS;
104614885eb1SEtienne Carriere }
104714885eb1SEtienne Carriere 
104814885eb1SEtienne Carriere static TEE_Result gic_probe(const void *fdt, int offs, const void *cd __unused)
104914885eb1SEtienne Carriere {
105014885eb1SEtienne Carriere 	if (interrupt_register_provider(fdt, offs, dt_get_gic_chip_cb,
105114885eb1SEtienne Carriere 					&gic_data.chip))
105214885eb1SEtienne Carriere 		panic();
105314885eb1SEtienne Carriere 
105414885eb1SEtienne Carriere 	return TEE_SUCCESS;
105514885eb1SEtienne Carriere }
105614885eb1SEtienne Carriere 
105714885eb1SEtienne Carriere static const struct dt_device_match gic_match_table[] = {
105814885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a15-gic" },
105914885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a7-gic" },
106014885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a5-gic" },
106114885eb1SEtienne Carriere 	{ .compatible = "arm,cortex-a9-gic" },
106214885eb1SEtienne Carriere 	{ .compatible = "arm,gic-400" },
106314885eb1SEtienne Carriere 	{ }
106414885eb1SEtienne Carriere };
106514885eb1SEtienne Carriere 
106614885eb1SEtienne Carriere DEFINE_DT_DRIVER(gic_dt_driver) = {
106714885eb1SEtienne Carriere 	.name = "gic",
106814885eb1SEtienne Carriere 	.match_table = gic_match_table,
106914885eb1SEtienne Carriere 	.probe = gic_probe,
107014885eb1SEtienne Carriere };
107114885eb1SEtienne Carriere #endif /*CFG_DT*/
10729e935234SJens Wiklander 
10739e935234SJens Wiklander static TEE_Result gic_set_primary_done(void)
10749e935234SJens Wiklander {
10759e935234SJens Wiklander 	gic_primary_done = true;
10769e935234SJens Wiklander 	return TEE_SUCCESS;
10779e935234SJens Wiklander }
10789e935234SJens Wiklander 
10799e935234SJens Wiklander nex_release_init_resource(gic_set_primary_done);
1080