|  | README on the Vectored Interrupt Controller of the LH7A404 | 
|  | ========================================================== | 
|  |  | 
|  | The 404 revision of the LH7A40X series comes with two vectored | 
|  | interrupts controllers.  While the kernel does use some of the | 
|  | features of these devices, it is far from the purpose for which they | 
|  | were designed. | 
|  |  | 
|  | When this README was written, the implementation of the VICs was in | 
|  | flux.  It is possible that some details, especially with priorities, | 
|  | will change. | 
|  |  | 
|  | The VIC support code is inspired by routines written by Sharp. | 
|  |  | 
|  |  | 
|  | Priority Control | 
|  | ---------------- | 
|  |  | 
|  | The significant reason for using the VIC's vectoring is to control | 
|  | interrupt priorities.  There are two tables in | 
|  | arch/arm/mach-lh7a40x/irq-lh7a404.c that look something like this. | 
|  |  | 
|  | static unsigned char irq_pri_vic1[] = { IRQ_GPIO3INTR, }; | 
|  | static unsigned char irq_pri_vic2[] = { | 
|  | IRQ_T3UI, IRQ_GPIO7INTR, | 
|  | IRQ_UART1INTR, IRQ_UART2INTR, IRQ_UART3INTR, }; | 
|  |  | 
|  | The initialization code reads these tables and inserts a vector | 
|  | address and enable for each indicated IRQ.  Vectored interrupts have | 
|  | higher priority than non-vectored interrupts.  So, on VIC1, | 
|  | IRQ_GPIO3INTR will be served before any other non-FIQ interrupt.  Due | 
|  | to the way that the vectoring works, IRQ_T3UI is the next highest | 
|  | priority followed by the other vectored interrupts on VIC2.  After | 
|  | that, the non-vectored interrupts are scanned in VIC1 then in VIC2. | 
|  |  | 
|  |  | 
|  | ISR | 
|  | --- | 
|  |  | 
|  | The interrupt service routine macro get_irqnr() in | 
|  | arch/arm/kernel/entry-armv.S scans the VICs for the next active | 
|  | interrupt.  The vectoring makes this code somewhat larger than it was | 
|  | before using vectoring (refer to the LH7A400 implementation).  In the | 
|  | case where an interrupt is vectored, the implementation will tend to | 
|  | be faster than the non-vectored version.  However, the worst-case path | 
|  | is longer. | 
|  |  | 
|  | It is worth noting that at present, there is no need to read | 
|  | VIC2_VECTADDR because the register appears to be shared between the | 
|  | controllers.  The code is written such that if this changes, it ought | 
|  | to still work properly. | 
|  |  | 
|  |  | 
|  | Vector Addresses | 
|  | ---------------- | 
|  |  | 
|  | The proper use of the vectoring hardware would jump to the ISR | 
|  | specified by the vectoring address.  Linux isn't structured to take | 
|  | advantage of this feature, though it might be possible to change | 
|  | things to support it. | 
|  |  | 
|  | In this implementation, the vectoring address is used to speed the | 
|  | search for the active IRQ.  The address is coded such that the lowest | 
|  | 6 bits store the IRQ number for vectored interrupts.  These numbers | 
|  | correspond to the bits in the interrupt status registers.  IRQ zero is | 
|  | the lowest interrupt bit in VIC1.  IRQ 32 is the lowest interrupt bit | 
|  | in VIC2.  Because zero is a valid IRQ number and because we cannot | 
|  | detect whether or not there is a valid vectoring address if that | 
|  | address is zero, the eigth bit (0x100) is set for vectored interrupts. | 
|  | The address for IRQ 0x18 (VIC2) is 0x118.  Only the ninth bit is set | 
|  | for the default handler on VIC1 and only the tenth bit is set for the | 
|  | default handler on VIC2. | 
|  |  | 
|  | In other words. | 
|  |  | 
|  | 0x000		- no active interrupt | 
|  | 0x1ii		- vectored interrupt 0xii | 
|  | 0x2xx		- unvectored interrupt on VIC1 (xx is don't care) | 
|  | 0x4xx		- unvectored interrupt on VIC2 (xx is don't care) | 
|  |  |