📄 ints-priority-sc.c
字号:
#define PINT_2_BANK(x) ((x) >> 5)#define PINT_2_BIT(x) ((x) & 0x1F)#define PINT_BIT(x) (1 << (PINT_2_BIT(x)))static unsigned char irq2pint_lut[NR_PINTS];static unsigned char pint2irq_lut[NR_PINT_SYS_IRQS * NR_PINT_BITS];struct pin_int_t { unsigned int mask_set; unsigned int mask_clear; unsigned int request; unsigned int assign; unsigned int edge_set; unsigned int edge_clear; unsigned int invert_set; unsigned int invert_clear; unsigned int pinstate; unsigned int latch;};static struct pin_int_t *pint[NR_PINT_SYS_IRQS] = { (struct pin_int_t *)PINT0_MASK_SET, (struct pin_int_t *)PINT1_MASK_SET, (struct pin_int_t *)PINT2_MASK_SET, (struct pin_int_t *)PINT3_MASK_SET,};unsigned short get_irq_base(u8 bank, u8 bmap){ u16 irq_base; if (bank < 2) { /*PA-PB */ irq_base = IRQ_PA0 + bmap * 16; } else { /*PC-PJ */ irq_base = IRQ_PC0 + bmap * 16; } return irq_base;} /* Whenever PINTx_ASSIGN is altered init_pint_lut() must be executed! */void init_pint_lut(void){ u16 bank, bit, irq_base, bit_pos; u32 pint_assign; u8 bmap; memset(irq2pint_lut, IRQ_NOT_AVAIL, sizeof(irq2pint_lut)); for (bank = 0; bank < NR_PINT_SYS_IRQS; bank++) { pint_assign = pint[bank]->assign; for (bit = 0; bit < NR_PINT_BITS; bit++) { bmap = (pint_assign >> ((bit / 8) * 8)) & 0xFF; irq_base = get_irq_base(bank, bmap); irq_base += (bit % 8) + ((bit / 8) & 1 ? 8 : 0); bit_pos = bit + bank * NR_PINT_BITS; pint2irq_lut[bit_pos] = irq_base - SYS_IRQS; irq2pint_lut[irq_base - SYS_IRQS] = bit_pos; } }}static unsigned short gpio_enabled[gpio_bank(MAX_BLACKFIN_GPIOS)];static void bfin_gpio_ack_irq(unsigned int irq){ u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; pint[PINT_2_BANK(pint_val)]->request = PINT_BIT(pint_val); SSYNC();}static void bfin_gpio_mask_ack_irq(unsigned int irq){ u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; u32 pintbit = PINT_BIT(pint_val); u8 bank = PINT_2_BANK(pint_val); pint[bank]->request = pintbit; pint[bank]->mask_clear = pintbit; SSYNC();}static void bfin_gpio_mask_irq(unsigned int irq){ u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; pint[PINT_2_BANK(pint_val)]->mask_clear = PINT_BIT(pint_val); SSYNC();}static void bfin_gpio_unmask_irq(unsigned int irq){ u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; u32 pintbit = PINT_BIT(pint_val); u8 bank = PINT_2_BANK(pint_val); pint[bank]->request = pintbit; pint[bank]->mask_set = pintbit; SSYNC();}static unsigned int bfin_gpio_irq_startup(unsigned int irq){ unsigned int ret; u16 gpionr = irq - IRQ_PA0; u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; if (pint_val == IRQ_NOT_AVAIL) { printk(KERN_ERR "GPIO IRQ %d :Not in PINT Assign table " "Reconfigure Interrupt to Port Assignemt\n", irq); return -ENODEV; } if (!(gpio_enabled[gpio_bank(gpionr)] & gpio_bit(gpionr))) { ret = gpio_request(gpionr, "IRQ"); if (ret) return ret; } gpio_enabled[gpio_bank(gpionr)] |= gpio_bit(gpionr); bfin_gpio_unmask_irq(irq); return ret;}static void bfin_gpio_irq_shutdown(unsigned int irq){ bfin_gpio_mask_irq(irq); gpio_free(irq - IRQ_PA0); gpio_enabled[gpio_bank(irq - IRQ_PA0)] &= ~gpio_bit(irq - IRQ_PA0);}static int bfin_gpio_irq_type(unsigned int irq, unsigned int type){ unsigned int ret; u16 gpionr = irq - IRQ_PA0; u8 pint_val = irq2pint_lut[irq - SYS_IRQS]; u32 pintbit = PINT_BIT(pint_val); u8 bank = PINT_2_BANK(pint_val); if (pint_val == IRQ_NOT_AVAIL) return -ENODEV; if (type == IRQ_TYPE_PROBE) { /* only probe unenabled GPIO interrupt lines */ if (gpio_enabled[gpio_bank(gpionr)] & gpio_bit(gpionr)) return 0; type = IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING; } if (type & (IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_LEVEL_HIGH | IRQ_TYPE_LEVEL_LOW)) { if (!(gpio_enabled[gpio_bank(gpionr)] & gpio_bit(gpionr))) { ret = gpio_request(gpionr, "IRQ"); if (ret) return ret; } gpio_enabled[gpio_bank(gpionr)] |= gpio_bit(gpionr); } else { gpio_enabled[gpio_bank(gpionr)] &= ~gpio_bit(gpionr); return 0; } gpio_direction_input(gpionr); if (type & (IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING)) { pint[bank]->edge_set = pintbit; } else { pint[bank]->edge_clear = pintbit; } if ((type & (IRQ_TYPE_EDGE_FALLING | IRQ_TYPE_LEVEL_LOW))) pint[bank]->invert_set = pintbit; /* low or falling edge denoted by one */ else pint[bank]->invert_set = pintbit; /* high or rising edge denoted by zero */ if (type & (IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING)) pint[bank]->invert_set = pintbit; else pint[bank]->invert_set = pintbit; SSYNC(); if (type & (IRQ_TYPE_EDGE_RISING | IRQ_TYPE_EDGE_FALLING)) set_irq_handler(irq, handle_edge_irq); else set_irq_handler(irq, handle_level_irq); return 0;}static struct irq_chip bfin_gpio_irqchip = { .ack = bfin_gpio_ack_irq, .mask = bfin_gpio_mask_irq, .mask_ack = bfin_gpio_mask_ack_irq, .unmask = bfin_gpio_unmask_irq, .set_type = bfin_gpio_irq_type, .startup = bfin_gpio_irq_startup, .shutdown = bfin_gpio_irq_shutdown};static void bfin_demux_gpio_irq(unsigned int intb_irq, struct irq_desc *intb_desc){ u8 bank, pint_val; u32 request, irq; struct irq_desc *desc; switch (intb_irq) { case IRQ_PINT0: bank = 0; break; case IRQ_PINT2: bank = 2; break; case IRQ_PINT3: bank = 3; break; case IRQ_PINT1: bank = 1; break; default: return; } pint_val = bank * NR_PINT_BITS; request = pint[bank]->request; while (request) { if (request & 1) { irq = pint2irq_lut[pint_val] + SYS_IRQS; desc = irq_desc + irq; desc->handle_irq(irq, desc); } pint_val++; request >>= 1; }}#endifvoid __init init_exception_vectors(void){ SSYNC(); /* cannot program in software: * evt0 - emulation (jtag) * evt1 - reset */ bfin_write_EVT2(evt_nmi); bfin_write_EVT3(trap); bfin_write_EVT5(evt_ivhw); bfin_write_EVT6(evt_timer); bfin_write_EVT7(evt_evt7); bfin_write_EVT8(evt_evt8); bfin_write_EVT9(evt_evt9); bfin_write_EVT10(evt_evt10); bfin_write_EVT11(evt_evt11); bfin_write_EVT12(evt_evt12); bfin_write_EVT13(evt_evt13); bfin_write_EVT14(evt14_softirq); bfin_write_EVT15(evt_system_call); CSYNC();}/* * This function should be called during kernel startup to initialize * the BFin IRQ handling routines. */int __init init_arch_irq(void){ int irq; unsigned long ilat = 0; /* Disable all the peripheral intrs - page 4-29 HW Ref manual */#if defined(CONFIG_BF54x) || defined(CONFIG_BF52x) bfin_write_SIC_IMASK0(SIC_UNMASK_ALL); bfin_write_SIC_IMASK1(SIC_UNMASK_ALL); bfin_write_SIC_IWR0(IWR_ENABLE_ALL); bfin_write_SIC_IWR1(IWR_ENABLE_ALL);# ifdef CONFIG_BF54x bfin_write_SIC_IMASK2(SIC_UNMASK_ALL); bfin_write_SIC_IWR2(IWR_ENABLE_ALL);# endif#else bfin_write_SIC_IMASK(SIC_UNMASK_ALL); bfin_write_SIC_IWR(IWR_ENABLE_ALL);#endif SSYNC(); local_irq_disable();#ifdef CONFIG_BF54x# ifdef CONFIG_PINTx_REASSIGN pint[0]->assign = CONFIG_PINT0_ASSIGN; pint[1]->assign = CONFIG_PINT1_ASSIGN; pint[2]->assign = CONFIG_PINT2_ASSIGN; pint[3]->assign = CONFIG_PINT3_ASSIGN;# endif /* Whenever PINTx_ASSIGN is altered init_pint_lut() must be executed! */ init_pint_lut();#endif for (irq = 0; irq <= SYS_IRQS; irq++) { if (irq <= IRQ_CORETMR) set_irq_chip(irq, &bfin_core_irqchip); else set_irq_chip(irq, &bfin_internal_irqchip);#ifdef BF537_GENERIC_ERROR_INT_DEMUX if (irq != IRQ_GENERIC_ERROR) {#endif switch (irq) {#if defined(CONFIG_BF53x) case IRQ_PROG_INTA: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break;# if defined(BF537_FAMILY) && !(defined(CONFIG_BFIN_MAC) || defined(CONFIG_BFIN_MAC_MODULE)) case IRQ_MAC_RX: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break;# endif#elif defined(CONFIG_BF54x) case IRQ_PINT0: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break; case IRQ_PINT1: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break; case IRQ_PINT2: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break; case IRQ_PINT3: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break;#elif defined(CONFIG_BF52x) case IRQ_PORTF_INTA: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break; case IRQ_PORTG_INTA: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break; case IRQ_PORTH_INTA: set_irq_chained_handler(irq, bfin_demux_gpio_irq); break;#endif default: set_irq_handler(irq, handle_simple_irq); break; }#ifdef BF537_GENERIC_ERROR_INT_DEMUX } else { set_irq_handler(irq, bfin_demux_error_irq); }#endif }#ifdef BF537_GENERIC_ERROR_INT_DEMUX for (irq = IRQ_PPI_ERROR; irq <= IRQ_UART1_ERROR; irq++) { set_irq_chip(irq, &bfin_generic_error_irqchip); set_irq_handler(irq, handle_level_irq); }#endif#ifndef CONFIG_BF54x for (irq = IRQ_PF0; irq < NR_IRQS; irq++) {#else for (irq = IRQ_PA0; irq < NR_IRQS; irq++) {#endif set_irq_chip(irq, &bfin_gpio_irqchip); /* if configured as edge, then will be changed to do_edge_IRQ */ set_irq_handler(irq, handle_level_irq); } bfin_write_IMASK(0); CSYNC(); ilat = bfin_read_ILAT(); CSYNC(); bfin_write_ILAT(ilat); CSYNC(); printk(KERN_INFO "Configuring Blackfin Priority Driven Interrupts\n"); /* IMASK=xxx is equivalent to STI xx or irq_flags=xx, * local_irq_enable() */ program_IAR(); /* Therefore it's better to setup IARs before interrupts enabled */ search_IAR(); /* Enable interrupts IVG7-15 */ irq_flags = irq_flags | IMASK_IVG15 | IMASK_IVG14 | IMASK_IVG13 | IMASK_IVG12 | IMASK_IVG11 | IMASK_IVG10 | IMASK_IVG9 | IMASK_IVG8 | IMASK_IVG7 | IMASK_IVGHW; return 0;}#ifdef CONFIG_DO_IRQ_L1__attribute__((l1_text))#endifvoid do_irq(int vec, struct pt_regs *fp){ if (vec == EVT_IVTMR_P) { vec = IRQ_CORETMR; } else { struct ivgx *ivg = ivg7_13[vec - IVG7].ifirst; struct ivgx *ivg_stop = ivg7_13[vec - IVG7].istop;#if defined(CONFIG_BF54x) || defined(CONFIG_BF52x) unsigned long sic_status[3]; SSYNC(); sic_status[0] = bfin_read_SIC_ISR0() & bfin_read_SIC_IMASK0(); sic_status[1] = bfin_read_SIC_ISR1() & bfin_read_SIC_IMASK1();#ifdef CONFIG_BF54x sic_status[2] = bfin_read_SIC_ISR2() & bfin_read_SIC_IMASK2();#endif for (;; ivg++) { if (ivg >= ivg_stop) { atomic_inc(&num_spurious); return; } if (sic_status[(ivg->irqno - IVG7) / 32] & ivg->isrflag) break; }#else unsigned long sic_status; SSYNC(); sic_status = bfin_read_SIC_IMASK() & bfin_read_SIC_ISR(); for (;; ivg++) { if (ivg >= ivg_stop) { atomic_inc(&num_spurious); return; } else if (sic_status & ivg->isrflag) break; }#endif vec = ivg->irqno; } asm_do_IRQ(vec, fp);#ifdef CONFIG_KGDB kgdb_process_breakpoint();#endif}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -