📄 open_pic.c
字号:
/* * arch/ppc/kernel/open_pic.c -- OpenPIC Interrupt Handling * * Copyright (C) 1997 Geert Uytterhoeven * * 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. */#include <linux/config.h>#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 <asm/ptrace.h>#include <asm/signal.h>#include <asm/io.h>#include <asm/pgtable.h>#include <asm/irq.h>#include <asm/prom.h>#include <asm/machdep.h>#include "open_pic.h"#include "open_pic_defs.h"#include "i8259.h"#include <asm/ppcdebug.h>void* OpenPIC_Addr;static volatile struct OpenPIC *OpenPIC = NULL;u_int OpenPIC_NumInitSenses __initdata = 0;u_char *OpenPIC_InitSenses __initdata = NULL;/* * Local (static) OpenPIC Operations *//* Global Operations */static void openpic_reset(void);static void openpic_enable_8259_pass_through(void);static void openpic_disable_8259_pass_through(void);static u_int openpic_irq(void);static void openpic_eoi(void);static u_int openpic_get_priority(void);static void openpic_set_priority(u_int pri);static u_int openpic_get_spurious(void);static void openpic_set_spurious(u_int vector);#ifdef CONFIG_SMP/* Interprocessor Interrupts */static void openpic_initipi(u_int ipi, u_int pri, u_int vector);static irqreturn_t openpic_ipi_action(int cpl, void *dev_id, struct pt_regs *regs);#endif/* Timer Interrupts */static void openpic_inittimer(u_int timer, u_int pri, u_int vector);static void openpic_maptimer(u_int timer, u_int cpumask);/* Interrupt Sources */static void openpic_enable_irq(u_int irq);static void openpic_disable_irq(u_int irq);static void openpic_initirq(u_int irq, u_int pri, u_int vector, int polarity, int is_level);static void openpic_mapirq(u_int irq, u_int cpumask);static void find_ISUs(void);static u_int NumProcessors;static u_int NumSources;static int NumISUs;static int open_pic_irq_offset;static volatile unsigned char* chrp_int_ack_special;OpenPIC_SourcePtr ISU[OPENPIC_MAX_ISU];static void openpic_end_irq(unsigned int irq_nr);static void openpic_set_affinity(unsigned int irq_nr, cpumask_t cpumask);struct hw_interrupt_type open_pic = { " OpenPIC ", NULL, NULL, openpic_enable_irq, openpic_disable_irq, NULL, openpic_end_irq, openpic_set_affinity};#ifdef CONFIG_SMPstatic void openpic_end_ipi(unsigned int irq_nr);static void openpic_enable_ipi(unsigned int irq_nr);static void openpic_disable_ipi(unsigned int irq_nr);struct hw_interrupt_type open_pic_ipi = { " OpenPIC ", NULL, NULL, openpic_enable_ipi, openpic_disable_ipi, NULL, openpic_end_ipi, NULL};#endif /* CONFIG_SMP */unsigned int openpic_vec_ipi;unsigned int openpic_vec_timer;unsigned int openpic_vec_spurious;/* * Accesses to the current processor's openpic registers */#ifdef CONFIG_SMP#define THIS_CPU Processor[cpu]#define DECL_THIS_CPU int cpu = hard_smp_processor_id()#define CHECK_THIS_CPU check_arg_cpu(cpu)#else#define THIS_CPU Processor[hard_smp_processor_id()]#define DECL_THIS_CPU#define CHECK_THIS_CPU#endif /* CONFIG_SMP */#if 0#define check_arg_ipi(ipi) \ if (ipi < 0 || ipi >= OPENPIC_NUM_IPI) \ printk(KERN_ERR "open_pic.c:%d: invalid ipi %d\n", __LINE__, ipi);#define check_arg_timer(timer) \ if (timer < 0 || timer >= OPENPIC_NUM_TIMERS) \ printk(KERN_ERR "open_pic.c:%d: invalid timer %d\n", __LINE__, timer);#define check_arg_vec(vec) \ if (vec < 0 || vec >= OPENPIC_NUM_VECTORS) \ printk(KERN_ERR "open_pic.c:%d: invalid vector %d\n", __LINE__, vec);#define check_arg_pri(pri) \ if (pri < 0 || pri >= OPENPIC_NUM_PRI) \ printk(KERN_ERR "open_pic.c:%d: invalid priority %d\n", __LINE__, pri);/* * Print out a backtrace if it's out of range, since if it's larger than NR_IRQ's * data has probably been corrupted and we're going to panic or deadlock later * anyway --Troy */#define check_arg_irq(irq) \ if (irq < open_pic_irq_offset || irq >= (NumSources+open_pic_irq_offset)){ \ printk(KERN_ERR "open_pic.c:%d: invalid irq %d\n", __LINE__, irq); \ dump_stack(); }#define check_arg_cpu(cpu) \ if (cpu < 0 || cpu >= OPENPIC_MAX_PROCESSORS){ \ printk(KERN_ERR "open_pic.c:%d: invalid cpu %d\n", __LINE__, cpu); \ dump_stack(); }#else#define check_arg_ipi(ipi) do {} while (0)#define check_arg_timer(timer) do {} while (0)#define check_arg_vec(vec) do {} while (0)#define check_arg_pri(pri) do {} while (0)#define check_arg_irq(irq) do {} while (0)#define check_arg_cpu(cpu) do {} while (0)#endif#define GET_ISU(source) ISU[(source) >> 4][(source) & 0xf]void __init pSeries_init_openpic(void){ struct device_node *np; int i; unsigned int *addrp; unsigned char* chrp_int_ack_special = NULL; unsigned char init_senses[NR_IRQS - NUM_ISA_INTERRUPTS]; int nmi_irq = -1;#if defined(CONFIG_VT) && defined(CONFIG_ADB_KEYBOARD) && defined(XMON) struct device_node *kbd;#endif if (!(np = of_find_node_by_name(NULL, "pci")) || !(addrp = (unsigned int *) get_property(np, "8259-interrupt-acknowledge", NULL))) printk(KERN_ERR "Cannot find pci to get ack address\n"); else chrp_int_ack_special = (unsigned char *) __ioremap(addrp[prom_n_addr_cells(np)-1], 1, _PAGE_NO_CACHE); /* hydra still sets OpenPIC_InitSenses to a static set of values */ if (OpenPIC_InitSenses == NULL) { prom_get_irq_senses(init_senses, NUM_ISA_INTERRUPTS, NR_IRQS); OpenPIC_InitSenses = init_senses; OpenPIC_NumInitSenses = NR_IRQS - NUM_ISA_INTERRUPTS; } openpic_init(1, NUM_ISA_INTERRUPTS, chrp_int_ack_special, nmi_irq); for (i = 0; i < NUM_ISA_INTERRUPTS; i++) irq_desc[i].handler = &i8259_pic; of_node_put(np);}static inline u_int openpic_read(volatile u_int *addr){ u_int val; val = in_le32(addr); return val;}static inline void openpic_write(volatile u_int *addr, u_int val){ out_le32(addr, val);}static inline u_int openpic_readfield(volatile u_int *addr, u_int mask){ u_int val = openpic_read(addr); return val & mask;}static inline void openpic_writefield(volatile u_int *addr, u_int mask, u_int field){ u_int val = openpic_read(addr); openpic_write(addr, (val & ~mask) | (field & mask));}static inline void openpic_clearfield(volatile u_int *addr, u_int mask){ openpic_writefield(addr, mask, 0);}static inline void openpic_setfield(volatile u_int *addr, u_int mask){ openpic_writefield(addr, mask, mask);}static void openpic_safe_writefield(volatile u_int *addr, u_int mask, u_int field){ unsigned int loops = 100000; openpic_setfield(addr, OPENPIC_MASK); while (openpic_read(addr) & OPENPIC_ACTIVITY) { if (!loops--) { printk(KERN_ERR "openpic_safe_writefield timeout\n"); break; } } openpic_writefield(addr, mask | OPENPIC_MASK, field | OPENPIC_MASK);}#ifdef CONFIG_SMPstatic int broken_ipi_registers;static u_int openpic_read_IPI(volatile u_int* addr){ u_int val = 0; if (broken_ipi_registers) /* yes this is right ... bug, feature, you decide! -- tgall */ val = in_be32(addr); else val = in_le32(addr); return val;}static void openpic_test_broken_IPI(void){ u_int t; openpic_write(&OpenPIC->Global.IPI_Vector_Priority(0), OPENPIC_MASK); t = openpic_read(&OpenPIC->Global.IPI_Vector_Priority(0)); if (t == le32_to_cpu(OPENPIC_MASK)) { printk(KERN_INFO "OpenPIC reversed IPI registers detected\n"); broken_ipi_registers = 1; }}/* because of the power3 be / le above, this is needed */static inline void openpic_writefield_IPI(volatile u_int* addr, u_int mask, u_int field){ u_int val = openpic_read_IPI(addr); openpic_write(addr, (val & ~mask) | (field & mask));}static inline void openpic_clearfield_IPI(volatile u_int *addr, u_int mask){ openpic_writefield_IPI(addr, mask, 0);}static inline void openpic_setfield_IPI(volatile u_int *addr, u_int mask){ openpic_writefield_IPI(addr, mask, mask);}static void openpic_safe_writefield_IPI(volatile u_int *addr, u_int mask, u_int field){ unsigned int loops = 100000; openpic_setfield_IPI(addr, OPENPIC_MASK); /* wait until it's not in use */ /* BenH: Is this code really enough ? I would rather check the result * and eventually retry ... */ while(openpic_read_IPI(addr) & OPENPIC_ACTIVITY) { if (!loops--) { printk(KERN_ERR "openpic_safe_writefield timeout\n"); break; } } openpic_writefield_IPI(addr, mask, field | OPENPIC_MASK);}#endif /* CONFIG_SMP */void __init openpic_init(int main_pic, int offset, unsigned char* chrp_ack, int programmer_switch_irq){ u_int t, i; u_int timerfreq; const char *version; if (!OpenPIC_Addr) { printk(KERN_INFO "No OpenPIC found !\n"); return; } OpenPIC = (volatile struct OpenPIC *)OpenPIC_Addr; ppc64_boot_msg(0x20, "OpenPic Init"); t = openpic_read(&OpenPIC->Global.Feature_Reporting0); switch (t & OPENPIC_FEATURE_VERSION_MASK) { case 1: version = "1.0"; break; case 2: version = "1.2"; break; case 3: version = "1.3"; break; default: version = "?"; break; } NumProcessors = ((t & OPENPIC_FEATURE_LAST_PROCESSOR_MASK) >> OPENPIC_FEATURE_LAST_PROCESSOR_SHIFT) + 1; NumSources = ((t & OPENPIC_FEATURE_LAST_SOURCE_MASK) >> OPENPIC_FEATURE_LAST_SOURCE_SHIFT) + 1; printk(KERN_INFO "OpenPIC Version %s (%d CPUs and %d IRQ sources) at %p\n", version, NumProcessors, NumSources, OpenPIC); timerfreq = openpic_read(&OpenPIC->Global.Timer_Frequency); if (timerfreq) printk(KERN_INFO "OpenPIC timer frequency is %d.%06d MHz\n", timerfreq / 1000000, timerfreq % 1000000); if (!main_pic) return; open_pic_irq_offset = offset; chrp_int_ack_special = (volatile unsigned char*)chrp_ack; find_ISUs(); /* Initialize timer interrupts */ ppc64_boot_msg(0x21, "OpenPic Timer"); for (i = 0; i < OPENPIC_NUM_TIMERS; i++) { /* Disabled, Priority 0 */ openpic_inittimer(i, 0, openpic_vec_timer+i); /* No processor */ openpic_maptimer(i, 0); }#ifdef CONFIG_SMP /* Initialize IPI interrupts */ ppc64_boot_msg(0x22, "OpenPic IPI"); openpic_test_broken_IPI(); for (i = 0; i < OPENPIC_NUM_IPI; i++) { /* Disabled, Priority 10..13 */ openpic_initipi(i, 10+i, openpic_vec_ipi+i); /* IPIs are per-CPU */ irq_desc[openpic_vec_ipi+i].status |= IRQ_PER_CPU; irq_desc[openpic_vec_ipi+i].handler = &open_pic_ipi; }#endif /* Initialize external interrupts */ ppc64_boot_msg(0x23, "OpenPic Ext"); openpic_set_priority(0xf); /* SIOint (8259 cascade) is special */ if (offset) { openpic_initirq(0, 8, offset, 1, 1); openpic_mapirq(0, 1 << get_hard_smp_processor_id(boot_cpuid)); } /* Init all external sources */ for (i = 0; i < NumSources; i++) { int pri, sense; /* skip cascade if any */ if (offset && i == 0) continue; /* the bootloader may have left it enabled (bad !) */ openpic_disable_irq(i+offset); pri = (i == programmer_switch_irq)? 9: 8; sense = (i < OpenPIC_NumInitSenses)? OpenPIC_InitSenses[i]: 1; if (sense) irq_desc[i+offset].status = IRQ_LEVEL; /* Enabled, Priority 8 or 9 */ openpic_initirq(i, pri, i+offset, !sense, sense); /* Processor 0 */ openpic_mapirq(i, 1 << get_hard_smp_processor_id(boot_cpuid)); } /* Init descriptors */ for (i = offset; i < NumSources + offset; i++) irq_desc[i].handler = &open_pic; /* Initialize the spurious interrupt */ ppc64_boot_msg(0x24, "OpenPic Spurious"); openpic_set_spurious(openpic_vec_spurious); openpic_set_priority(0); openpic_disable_8259_pass_through(); ppc64_boot_msg(0x25, "OpenPic Done");}/* * We cant do this in init_IRQ because we need the memory subsystem up for * request_irq() */static int __init openpic_setup_i8259(void){ if (systemcfg->platform == PLATFORM_POWERMAC) return 0; if (naca->interrupt_controller == IC_OPEN_PIC) { /* Initialize the cascade */ if (request_irq(NUM_ISA_INTERRUPTS, no_action, SA_INTERRUPT, "82c59 cascade", NULL)) printk(KERN_ERR "Unable to get OpenPIC IRQ 0 for cascade\n"); i8259_init();
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -