2007-10-27 11:13:04 +04:00
/* irq.h: IRQ registers on the 64-bit Sparc.
2005-04-17 02:20:36 +04:00
*
2007-10-27 11:13:04 +04:00
* Copyright ( C ) 1996 David S . Miller ( davem @ davemloft . net )
2005-04-17 02:20:36 +04:00
* Copyright ( C ) 1998 Jakub Jelinek ( jj @ ultra . linux . cz )
*/
# ifndef _SPARC64_IRQ_H
# define _SPARC64_IRQ_H
# include <linux/linkage.h>
# include <linux/kernel.h>
# include <linux/errno.h>
# include <linux/interrupt.h>
# include <asm/pil.h>
# include <asm/ptrace.h>
/* IMAP/ICLR register defines */
2007-08-29 01:25:32 +04:00
# define IMAP_VALID 0x80000000UL /* IRQ Enabled */
# define IMAP_TID_UPA 0x7c000000UL /* UPA TargetID */
# define IMAP_TID_JBUS 0x7c000000UL /* JBUS TargetID */
2006-02-17 19:38:06 +03:00
# define IMAP_TID_SHIFT 26
2007-08-29 01:25:32 +04:00
# define IMAP_AID_SAFARI 0x7c000000UL /* Safari AgentID */
2006-02-17 19:38:06 +03:00
# define IMAP_AID_SHIFT 26
2007-08-29 01:25:32 +04:00
# define IMAP_NID_SAFARI 0x03e00000UL /* Safari NodeID */
2006-02-17 19:38:06 +03:00
# define IMAP_NID_SHIFT 21
2007-08-29 01:25:32 +04:00
# define IMAP_IGN 0x000007c0UL /* IRQ Group Number */
# define IMAP_INO 0x0000003fUL /* IRQ Number */
# define IMAP_INR 0x000007ffUL /* Full interrupt number*/
2005-04-17 02:20:36 +04:00
2007-08-29 01:25:32 +04:00
# define ICLR_IDLE 0x00000000UL /* Idle state */
# define ICLR_TRANSMIT 0x00000001UL /* Transmit state */
# define ICLR_PENDING 0x00000003UL /* Pending state */
2005-04-17 02:20:36 +04:00
2006-06-20 12:22:35 +04:00
/* The largest number of unique interrupt sources we support.
* If this needs to ever be larger than 255 , you need to change
* the type of ino_bucket - > virt_irq as appropriate .
*
* ino_bucket - > virt_irq allocation is made during { sun4v_ , } build_irq ( ) .
*/
# define NR_IRQS 255
2005-04-17 02:20:36 +04:00
2006-06-20 12:22:35 +04:00
extern void irq_install_pre_handler ( int virt_irq ,
2006-06-20 12:23:32 +04:00
void ( * func ) ( unsigned int , void * , void * ) ,
2006-06-20 12:22:35 +04:00
void * arg1 , void * arg2 ) ;
2005-04-17 02:20:36 +04:00
# define irq_canonicalize(irq) (irq)
2006-06-20 12:23:32 +04:00
extern unsigned int build_irq ( int inofixup , unsigned long iclr , unsigned long imap ) ;
extern unsigned int sun4v_build_irq ( u32 devhandle , unsigned int devino ) ;
2007-06-13 11:01:04 +04:00
extern unsigned int sun4v_build_virq ( u32 devhandle , unsigned int devino ) ;
[SPARC64]: Add PCI MSI support on Niagara.
This is kind of hokey, we could use the hardware provided facilities
much better.
MSIs are assosciated with MSI Queues. MSI Queues generate interrupts
when any MSI assosciated with it is signalled. This suggests a
two-tiered IRQ dispatch scheme:
MSI Queue interrupt --> queue interrupt handler
MSI dispatch --> driver interrupt handler
But we just get one-level under Linux currently. What I'd like to do
is possibly stick the IRQ actions into a per-MSI-Queue data structure,
and dispatch them form there, but the generic IRQ layer doesn't
provide a way to do that right now.
So, the current kludge is to "ACK" the interrupt by processing the
MSI Queue data structures and ACK'ing them, then we run the actual
handler like normal.
We are wasting a lot of useful information, for example the MSI data
and address are provided with ever MSI, as well as a system tick if
available. If we could pass this into the IRQ handler it could help
with certain things, in particular for PCI-Express error messages.
The MSI entries on sparc64 also tell you exactly which bus/device/fn
sent the MSI, which would be great for error handling when no
registered IRQ handler can service the interrupt.
We override the disable/enable IRQ chip methods in sun4v_msi, so we
have to call {mask,unmask}_msi_irq() directly from there. This is
another ugly wart.
Signed-off-by: David S. Miller <davem@davemloft.net>
2007-02-11 04:41:02 +03:00
extern unsigned int sun4v_build_msi ( u32 devhandle , unsigned int * virt_irq_p ,
unsigned int msi_devino_start ,
unsigned int msi_devino_end ) ;
extern void sun4v_destroy_msi ( unsigned int virt_irq ) ;
2007-08-31 09:33:25 +04:00
extern unsigned int sun4u_build_msi ( u32 portid , unsigned int * virt_irq_p ,
unsigned int msi_devino_start ,
unsigned int msi_devino_end ,
unsigned long imap_base ,
unsigned long iclr_base ) ;
extern void sun4u_destroy_msi ( unsigned int virt_irq ) ;
2005-04-17 02:20:36 +04:00
extern unsigned int sbus_build_irq ( void * sbus , unsigned int ino ) ;
2007-10-14 10:50:38 +04:00
extern unsigned char virt_irq_alloc ( unsigned int dev_handle ,
2007-10-14 10:27:48 +04:00
unsigned int dev_ino ) ;
2007-10-11 14:16:13 +04:00
# ifdef CONFIG_PCI_MSI
extern void virt_irq_free ( unsigned int virt_irq ) ;
# endif
2007-08-31 09:27:28 +04:00
2007-07-16 14:49:40 +04:00
extern void fixup_irqs ( void ) ;
2007-10-27 11:13:04 +04:00
static inline void set_softint ( unsigned long bits )
2005-04-17 02:20:36 +04:00
{
__asm__ __volatile__ ( " wr %0, 0x0, %%set_softint "
: /* No outputs */
: " r " ( bits ) ) ;
}
2007-10-27 11:13:04 +04:00
static inline void clear_softint ( unsigned long bits )
2005-04-17 02:20:36 +04:00
{
__asm__ __volatile__ ( " wr %0, 0x0, %%clear_softint "
: /* No outputs */
: " r " ( bits ) ) ;
}
2007-10-27 11:13:04 +04:00
static inline unsigned long get_softint ( void )
2005-04-17 02:20:36 +04:00
{
unsigned long retval ;
__asm__ __volatile__ ( " rd %%softint, %0 "
: " =r " ( retval ) ) ;
return retval ;
}
# endif