/arch/powerpc/sysdev/mpic.c
C | 1715 lines | 1273 code | 308 blank | 134 comment | 190 complexity | 684a100e713318dd71b4e85b5314fe28 MD5 | raw file
Possible License(s): LGPL-2.0, AGPL-1.0, GPL-2.0
- /*
- * arch/powerpc/kernel/mpic.c
- *
- * Driver for interrupt controllers following the OpenPIC standard, the
- * common implementation beeing IBM's MPIC. This driver also can deal
- * with various broken implementations of this HW.
- *
- * Copyright (C) 2004 Benjamin Herrenschmidt, IBM Corp.
- *
- * 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.
- */
- #undef DEBUG
- #undef DEBUG_IPI
- #undef DEBUG_IRQ
- #undef DEBUG_LOW
- #include <linux/types.h>
- #include <linux/kernel.h>
- #include <linux/init.h>
- #include <linux/irq.h>
- #include <linux/smp.h>
- #include <linux/interrupt.h>
- #include <linux/bootmem.h>
- #include <linux/spinlock.h>
- #include <linux/pci.h>
- #include <linux/slab.h>
- #include <asm/ptrace.h>
- #include <asm/signal.h>
- #include <asm/io.h>
- #include <asm/pgtable.h>
- #include <asm/irq.h>
- #include <asm/machdep.h>
- #include <asm/mpic.h>
- #include <asm/smp.h>
- #include "mpic.h"
- #ifdef DEBUG
- #define DBG(fmt...) printk(fmt)
- #else
- #define DBG(fmt...)
- #endif
- static struct mpic *mpics;
- static struct mpic *mpic_primary;
- static DEFINE_RAW_SPINLOCK(mpic_lock);
- #ifdef CONFIG_PPC32 /* XXX for now */
- #ifdef CONFIG_IRQ_ALL_CPUS
- #define distribute_irqs (1)
- #else
- #define distribute_irqs (0)
- #endif
- #endif
- #ifdef CONFIG_MPIC_WEIRD
- static u32 mpic_infos[][MPIC_IDX_END] = {
- [0] = { /* Original OpenPIC compatible MPIC */
- MPIC_GREG_BASE,
- MPIC_GREG_FEATURE_0,
- MPIC_GREG_GLOBAL_CONF_0,
- MPIC_GREG_VENDOR_ID,
- MPIC_GREG_IPI_VECTOR_PRI_0,
- MPIC_GREG_IPI_STRIDE,
- MPIC_GREG_SPURIOUS,
- MPIC_GREG_TIMER_FREQ,
- MPIC_TIMER_BASE,
- MPIC_TIMER_STRIDE,
- MPIC_TIMER_CURRENT_CNT,
- MPIC_TIMER_BASE_CNT,
- MPIC_TIMER_VECTOR_PRI,
- MPIC_TIMER_DESTINATION,
- MPIC_CPU_BASE,
- MPIC_CPU_STRIDE,
- MPIC_CPU_IPI_DISPATCH_0,
- MPIC_CPU_IPI_DISPATCH_STRIDE,
- MPIC_CPU_CURRENT_TASK_PRI,
- MPIC_CPU_WHOAMI,
- MPIC_CPU_INTACK,
- MPIC_CPU_EOI,
- MPIC_CPU_MCACK,
- MPIC_IRQ_BASE,
- MPIC_IRQ_STRIDE,
- MPIC_IRQ_VECTOR_PRI,
- MPIC_VECPRI_VECTOR_MASK,
- MPIC_VECPRI_POLARITY_POSITIVE,
- MPIC_VECPRI_POLARITY_NEGATIVE,
- MPIC_VECPRI_SENSE_LEVEL,
- MPIC_VECPRI_SENSE_EDGE,
- MPIC_VECPRI_POLARITY_MASK,
- MPIC_VECPRI_SENSE_MASK,
- MPIC_IRQ_DESTINATION
- },
- [1] = { /* Tsi108/109 PIC */
- TSI108_GREG_BASE,
- TSI108_GREG_FEATURE_0,
- TSI108_GREG_GLOBAL_CONF_0,
- TSI108_GREG_VENDOR_ID,
- TSI108_GREG_IPI_VECTOR_PRI_0,
- TSI108_GREG_IPI_STRIDE,
- TSI108_GREG_SPURIOUS,
- TSI108_GREG_TIMER_FREQ,
- TSI108_TIMER_BASE,
- TSI108_TIMER_STRIDE,
- TSI108_TIMER_CURRENT_CNT,
- TSI108_TIMER_BASE_CNT,
- TSI108_TIMER_VECTOR_PRI,
- TSI108_TIMER_DESTINATION,
- TSI108_CPU_BASE,
- TSI108_CPU_STRIDE,
- TSI108_CPU_IPI_DISPATCH_0,
- TSI108_CPU_IPI_DISPATCH_STRIDE,
- TSI108_CPU_CURRENT_TASK_PRI,
- TSI108_CPU_WHOAMI,
- TSI108_CPU_INTACK,
- TSI108_CPU_EOI,
- TSI108_CPU_MCACK,
- TSI108_IRQ_BASE,
- TSI108_IRQ_STRIDE,
- TSI108_IRQ_VECTOR_PRI,
- TSI108_VECPRI_VECTOR_MASK,
- TSI108_VECPRI_POLARITY_POSITIVE,
- TSI108_VECPRI_POLARITY_NEGATIVE,
- TSI108_VECPRI_SENSE_LEVEL,
- TSI108_VECPRI_SENSE_EDGE,
- TSI108_VECPRI_POLARITY_MASK,
- TSI108_VECPRI_SENSE_MASK,
- TSI108_IRQ_DESTINATION
- },
- };
- #define MPIC_INFO(name) mpic->hw_set[MPIC_IDX_##name]
- #else /* CONFIG_MPIC_WEIRD */
- #define MPIC_INFO(name) MPIC_##name
- #endif /* CONFIG_MPIC_WEIRD */
- /*
- * Register accessor functions
- */
- static inline u32 _mpic_read(enum mpic_reg_type type,
- struct mpic_reg_bank *rb,
- unsigned int reg)
- {
- switch(type) {
- #ifdef CONFIG_PPC_DCR
- case mpic_access_dcr:
- return dcr_read(rb->dhost, reg);
- #endif
- case mpic_access_mmio_be:
- return in_be32(rb->base + (reg >> 2));
- case mpic_access_mmio_le:
- default:
- return in_le32(rb->base + (reg >> 2));
- }
- }
- static inline void _mpic_write(enum mpic_reg_type type,
- struct mpic_reg_bank *rb,
- unsigned int reg, u32 value)
- {
- switch(type) {
- #ifdef CONFIG_PPC_DCR
- case mpic_access_dcr:
- dcr_write(rb->dhost, reg, value);
- break;
- #endif
- case mpic_access_mmio_be:
- out_be32(rb->base + (reg >> 2), value);
- break;
- case mpic_access_mmio_le:
- default:
- out_le32(rb->base + (reg >> 2), value);
- break;
- }
- }
- static inline u32 _mpic_ipi_read(struct mpic *mpic, unsigned int ipi)
- {
- enum mpic_reg_type type = mpic->reg_type;
- unsigned int offset = MPIC_INFO(GREG_IPI_VECTOR_PRI_0) +
- (ipi * MPIC_INFO(GREG_IPI_STRIDE));
- if ((mpic->flags & MPIC_BROKEN_IPI) && type == mpic_access_mmio_le)
- type = mpic_access_mmio_be;
- return _mpic_read(type, &mpic->gregs, offset);
- }
- static inline void _mpic_ipi_write(struct mpic *mpic, unsigned int ipi, u32 value)
- {
- unsigned int offset = MPIC_INFO(GREG_IPI_VECTOR_PRI_0) +
- (ipi * MPIC_INFO(GREG_IPI_STRIDE));
- _mpic_write(mpic->reg_type, &mpic->gregs, offset, value);
- }
- static inline u32 _mpic_cpu_read(struct mpic *mpic, unsigned int reg)
- {
- unsigned int cpu = 0;
- if (mpic->flags & MPIC_PRIMARY)
- cpu = hard_smp_processor_id();
- return _mpic_read(mpic->reg_type, &mpic->cpuregs[cpu], reg);
- }
- static inline void _mpic_cpu_write(struct mpic *mpic, unsigned int reg, u32 value)
- {
- unsigned int cpu = 0;
- if (mpic->flags & MPIC_PRIMARY)
- cpu = hard_smp_processor_id();
- _mpic_write(mpic->reg_type, &mpic->cpuregs[cpu], reg, value);
- }
- static inline u32 _mpic_irq_read(struct mpic *mpic, unsigned int src_no, unsigned int reg)
- {
- unsigned int isu = src_no >> mpic->isu_shift;
- unsigned int idx = src_no & mpic->isu_mask;
- unsigned int val;
- val = _mpic_read(mpic->reg_type, &mpic->isus[isu],
- reg + (idx * MPIC_INFO(IRQ_STRIDE)));
- #ifdef CONFIG_MPIC_BROKEN_REGREAD
- if (reg == 0)
- val = (val & (MPIC_VECPRI_MASK | MPIC_VECPRI_ACTIVITY)) |
- mpic->isu_reg0_shadow[src_no];
- #endif
- return val;
- }
- static inline void _mpic_irq_write(struct mpic *mpic, unsigned int src_no,
- unsigned int reg, u32 value)