2014-06-30 16:01:30 +01:00
/*
* Copyright ( C ) 2002 ARM Limited , All Rights Reserved .
*
* This program is free software ; you can redistribute it and / or modify
* it under the terms of the GNU General Public License version 2 as
* published by the Free Software Foundation .
*
* 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 , see < http : //www.gnu.org/licenses/>.
*/
# include <linux/interrupt.h>
# include <linux/io.h>
# include <linux/irq.h>
# include <linux/irqchip/arm-gic.h>
# include "irq-gic-common.h"
2016-04-11 16:32:54 +01:00
static const struct gic_kvm_info * gic_kvm_info ;
const struct gic_kvm_info * gic_get_kvm_info ( void )
{
return gic_kvm_info ;
}
void gic_set_kvm_info ( const struct gic_kvm_info * info )
{
BUG_ON ( gic_kvm_info ! = NULL ) ;
gic_kvm_info = info ;
}
2015-09-21 22:58:37 +02:00
void gic_enable_quirks ( u32 iidr , const struct gic_quirk * quirks ,
void * data )
{
for ( ; quirks - > desc ; quirks + + ) {
if ( quirks - > iidr ! = ( quirks - > mask & iidr ) )
continue ;
2017-10-17 17:55:55 +01:00
if ( quirks - > init ( data ) )
pr_info ( " GIC: enabling workaround for %s \n " ,
quirks - > desc ) ;
2015-09-21 22:58:37 +02:00
}
}
2015-01-20 16:52:59 +00:00
int gic_configure_irq ( unsigned int irq , unsigned int type ,
2014-06-30 16:01:30 +01:00
void __iomem * base , void ( * sync_access ) ( void ) )
{
u32 confmask = 0x2 < < ( ( irq % 16 ) * 2 ) ;
u32 confoff = ( irq / 16 ) * 4 ;
2015-01-20 16:52:59 +00:00
u32 val , oldval ;
int ret = 0 ;
2014-06-30 16:01:30 +01:00
/*
* Read current configuration register , and insert the config
* for " irq " , depending on " type " .
*/
2015-01-20 16:52:59 +00:00
val = oldval = readl_relaxed ( base + GIC_DIST_CONFIG + confoff ) ;
if ( type & IRQ_TYPE_LEVEL_MASK )
2014-06-30 16:01:30 +01:00
val & = ~ confmask ;
2015-01-20 16:52:59 +00:00
else if ( type & IRQ_TYPE_EDGE_BOTH )
2014-06-30 16:01:30 +01:00
val | = confmask ;
2016-05-10 16:14:38 +01:00
/* If the current configuration is the same, then we are done */
if ( val = = oldval )
return 0 ;
2014-06-30 16:01:30 +01:00
/*
* Write back the new configuration , and possibly re - enable
2016-05-10 16:14:39 +01:00
* the interrupt . If we fail to write a new configuration for
* an SPI then WARN and return an error . If we fail to write the
* configuration for a PPI this is most likely because the GIC
* does not allow us to set the configuration or we are in a
* non - secure mode , and hence it may not be catastrophic .
2014-06-30 16:01:30 +01:00
*/
writel_relaxed ( val , base + GIC_DIST_CONFIG + confoff ) ;
2016-05-10 16:14:39 +01:00
if ( readl_relaxed ( base + GIC_DIST_CONFIG + confoff ) ! = val ) {
if ( WARN_ON ( irq > = 32 ) )
ret = - EINVAL ;
else
pr_warn ( " GIC: PPI%d is secure or misconfigured \n " ,
irq - 16 ) ;
}
2014-06-30 16:01:30 +01:00
if ( sync_access )
sync_access ( ) ;
2015-01-20 16:52:59 +00:00
return ret ;
2014-06-30 16:01:30 +01:00
}
2016-06-07 16:12:32 +01:00
void gic_dist_config ( void __iomem * base , int gic_irqs ,
void ( * sync_access ) ( void ) )
2014-06-30 16:01:30 +01:00
{
unsigned int i ;
/*
* Set all global interrupts to be level triggered , active low .
*/
for ( i = 32 ; i < gic_irqs ; i + = 16 )
2014-07-30 14:56:58 -07:00
writel_relaxed ( GICD_INT_ACTLOW_LVLTRIG ,
base + GIC_DIST_CONFIG + i / 4 ) ;
2014-06-30 16:01:30 +01:00
/*
* Set priority on all global interrupts .
*/
for ( i = 32 ; i < gic_irqs ; i + = 4 )
2014-07-30 14:56:58 -07:00
writel_relaxed ( GICD_INT_DEF_PRI_X4 , base + GIC_DIST_PRI + i ) ;
2014-06-30 16:01:30 +01:00
/*
2015-11-16 19:13:26 +00:00
* Deactivate and disable all SPIs . Leave the PPI and SGIs
* alone as they are in the redistributor registers on GICv3 .
2014-06-30 16:01:30 +01:00
*/
2015-11-16 19:13:26 +00:00
for ( i = 32 ; i < gic_irqs ; i + = 32 ) {
2014-07-30 14:56:58 -07:00
writel_relaxed ( GICD_INT_EN_CLR_X32 ,
2015-11-16 19:13:26 +00:00
base + GIC_DIST_ACTIVE_CLEAR + i / 8 ) ;
writel_relaxed ( GICD_INT_EN_CLR_X32 ,
base + GIC_DIST_ENABLE_CLEAR + i / 8 ) ;
}
2014-06-30 16:01:30 +01:00
if ( sync_access )
sync_access ( ) ;
}
void gic_cpu_config ( void __iomem * base , void ( * sync_access ) ( void ) )
{
int i ;
/*
* Deal with the banked PPI and SGI interrupts - disable all
* PPI interrupts , ensure all SGI interrupts are enabled .
2015-11-16 19:13:26 +00:00
* Make sure everything is deactivated .
2014-06-30 16:01:30 +01:00
*/
2015-11-16 19:13:26 +00:00
writel_relaxed ( GICD_INT_EN_CLR_X32 , base + GIC_DIST_ACTIVE_CLEAR ) ;
2014-07-30 14:56:58 -07:00
writel_relaxed ( GICD_INT_EN_CLR_PPI , base + GIC_DIST_ENABLE_CLEAR ) ;
writel_relaxed ( GICD_INT_EN_SET_SGI , base + GIC_DIST_ENABLE_SET ) ;
2014-06-30 16:01:30 +01:00
/*
* Set priority on PPI and SGI interrupts
*/
for ( i = 0 ; i < 32 ; i + = 4 )
2014-07-30 14:56:58 -07:00
writel_relaxed ( GICD_INT_DEF_PRI_X4 ,
base + GIC_DIST_PRI + i * 4 / 4 ) ;
2014-06-30 16:01:30 +01:00
if ( sync_access )
sync_access ( ) ;
}