2015-10-24 00:15:52 +02:00
/*
* Special GIC quirks for the ARM RealView
* Copyright ( C ) 2015 Linus Walleij
*/
# include <linux/of.h>
# include <linux/regmap.h>
# include <linux/mfd/syscon.h>
# include <linux/bitops.h>
# include <linux/irqchip.h>
# include <linux/irqchip/arm-gic.h>
# define REALVIEW_SYS_LOCK_OFFSET 0x20
2016-02-22 15:01:01 +01:00
# define REALVIEW_SYS_PLD_CTRL1 0x74
# define REALVIEW_EB_REVB_SYS_PLD_CTRL1 0xD8
2015-10-24 00:15:52 +02:00
# define VERSATILE_LOCK_VAL 0xA05F
# define PLD_INTMODE_MASK BIT(22)|BIT(23)|BIT(24)
# define PLD_INTMODE_LEGACY 0x0
# define PLD_INTMODE_NEW_DCC BIT(22)
# define PLD_INTMODE_NEW_NO_DCC BIT(23)
# define PLD_INTMODE_FIQ_ENABLE BIT(24)
2016-02-22 15:01:01 +01:00
/* For some reason RealView EB Rev B moved this register */
static const struct of_device_id syscon_pldset_of_match [ ] = {
{
. compatible = " arm,realview-eb11mp-revb-syscon " ,
. data = ( void * ) REALVIEW_EB_REVB_SYS_PLD_CTRL1 ,
} ,
{
. compatible = " arm,realview-eb11mp-revc-syscon " ,
. data = ( void * ) REALVIEW_SYS_PLD_CTRL1 ,
} ,
{
. compatible = " arm,realview-eb-syscon " ,
. data = ( void * ) REALVIEW_SYS_PLD_CTRL1 ,
} ,
{
. compatible = " arm,realview-pb11mp-syscon " ,
. data = ( void * ) REALVIEW_SYS_PLD_CTRL1 ,
} ,
{ } ,
} ;
2015-10-24 00:15:52 +02:00
static int __init
realview_gic_of_init ( struct device_node * node , struct device_node * parent )
{
2017-07-15 22:07:40 +02:00
struct regmap * map ;
2016-02-22 15:01:01 +01:00
struct device_node * np ;
const struct of_device_id * gic_id ;
u32 pld1_ctrl ;
np = of_find_matching_node_and_match ( NULL , syscon_pldset_of_match ,
& gic_id ) ;
if ( ! np )
return - ENODEV ;
pld1_ctrl = ( u32 ) gic_id - > data ;
2015-10-24 00:15:52 +02:00
/* The PB11MPCore GIC needs to be configured in the syscon */
2016-02-22 15:01:01 +01:00
map = syscon_node_to_regmap ( np ) ;
2015-10-24 00:15:52 +02:00
if ( ! IS_ERR ( map ) ) {
/* new irq mode with no DCC */
regmap_write ( map , REALVIEW_SYS_LOCK_OFFSET ,
VERSATILE_LOCK_VAL ) ;
2016-02-22 15:01:01 +01:00
regmap_update_bits ( map , pld1_ctrl ,
2015-10-24 00:15:52 +02:00
PLD_INTMODE_NEW_NO_DCC ,
PLD_INTMODE_MASK ) ;
regmap_write ( map , REALVIEW_SYS_LOCK_OFFSET , 0x0000 ) ;
2016-02-22 15:01:01 +01:00
pr_info ( " RealView GIC: set up interrupt controller to NEW mode, no DCC \n " ) ;
2015-10-24 00:15:52 +02:00
} else {
2016-02-22 15:01:01 +01:00
pr_err ( " RealView GIC setup: could not find syscon \n " ) ;
return - ENODEV ;
2015-10-24 00:15:52 +02:00
}
return gic_of_init ( node , parent ) ;
}
IRQCHIP_DECLARE ( armtc11mp_gic , " arm,tc11mp-gic " , realview_gic_of_init ) ;
2016-02-22 15:01:01 +01:00
IRQCHIP_DECLARE ( armeb11mp_gic , " arm,eb11mp-gic " , realview_gic_of_init ) ;