| /* | 
 |  * This file is subject to the terms and conditions of the GNU General Public | 
 |  * License.  See the file "COPYING" in the main directory of this archive | 
 |  * for more details. | 
 |  * | 
 |  * Copyright (C) 2008 Ralf Baechle (ralf@linux-mips.org) | 
 |  * Copyright (C) 2012 MIPS Technologies, Inc.  All rights reserved. | 
 |  */ | 
 | #include <linux/bitmap.h> | 
 | #include <linux/clocksource.h> | 
 | #include <linux/init.h> | 
 | #include <linux/interrupt.h> | 
 | #include <linux/irq.h> | 
 | #include <linux/irqchip.h> | 
 | #include <linux/irqchip/mips-gic.h> | 
 | #include <linux/of_address.h> | 
 | #include <linux/sched.h> | 
 | #include <linux/smp.h> | 
 |  | 
 | #include <asm/mips-cm.h> | 
 | #include <asm/setup.h> | 
 | #include <asm/traps.h> | 
 |  | 
 | #include <dt-bindings/interrupt-controller/mips-gic.h> | 
 |  | 
 | unsigned int gic_present; | 
 |  | 
 | struct gic_pcpu_mask { | 
 | 	DECLARE_BITMAP(pcpu_mask, GIC_MAX_INTRS); | 
 | }; | 
 |  | 
 | static unsigned long __gic_base_addr; | 
 |  | 
 | static void __iomem *gic_base; | 
 | static struct gic_pcpu_mask pcpu_masks[NR_CPUS]; | 
 | static DEFINE_SPINLOCK(gic_lock); | 
 | static struct irq_domain *gic_irq_domain; | 
 | static struct irq_domain *gic_ipi_domain; | 
 | static int gic_shared_intrs; | 
 | static int gic_vpes; | 
 | static unsigned int gic_cpu_pin; | 
 | static unsigned int timer_cpu_pin; | 
 | static struct irq_chip gic_level_irq_controller, gic_edge_irq_controller; | 
 | DECLARE_BITMAP(ipi_resrv, GIC_MAX_INTRS); | 
 | DECLARE_BITMAP(ipi_available, GIC_MAX_INTRS); | 
 |  | 
 | static void __gic_irq_dispatch(void); | 
 |  | 
 | static inline u32 gic_read32(unsigned int reg) | 
 | { | 
 | 	return __raw_readl(gic_base + reg); | 
 | } | 
 |  | 
 | static inline u64 gic_read64(unsigned int reg) | 
 | { | 
 | 	return __raw_readq(gic_base + reg); | 
 | } | 
 |  | 
 | static inline unsigned long gic_read(unsigned int reg) | 
 | { | 
 | 	if (!mips_cm_is64) | 
 | 		return gic_read32(reg); | 
 | 	else | 
 | 		return gic_read64(reg); | 
 | } | 
 |  | 
 | static inline void gic_write32(unsigned int reg, u32 val) | 
 | { | 
 | 	return __raw_writel(val, gic_base + reg); | 
 | } | 
 |  | 
 | static inline void gic_write64(unsigned int reg, u64 val) | 
 | { | 
 | 	return __raw_writeq(val, gic_base + reg); | 
 | } | 
 |  | 
 | static inline void gic_write(unsigned int reg, unsigned long val) | 
 | { | 
 | 	if (!mips_cm_is64) | 
 | 		return gic_write32(reg, (u32)val); | 
 | 	else | 
 | 		return gic_write64(reg, (u64)val); | 
 | } | 
 |  | 
 | static inline void gic_update_bits(unsigned int reg, unsigned long mask, | 
 | 				   unsigned long val) | 
 | { | 
 | 	unsigned long regval; | 
 |  | 
 | 	regval = gic_read(reg); | 
 | 	regval &= ~mask; | 
 | 	regval |= val; | 
 | 	gic_write(reg, regval); | 
 | } | 
 |  | 
 | static inline void gic_reset_mask(unsigned int intr) | 
 | { | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_RMASK) + GIC_INTR_OFS(intr), | 
 | 		  1ul << GIC_INTR_BIT(intr)); | 
 | } | 
 |  | 
 | static inline void gic_set_mask(unsigned int intr) | 
 | { | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_SMASK) + GIC_INTR_OFS(intr), | 
 | 		  1ul << GIC_INTR_BIT(intr)); | 
 | } | 
 |  | 
 | static inline void gic_set_polarity(unsigned int intr, unsigned int pol) | 
 | { | 
 | 	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_POLARITY) + | 
 | 			GIC_INTR_OFS(intr), 1ul << GIC_INTR_BIT(intr), | 
 | 			(unsigned long)pol << GIC_INTR_BIT(intr)); | 
 | } | 
 |  | 
 | static inline void gic_set_trigger(unsigned int intr, unsigned int trig) | 
 | { | 
 | 	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_TRIGGER) + | 
 | 			GIC_INTR_OFS(intr), 1ul << GIC_INTR_BIT(intr), | 
 | 			(unsigned long)trig << GIC_INTR_BIT(intr)); | 
 | } | 
 |  | 
 | static inline void gic_set_dual_edge(unsigned int intr, unsigned int dual) | 
 | { | 
 | 	gic_update_bits(GIC_REG(SHARED, GIC_SH_SET_DUAL) + GIC_INTR_OFS(intr), | 
 | 			1ul << GIC_INTR_BIT(intr), | 
 | 			(unsigned long)dual << GIC_INTR_BIT(intr)); | 
 | } | 
 |  | 
 | static inline void gic_map_to_pin(unsigned int intr, unsigned int pin) | 
 | { | 
 | 	gic_write32(GIC_REG(SHARED, GIC_SH_INTR_MAP_TO_PIN_BASE) + | 
 | 		    GIC_SH_MAP_TO_PIN(intr), GIC_MAP_TO_PIN_MSK | pin); | 
 | } | 
 |  | 
 | static inline void gic_map_to_vpe(unsigned int intr, unsigned int vpe) | 
 | { | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_INTR_MAP_TO_VPE_BASE) + | 
 | 		  GIC_SH_MAP_TO_VPE_REG_OFF(intr, vpe), | 
 | 		  GIC_SH_MAP_TO_VPE_REG_BIT(vpe)); | 
 | } | 
 |  | 
 | #ifdef CONFIG_CLKSRC_MIPS_GIC | 
 | u64 notrace gic_read_count(void) | 
 | { | 
 | 	unsigned int hi, hi2, lo; | 
 |  | 
 | 	if (mips_cm_is64) | 
 | 		return (u64)gic_read(GIC_REG(SHARED, GIC_SH_COUNTER)); | 
 |  | 
 | 	do { | 
 | 		hi = gic_read32(GIC_REG(SHARED, GIC_SH_COUNTER_63_32)); | 
 | 		lo = gic_read32(GIC_REG(SHARED, GIC_SH_COUNTER_31_00)); | 
 | 		hi2 = gic_read32(GIC_REG(SHARED, GIC_SH_COUNTER_63_32)); | 
 | 	} while (hi2 != hi); | 
 |  | 
 | 	return (((u64) hi) << 32) + lo; | 
 | } | 
 |  | 
 | unsigned int gic_get_count_width(void) | 
 | { | 
 | 	unsigned int bits, config; | 
 |  | 
 | 	config = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG)); | 
 | 	bits = 32 + 4 * ((config & GIC_SH_CONFIG_COUNTBITS_MSK) >> | 
 | 			 GIC_SH_CONFIG_COUNTBITS_SHF); | 
 |  | 
 | 	return bits; | 
 | } | 
 |  | 
 | void notrace gic_write_compare(u64 cnt) | 
 | { | 
 | 	if (mips_cm_is64) { | 
 | 		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE), cnt); | 
 | 	} else { | 
 | 		gic_write32(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_HI), | 
 | 					(int)(cnt >> 32)); | 
 | 		gic_write32(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_LO), | 
 | 					(int)(cnt & 0xffffffff)); | 
 | 	} | 
 | } | 
 |  | 
 | void notrace gic_write_cpu_compare(u64 cnt, int cpu) | 
 | { | 
 | 	unsigned long flags; | 
 |  | 
 | 	local_irq_save(flags); | 
 |  | 
 | 	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), mips_cm_vp_id(cpu)); | 
 |  | 
 | 	if (mips_cm_is64) { | 
 | 		gic_write(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE), cnt); | 
 | 	} else { | 
 | 		gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_HI), | 
 | 					(int)(cnt >> 32)); | 
 | 		gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_LO), | 
 | 					(int)(cnt & 0xffffffff)); | 
 | 	} | 
 |  | 
 | 	local_irq_restore(flags); | 
 | } | 
 |  | 
 | u64 gic_read_compare(void) | 
 | { | 
 | 	unsigned int hi, lo; | 
 |  | 
 | 	if (mips_cm_is64) | 
 | 		return (u64)gic_read(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE)); | 
 |  | 
 | 	hi = gic_read32(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_HI)); | 
 | 	lo = gic_read32(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_LO)); | 
 |  | 
 | 	return (((u64) hi) << 32) + lo; | 
 | } | 
 |  | 
 | void gic_start_count(void) | 
 | { | 
 | 	u32 gicconfig; | 
 |  | 
 | 	/* Start the counter */ | 
 | 	gicconfig = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG)); | 
 | 	gicconfig &= ~(1 << GIC_SH_CONFIG_COUNTSTOP_SHF); | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_CONFIG), gicconfig); | 
 | } | 
 |  | 
 | void gic_stop_count(void) | 
 | { | 
 | 	u32 gicconfig; | 
 |  | 
 | 	/* Stop the counter */ | 
 | 	gicconfig = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG)); | 
 | 	gicconfig |= 1 << GIC_SH_CONFIG_COUNTSTOP_SHF; | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_CONFIG), gicconfig); | 
 | } | 
 |  | 
 | #endif | 
 |  | 
 | unsigned gic_read_local_vp_id(void) | 
 | { | 
 | 	unsigned long ident; | 
 |  | 
 | 	ident = gic_read(GIC_REG(VPE_LOCAL, GIC_VP_IDENT)); | 
 | 	return ident & GIC_VP_IDENT_VCNUM_MSK; | 
 | } | 
 |  | 
 | static bool gic_local_irq_is_routable(int intr) | 
 | { | 
 | 	u32 vpe_ctl; | 
 |  | 
 | 	/* All local interrupts are routable in EIC mode. */ | 
 | 	if (cpu_has_veic) | 
 | 		return true; | 
 |  | 
 | 	vpe_ctl = gic_read32(GIC_REG(VPE_LOCAL, GIC_VPE_CTL)); | 
 | 	switch (intr) { | 
 | 	case GIC_LOCAL_INT_TIMER: | 
 | 		return vpe_ctl & GIC_VPE_CTL_TIMER_RTBL_MSK; | 
 | 	case GIC_LOCAL_INT_PERFCTR: | 
 | 		return vpe_ctl & GIC_VPE_CTL_PERFCNT_RTBL_MSK; | 
 | 	case GIC_LOCAL_INT_FDC: | 
 | 		return vpe_ctl & GIC_VPE_CTL_FDC_RTBL_MSK; | 
 | 	case GIC_LOCAL_INT_SWINT0: | 
 | 	case GIC_LOCAL_INT_SWINT1: | 
 | 		return vpe_ctl & GIC_VPE_CTL_SWINT_RTBL_MSK; | 
 | 	default: | 
 | 		return true; | 
 | 	} | 
 | } | 
 |  | 
 | static void gic_bind_eic_interrupt(int irq, int set) | 
 | { | 
 | 	/* Convert irq vector # to hw int # */ | 
 | 	irq -= GIC_PIN_TO_VEC_OFFSET; | 
 |  | 
 | 	/* Set irq to use shadow set */ | 
 | 	gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_EIC_SHADOW_SET_BASE) + | 
 | 		  GIC_VPE_EIC_SS(irq), set); | 
 | } | 
 |  | 
 | static void gic_send_ipi(struct irq_data *d, unsigned int cpu) | 
 | { | 
 | 	irq_hw_number_t hwirq = GIC_HWIRQ_TO_SHARED(irqd_to_hwirq(d)); | 
 |  | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_WEDGE), GIC_SH_WEDGE_SET(hwirq)); | 
 | } | 
 |  | 
 | int gic_get_c0_compare_int(void) | 
 | { | 
 | 	if (!gic_local_irq_is_routable(GIC_LOCAL_INT_TIMER)) | 
 | 		return MIPS_CPU_IRQ_BASE + cp0_compare_irq; | 
 | 	return irq_create_mapping(gic_irq_domain, | 
 | 				  GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_TIMER)); | 
 | } | 
 |  | 
 | int gic_get_c0_perfcount_int(void) | 
 | { | 
 | 	if (!gic_local_irq_is_routable(GIC_LOCAL_INT_PERFCTR)) { | 
 | 		/* Is the performance counter shared with the timer? */ | 
 | 		if (cp0_perfcount_irq < 0) | 
 | 			return -1; | 
 | 		return MIPS_CPU_IRQ_BASE + cp0_perfcount_irq; | 
 | 	} | 
 | 	return irq_create_mapping(gic_irq_domain, | 
 | 				  GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_PERFCTR)); | 
 | } | 
 |  | 
 | int gic_get_c0_fdc_int(void) | 
 | { | 
 | 	if (!gic_local_irq_is_routable(GIC_LOCAL_INT_FDC)) { | 
 | 		/* Is the FDC IRQ even present? */ | 
 | 		if (cp0_fdc_irq < 0) | 
 | 			return -1; | 
 | 		return MIPS_CPU_IRQ_BASE + cp0_fdc_irq; | 
 | 	} | 
 |  | 
 | 	return irq_create_mapping(gic_irq_domain, | 
 | 				  GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_FDC)); | 
 | } | 
 |  | 
 | int gic_get_usm_range(struct resource *gic_usm_res) | 
 | { | 
 | 	if (!gic_present) | 
 | 		return -1; | 
 |  | 
 | 	gic_usm_res->start = __gic_base_addr + USM_VISIBLE_SECTION_OFS; | 
 | 	gic_usm_res->end = gic_usm_res->start + (USM_VISIBLE_SECTION_SIZE - 1); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static void gic_handle_shared_int(bool chained) | 
 | { | 
 | 	unsigned int i, intr, virq, gic_reg_step = mips_cm_is64 ? 8 : 4; | 
 | 	unsigned long *pcpu_mask; | 
 | 	unsigned long pending_reg, intrmask_reg; | 
 | 	DECLARE_BITMAP(pending, GIC_MAX_INTRS); | 
 | 	DECLARE_BITMAP(intrmask, GIC_MAX_INTRS); | 
 |  | 
 | 	/* Get per-cpu bitmaps */ | 
 | 	pcpu_mask = pcpu_masks[smp_processor_id()].pcpu_mask; | 
 |  | 
 | 	pending_reg = GIC_REG(SHARED, GIC_SH_PEND); | 
 | 	intrmask_reg = GIC_REG(SHARED, GIC_SH_MASK); | 
 |  | 
 | 	for (i = 0; i < BITS_TO_LONGS(gic_shared_intrs); i++) { | 
 | 		pending[i] = gic_read(pending_reg); | 
 | 		intrmask[i] = gic_read(intrmask_reg); | 
 | 		pending_reg += gic_reg_step; | 
 | 		intrmask_reg += gic_reg_step; | 
 |  | 
 | 		if (!IS_ENABLED(CONFIG_64BIT) || mips_cm_is64) | 
 | 			continue; | 
 |  | 
 | 		pending[i] |= (u64)gic_read(pending_reg) << 32; | 
 | 		intrmask[i] |= (u64)gic_read(intrmask_reg) << 32; | 
 | 		pending_reg += gic_reg_step; | 
 | 		intrmask_reg += gic_reg_step; | 
 | 	} | 
 |  | 
 | 	bitmap_and(pending, pending, intrmask, gic_shared_intrs); | 
 | 	bitmap_and(pending, pending, pcpu_mask, gic_shared_intrs); | 
 |  | 
 | 	for_each_set_bit(intr, pending, gic_shared_intrs) { | 
 | 		virq = irq_linear_revmap(gic_irq_domain, | 
 | 					 GIC_SHARED_TO_HWIRQ(intr)); | 
 | 		if (chained) | 
 | 			generic_handle_irq(virq); | 
 | 		else | 
 | 			do_IRQ(virq); | 
 | 	} | 
 | } | 
 |  | 
 | static void gic_mask_irq(struct irq_data *d) | 
 | { | 
 | 	gic_reset_mask(GIC_HWIRQ_TO_SHARED(d->hwirq)); | 
 | } | 
 |  | 
 | static void gic_unmask_irq(struct irq_data *d) | 
 | { | 
 | 	gic_set_mask(GIC_HWIRQ_TO_SHARED(d->hwirq)); | 
 | } | 
 |  | 
 | static void gic_ack_irq(struct irq_data *d) | 
 | { | 
 | 	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq); | 
 |  | 
 | 	gic_write(GIC_REG(SHARED, GIC_SH_WEDGE), GIC_SH_WEDGE_CLR(irq)); | 
 | } | 
 |  | 
 | static int gic_set_type(struct irq_data *d, unsigned int type) | 
 | { | 
 | 	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq); | 
 | 	unsigned long flags; | 
 | 	bool is_edge; | 
 |  | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 | 	switch (type & IRQ_TYPE_SENSE_MASK) { | 
 | 	case IRQ_TYPE_EDGE_FALLING: | 
 | 		gic_set_polarity(irq, GIC_POL_NEG); | 
 | 		gic_set_trigger(irq, GIC_TRIG_EDGE); | 
 | 		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE); | 
 | 		is_edge = true; | 
 | 		break; | 
 | 	case IRQ_TYPE_EDGE_RISING: | 
 | 		gic_set_polarity(irq, GIC_POL_POS); | 
 | 		gic_set_trigger(irq, GIC_TRIG_EDGE); | 
 | 		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE); | 
 | 		is_edge = true; | 
 | 		break; | 
 | 	case IRQ_TYPE_EDGE_BOTH: | 
 | 		/* polarity is irrelevant in this case */ | 
 | 		gic_set_trigger(irq, GIC_TRIG_EDGE); | 
 | 		gic_set_dual_edge(irq, GIC_TRIG_DUAL_ENABLE); | 
 | 		is_edge = true; | 
 | 		break; | 
 | 	case IRQ_TYPE_LEVEL_LOW: | 
 | 		gic_set_polarity(irq, GIC_POL_NEG); | 
 | 		gic_set_trigger(irq, GIC_TRIG_LEVEL); | 
 | 		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE); | 
 | 		is_edge = false; | 
 | 		break; | 
 | 	case IRQ_TYPE_LEVEL_HIGH: | 
 | 	default: | 
 | 		gic_set_polarity(irq, GIC_POL_POS); | 
 | 		gic_set_trigger(irq, GIC_TRIG_LEVEL); | 
 | 		gic_set_dual_edge(irq, GIC_TRIG_DUAL_DISABLE); | 
 | 		is_edge = false; | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	if (is_edge) | 
 | 		irq_set_chip_handler_name_locked(d, &gic_edge_irq_controller, | 
 | 						 handle_edge_irq, NULL); | 
 | 	else | 
 | 		irq_set_chip_handler_name_locked(d, &gic_level_irq_controller, | 
 | 						 handle_level_irq, NULL); | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | #ifdef CONFIG_SMP | 
 | static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask, | 
 | 			    bool force) | 
 | { | 
 | 	unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq); | 
 | 	cpumask_t	tmp = CPU_MASK_NONE; | 
 | 	unsigned long	flags; | 
 | 	int		i; | 
 |  | 
 | 	cpumask_and(&tmp, cpumask, cpu_online_mask); | 
 | 	if (cpumask_empty(&tmp)) | 
 | 		return -EINVAL; | 
 |  | 
 | 	/* Assumption : cpumask refers to a single CPU */ | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 |  | 
 | 	/* Re-route this IRQ */ | 
 | 	gic_map_to_vpe(irq, mips_cm_vp_id(cpumask_first(&tmp))); | 
 |  | 
 | 	/* Update the pcpu_masks */ | 
 | 	for (i = 0; i < min(gic_vpes, NR_CPUS); i++) | 
 | 		clear_bit(irq, pcpu_masks[i].pcpu_mask); | 
 | 	set_bit(irq, pcpu_masks[cpumask_first(&tmp)].pcpu_mask); | 
 |  | 
 | 	cpumask_copy(irq_data_get_affinity_mask(d), cpumask); | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 |  | 
 | 	return IRQ_SET_MASK_OK_NOCOPY; | 
 | } | 
 | #endif | 
 |  | 
 | static struct irq_chip gic_level_irq_controller = { | 
 | 	.name			=	"MIPS GIC", | 
 | 	.irq_mask		=	gic_mask_irq, | 
 | 	.irq_unmask		=	gic_unmask_irq, | 
 | 	.irq_set_type		=	gic_set_type, | 
 | #ifdef CONFIG_SMP | 
 | 	.irq_set_affinity	=	gic_set_affinity, | 
 | #endif | 
 | }; | 
 |  | 
 | static struct irq_chip gic_edge_irq_controller = { | 
 | 	.name			=	"MIPS GIC", | 
 | 	.irq_ack		=	gic_ack_irq, | 
 | 	.irq_mask		=	gic_mask_irq, | 
 | 	.irq_unmask		=	gic_unmask_irq, | 
 | 	.irq_set_type		=	gic_set_type, | 
 | #ifdef CONFIG_SMP | 
 | 	.irq_set_affinity	=	gic_set_affinity, | 
 | #endif | 
 | 	.ipi_send_single	=	gic_send_ipi, | 
 | }; | 
 |  | 
 | static void gic_handle_local_int(bool chained) | 
 | { | 
 | 	unsigned long pending, masked; | 
 | 	unsigned int intr, virq; | 
 |  | 
 | 	pending = gic_read32(GIC_REG(VPE_LOCAL, GIC_VPE_PEND)); | 
 | 	masked = gic_read32(GIC_REG(VPE_LOCAL, GIC_VPE_MASK)); | 
 |  | 
 | 	bitmap_and(&pending, &pending, &masked, GIC_NUM_LOCAL_INTRS); | 
 |  | 
 | 	for_each_set_bit(intr, &pending, GIC_NUM_LOCAL_INTRS) { | 
 | 		virq = irq_linear_revmap(gic_irq_domain, | 
 | 					 GIC_LOCAL_TO_HWIRQ(intr)); | 
 | 		if (chained) | 
 | 			generic_handle_irq(virq); | 
 | 		else | 
 | 			do_IRQ(virq); | 
 | 	} | 
 | } | 
 |  | 
 | static void gic_mask_local_irq(struct irq_data *d) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq); | 
 |  | 
 | 	gic_write32(GIC_REG(VPE_LOCAL, GIC_VPE_RMASK), 1 << intr); | 
 | } | 
 |  | 
 | static void gic_unmask_local_irq(struct irq_data *d) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq); | 
 |  | 
 | 	gic_write32(GIC_REG(VPE_LOCAL, GIC_VPE_SMASK), 1 << intr); | 
 | } | 
 |  | 
 | static struct irq_chip gic_local_irq_controller = { | 
 | 	.name			=	"MIPS GIC Local", | 
 | 	.irq_mask		=	gic_mask_local_irq, | 
 | 	.irq_unmask		=	gic_unmask_local_irq, | 
 | }; | 
 |  | 
 | static void gic_mask_local_irq_all_vpes(struct irq_data *d) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq); | 
 | 	int i; | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 | 	for (i = 0; i < gic_vpes; i++) { | 
 | 		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), | 
 | 			  mips_cm_vp_id(i)); | 
 | 		gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_RMASK), 1 << intr); | 
 | 	} | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 | } | 
 |  | 
 | static void gic_unmask_local_irq_all_vpes(struct irq_data *d) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq); | 
 | 	int i; | 
 | 	unsigned long flags; | 
 |  | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 | 	for (i = 0; i < gic_vpes; i++) { | 
 | 		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), | 
 | 			  mips_cm_vp_id(i)); | 
 | 		gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_SMASK), 1 << intr); | 
 | 	} | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 | } | 
 |  | 
 | static struct irq_chip gic_all_vpes_local_irq_controller = { | 
 | 	.name			=	"MIPS GIC Local", | 
 | 	.irq_mask		=	gic_mask_local_irq_all_vpes, | 
 | 	.irq_unmask		=	gic_unmask_local_irq_all_vpes, | 
 | }; | 
 |  | 
 | static void __gic_irq_dispatch(void) | 
 | { | 
 | 	gic_handle_local_int(false); | 
 | 	gic_handle_shared_int(false); | 
 | } | 
 |  | 
 | static void gic_irq_dispatch(struct irq_desc *desc) | 
 | { | 
 | 	gic_handle_local_int(true); | 
 | 	gic_handle_shared_int(true); | 
 | } | 
 |  | 
 | static void __init gic_basic_init(void) | 
 | { | 
 | 	unsigned int i; | 
 |  | 
 | 	board_bind_eic_interrupt = &gic_bind_eic_interrupt; | 
 |  | 
 | 	/* Setup defaults */ | 
 | 	for (i = 0; i < gic_shared_intrs; i++) { | 
 | 		gic_set_polarity(i, GIC_POL_POS); | 
 | 		gic_set_trigger(i, GIC_TRIG_LEVEL); | 
 | 		gic_reset_mask(i); | 
 | 	} | 
 |  | 
 | 	for (i = 0; i < gic_vpes; i++) { | 
 | 		unsigned int j; | 
 |  | 
 | 		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), | 
 | 			  mips_cm_vp_id(i)); | 
 | 		for (j = 0; j < GIC_NUM_LOCAL_INTRS; j++) { | 
 | 			if (!gic_local_irq_is_routable(j)) | 
 | 				continue; | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_RMASK), 1 << j); | 
 | 		} | 
 | 	} | 
 | } | 
 |  | 
 | static int gic_local_irq_domain_map(struct irq_domain *d, unsigned int virq, | 
 | 				    irq_hw_number_t hw) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_LOCAL(hw); | 
 | 	int ret = 0; | 
 | 	int i; | 
 | 	unsigned long flags; | 
 |  | 
 | 	if (!gic_local_irq_is_routable(intr)) | 
 | 		return -EPERM; | 
 |  | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 | 	for (i = 0; i < gic_vpes; i++) { | 
 | 		u32 val = GIC_MAP_TO_PIN_MSK | gic_cpu_pin; | 
 |  | 
 | 		gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), | 
 | 			  mips_cm_vp_id(i)); | 
 |  | 
 | 		switch (intr) { | 
 | 		case GIC_LOCAL_INT_WD: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_WD_MAP), val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_COMPARE: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_MAP), | 
 | 				    val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_TIMER: | 
 | 			/* CONFIG_MIPS_CMP workaround (see __gic_init) */ | 
 | 			val = GIC_MAP_TO_PIN_MSK | timer_cpu_pin; | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_TIMER_MAP), | 
 | 				    val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_PERFCTR: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_PERFCTR_MAP), | 
 | 				    val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_SWINT0: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_SWINT0_MAP), | 
 | 				    val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_SWINT1: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_SWINT1_MAP), | 
 | 				    val); | 
 | 			break; | 
 | 		case GIC_LOCAL_INT_FDC: | 
 | 			gic_write32(GIC_REG(VPE_OTHER, GIC_VPE_FDC_MAP), val); | 
 | 			break; | 
 | 		default: | 
 | 			pr_err("Invalid local IRQ %d\n", intr); | 
 | 			ret = -EINVAL; | 
 | 			break; | 
 | 		} | 
 | 	} | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 |  | 
 | 	return ret; | 
 | } | 
 |  | 
 | static int gic_shared_irq_domain_map(struct irq_domain *d, unsigned int virq, | 
 | 				     irq_hw_number_t hw, unsigned int vpe) | 
 | { | 
 | 	int intr = GIC_HWIRQ_TO_SHARED(hw); | 
 | 	unsigned long flags; | 
 | 	int i; | 
 |  | 
 | 	spin_lock_irqsave(&gic_lock, flags); | 
 | 	gic_map_to_pin(intr, gic_cpu_pin); | 
 | 	gic_map_to_vpe(intr, mips_cm_vp_id(vpe)); | 
 | 	for (i = 0; i < min(gic_vpes, NR_CPUS); i++) | 
 | 		clear_bit(intr, pcpu_masks[i].pcpu_mask); | 
 | 	set_bit(intr, pcpu_masks[vpe].pcpu_mask); | 
 | 	spin_unlock_irqrestore(&gic_lock, flags); | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int gic_irq_domain_xlate(struct irq_domain *d, struct device_node *ctrlr, | 
 | 				const u32 *intspec, unsigned int intsize, | 
 | 				irq_hw_number_t *out_hwirq, | 
 | 				unsigned int *out_type) | 
 | { | 
 | 	if (intsize != 3) | 
 | 		return -EINVAL; | 
 |  | 
 | 	if (intspec[0] == GIC_SHARED) | 
 | 		*out_hwirq = GIC_SHARED_TO_HWIRQ(intspec[1]); | 
 | 	else if (intspec[0] == GIC_LOCAL) | 
 | 		*out_hwirq = GIC_LOCAL_TO_HWIRQ(intspec[1]); | 
 | 	else | 
 | 		return -EINVAL; | 
 | 	*out_type = intspec[2] & IRQ_TYPE_SENSE_MASK; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int gic_irq_domain_map(struct irq_domain *d, unsigned int virq, | 
 | 			      irq_hw_number_t hwirq) | 
 | { | 
 | 	int err; | 
 |  | 
 | 	if (hwirq >= GIC_SHARED_HWIRQ_BASE) { | 
 | 		/* verify that shared irqs don't conflict with an IPI irq */ | 
 | 		if (test_bit(GIC_HWIRQ_TO_SHARED(hwirq), ipi_resrv)) | 
 | 			return -EBUSY; | 
 |  | 
 | 		err = irq_domain_set_hwirq_and_chip(d, virq, hwirq, | 
 | 						    &gic_level_irq_controller, | 
 | 						    NULL); | 
 | 		if (err) | 
 | 			return err; | 
 |  | 
 | 		return gic_shared_irq_domain_map(d, virq, hwirq, 0); | 
 | 	} | 
 |  | 
 | 	switch (GIC_HWIRQ_TO_LOCAL(hwirq)) { | 
 | 	case GIC_LOCAL_INT_TIMER: | 
 | 	case GIC_LOCAL_INT_PERFCTR: | 
 | 	case GIC_LOCAL_INT_FDC: | 
 | 		/* | 
 | 		 * HACK: These are all really percpu interrupts, but | 
 | 		 * the rest of the MIPS kernel code does not use the | 
 | 		 * percpu IRQ API for them. | 
 | 		 */ | 
 | 		err = irq_domain_set_hwirq_and_chip(d, virq, hwirq, | 
 | 						    &gic_all_vpes_local_irq_controller, | 
 | 						    NULL); | 
 | 		if (err) | 
 | 			return err; | 
 |  | 
 | 		irq_set_handler(virq, handle_percpu_irq); | 
 | 		break; | 
 |  | 
 | 	default: | 
 | 		err = irq_domain_set_hwirq_and_chip(d, virq, hwirq, | 
 | 						    &gic_local_irq_controller, | 
 | 						    NULL); | 
 | 		if (err) | 
 | 			return err; | 
 |  | 
 | 		irq_set_handler(virq, handle_percpu_devid_irq); | 
 | 		irq_set_percpu_devid(virq); | 
 | 		break; | 
 | 	} | 
 |  | 
 | 	return gic_local_irq_domain_map(d, virq, hwirq); | 
 | } | 
 |  | 
 | static int gic_irq_domain_alloc(struct irq_domain *d, unsigned int virq, | 
 | 				unsigned int nr_irqs, void *arg) | 
 | { | 
 | 	struct irq_fwspec *fwspec = arg; | 
 | 	irq_hw_number_t hwirq; | 
 |  | 
 | 	if (fwspec->param[0] == GIC_SHARED) | 
 | 		hwirq = GIC_SHARED_TO_HWIRQ(fwspec->param[1]); | 
 | 	else | 
 | 		hwirq = GIC_LOCAL_TO_HWIRQ(fwspec->param[1]); | 
 |  | 
 | 	return gic_irq_domain_map(d, virq, hwirq); | 
 | } | 
 |  | 
 | void gic_irq_domain_free(struct irq_domain *d, unsigned int virq, | 
 | 			 unsigned int nr_irqs) | 
 | { | 
 | } | 
 |  | 
 | static const struct irq_domain_ops gic_irq_domain_ops = { | 
 | 	.xlate = gic_irq_domain_xlate, | 
 | 	.alloc = gic_irq_domain_alloc, | 
 | 	.free = gic_irq_domain_free, | 
 | 	.map = gic_irq_domain_map, | 
 | }; | 
 |  | 
 | static int gic_ipi_domain_xlate(struct irq_domain *d, struct device_node *ctrlr, | 
 | 				const u32 *intspec, unsigned int intsize, | 
 | 				irq_hw_number_t *out_hwirq, | 
 | 				unsigned int *out_type) | 
 | { | 
 | 	/* | 
 | 	 * There's nothing to translate here. hwirq is dynamically allocated and | 
 | 	 * the irq type is always edge triggered. | 
 | 	 * */ | 
 | 	*out_hwirq = 0; | 
 | 	*out_type = IRQ_TYPE_EDGE_RISING; | 
 |  | 
 | 	return 0; | 
 | } | 
 |  | 
 | static int gic_ipi_domain_alloc(struct irq_domain *d, unsigned int virq, | 
 | 				unsigned int nr_irqs, void *arg) | 
 | { | 
 | 	struct cpumask *ipimask = arg; | 
 | 	irq_hw_number_t hwirq, base_hwirq; | 
 | 	int cpu, ret, i; | 
 |  | 
 | 	base_hwirq = find_first_bit(ipi_available, gic_shared_intrs); | 
 | 	if (base_hwirq == gic_shared_intrs) | 
 | 		return -ENOMEM; | 
 |  | 
 | 	/* check that we have enough space */ | 
 | 	for (i = base_hwirq; i < nr_irqs; i++) { | 
 | 		if (!test_bit(i, ipi_available)) | 
 | 			return -EBUSY; | 
 | 	} | 
 | 	bitmap_clear(ipi_available, base_hwirq, nr_irqs); | 
 |  | 
 | 	/* map the hwirq for each cpu consecutively */ | 
 | 	i = 0; | 
 | 	for_each_cpu(cpu, ipimask) { | 
 | 		hwirq = GIC_SHARED_TO_HWIRQ(base_hwirq + i); | 
 |  | 
 | 		ret = irq_domain_set_hwirq_and_chip(d, virq + i, hwirq, | 
 | 						    &gic_edge_irq_controller, | 
 | 						    NULL); | 
 | 		if (ret) | 
 | 			goto error; | 
 |  | 
 | 		ret = irq_domain_set_hwirq_and_chip(d->parent, virq + i, hwirq, | 
 | 						    &gic_edge_irq_controller, | 
 | 						    NULL); | 
 | 		if (ret) | 
 | 			goto error; | 
 |  | 
 | 		ret = irq_set_irq_type(virq + i, IRQ_TYPE_EDGE_RISING); | 
 | 		if (ret) | 
 | 			goto error; | 
 |  | 
 | 		ret = gic_shared_irq_domain_map(d, virq + i, hwirq, cpu); | 
 | 		if (ret) | 
 | 			goto error; | 
 |  | 
 | 		i++; | 
 | 	} | 
 |  | 
 | 	return 0; | 
 | error: | 
 | 	bitmap_set(ipi_available, base_hwirq, nr_irqs); | 
 | 	return ret; | 
 | } | 
 |  | 
 | void gic_ipi_domain_free(struct irq_domain *d, unsigned int virq, | 
 | 			 unsigned int nr_irqs) | 
 | { | 
 | 	irq_hw_number_t base_hwirq; | 
 | 	struct irq_data *data; | 
 |  | 
 | 	data = irq_get_irq_data(virq); | 
 | 	if (!data) | 
 | 		return; | 
 |  | 
 | 	base_hwirq = GIC_HWIRQ_TO_SHARED(irqd_to_hwirq(data)); | 
 | 	bitmap_set(ipi_available, base_hwirq, nr_irqs); | 
 | } | 
 |  | 
 | int gic_ipi_domain_match(struct irq_domain *d, struct device_node *node, | 
 | 			 enum irq_domain_bus_token bus_token) | 
 | { | 
 | 	bool is_ipi; | 
 |  | 
 | 	switch (bus_token) { | 
 | 	case DOMAIN_BUS_IPI: | 
 | 		is_ipi = d->bus_token == bus_token; | 
 | 		return (!node || to_of_node(d->fwnode) == node) && is_ipi; | 
 | 		break; | 
 | 	default: | 
 | 		return 0; | 
 | 	} | 
 | } | 
 |  | 
 | static const struct irq_domain_ops gic_ipi_domain_ops = { | 
 | 	.xlate = gic_ipi_domain_xlate, | 
 | 	.alloc = gic_ipi_domain_alloc, | 
 | 	.free = gic_ipi_domain_free, | 
 | 	.match = gic_ipi_domain_match, | 
 | }; | 
 |  | 
 | static void __init __gic_init(unsigned long gic_base_addr, | 
 | 			      unsigned long gic_addrspace_size, | 
 | 			      unsigned int cpu_vec, unsigned int irqbase, | 
 | 			      struct device_node *node) | 
 | { | 
 | 	unsigned int gicconfig, cpu; | 
 | 	unsigned int v[2]; | 
 |  | 
 | 	__gic_base_addr = gic_base_addr; | 
 |  | 
 | 	gic_base = ioremap_nocache(gic_base_addr, gic_addrspace_size); | 
 |  | 
 | 	gicconfig = gic_read(GIC_REG(SHARED, GIC_SH_CONFIG)); | 
 | 	gic_shared_intrs = (gicconfig & GIC_SH_CONFIG_NUMINTRS_MSK) >> | 
 | 		   GIC_SH_CONFIG_NUMINTRS_SHF; | 
 | 	gic_shared_intrs = ((gic_shared_intrs + 1) * 8); | 
 |  | 
 | 	gic_vpes = (gicconfig & GIC_SH_CONFIG_NUMVPES_MSK) >> | 
 | 		  GIC_SH_CONFIG_NUMVPES_SHF; | 
 | 	gic_vpes = gic_vpes + 1; | 
 |  | 
 | 	if (cpu_has_veic) { | 
 | 		/* Set EIC mode for all VPEs */ | 
 | 		for_each_present_cpu(cpu) { | 
 | 			gic_write(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), | 
 | 				  mips_cm_vp_id(cpu)); | 
 | 			gic_write(GIC_REG(VPE_OTHER, GIC_VPE_CTL), | 
 | 				  GIC_VPE_CTL_EIC_MODE_MSK); | 
 | 		} | 
 |  | 
 | 		/* Always use vector 1 in EIC mode */ | 
 | 		gic_cpu_pin = 0; | 
 | 		timer_cpu_pin = gic_cpu_pin; | 
 | 		set_vi_handler(gic_cpu_pin + GIC_PIN_TO_VEC_OFFSET, | 
 | 			       __gic_irq_dispatch); | 
 | 	} else { | 
 | 		gic_cpu_pin = cpu_vec - GIC_CPU_PIN_OFFSET; | 
 | 		irq_set_chained_handler(MIPS_CPU_IRQ_BASE + cpu_vec, | 
 | 					gic_irq_dispatch); | 
 | 		/* | 
 | 		 * With the CMP implementation of SMP (deprecated), other CPUs | 
 | 		 * are started by the bootloader and put into a timer based | 
 | 		 * waiting poll loop. We must not re-route those CPU's local | 
 | 		 * timer interrupts as the wait instruction will never finish, | 
 | 		 * so just handle whatever CPU interrupt it is routed to by | 
 | 		 * default. | 
 | 		 * | 
 | 		 * This workaround should be removed when CMP support is | 
 | 		 * dropped. | 
 | 		 */ | 
 | 		if (IS_ENABLED(CONFIG_MIPS_CMP) && | 
 | 		    gic_local_irq_is_routable(GIC_LOCAL_INT_TIMER)) { | 
 | 			timer_cpu_pin = gic_read32(GIC_REG(VPE_LOCAL, | 
 | 							 GIC_VPE_TIMER_MAP)) & | 
 | 					GIC_MAP_MSK; | 
 | 			irq_set_chained_handler(MIPS_CPU_IRQ_BASE + | 
 | 						GIC_CPU_PIN_OFFSET + | 
 | 						timer_cpu_pin, | 
 | 						gic_irq_dispatch); | 
 | 		} else { | 
 | 			timer_cpu_pin = gic_cpu_pin; | 
 | 		} | 
 | 	} | 
 |  | 
 | 	gic_irq_domain = irq_domain_add_simple(node, GIC_NUM_LOCAL_INTRS + | 
 | 					       gic_shared_intrs, irqbase, | 
 | 					       &gic_irq_domain_ops, NULL); | 
 | 	if (!gic_irq_domain) | 
 | 		panic("Failed to add GIC IRQ domain"); | 
 | 	gic_irq_domain->name = "mips-gic-irq"; | 
 |  | 
 | 	gic_ipi_domain = irq_domain_add_hierarchy(gic_irq_domain, | 
 | 						  IRQ_DOMAIN_FLAG_IPI_PER_CPU, | 
 | 						  GIC_NUM_LOCAL_INTRS + gic_shared_intrs, | 
 | 						  node, &gic_ipi_domain_ops, NULL); | 
 | 	if (!gic_ipi_domain) | 
 | 		panic("Failed to add GIC IPI domain"); | 
 |  | 
 | 	gic_ipi_domain->name = "mips-gic-ipi"; | 
 | 	irq_domain_update_bus_token(gic_ipi_domain, DOMAIN_BUS_IPI); | 
 |  | 
 | 	if (node && | 
 | 	    !of_property_read_u32_array(node, "mti,reserved-ipi-vectors", v, 2)) { | 
 | 		bitmap_set(ipi_resrv, v[0], v[1]); | 
 | 	} else { | 
 | 		/* Make the last 2 * gic_vpes available for IPIs */ | 
 | 		bitmap_set(ipi_resrv, | 
 | 			   gic_shared_intrs - 2 * gic_vpes, | 
 | 			   2 * gic_vpes); | 
 | 	} | 
 |  | 
 | 	bitmap_copy(ipi_available, ipi_resrv, GIC_MAX_INTRS); | 
 | 	gic_basic_init(); | 
 | } | 
 |  | 
 | void __init gic_init(unsigned long gic_base_addr, | 
 | 		     unsigned long gic_addrspace_size, | 
 | 		     unsigned int cpu_vec, unsigned int irqbase) | 
 | { | 
 | 	__gic_init(gic_base_addr, gic_addrspace_size, cpu_vec, irqbase, NULL); | 
 | } | 
 |  | 
 | static int __init gic_of_init(struct device_node *node, | 
 | 			      struct device_node *parent) | 
 | { | 
 | 	struct resource res; | 
 | 	unsigned int cpu_vec, i = 0, reserved = 0; | 
 | 	phys_addr_t gic_base; | 
 | 	size_t gic_len; | 
 |  | 
 | 	/* Find the first available CPU vector. */ | 
 | 	while (!of_property_read_u32_index(node, "mti,reserved-cpu-vectors", | 
 | 					   i++, &cpu_vec)) | 
 | 		reserved |= BIT(cpu_vec); | 
 | 	for (cpu_vec = 2; cpu_vec < 8; cpu_vec++) { | 
 | 		if (!(reserved & BIT(cpu_vec))) | 
 | 			break; | 
 | 	} | 
 | 	if (cpu_vec == 8) { | 
 | 		pr_err("No CPU vectors available for GIC\n"); | 
 | 		return -ENODEV; | 
 | 	} | 
 |  | 
 | 	if (of_address_to_resource(node, 0, &res)) { | 
 | 		/* | 
 | 		 * Probe the CM for the GIC base address if not specified | 
 | 		 * in the device-tree. | 
 | 		 */ | 
 | 		if (mips_cm_present()) { | 
 | 			gic_base = read_gcr_gic_base() & | 
 | 				~CM_GCR_GIC_BASE_GICEN_MSK; | 
 | 			gic_len = 0x20000; | 
 | 		} else { | 
 | 			pr_err("Failed to get GIC memory range\n"); | 
 | 			return -ENODEV; | 
 | 		} | 
 | 	} else { | 
 | 		gic_base = res.start; | 
 | 		gic_len = resource_size(&res); | 
 | 	} | 
 |  | 
 | 	if (mips_cm_present()) | 
 | 		write_gcr_gic_base(gic_base | CM_GCR_GIC_BASE_GICEN_MSK); | 
 | 	gic_present = true; | 
 |  | 
 | 	__gic_init(gic_base, gic_len, cpu_vec, 0, node); | 
 |  | 
 | 	return 0; | 
 | } | 
 | IRQCHIP_DECLARE(mips_gic, "mti,gic", gic_of_init); |