[<prev] [next>] [<thread-prev] [thread-next>] [day] [month] [year] [list]
Message-ID: <54195975.1010209@imgtec.com>
Date: Wed, 17 Sep 2014 10:50:45 +0100
From: Qais Yousef <qais.yousef@...tec.com>
To: Andrew Bresticker <abrestic@...omium.org>,
Ralf Baechle <ralf@...ux-mips.org>,
Thomas Gleixner <tglx@...utronix.de>,
Jason Cooper <jason@...edaemon.net>
CC: Jeffrey Deans <jeffrey.deans@...tec.com>,
Markos Chandras <markos.chandras@...tec.com>,
Paul Burton <paul.burton@...tec.com>,
"Jonas Gorski" <jogo@...nwrt.org>,
John Crispin <blogic@...nwrt.org>,
David Daney <ddaney.cavm@...il.com>,
<linux-mips@...ux-mips.org>, <linux-kernel@...r.kernel.org>
Subject: Re: [PATCH 21/24] irqchip: mips-gic: Support local interrupts
On 09/16/2014 12:51 AM, Andrew Bresticker wrote:
> 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.
>
> 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: Andrew Bresticker <abrestic@...omium.org>
> ---
> arch/mips/include/asm/gic.h | 29 +++-
> arch/mips/include/asm/mips-boards/maltaint.h | 4 +-
> arch/mips/include/asm/mips-boards/sead3int.h | 10 +-
> arch/mips/kernel/cevt-gic.c | 15 +-
> arch/mips/kernel/cevt-r4k.c | 2 +-
> arch/mips/mti-malta/malta-int.c | 6 +-
> arch/mips/mti-malta/malta-time.c | 13 +-
> arch/mips/mti-sead3/sead3-time.c | 34 +---
> drivers/irqchip/irq-mips-gic.c | 244 +++++++++++++++++++--------
> 9 files changed, 232 insertions(+), 125 deletions(-)
>
> diff --git a/arch/mips/include/asm/gic.h b/arch/mips/include/asm/gic.h
> index f245395..6b99610 100644
> --- a/arch/mips/include/asm/gic.h
> +++ b/arch/mips/include/asm/gic.h
> @@ -209,6 +209,7 @@
> #define GIC_VPE_WD_MAP_OFS 0x0040
> #define GIC_VPE_COMPARE_MAP_OFS 0x0044
> #define GIC_VPE_TIMER_MAP_OFS 0x0048
> +#define GIC_VPE_FDC_MAP_OFS 0x004c
> #define GIC_VPE_PERFCTR_MAP_OFS 0x0050
> #define GIC_VPE_SWINT0_MAP_OFS 0x0054
> #define GIC_VPE_SWINT1_MAP_OFS 0x0058
> @@ -262,6 +263,10 @@
> #define GIC_MAP_MSK (MSK(6) << GIC_MAP_SHF)
>
> /* 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_MSK (MSK(1) << GIC_VPE_CTL_PERFCNT_RTBL_SHF)
> #define GIC_VPE_CTL_TIMER_RTBL_SHF 1
> @@ -329,16 +334,30 @@
> /* Add 2 to convert GIC CPU pin to core interrupt */
> #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. */
> #define GIC_CPU_TO_VEC_OFFSET (2)
>
> /* Mapped interrupt to pin X, then GIC will generate the vector (X+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/irq.h>
>
> @@ -363,4 +382,6 @@ extern void gic_bind_eic_interrupt(int irq, int set);
> extern unsigned int gic_get_timer_pending(void);
> extern void gic_get_int_mask(unsigned long *dst, const unsigned long *src);
> 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 */
> diff --git a/arch/mips/include/asm/mips-boards/maltaint.h b/arch/mips/include/asm/mips-boards/maltaint.h
> index bdd6f39..38b06a0 100644
> --- a/arch/mips/include/asm/mips-boards/maltaint.h
> +++ b/arch/mips/include/asm/mips-boards/maltaint.h
> @@ -10,6 +10,8 @@
> #ifndef _MIPS_MALTAINT_H
> #define _MIPS_MALTAINT_H
>
> +#include <asm/gic.h>
> +
nit: I think gic.h should be split to driver/irqchip/irq-mips-gic.h for
private definitions and include/linux/irqchip/irq-mips-gic.h for
exported/public definitions.
> /*
> * Interrupts 0..15 are used for Malta ISA compatible interrupts
> */
> @@ -61,6 +63,6 @@
> #define MSC01E_INT_CPUCTR 11
>
> /* GIC external interrupts */
> -#define GIC_INT_I8259A 3
> +#define GIC_INT_I8259A GIC_SHARED_TO_HWIRQ(3)
>
> #endif /* !(_MIPS_MALTAINT_H) */
> diff --git a/arch/mips/include/asm/mips-boards/sead3int.h b/arch/mips/include/asm/mips-boards/sead3int.h
> index a2e0095..59d6c32 100644
> --- a/arch/mips/include/asm/mips-boards/sead3int.h
> +++ b/arch/mips/include/asm/mips-boards/sead3int.h
> @@ -10,6 +10,8 @@
> #ifndef _MIPS_SEAD3INT_H
> #define _MIPS_SEAD3INT_H
>
> +#include <asm/gic.h>
> +
> /* SEAD-3 GIC address space definitions. */
> #define GIC_BASE_ADDR 0x1b1c0000
> #define GIC_ADDRSPACE_SZ (128 * 1024)
> @@ -22,9 +24,9 @@
> #define CPU_INT_NET 6
>
> /* GIC interrupt offsets */
> -#define GIC_INT_NET 0
> -#define GIC_INT_UART1 2
> -#define GIC_INT_UART0 3
> -#define GIC_INT_EHCI 5
> +#define GIC_INT_NET GIC_SHARED_TO_HWIRQ(0)
> +#define GIC_INT_UART1 GIC_SHARED_TO_HWIRQ(2)
> +#define GIC_INT_UART0 GIC_SHARED_TO_HWIRQ(3)
> +#define GIC_INT_EHCI GIC_SHARED_TO_HWIRQ(5)
>
> #endif /* !(_MIPS_SEAD3INT_H) */
> diff --git a/arch/mips/kernel/cevt-gic.c b/arch/mips/kernel/cevt-gic.c
> index a90bd4c..4f9262a 100644
> --- a/arch/mips/kernel/cevt-gic.c
> +++ b/arch/mips/kernel/cevt-gic.c
> @@ -68,7 +68,7 @@ int gic_clockevent_init(void)
> if (!cpu_has_counter || !gic_frequency)
> 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);
>
> @@ -91,16 +91,13 @@ int gic_clockevent_init(void)
>
> clockevents_register_device(cd);
>
> - GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_COMPARE_MAP),
> - GIC_MAP_TO_PIN_MSK | gic_cpu_pin);
> - GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_SMASK), GIC_VPE_SMASK_CMP_MSK);
> + if (!gic_timer_irq_installed) {
> + setup_percpu_irq(irq, &gic_compare_irqaction);
> + gic_timer_irq_installed = 1;
> + }
>
> - if (gic_timer_irq_installed)
> - return 0;
> + enable_percpu_irq(irq, IRQ_TYPE_NONE);
>
> - gic_timer_irq_installed = 1;
>
> - setup_irq(irq, &gic_compare_irqaction);
> - irq_set_handler(irq, handle_percpu_irq);
> return 0;
> }
> diff --git a/arch/mips/kernel/cevt-r4k.c b/arch/mips/kernel/cevt-r4k.c
> index 5b8f8e3..fd0ef8d 100644
> --- a/arch/mips/kernel/cevt-r4k.c
> +++ b/arch/mips/kernel/cevt-r4k.c
> @@ -86,7 +86,7 @@ void mips_event_handler(struct clock_event_device *dev)
> static int c0_compare_int_pending(void)
> {
> #ifdef CONFIG_MIPS_GIC
> - if (cpu_has_veic)
> + if (gic_present)
> return gic_get_timer_pending();
> #endif
> return (read_c0_cause() >> cp0_compare_irq_shift) & (1ul << CAUSEB_IP);
> diff --git a/arch/mips/mti-malta/malta-int.c b/arch/mips/mti-malta/malta-int.c
> index 3b3bc1d..c6b3548 100644
> --- a/arch/mips/mti-malta/malta-int.c
> +++ b/arch/mips/mti-malta/malta-int.c
> @@ -273,11 +273,7 @@ asmlinkage void plat_irq_dispatch(void)
>
> 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);
> }
>
> #ifdef CONFIG_MIPS_MT_SMP
> diff --git a/arch/mips/mti-malta/malta-time.c b/arch/mips/mti-malta/malta-time.c
> index 17cfc8a..f6ca8ea 100644
> --- a/arch/mips/mti-malta/malta-time.c
> +++ b/arch/mips/mti-malta/malta-time.c
> @@ -126,9 +126,9 @@ int get_c0_perfcount_int(void)
> if (cpu_has_veic) {
> set_vi_handler(MSC01E_INT_PERFCTR, mips_perf_dispatch);
> 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) {
> - if (cpu_has_vint)
> - set_vi_handler(cp0_perfcount_irq, mips_perf_dispatch);
> mips_cpu_perf_irq = MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
> } else {
> mips_cpu_perf_irq = -1;
> @@ -139,15 +139,12 @@ int get_c0_perfcount_int(void)
>
> unsigned int get_c0_compare_int(void)
> {
> -#ifdef MSC01E_INT_BASE
> if (cpu_has_veic) {
> set_vi_handler(MSC01E_INT_CPUCTR, mips_timer_dispatch);
> mips_cpu_timer_irq = MSC01E_INT_BASE + MSC01E_INT_CPUCTR;
> - } else
> -#endif
> - {
> - if (cpu_has_vint)
> - set_vi_handler(cp0_compare_irq, mips_timer_dispatch);
> + } else if (gic_present) {
> + mips_cpu_timer_irq = gic_get_c0_compare_int();
> + } else {
> mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
> }
>
> diff --git a/arch/mips/mti-sead3/sead3-time.c b/arch/mips/mti-sead3/sead3-time.c
> index f090c51..fd40de3 100644
> --- a/arch/mips/mti-sead3/sead3-time.c
> +++ b/arch/mips/mti-sead3/sead3-time.c
> @@ -8,24 +8,12 @@
> #include <linux/init.h>
>
> #include <asm/cpu.h>
> +#include <asm/gic.h>
> #include <asm/setup.h>
> #include <asm/time.h>
> #include <asm/irq.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;
>
> /*
> @@ -83,22 +71,18 @@ void read_persistent_clock(struct timespec *ts)
>
> int get_c0_perfcount_int(void)
> {
> - 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;
> - } else {
> - mips_cpu_perf_irq = -1;
> - }
> - return mips_cpu_perf_irq;
> + if (gic_present)
> + return gic_get_c0_compare_int();
> + if (cp0_perfcount_irq >= 0)
> + return MIPS_CPU_IRQ_BASE + cp0_perfcount_irq;
> + return -1;
> }
>
> unsigned int get_c0_compare_int(void)
> {
> - if (cpu_has_vint)
> - set_vi_handler(cp0_compare_irq, mips_timer_dispatch);
> - mips_cpu_timer_irq = MIPS_CPU_IRQ_BASE + cp0_compare_irq;
> - return mips_cpu_timer_irq;
> + if (gic_present)
> + return gic_get_c0_compare_int();
> + return MIPS_CPU_IRQ_BASE + cp0_compare_irq;
> }
>
> void __init plat_time_init(void)
> diff --git a/drivers/irqchip/irq-mips-gic.c b/drivers/irqchip/irq-mips-gic.c
> index 6682a4e..3abe310 100644
> --- a/drivers/irqchip/irq-mips-gic.c
> +++ b/drivers/irqchip/irq-mips-gic.c
> @@ -43,6 +43,7 @@ static struct gic_pending_regs pending_regs[NR_CPUS];
> static struct gic_intrmask_regs intrmask_regs[NR_CPUS];
> static struct irq_domain *gic_irq_domain;
> static int gic_shared_intrs;
> +static int gic_vpes;
> static struct irq_chip gic_level_irq_controller, gic_edge_irq_controller;
>
> static void __gic_irq_dispatch(void);
> @@ -95,12 +96,39 @@ cycle_t gic_read_compare(void)
> }
> #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:
> + /*
> + * SWINT{0,1} are not routable in non-EIC mode, regardless
> + * of the setting of SWINT_ROUTABLE.
> + */
> + return false;
Hmm AFAIK they are routable. Actually from hard reset they're
automatically routed to vpe0 pin 0 which caught me a number of times
when trying to use software interrupt on hardware that has GIC. When
setting software interrupt I was seeing pin 0 going high too and thought
it's a hardware bug for a while.
I think all local interrupts should be masked at GIC initialisation
except for timer interrupt. I was preparing a set of patches for GIC but
you beat me into it :)
> + default:
> + return true;
> + }
> +}
> +
> unsigned int gic_get_timer_pending(void)
> {
> unsigned int vpe_pending;
>
> - GICWRITE(GIC_REG(VPE_LOCAL, GIC_VPE_OTHER_ADDR), 0);
> - GICREAD(GIC_REG(VPE_OTHER, GIC_VPE_PEND), vpe_pending);
> + GICREAD(GIC_REG(VPE_LOCAL, GIC_VPE_PEND), vpe_pending);
> return (vpe_pending & GIC_VPE_PEND_TIMER_MSK);
> }
>
> @@ -118,53 +146,6 @@ void gic_send_ipi(unsigned int 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 pending;
> @@ -176,6 +157,26 @@ unsigned int gic_compare_int(void)
> 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)
> {
> unsigned int i;
> @@ -216,24 +217,24 @@ unsigned int gic_get_int(void)
>
> 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)
> {
> - 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)
> {
> - unsigned int irq = d->hwirq;
> + unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
>
> GICWRITE(GIC_REG(SHARED, GIC_SH_WEDGE), irq);
> }
>
> 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);
> bool is_edge;
>
> switch (type & IRQ_TYPE_SENSE_MASK) {
> @@ -289,7 +290,7 @@ static DEFINE_SPINLOCK(gic_lock);
> static int gic_set_affinity(struct irq_data *d, const struct cpumask *cpumask,
> bool force)
> {
> - unsigned int irq = d->hwirq;
> + unsigned int irq = GIC_HWIRQ_TO_SHARED(d->hwirq);
> cpumask_t tmp = CPU_MASK_NONE;
> unsigned long flags;
> int i;
> @@ -341,12 +342,54 @@ static struct irq_chip gic_edge_irq_controller = {
> #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_ack = gic_mask_local_irq,
> + .irq_mask = gic_mask_local_irq,
> + .irq_mask_ack = gic_mask_local_irq,
> + .irq_unmask = gic_unmask_local_irq,
> + .irq_eoi = gic_unmask_local_irq,
> +};
> +
again I don't think irq_ack, irq_mask_ack and irq_eoi are needed.
nice clean up here too :)
Qais
> static void __gic_irq_dispatch(void)
> {
> 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) {
> - virq = irq_linear_revmap(gic_irq_domain, intr);
> + virq = irq_linear_revmap(gic_irq_domain,
> + GIC_SHARED_TO_HWIRQ(intr));
> do_IRQ(virq);
> }
> }
> @@ -399,7 +442,8 @@ static struct irqaction irq_call = {
> static __init void gic_ipi_init_one(unsigned int intr, int cpu,
> 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;
>
> GIC_SH_MAP_TO_VPE_SMASK(intr, cpu);
> @@ -432,7 +476,7 @@ static inline void gic_ipi_init(void)
> }
> #endif
>
> -static void __init gic_basic_init(int numvpes)
> +static void __init gic_basic_init(void)
> {
> unsigned int i;
>
> @@ -444,25 +488,89 @@ static void __init gic_basic_init(int numvpes)
> GIC_SET_TRIGGER(i, GIC_TRIG_LEVEL);
> GIC_CLR_INTR_MASK(i);
> }
> +}
> +
> +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 i;
> +
> + if (!gic_local_irq_is_routable(intr))
> + return -EPERM;
>
> - vpe_local_setup(numvpes);
> + irq_set_chip_and_handler(virq, &gic_local_irq_controller,
> + handle_percpu_irq);
> +
> + /*
> + * 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_percpu_devid(virq);
> +
> + 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);
> + return -EINVAL;
> + }
> + }
> +
> + return 0;
> }
>
> -static int gic_irq_domain_map(struct irq_domain *d, unsigned int virq,
> - irq_hw_number_t hw)
> +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);
> +
> irq_set_chip_and_handler(virq, &gic_level_irq_controller,
> handle_level_irq);
>
> - 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);
> /* Map to VPE 0 by default */
> - GIC_SH_MAP_TO_VPE_SMASK(hw, 0);
> - set_bit(hw, pcpu_masks[0].pcpu_mask);
> + GIC_SH_MAP_TO_VPE_SMASK(intr, 0);
> + set_bit(intr, pcpu_masks[0].pcpu_mask);
>
> 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 = {
> .map = gic_irq_domain_map,
> .xlate = irq_domain_xlate_twocell,
> @@ -473,7 +581,6 @@ void __init gic_init(unsigned long gic_base_addr,
> unsigned int irqbase)
> {
> unsigned int gicconfig;
> - int numvpes, numintrs;
>
> _gic_base = (unsigned long) ioremap_nocache(gic_base_addr,
> gic_addrspace_size);
> @@ -483,9 +590,9 @@ void __init gic_init(unsigned long gic_base_addr,
> GIC_SH_CONFIG_NUMINTRS_SHF;
> 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;
> - numvpes = numvpes + 1;
> + gic_vpes = gic_vpes + 1;
>
> if (cpu_has_veic) {
> /* Always use vector 1 in EIC mode */
> @@ -498,12 +605,13 @@ void __init gic_init(unsigned long gic_base_addr,
> 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);
> if (!gic_irq_domain)
> panic("Failed to add GIC IRQ domain");
>
> - gic_basic_init(numvpes);
> + gic_basic_init();
>
> gic_ipi_init();
> }
--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@...r.kernel.org
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/
Powered by blists - more mailing lists