Commit e9de688d authored by Andrew Bresticker's avatar Andrew Bresticker Committed by Ralf Baechle

irqchip: mips-gic: Support local interrupts

The MIPS GIC supports 7 local interrupts, 2 of which are the GIC
local watchdog and count/compare timer.  The remainder are CPU
interrupts which may optionally be re-routed through the GIC.
GIC hardware IRQs 0-6 are now used for local interrupts while
hardware IRQs 7+ are used for external (shared) interrupts.

Note that the 5 CPU interrupts may not be re-routable through
the GIC.  In that case mapping will fail and the vectors reported
in C0_IntCtl should be used instead.  gic_get_c0_compare_int() and
gic_get_c0_perfcount_int() will return the correct IRQ number to
use for the C0 timer and perfcounter interrupts based on the
routability of those interrupts through the GIC.

A separate irq_chip, with callbacks that mask/unmask the local
interrupt on all CPUs, is used for the C0 timer and performance
counter interrupts since all other platforms do not use the percpu
IRQ API for those interrupts.

Malta, SEAD-3, and the GIC clockevent driver have been updated
to use local interrupts and the R4K clockevent driver has been
updated to poll for C0 timer interrupts through the GIC when
the GIC is present.
Signed-off-by: default avatarAndrew Bresticker <abrestic@chromium.org>
Acked-by: default avatarJason Cooper <jason@lakedaemon.net>
Reviewed-by: default avatarQais Yousef <qais.yousef@imgtec.com>
Tested-by: default avatarQais Yousef <qais.yousef@imgtec.com>
Cc: Thomas Gleixner <tglx@linutronix.de>
Cc: Jeffrey Deans <jeffrey.deans@imgtec.com>
Cc: Markos Chandras <markos.chandras@imgtec.com>
Cc: Paul Burton <paul.burton@imgtec.com>
Cc: Jonas Gorski <jogo@openwrt.org>
Cc: John Crispin <blogic@openwrt.org>
Cc: David Daney <ddaney.cavm@gmail.com>
Cc: linux-mips@linux-mips.org
Cc: linux-kernel@vger.kernel.org
Patchwork: https://patchwork.linux-mips.org/patch/7819/Signed-off-by: default avatarRalf Baechle <ralf@linux-mips.org>
parent 4a6a3ea3
...@@ -209,6 +209,7 @@ ...@@ -209,6 +209,7 @@
#define GIC_VPE_WD_MAP_OFS 0x0040 #define GIC_VPE_WD_MAP_OFS 0x0040
#define GIC_VPE_COMPARE_MAP_OFS 0x0044 #define GIC_VPE_COMPARE_MAP_OFS 0x0044
#define GIC_VPE_TIMER_MAP_OFS 0x0048 #define GIC_VPE_TIMER_MAP_OFS 0x0048
#define GIC_VPE_FDC_MAP_OFS 0x004c
#define GIC_VPE_PERFCTR_MAP_OFS 0x0050 #define GIC_VPE_PERFCTR_MAP_OFS 0x0050
#define GIC_VPE_SWINT0_MAP_OFS 0x0054 #define GIC_VPE_SWINT0_MAP_OFS 0x0054
#define GIC_VPE_SWINT1_MAP_OFS 0x0058 #define GIC_VPE_SWINT1_MAP_OFS 0x0058
...@@ -262,6 +263,10 @@ ...@@ -262,6 +263,10 @@
#define GIC_MAP_MSK (MSK(6) << GIC_MAP_SHF) #define GIC_MAP_MSK (MSK(6) << GIC_MAP_SHF)
/* GIC_VPE_CTL Masks */ /* GIC_VPE_CTL Masks */
#define GIC_VPE_CTL_FDC_RTBL_SHF 4
#define GIC_VPE_CTL_FDC_RTBL_MSK (MSK(1) << GIC_VPE_CTL_FDC_RTBL_SHF)
#define GIC_VPE_CTL_SWINT_RTBL_SHF 3
#define GIC_VPE_CTL_SWINT_RTBL_MSK (MSK(1) << GIC_VPE_CTL_SWINT_RTBL_SHF)
#define GIC_VPE_CTL_PERFCNT_RTBL_SHF 2 #define GIC_VPE_CTL_PERFCNT_RTBL_SHF 2
#define GIC_VPE_CTL_PERFCNT_RTBL_MSK (MSK(1) << GIC_VPE_CTL_PERFCNT_RTBL_SHF) #define GIC_VPE_CTL_PERFCNT_RTBL_MSK (MSK(1) << GIC_VPE_CTL_PERFCNT_RTBL_SHF)
#define GIC_VPE_CTL_TIMER_RTBL_SHF 1 #define GIC_VPE_CTL_TIMER_RTBL_SHF 1
...@@ -329,16 +334,30 @@ ...@@ -329,16 +334,30 @@
/* Add 2 to convert GIC CPU pin to core interrupt */ /* Add 2 to convert GIC CPU pin to core interrupt */
#define GIC_CPU_PIN_OFFSET 2 #define GIC_CPU_PIN_OFFSET 2
/* Local GIC interrupts. */
#define GIC_INT_TMR (GIC_CPU_INT5)
#define GIC_INT_PERFCTR (GIC_CPU_INT5)
/* Add 2 to convert non-EIC hardware interrupt to EIC vector number. */ /* Add 2 to convert non-EIC hardware interrupt to EIC vector number. */
#define GIC_CPU_TO_VEC_OFFSET (2) #define GIC_CPU_TO_VEC_OFFSET (2)
/* Mapped interrupt to pin X, then GIC will generate the vector (X+1). */ /* Mapped interrupt to pin X, then GIC will generate the vector (X+1). */
#define GIC_PIN_TO_VEC_OFFSET (1) #define GIC_PIN_TO_VEC_OFFSET (1)
/* Local GIC interrupts. */
#define GIC_LOCAL_INT_WD 0 /* GIC watchdog */
#define GIC_LOCAL_INT_COMPARE 1 /* GIC count and compare timer */
#define GIC_LOCAL_INT_TIMER 2 /* CPU timer interrupt */
#define GIC_LOCAL_INT_PERFCTR 3 /* CPU performance counter */
#define GIC_LOCAL_INT_SWINT0 4 /* CPU software interrupt 0 */
#define GIC_LOCAL_INT_SWINT1 5 /* CPU software interrupt 1 */
#define GIC_LOCAL_INT_FDC 6 /* CPU fast debug channel */
#define GIC_NUM_LOCAL_INTRS 7
/* Convert between local/shared IRQ number and GIC HW IRQ number. */
#define GIC_LOCAL_HWIRQ_BASE 0
#define GIC_LOCAL_TO_HWIRQ(x) (GIC_LOCAL_HWIRQ_BASE + (x))
#define GIC_HWIRQ_TO_LOCAL(x) ((x) - GIC_LOCAL_HWIRQ_BASE)
#define GIC_SHARED_HWIRQ_BASE GIC_NUM_LOCAL_INTRS
#define GIC_SHARED_TO_HWIRQ(x) (GIC_SHARED_HWIRQ_BASE + (x))
#define GIC_HWIRQ_TO_SHARED(x) ((x) - GIC_SHARED_HWIRQ_BASE)
#include <linux/clocksource.h> #include <linux/clocksource.h>
#include <linux/irq.h> #include <linux/irq.h>
...@@ -363,4 +382,6 @@ extern void gic_bind_eic_interrupt(int irq, int set); ...@@ -363,4 +382,6 @@ extern void gic_bind_eic_interrupt(int irq, int set);
extern unsigned int gic_get_timer_pending(void); extern unsigned int gic_get_timer_pending(void);
extern void gic_get_int_mask(unsigned long *dst, const unsigned long *src); extern void gic_get_int_mask(unsigned long *dst, const unsigned long *src);
extern unsigned int gic_get_int(void); extern unsigned int gic_get_int(void);
extern int gic_get_c0_compare_int(void);
extern int gic_get_c0_perfcount_int(void);
#endif /* _ASM_GICREGS_H */ #endif /* _ASM_GICREGS_H */
...@@ -10,6 +10,8 @@ ...@@ -10,6 +10,8 @@
#ifndef _MIPS_MALTAINT_H #ifndef _MIPS_MALTAINT_H
#define _MIPS_MALTAINT_H #define _MIPS_MALTAINT_H
#include <asm/gic.h>
/* /*
* Interrupts 0..15 are used for Malta ISA compatible interrupts * Interrupts 0..15 are used for Malta ISA compatible interrupts
*/ */
...@@ -61,6 +63,6 @@ ...@@ -61,6 +63,6 @@
#define MSC01E_INT_CPUCTR 11 #define MSC01E_INT_CPUCTR 11
/* GIC external interrupts */ /* GIC external interrupts */
#define GIC_INT_I8259A 3 #define GIC_INT_I8259A GIC_SHARED_TO_HWIRQ(3)
#endif /* !(_MIPS_MALTAINT_H) */ #endif /* !(_MIPS_MALTAINT_H) */
...@@ -10,6 +10,8 @@ ...@@ -10,6 +10,8 @@
#ifndef _MIPS_SEAD3INT_H #ifndef _MIPS_SEAD3INT_H
#define _MIPS_SEAD3INT_H #define _MIPS_SEAD3INT_H
#include <asm/gic.h>
/* SEAD-3 GIC address space definitions. */ /* SEAD-3 GIC address space definitions. */
#define GIC_BASE_ADDR 0x1b1c0000 #define GIC_BASE_ADDR 0x1b1c0000
#define GIC_ADDRSPACE_SZ (128 * 1024) #define GIC_ADDRSPACE_SZ (128 * 1024)
...@@ -22,9 +24,9 @@ ...@@ -22,9 +24,9 @@
#define CPU_INT_NET 6 #define CPU_INT_NET 6
/* GIC interrupt offsets */ /* GIC interrupt offsets */
#define GIC_INT_NET 0 #define GIC_INT_NET GIC_SHARED_TO_HWIRQ(0)
#define GIC_INT_UART1 2 #define GIC_INT_UART1 GIC_SHARED_TO_HWIRQ(2)
#define GIC_INT_UART0 3 #define GIC_INT_UART0 GIC_SHARED_TO_HWIRQ(3)
#define GIC_INT_EHCI 5 #define GIC_INT_EHCI GIC_SHARED_TO_HWIRQ(5)
#endif /* !(_MIPS_SEAD3INT_H) */ #endif /* !(_MIPS_SEAD3INT_H) */
...@@ -68,7 +68,7 @@ int gic_clockevent_init(void) ...@@ -68,7 +68,7 @@ int gic_clockevent_init(void)
if (!cpu_has_counter || !gic_frequency) if (!cpu_has_counter || !gic_frequency)
return -ENXIO; return -ENXIO;
irq = MIPS_GIC_IRQ_BASE; irq = MIPS_GIC_IRQ_BASE + GIC_LOCAL_TO_HWIRQ(GIC_LOCAL_INT_COMPARE);
cd = &per_cpu(gic_clockevent_device, cpu); cd = &per_cpu(gic_clockevent_device, cpu);
...@@ -91,16 +91,13 @@ int gic_clockevent_init(void) ...@@ -91,16 +91,13 @@ int gic_clockevent_init(void)
clockevents_register_device(cd); clockevents_register_device(cd);
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_MAP), if (!gic_timer_irq_installed) {
GIC_MAP_TO_PIN_MSK | gic_cpu_pin); setup_percpu_irq(irq, &gic_compare_irqaction);
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_SMASK), GIC_VPE_SMASK_CMP_MSK); gic_timer_irq_installed = 1;
}
if (gic_timer_irq_installed) enable_percpu_irq(irq, IRQ_TYPE_NONE);
return 0;
gic_timer_irq_installed = 1;
setup_irq(irq, &gic_compare_irqaction);
irq_set_handler(irq, handle_percpu_irq);
return 0; return 0;
} }
...@@ -86,7 +86,7 @@ void mips_event_handler(struct clock_event_device *dev) ...@@ -86,7 +86,7 @@ void mips_event_handler(struct clock_event_device *dev)
static int c0_compare_int_pending(void) static int c0_compare_int_pending(void)
{ {
#ifdef CONFIG_MIPS_GIC #ifdef CONFIG_MIPS_GIC
if (cpu_has_veic) if (gic_present)
return gic_get_timer_pending(); return gic_get_timer_pending();
#endif #endif
return (read_c0_cause() >> cp0_compare_irq_shift) & (1ul << CAUSEB_IP); return (read_c0_cause() >> cp0_compare_irq_shift) & (1ul << CAUSEB_IP);
......
...@@ -273,10 +273,6 @@ asmlinkage void plat_irq_dispatch(void) ...@@ -273,10 +273,6 @@ asmlinkage void plat_irq_dispatch(void)
irq = irq_ffs(pending); irq = irq_ffs(pending);
/* HACK: GIC doesn't properly dispatch local interrupts yet */
if (gic_present && irq == MIPSCPU_INT_GIC && gic_compare_int())
do_IRQ(MIPS_GIC_IRQ_BASE);
else
do_IRQ(MIPS_CPU_IRQ_BASE + irq); do_IRQ(MIPS_CPU_IRQ_BASE + irq);
} }
......
...@@ -126,9 +126,9 @@ int get_c0_perfcount_int(void) ...@@ -126,9 +126,9 @@ int get_c0_perfcount_int(void)
if (cpu_has_veic) { if (cpu_has_veic) {
set_vi_handler(MSC01E_INT_PERFCTR, mips_perf_dispatch); set_vi_handler(MSC01E_INT_PERFCTR, mips_perf_dispatch);
mips_cpu_perf_irq = MSC01E_INT_BASE + MSC01E_INT_PERFCTR; mips_cpu_perf_irq = MSC01E_INT_BASE + MSC01E_INT_PERFCTR;
} else if (gic_present) {
mips_cpu_perf_irq = gic_get_c0_perfcount_int();
} else if (cp0_perfcount_irq >= 0) { } else if (cp0_perfcount_irq >= 0) {
if (cpu_has_vint)
set_vi_handler(cp0_perfcount_irq, mips_perf_dispatch);
mips_cpu_perf_irq = MIPS_CPU_IRQ_BASE + cp0_perfcount_irq; mips_cpu_perf_irq = MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
} else { } else {
mips_cpu_perf_irq = -1; mips_cpu_perf_irq = -1;
...@@ -139,15 +139,12 @@ int get_c0_perfcount_int(void) ...@@ -139,15 +139,12 @@ int get_c0_perfcount_int(void)
unsigned int get_c0_compare_int(void) unsigned int get_c0_compare_int(void)
{ {
#ifdef MSC01E_INT_BASE
if (cpu_has_veic) { if (cpu_has_veic) {
set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch); set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch);
mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR; mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR;
} else } else if (gic_present) {
#endif mips_cpu_timer_irq = gic_get_c0_compare_int();
{ } else {
if (cpu_has_vint)
set_vi_handler(cp0_compare_irq, mips_timer_dispatch);
mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq; mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
} }
......
...@@ -8,24 +8,12 @@ ...@@ -8,24 +8,12 @@
#include <linux/init.h> #include <linux/init.h>
#include <asm/cpu.h> #include <asm/cpu.h>
#include <asm/gic.h>
#include <asm/setup.h> #include <asm/setup.h>
#include <asm/time.h> #include <asm/time.h>
#include <asm/irq.h> #include <asm/irq.h>
#include <asm/mips-boards/generic.h> #include <asm/mips-boards/generic.h>
static int mips_cpu_timer_irq;
static int mips_cpu_perf_irq;
static void mips_timer_dispatch(void)
{
do_IRQ(mips_cpu_timer_irq);
}
static void mips_perf_dispatch(void)
{
do_IRQ(mips_cpu_perf_irq);
}
static void __iomem *status_reg = (void __iomem *)0xbf000410; static void __iomem *status_reg = (void __iomem *)0xbf000410;
/* /*
...@@ -83,22 +71,18 @@ void read_persistent_clock(struct timespec *ts) ...@@ -83,22 +71,18 @@ void read_persistent_clock(struct timespec *ts)
int get_c0_perfcount_int(void) int get_c0_perfcount_int(void)
{ {
if (cp0_perfcount_irq >= 0) { if (gic_present)
if (cpu_has_vint) return gic_get_c0_compare_int();
set_vi_handler(cp0_perfcount_irq, mips_perf_dispatch); if (cp0_perfcount_irq >= 0)
mips_cpu_perf_irq = MIPS_CPU_IRQ_BASE + cp0_perfcount_irq; return MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
} else { return -1;
mips_cpu_perf_irq = -1;
}
return mips_cpu_perf_irq;
} }
unsigned int get_c0_compare_int(void) unsigned int get_c0_compare_int(void)
{ {
if (cpu_has_vint) if (gic_present)
set_vi_handler(cp0_compare_irq, mips_timer_dispatch); return gic_get_c0_compare_int();
mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq; return MIPS_CPU_IRQ_BASE + cp0_compare_irq;
return mips_cpu_timer_irq;
} }
void __init plat_time_init(void) void __init plat_time_init(void)
......
...@@ -44,6 +44,7 @@ static struct gic_intrmask_regs intrmask_regs[NR_CPUS]; ...@@ -44,6 +44,7 @@ static struct gic_intrmask_regs intrmask_regs[NR_CPUS];
static DEFINE_SPINLOCK(gic_lock); static DEFINE_SPINLOCK(gic_lock);
static struct irq_domain *gic_irq_domain; static struct irq_domain *gic_irq_domain;
static int gic_shared_intrs; static int gic_shared_intrs;
static int gic_vpes;
static struct irq_chip gic_level_irq_controller, gic_edge_irq_controller; static struct irq_chip gic_level_irq_controller, gic_edge_irq_controller;
static void __gic_irq_dispatch(void); static void __gic_irq_dispatch(void);
...@@ -96,12 +97,35 @@ cycle_t gic_read_compare(void) ...@@ -96,12 +97,35 @@ cycle_t gic_read_compare(void)
} }
#endif #endif
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;
GICREAD(GIC_REG(VPE_LOCAL, GIC_VPE_CTL), 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;
}
}
unsigned int gic_get_timer_pending(void) unsigned int gic_get_timer_pending(void)
{ {
unsigned int vpe_pending; unsigned int vpe_pending;
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), 0); GICREAD(GIC_REG(VPE_LOCAL, GIC_VPE_PEND), vpe_pending);
GICREAD(GIC_REG(VPE_OTHER, GIC_VPE_PEND), vpe_pending);
return vpe_pending & GIC_VPE_PEND_TIMER_MSK; return vpe_pending & GIC_VPE_PEND_TIMER_MSK;
} }
...@@ -119,53 +143,6 @@ void gic_send_ipi(unsigned int intr) ...@@ -119,53 +143,6 @@ void gic_send_ipi(unsigned int intr)
GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), 0x80000000 | intr); GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), 0x80000000 | intr);
} }
static void __init vpe_local_setup(unsigned int numvpes)
{
unsigned long timer_intr = GIC_INT_TMR;
unsigned long perf_intr = GIC_INT_PERFCTR;
unsigned int vpe_ctl;
int i;
if (cpu_has_veic) {
/*
* GIC timer interrupt -> CPU HW Int X (vector X+2) ->
* map to pin X+2-1 (since GIC adds 1)
*/
timer_intr += (GIC_CPU_TO_VEC_OFFSET - GIC_PIN_TO_VEC_OFFSET);
/*
* GIC perfcnt interrupt -> CPU HW Int X (vector X+2) ->
* map to pin X+2-1 (since GIC adds 1)
*/
perf_intr += (GIC_CPU_TO_VEC_OFFSET - GIC_PIN_TO_VEC_OFFSET);
}
/*
* Setup the default performance counter timer interrupts
* for all VPEs
*/
for (i = 0; i < numvpes; i++) {
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
/* Are Interrupts locally routable? */
GICREAD(GIC_REG(VPE_OTHER, GIC_VPE_CTL), vpe_ctl);
if (vpe_ctl & GIC_VPE_CTL_TIMER_RTBL_MSK)
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_TIMER_MAP),
GIC_MAP_TO_PIN_MSK | timer_intr);
if (cpu_has_veic) {
set_vi_handler(timer_intr + GIC_PIN_TO_VEC_OFFSET,
__gic_irq_dispatch);
}
if (vpe_ctl & GIC_VPE_CTL_PERFCNT_RTBL_MSK)
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_PERFCTR_MAP),
GIC_MAP_TO_PIN_MSK | perf_intr);
if (cpu_has_veic) {
set_vi_handler(perf_intr + GIC_PIN_TO_VEC_OFFSET,
__gic_irq_dispatch);
}
}
}
unsigned int gic_compare_int(void) unsigned int gic_compare_int(void)
{ {
unsigned int pending; unsigned int pending;
...@@ -177,6 +154,26 @@ unsigned int gic_compare_int(void) ...@@ -177,6 +154,26 @@ unsigned int gic_compare_int(void)
return 0; return 0;
} }
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 erformance 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));
}
void gic_get_int_mask(unsigned long *dst, const unsigned long *src) void gic_get_int_mask(unsigned long *dst, const unsigned long *src)
{ {
unsigned int i; unsigned int i;
...@@ -217,24 +214,24 @@ unsigned int gic_get_int(void) ...@@ -217,24 +214,24 @@ unsigned int gic_get_int(void)
static void gic_mask_irq(struct irq_data *d) static void gic_mask_irq(struct irq_data *d)
{ {
GIC_CLR_INTR_MASK(d->hwirq); GIC_CLR_INTR_MASK(GIC_HWIRQ_TO_SHARED(d->hwirq));
} }
static void gic_unmask_irq(struct irq_data *d) static void gic_unmask_irq(struct irq_data *d)
{ {
GIC_SET_INTR_MASK(d->hwirq); GIC_SET_INTR_MASK(GIC_HWIRQ_TO_SHARED(d->hwirq));
} }
static void gic_ack_irq(struct irq_data *d) static void gic_ack_irq(struct irq_data *d)
{ {
unsigned int irq = d->hwirq; unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq); GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq);
} }
static int gic_set_type(struct irq_data *d, unsigned int type) static int gic_set_type(struct irq_data *d, unsigned int type)
{ {
unsigned int irq = d->hwirq; unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
unsigned long flags; unsigned long flags;
bool is_edge; bool is_edge;
...@@ -291,7 +288,7 @@ static int gic_set_type(struct irq_data *d, unsigned int type) ...@@ -291,7 +288,7 @@ static int gic_set_type(struct irq_data *d, unsigned int type)
static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask, static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask,
bool force) bool force)
{ {
unsigned int irq = d->hwirq; unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
cpumask_t tmp = CPU_MASK_NONE; cpumask_t tmp = CPU_MASK_NONE;
unsigned long flags; unsigned long flags;
int i; int i;
...@@ -339,12 +336,85 @@ static struct irq_chip gic_edge_irq_controller = { ...@@ -339,12 +336,85 @@ static struct irq_chip gic_edge_irq_controller = {
#endif #endif
}; };
static unsigned int gic_get_local_int(void)
{
unsigned long pending, masked;
GICREAD(GIC_REG(VPE_LOCAL, GIC_VPE_PEND), pending);
GICREAD(GIC_REG(VPE_LOCAL, GIC_VPE_MASK), masked);
bitmap_and(&pending, &pending, &masked, GIC_NUM_LOCAL_INTRS);
return find_first_bit(&pending, GIC_NUM_LOCAL_INTRS);
}
static void gic_mask_local_irq(struct irq_data *d)
{
int intr = GIC_HWIRQ_TO_LOCAL(d->hwirq);
GICWRITE(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);
GICWRITE(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++) {
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
GICWRITE(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++) {
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
GICWRITE(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) static void __gic_irq_dispatch(void)
{ {
unsigned int intr, virq; unsigned int intr, virq;
while ((intr = gic_get_local_int()) != GIC_NUM_LOCAL_INTRS) {
virq = irq_linear_revmap(gic_irq_domain,
GIC_LOCAL_TO_HWIRQ(intr));
do_IRQ(virq);
}
while ((intr = gic_get_int()) != gic_shared_intrs) { while ((intr = gic_get_int()) != gic_shared_intrs) {
virq = irq_linear_revmap(gic_irq_domain, intr); virq = irq_linear_revmap(gic_irq_domain,
GIC_SHARED_TO_HWIRQ(intr));
do_IRQ(virq); do_IRQ(virq);
} }
} }
...@@ -397,7 +467,8 @@ static struct irqaction irq_call = { ...@@ -397,7 +467,8 @@ static struct irqaction irq_call = {
static __init void gic_ipi_init_one(unsigned int intr, int cpu, static __init void gic_ipi_init_one(unsigned int intr, int cpu,
struct irqaction *action) struct irqaction *action)
{ {
int virq = irq_create_mapping(gic_irq_domain, intr); int virq = irq_create_mapping(gic_irq_domain,
GIC_SHARED_TO_HWIRQ(intr));
int i; int i;
GIC_SH_MAP_TO_VPE_SMASK(intr, cpu); GIC_SH_MAP_TO_VPE_SMASK(intr, cpu);
...@@ -430,7 +501,7 @@ static inline void gic_ipi_init(void) ...@@ -430,7 +501,7 @@ static inline void gic_ipi_init(void)
} }
#endif #endif
static void __init gic_basic_init(int numvpes) static void __init gic_basic_init(void)
{ {
unsigned int i; unsigned int i;
...@@ -443,28 +514,112 @@ static void __init gic_basic_init(int numvpes) ...@@ -443,28 +514,112 @@ static void __init gic_basic_init(int numvpes)
GIC_CLR_INTR_MASK(i); GIC_CLR_INTR_MASK(i);
} }
vpe_local_setup(numvpes); for (i = 0; i < gic_vpes; i++) {
unsigned int j;
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
for (j = 0; j < GIC_NUM_LOCAL_INTRS; j++) {
if (!gic_local_irq_is_routable(j))
continue;
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_RMASK), 1 << j);
}
}
} }
static int gic_irq_domain_map(struct irq_domain *d, unsigned int virq, static int gic_local_irq_domain_map(struct irq_domain *d, unsigned int virq,
irq_hw_number_t hw) 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;
/*
* HACK: These are all really percpu interrupts, but the rest
* of the MIPS kernel code does not use the percpu IRQ API for
* the CP0 timer and performance counter interrupts.
*/
if (intr != GIC_LOCAL_INT_TIMER && intr != GIC_LOCAL_INT_PERFCTR) {
irq_set_chip_and_handler(virq,
&gic_local_irq_controller,
handle_percpu_devid_irq);
irq_set_percpu_devid(virq);
} else {
irq_set_chip_and_handler(virq,
&gic_all_vpes_local_irq_controller,
handle_percpu_irq);
}
spin_lock_irqsave(&gic_lock, flags);
for (i = 0; i < gic_vpes; i++) {
u32 val = GIC_MAP_TO_PIN_MSK | gic_cpu_pin;
GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), i);
switch (intr) {
case GIC_LOCAL_INT_WD:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_WD_MAP), val);
break;
case GIC_LOCAL_INT_COMPARE:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_COMPARE_MAP), val);
break;
case GIC_LOCAL_INT_TIMER:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_TIMER_MAP), val);
break;
case GIC_LOCAL_INT_PERFCTR:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_PERFCTR_MAP), val);
break;
case GIC_LOCAL_INT_SWINT0:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_SWINT0_MAP), val);
break;
case GIC_LOCAL_INT_SWINT1:
GICWRITE(GIC_REG(VPE_OTHER, GIC_VPE_SWINT1_MAP), val);
break;
case GIC_LOCAL_INT_FDC:
GICWRITE(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)
{
int intr = GIC_HWIRQ_TO_SHARED(hw);
unsigned long flags; unsigned long flags;
irq_set_chip_and_handler(virq, &gic_level_irq_controller, irq_set_chip_and_handler(virq, &gic_level_irq_controller,
handle_level_irq); handle_level_irq);
spin_lock_irqsave(&gic_lock, flags); spin_lock_irqsave(&gic_lock, flags);
GICWRITE(GIC_REG_ADDR(SHARED, GIC_SH_MAP_TO_PIN(hw)), GICWRITE(GIC_REG_ADDR(SHARED, GIC_SH_MAP_TO_PIN(intr)),
GIC_MAP_TO_PIN_MSK | gic_cpu_pin); GIC_MAP_TO_PIN_MSK | gic_cpu_pin);
/* Map to VPE 0 by default */ /* Map to VPE 0 by default */
GIC_SH_MAP_TO_VPE_SMASK(hw, 0); GIC_SH_MAP_TO_VPE_SMASK(intr, 0);
set_bit(hw, pcpu_masks[0].pcpu_mask); set_bit(intr, pcpu_masks[0].pcpu_mask);
spin_unlock_irqrestore(&gic_lock, flags); spin_unlock_irqrestore(&gic_lock, flags);
return 0; return 0;
} }
static int gic_irq_domain_map(struct irq_domain *d, unsigned int virq,
irq_hw_number_t hw)
{
if (GIC_HWIRQ_TO_LOCAL(hw) < GIC_NUM_LOCAL_INTRS)
return gic_local_irq_domain_map(d, virq, hw);
return gic_shared_irq_domain_map(d, virq, hw);
}
static struct irq_domain_ops gic_irq_domain_ops = { static struct irq_domain_ops gic_irq_domain_ops = {
.map = gic_irq_domain_map, .map = gic_irq_domain_map,
.xlate = irq_domain_xlate_twocell, .xlate = irq_domain_xlate_twocell,
...@@ -475,7 +630,6 @@ void __init gic_init(unsigned long gic_base_addr, ...@@ -475,7 +630,6 @@ void __init gic_init(unsigned long gic_base_addr,
unsigned int irqbase) unsigned int irqbase)
{ {
unsigned int gicconfig; unsigned int gicconfig;
int numvpes, numintrs;
_gic_base = (unsigned long) ioremap_nocache(gic_base_addr, _gic_base = (unsigned long) ioremap_nocache(gic_base_addr,
gic_addrspace_size); gic_addrspace_size);
...@@ -485,9 +639,9 @@ void __init gic_init(unsigned long gic_base_addr, ...@@ -485,9 +639,9 @@ void __init gic_init(unsigned long gic_base_addr,
GIC_SH_CONFIG_NUMINTRS_SHF; GIC_SH_CONFIG_NUMINTRS_SHF;
gic_shared_intrs = ((gic_shared_intrs + 1) * 8); gic_shared_intrs = ((gic_shared_intrs + 1) * 8);
numvpes = (gicconfig & GIC_SH_CONFIG_NUMVPES_MSK) >> gic_vpes = (gicconfig & GIC_SH_CONFIG_NUMVPES_MSK) >>
GIC_SH_CONFIG_NUMVPES_SHF; GIC_SH_CONFIG_NUMVPES_SHF;
numvpes = numvpes + 1; gic_vpes = gic_vpes + 1;
if (cpu_has_veic) { if (cpu_has_veic) {
/* Always use vector 1 in EIC mode */ /* Always use vector 1 in EIC mode */
...@@ -500,12 +654,13 @@ void __init gic_init(unsigned long gic_base_addr, ...@@ -500,12 +654,13 @@ void __init gic_init(unsigned long gic_base_addr,
gic_irq_dispatch); gic_irq_dispatch);
} }
gic_irq_domain = irq_domain_add_simple(NULL, gic_shared_intrs, irqbase, gic_irq_domain = irq_domain_add_simple(NULL, GIC_NUM_LOCAL_INTRS +
gic_shared_intrs, irqbase,
&gic_irq_domain_ops, NULL); &gic_irq_domain_ops, NULL);
if (!gic_irq_domain) if (!gic_irq_domain)
panic("Failed to add GIC IRQ domain"); panic("Failed to add GIC IRQ domain");
gic_basic_init(numvpes); gic_basic_init();
gic_ipi_init(); gic_ipi_init();
} }
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment