1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980 |
- 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)
|