2007-07-10 01:06:53 +04:00
/*
2008-07-05 12:02:54 +04:00
* Copyright 2004 - 2007 Freescale Semiconductor , Inc . All Rights Reserved .
* Copyright 2008 Juergen Beisert , kernel @ pengutronix . de
*
* This program is free software ; you can redistribute it and / or
* modify it under the terms of the GNU General Public License
* as published by the Free Software Foundation ; either version 2
* of the License , or ( at your option ) any later version .
* This program is distributed in the hope that it will be useful ,
* but WITHOUT ANY WARRANTY ; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE . See the
* GNU General Public License for more details .
*
* You should have received a copy of the GNU General Public License
* along with this program ; if not , write to the Free Software
* Foundation , Inc . , 51 Franklin Street , Fifth Floor , Boston ,
* MA 02110 - 1301 , USA .
2007-07-10 01:06:53 +04:00
*/
2008-11-14 13:01:39 +03:00
# include <linux/module.h>
2008-07-05 12:02:54 +04:00
# include <linux/irq.h>
2008-09-06 15:10:45 +04:00
# include <linux/io.h>
2008-08-05 19:14:15 +04:00
# include <mach/common.h>
2008-11-14 13:01:39 +03:00
# include <asm/mach/irq.h>
2008-12-18 13:51:57 +03:00
# include <mach/hardware.h>
2007-07-10 01:06:53 +04:00
2010-12-06 14:37:38 +03:00
# include "irq-common.h"
2009-02-18 22:59:04 +03:00
# define AVIC_INTCNTL 0x00 /* int control reg */
# define AVIC_NIMASK 0x04 /* int mask reg */
# define AVIC_INTENNUM 0x08 /* int enable number reg */
# define AVIC_INTDISNUM 0x0C /* int disable number reg */
# define AVIC_INTENABLEH 0x10 /* int enable reg high */
# define AVIC_INTENABLEL 0x14 /* int enable reg low */
# define AVIC_INTTYPEH 0x18 /* int type reg high */
# define AVIC_INTTYPEL 0x1C /* int type reg low */
# define AVIC_NIPRIORITY(x) (0x20 + 4 * (7 - (x))) /* int priority */
# define AVIC_NIVECSR 0x40 /* norm int vector/status */
# define AVIC_FIVECSR 0x44 /* fast int vector/status */
# define AVIC_INTSRCH 0x48 /* int source reg high */
# define AVIC_INTSRCL 0x4C /* int source reg low */
# define AVIC_INTFRCH 0x50 /* int force reg high */
# define AVIC_INTFRCL 0x54 /* int force reg low */
# define AVIC_NIPNDH 0x58 /* norm int pending high */
# define AVIC_NIPNDL 0x5C /* norm int pending low */
# define AVIC_FIPNDH 0x60 /* fast int pending high */
# define AVIC_FIPNDL 0x64 /* fast int pending low */
2009-05-25 12:50:52 +04:00
void __iomem * avic_base ;
2008-07-05 12:02:54 +04:00
2009-04-08 17:17:50 +04:00
# ifdef CONFIG_MXC_IRQ_PRIOR
2010-12-06 14:37:38 +03:00
static int avic_irq_set_priority ( unsigned char irq , unsigned char prio )
{
2008-09-09 13:29:41 +04:00
unsigned int temp ;
unsigned int mask = 0x0F < < irq % 8 * 4 ;
2009-04-08 17:17:50 +04:00
if ( irq > = MXC_INTERNAL_IRQS )
return - EINVAL ; ;
2008-09-09 13:29:41 +04:00
2009-02-18 22:59:04 +03:00
temp = __raw_readl ( avic_base + AVIC_NIPRIORITY ( irq / 8 ) ) ;
2008-09-09 13:29:41 +04:00
temp & = ~ mask ;
temp | = prio & mask ;
2009-02-18 22:59:04 +03:00
__raw_writel ( temp , avic_base + AVIC_NIPRIORITY ( irq / 8 ) ) ;
2009-04-08 17:17:50 +04:00
return 0 ;
2008-09-09 13:29:41 +04:00
}
2010-12-06 14:37:38 +03:00
# endif
2008-09-09 13:29:41 +04:00
2008-11-14 13:01:39 +03:00
# ifdef CONFIG_FIQ
2010-12-06 14:37:38 +03:00
static int avic_set_irq_fiq ( unsigned int irq , unsigned int type )
2008-11-14 13:01:39 +03:00
{
unsigned int irqt ;
2008-12-18 13:08:55 +03:00
if ( irq > = MXC_INTERNAL_IRQS )
2008-11-14 13:01:39 +03:00
return - EINVAL ;
2008-12-18 13:08:55 +03:00
if ( irq < MXC_INTERNAL_IRQS / 2 ) {
2009-02-18 22:59:04 +03:00
irqt = __raw_readl ( avic_base + AVIC_INTTYPEL ) & ~ ( 1 < < irq ) ;
__raw_writel ( irqt | ( ! ! type < < irq ) , avic_base + AVIC_INTTYPEL ) ;
2008-11-14 13:01:39 +03:00
} else {
2008-12-18 13:08:55 +03:00
irq - = MXC_INTERNAL_IRQS / 2 ;
2009-02-18 22:59:04 +03:00
irqt = __raw_readl ( avic_base + AVIC_INTTYPEH ) & ~ ( 1 < < irq ) ;
__raw_writel ( irqt | ( ! ! type < < irq ) , avic_base + AVIC_INTTYPEH ) ;
2008-11-14 13:01:39 +03:00
}
return 0 ;
}
# endif /* CONFIG_FIQ */
2008-03-28 13:02:13 +03:00
/* Disable interrupt number "irq" in the AVIC */
2010-11-29 13:16:23 +03:00
static void mxc_mask_irq ( struct irq_data * d )
2007-07-10 01:06:53 +04:00
{
2010-11-29 13:16:23 +03:00
__raw_writel ( d - > irq , avic_base + AVIC_INTDISNUM ) ;
2007-07-10 01:06:53 +04:00
}
2008-03-28 13:02:13 +03:00
/* Enable interrupt number "irq" in the AVIC */
2010-11-29 13:16:23 +03:00
static void mxc_unmask_irq ( struct irq_data * d )
2007-07-10 01:06:53 +04:00
{
2010-11-29 13:16:23 +03:00
__raw_writel ( d - > irq , avic_base + AVIC_INTENNUM ) ;
2007-07-10 01:06:53 +04:00
}
2010-12-06 14:37:38 +03:00
static struct mxc_irq_chip mxc_avic_chip = {
. base = {
2010-11-29 13:16:23 +03:00
. irq_ack = mxc_mask_irq ,
. irq_mask = mxc_mask_irq ,
. irq_unmask = mxc_unmask_irq ,
2010-12-06 14:37:38 +03:00
} ,
# ifdef CONFIG_MXC_IRQ_PRIOR
. set_priority = avic_irq_set_priority ,
# endif
# ifdef CONFIG_FIQ
. set_irq_fiq = avic_set_irq_fiq ,
# endif
2007-07-10 01:06:53 +04:00
} ;
2008-03-28 13:02:13 +03:00
/*
2007-07-10 01:06:53 +04:00
* This function initializes the AVIC hardware and disables all the
* interrupts . It registers the interrupt enable and disable functions
* to the kernel for each interrupt source .
*/
2009-05-25 19:36:19 +04:00
void __init mxc_init_irq ( void __iomem * irqbase )
2007-07-10 01:06:53 +04:00
{
int i ;
2009-05-25 19:36:19 +04:00
avic_base = irqbase ;
2009-02-18 22:59:04 +03:00
2007-07-10 01:06:53 +04:00
/* put the AVIC into the reset value with
* all interrupts disabled
*/
2009-02-18 22:59:04 +03:00
__raw_writel ( 0 , avic_base + AVIC_INTCNTL ) ;
__raw_writel ( 0x1f , avic_base + AVIC_NIMASK ) ;
2007-07-10 01:06:53 +04:00
/* disable all interrupts */
2009-02-18 22:59:04 +03:00
__raw_writel ( 0 , avic_base + AVIC_INTENABLEH ) ;
__raw_writel ( 0 , avic_base + AVIC_INTENABLEL ) ;
2007-07-10 01:06:53 +04:00
/* all IRQ no FIQ */
2009-02-18 22:59:04 +03:00
__raw_writel ( 0 , avic_base + AVIC_INTTYPEH ) ;
__raw_writel ( 0 , avic_base + AVIC_INTTYPEL ) ;
2008-12-18 13:08:55 +03:00
for ( i = 0 ; i < MXC_INTERNAL_IRQS ; i + + ) {
2010-12-06 14:37:38 +03:00
set_irq_chip ( i , & mxc_avic_chip . base ) ;
2007-07-10 01:06:53 +04:00
set_irq_handler ( i , handle_level_irq ) ;
set_irq_flags ( i , IRQF_VALID ) ;
}
2008-09-09 13:29:41 +04:00
/* Set default priority value (0) for all IRQ's */
for ( i = 0 ; i < 8 ; i + + )
2009-02-18 22:59:04 +03:00
__raw_writel ( 0 , avic_base + AVIC_NIPRIORITY ( i ) ) ;
2007-07-10 01:06:53 +04:00
2008-11-14 13:01:39 +03:00
# ifdef CONFIG_FIQ
/* Initialize FIQ */
init_FIQ ( ) ;
# endif
2007-07-10 01:06:53 +04:00
printk ( KERN_INFO " MXC IRQ initialized \n " ) ;
}
2009-02-18 22:59:04 +03:00