2007-10-02 12:15:23 +10:00
/*
* Interrupt controller driver for Xilinx Virtex FPGAs
*
* Copyright ( C ) 2007 Secret Lab Technologies Ltd .
*
* This file is licensed under the terms of the GNU General Public License
* version 2. This program is licensed " as is " without any warranty of any
* kind , whether express or implied .
*
*/
/*
* This is a driver for the interrupt controller typically found in
* Xilinx Virtex FPGA designs .
*
* The interrupt sense levels are hard coded into the FPGA design with
* typically a 1 : 1 relationship between irq lines and devices ( no shared
* irq lines ) . Therefore , this driver does not attempt to handle edge
* and level interrupts differently .
*/
# undef DEBUG
# include <linux/kernel.h>
# include <linux/irq.h>
# include <linux/of.h>
# include <asm/io.h>
# include <asm/processor.h>
2009-06-06 10:15:03 -06:00
# include <asm/i8259.h>
2007-10-02 12:15:23 +10:00
# include <asm/irq.h>
/*
* INTC Registers
*/
# define XINTC_ISR 0 /* Interrupt Status */
# define XINTC_IPR 4 /* Interrupt Pending */
# define XINTC_IER 8 /* Interrupt Enable */
# define XINTC_IAR 12 /* Interrupt Acknowledge */
# define XINTC_SIE 16 /* Set Interrupt Enable bits */
# define XINTC_CIE 20 /* Clear Interrupt Enable bits */
# define XINTC_IVR 24 /* Interrupt Vector */
# define XINTC_MER 28 /* Master Enable */
2012-02-14 14:06:50 -07:00
static struct irq_domain * master_irqhost ;
2007-10-02 12:15:23 +10:00
2009-05-14 10:23:11 -06:00
# define XILINX_INTC_MAXIRQS (32)
/* The following table allows the interrupt type, edge or level,
* to be cached after being read from the device tree until the interrupt
* is mapped
*/
static int xilinx_intc_typetable [ XILINX_INTC_MAXIRQS ] ;
/* Map the interrupt type from the device tree to the interrupt types
* used by the interrupt subsystem
*/
static unsigned char xilinx_intc_map_senses [ ] = {
IRQ_TYPE_EDGE_RISING ,
IRQ_TYPE_EDGE_FALLING ,
IRQ_TYPE_LEVEL_HIGH ,
IRQ_TYPE_LEVEL_LOW ,
} ;
2007-10-02 12:15:23 +10:00
/*
2009-05-14 10:23:11 -06:00
* The interrupt controller is setup such that it doesn ' t work well with
* the level interrupt handler in the kernel because the handler acks the
* interrupt before calling the application interrupt handler . To deal with
* that , we use 2 different irq chips so that different functions can be
* used for level and edge type interrupts .
*
* IRQ Chip common ( across level and edge ) operations
2007-10-02 12:15:23 +10:00
*/
2011-03-08 22:27:07 +00:00
static void xilinx_intc_mask ( struct irq_data * d )
2007-10-02 12:15:23 +10:00
{
2011-05-04 15:02:15 +10:00
int irq = irqd_to_hwirq ( d ) ;
2011-03-08 22:27:07 +00:00
void * regs = irq_data_get_irq_chip_data ( d ) ;
2007-10-02 12:15:23 +10:00
pr_debug ( " mask: %d \n " , irq ) ;
out_be32 ( regs + XINTC_CIE , 1 < < irq ) ;
}
2011-03-08 22:27:07 +00:00
static int xilinx_intc_set_type ( struct irq_data * d , unsigned int flow_type )
2009-05-14 10:23:11 -06:00
{
return 0 ;
}
/*
* IRQ Chip level operations
*/
2011-03-08 22:27:07 +00:00
static void xilinx_intc_level_unmask ( struct irq_data * d )
2007-10-02 12:15:23 +10:00
{
2011-05-04 15:02:15 +10:00
int irq = irqd_to_hwirq ( d ) ;
2011-03-08 22:27:07 +00:00
void * regs = irq_data_get_irq_chip_data ( d ) ;
2007-10-02 12:15:23 +10:00
pr_debug ( " unmask: %d \n " , irq ) ;
out_be32 ( regs + XINTC_SIE , 1 < < irq ) ;
2009-05-14 10:23:11 -06:00
/* ack level irqs because they can't be acked during
* ack function since the handle_level_irq function
* acks the irq before calling the inerrupt handler
*/
out_be32 ( regs + XINTC_IAR , 1 < < irq ) ;
2007-10-02 12:15:23 +10:00
}
2009-05-14 10:23:11 -06:00
static struct irq_chip xilinx_intc_level_irqchip = {
2009-11-18 23:44:21 +00:00
. name = " Xilinx Level INTC " ,
2011-03-08 22:27:07 +00:00
. irq_mask = xilinx_intc_mask ,
. irq_mask_ack = xilinx_intc_mask ,
. irq_unmask = xilinx_intc_level_unmask ,
. irq_set_type = xilinx_intc_set_type ,
2009-05-14 10:23:11 -06:00
} ;
/*
* IRQ Chip edge operations
*/
2011-03-08 22:27:07 +00:00
static void xilinx_intc_edge_unmask ( struct irq_data * d )
2009-05-14 10:23:11 -06:00
{
2011-05-04 15:02:15 +10:00
int irq = irqd_to_hwirq ( d ) ;
2011-03-08 22:27:07 +00:00
void * regs = irq_data_get_irq_chip_data ( d ) ;
2009-05-14 10:23:11 -06:00
pr_debug ( " unmask: %d \n " , irq ) ;
out_be32 ( regs + XINTC_SIE , 1 < < irq ) ;
}
2011-03-08 22:27:07 +00:00
static void xilinx_intc_edge_ack ( struct irq_data * d )
2007-10-02 12:15:23 +10:00
{
2011-05-04 15:02:15 +10:00
int irq = irqd_to_hwirq ( d ) ;
2011-03-08 22:27:07 +00:00
void * regs = irq_data_get_irq_chip_data ( d ) ;
2007-10-02 12:15:23 +10:00
pr_debug ( " ack: %d \n " , irq ) ;
out_be32 ( regs + XINTC_IAR , 1 < < irq ) ;
}
2009-05-14 10:23:11 -06:00
static struct irq_chip xilinx_intc_edge_irqchip = {
2009-11-18 23:44:21 +00:00
. name = " Xilinx Edge INTC " ,
2011-03-08 22:27:07 +00:00
. irq_mask = xilinx_intc_mask ,
. irq_unmask = xilinx_intc_edge_unmask ,
. irq_ack = xilinx_intc_edge_ack ,
. irq_set_type = xilinx_intc_set_type ,
2007-10-02 12:15:23 +10:00
} ;
/*
* IRQ Host operations
*/
2009-05-14 10:23:11 -06:00
/**
* xilinx_intc_xlate - translate virq # from device tree interrupts property
*/
2012-02-14 14:06:50 -07:00
static int xilinx_intc_xlate ( struct irq_domain * h , struct device_node * ct ,
2009-12-08 02:39:50 +00:00
const u32 * intspec , unsigned int intsize ,
2009-05-14 10:23:11 -06:00
irq_hw_number_t * out_hwirq ,
unsigned int * out_flags )
{
if ( ( intsize < 2 ) | | ( intspec [ 0 ] > = XILINX_INTC_MAXIRQS ) )
return - EINVAL ;
/* keep a copy of the interrupt type til the interrupt is mapped
*/
xilinx_intc_typetable [ intspec [ 0 ] ] = xilinx_intc_map_senses [ intspec [ 1 ] ] ;
/* Xilinx uses 2 interrupt entries, the 1st being the h/w
* interrupt number , the 2 nd being the interrupt type , edge or level
*/
* out_hwirq = intspec [ 0 ] ;
* out_flags = xilinx_intc_map_senses [ intspec [ 1 ] ] ;
return 0 ;
}
2012-02-14 14:06:50 -07:00
static int xilinx_intc_map ( struct irq_domain * h , unsigned int virq ,
2007-10-02 12:15:23 +10:00
irq_hw_number_t irq )
{
2011-03-25 16:45:20 +01:00
irq_set_chip_data ( virq , h - > host_data ) ;
2009-05-14 10:23:11 -06:00
if ( xilinx_intc_typetable [ irq ] = = IRQ_TYPE_LEVEL_HIGH | |
xilinx_intc_typetable [ irq ] = = IRQ_TYPE_LEVEL_LOW ) {
2011-03-25 16:45:20 +01:00
irq_set_chip_and_handler ( virq , & xilinx_intc_level_irqchip ,
handle_level_irq ) ;
2009-05-14 10:23:11 -06:00
} else {
2011-03-25 16:45:20 +01:00
irq_set_chip_and_handler ( virq , & xilinx_intc_edge_irqchip ,
handle_edge_irq ) ;
2009-05-14 10:23:11 -06:00
}
2007-10-02 12:15:23 +10:00
return 0 ;
}
2012-02-14 14:06:50 -07:00
static struct irq_domain_ops xilinx_intc_ops = {
2007-10-02 12:15:23 +10:00
. map = xilinx_intc_map ,
2009-05-14 10:23:11 -06:00
. xlate = xilinx_intc_xlate ,
2007-10-02 12:15:23 +10:00
} ;
2012-02-14 14:06:50 -07:00
struct irq_domain * __init
2007-10-02 12:15:23 +10:00
xilinx_intc_init ( struct device_node * np )
{
2012-02-14 14:06:50 -07:00
struct irq_domain * irq ;
2007-10-02 12:15:23 +10:00
void * regs ;
/* Find and map the intc registers */
2009-06-06 10:15:03 -06:00
regs = of_iomap ( np , 0 ) ;
if ( ! regs ) {
pr_err ( " xilinx_intc: could not map registers \n " ) ;
2007-10-02 12:15:23 +10:00
return NULL ;
}
/* Setup interrupt controller */
out_be32 ( regs + XINTC_IER , 0 ) ; /* disable all irqs */
out_be32 ( regs + XINTC_IAR , ~ ( u32 ) 0 ) ; /* Acknowledge pending irqs */
out_be32 ( regs + XINTC_MER , 0x3UL ) ; /* Turn on the Master Enable. */
2012-02-14 14:06:50 -07:00
/* Allocate and initialize an irq_domain structure. */
2012-02-14 14:06:54 -07:00
irq = irq_domain_add_linear ( np , XILINX_INTC_MAXIRQS , & xilinx_intc_ops ,
regs ) ;
2007-10-02 12:15:23 +10:00
if ( ! irq )
panic ( __FILE__ " : Cannot allocate IRQ host \n " ) ;
2009-06-06 10:15:03 -06:00
2007-10-02 12:15:23 +10:00
return irq ;
}
int xilinx_intc_get_irq ( void )
{
void * regs = master_irqhost - > host_data ;
pr_debug ( " get_irq: \n " ) ;
return irq_linear_revmap ( master_irqhost , in_be32 ( regs + XINTC_IVR ) ) ;
}
2009-06-06 10:15:03 -06:00
# if defined(CONFIG_PPC_I8259)
/*
* Support code for cascading to 8259 interrupt controllers
*/
static void xilinx_i8259_cascade ( unsigned int irq , struct irq_desc * desc )
{
2011-03-25 16:45:20 +01:00
struct irq_chip * chip = irq_desc_get_chip ( desc ) ;
2009-06-06 10:15:03 -06:00
unsigned int cascade_irq = i8259_irq ( ) ;
2011-03-08 22:27:07 +00:00
2009-06-06 10:15:03 -06:00
if ( cascade_irq )
generic_handle_irq ( cascade_irq ) ;
/* Let xilinx_intc end the interrupt */
2011-03-08 22:27:07 +00:00
chip - > irq_unmask ( & desc - > irq_data ) ;
2009-06-06 10:15:03 -06:00
}
static void __init xilinx_i8259_setup_cascade ( void )
{
struct device_node * cascade_node ;
int cascade_irq ;
/* Initialize i8259 controller */
cascade_node = of_find_compatible_node ( NULL , NULL , " chrp,iic " ) ;
if ( ! cascade_node )
return ;
cascade_irq = irq_of_parse_and_map ( cascade_node , 0 ) ;
if ( ! cascade_irq ) {
pr_err ( " virtex_ml510: Failed to map cascade interrupt \n " ) ;
goto out ;
}
i8259_init ( cascade_node , 0 ) ;
2011-03-25 16:45:20 +01:00
irq_set_chained_handler ( cascade_irq , xilinx_i8259_cascade ) ;
2009-06-06 10:15:03 -06:00
2009-06-06 10:15:24 -06:00
/* Program irq 7 (usb/audio), 14/15 (ide) to level sensitive */
/* This looks like a dirty hack to me --gcl */
outb ( 0xc0 , 0x4d0 ) ;
outb ( 0xc0 , 0x4d1 ) ;
2009-06-06 10:15:03 -06:00
out :
of_node_put ( cascade_node ) ;
}
# else
static inline void xilinx_i8259_setup_cascade ( void ) { return ; }
# endif /* defined(CONFIG_PPC_I8259) */
static struct of_device_id xilinx_intc_match [ ] __initconst = {
{ . compatible = " xlnx,opb-intc-1.00.c " , } ,
{ . compatible = " xlnx,xps-intc-1.00.a " , } ,
{ }
} ;
/*
* Initialize master Xilinx interrupt controller
*/
2007-10-02 12:15:23 +10:00
void __init xilinx_intc_init_tree ( void )
{
struct device_node * np ;
/* find top level interrupt controller */
2009-06-06 10:15:03 -06:00
for_each_matching_node ( np , xilinx_intc_match ) {
2007-10-02 12:15:23 +10:00
if ( ! of_get_property ( np , " interrupts " , NULL ) )
break ;
}
BUG_ON ( ! np ) ;
master_irqhost = xilinx_intc_init ( np ) ;
BUG_ON ( ! master_irqhost ) ;
irq_set_default_host ( master_irqhost ) ;
of_node_put ( np ) ;
2009-06-06 10:15:03 -06:00
xilinx_i8259_setup_cascade ( ) ;
2007-10-02 12:15:23 +10:00
}