2018-03-15 00:15:19 +03:00
// SPDX-License-Identifier: GPL-2.0
2011-04-03 13:42:53 +04:00
/*
* Library implementing the most common irq chip callback functions
*
* Copyright ( C ) 2011 , Thomas Gleixner
*/
# include <linux/io.h>
# include <linux/irq.h>
# include <linux/slab.h>
2011-05-27 00:00:52 +04:00
# include <linux/export.h>
2013-05-06 18:30:27 +04:00
# include <linux/irqdomain.h>
2011-04-03 13:42:53 +04:00
# include <linux/interrupt.h>
# include <linux/kernel_stat.h>
2011-04-16 00:36:08 +04:00
# include <linux/syscore_ops.h>
2011-04-03 13:42:53 +04:00
# include "internals.h"
2011-04-16 00:36:08 +04:00
static LIST_HEAD ( gc_list ) ;
static DEFINE_RAW_SPINLOCK ( gc_lock ) ;
2011-04-03 13:42:53 +04:00
/**
* irq_gc_noop - NOOP function
* @ d : irq_data
*/
void irq_gc_noop ( struct irq_data * d )
{
}
/**
* irq_gc_mask_disable_reg - Mask chip via disable register
* @ d : irq_data
*
* Chip has separate enable / disable registers instead of a single mask
* register .
*/
void irq_gc_mask_disable_reg ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , mask , ct - > regs . disable ) ;
2013-05-06 18:30:19 +04:00
* ct - > mask_cache & = ~ mask ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
/**
2013-06-28 13:45:15 +04:00
* irq_gc_mask_set_bit - Mask chip via setting bit in mask register
2011-04-03 13:42:53 +04:00
* @ d : irq_data
*
* Chip has a single mask register . Values of this register are cached
* and protected by gc - > lock
*/
void irq_gc_mask_set_bit ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2013-05-06 18:30:19 +04:00
* ct - > mask_cache | = mask ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , * ct - > mask_cache , ct - > regs . mask ) ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
2013-06-28 07:23:09 +04:00
EXPORT_SYMBOL_GPL ( irq_gc_mask_set_bit ) ;
2011-04-03 13:42:53 +04:00
/**
2013-06-28 13:45:15 +04:00
* irq_gc_mask_clr_bit - Mask chip via clearing bit in mask register
2011-04-03 13:42:53 +04:00
* @ d : irq_data
*
* Chip has a single mask register . Values of this register are cached
* and protected by gc - > lock
*/
void irq_gc_mask_clr_bit ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2013-05-06 18:30:19 +04:00
* ct - > mask_cache & = ~ mask ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , * ct - > mask_cache , ct - > regs . mask ) ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
2013-06-28 07:23:09 +04:00
EXPORT_SYMBOL_GPL ( irq_gc_mask_clr_bit ) ;
2011-04-03 13:42:53 +04:00
/**
* irq_gc_unmask_enable_reg - Unmask chip via enable register
* @ d : irq_data
*
* Chip has separate enable / disable registers instead of a single mask
* register .
*/
void irq_gc_unmask_enable_reg ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , mask , ct - > regs . enable ) ;
2013-05-06 18:30:19 +04:00
* ct - > mask_cache | = mask ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
/**
2011-07-06 20:41:31 +04:00
* irq_gc_ack_set_bit - Ack pending interrupt via setting bit
2011-04-03 13:42:53 +04:00
* @ d : irq_data
*/
2011-07-06 20:41:31 +04:00
void irq_gc_ack_set_bit ( struct irq_data * d )
2011-04-03 13:42:53 +04:00
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , mask , ct - > regs . ack ) ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
2013-06-28 07:23:09 +04:00
EXPORT_SYMBOL_GPL ( irq_gc_ack_set_bit ) ;
2011-04-03 13:42:53 +04:00
2011-07-06 20:41:31 +04:00
/**
* irq_gc_ack_clr_bit - Ack pending interrupt via clearing bit
* @ d : irq_data
*/
void irq_gc_ack_clr_bit ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = ~ d - > mask ;
2011-07-06 20:41:31 +04:00
irq_gc_lock ( gc ) ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , mask , ct - > regs . ack ) ;
2011-07-06 20:41:31 +04:00
irq_gc_unlock ( gc ) ;
}
2017-10-04 15:26:26 +03:00
/**
* irq_gc_mask_disable_and_ack_set - Mask and ack pending interrupt
* @ d : irq_data
*
* This generic implementation of the irq_mask_ack method is for chips
* with separate enable / disable registers instead of a single mask
* register and where a pending interrupt is acknowledged by setting a
* bit .
*
* Note : This is the only permutation currently used . Similar generic
* functions should be added here if other permutations are required .
*/
void irq_gc_mask_disable_and_ack_set ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
u32 mask = d - > mask ;
irq_gc_lock ( gc ) ;
irq_reg_writel ( gc , mask , ct - > regs . disable ) ;
* ct - > mask_cache & = ~ mask ;
irq_reg_writel ( gc , mask , ct - > regs . ack ) ;
irq_gc_unlock ( gc ) ;
}
2011-04-03 13:42:53 +04:00
/**
* irq_gc_eoi - EOI interrupt
* @ d : irq_data
*/
void irq_gc_eoi ( struct irq_data * d )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:17 +04:00
struct irq_chip_type * ct = irq_data_get_chip_type ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
irq_gc_lock ( gc ) ;
2014-11-07 09:44:17 +03:00
irq_reg_writel ( gc , mask , ct - > regs . eoi ) ;
2011-04-03 13:42:53 +04:00
irq_gc_unlock ( gc ) ;
}
/**
* irq_gc_set_wake - Set / clr wake bit for an interrupt
2013-06-28 13:45:15 +04:00
* @ d : irq_data
* @ on : Indicates whether the wake bit should be set or cleared
2011-04-03 13:42:53 +04:00
*
* For chips where the wake from suspend functionality is not
* configured in a separate register and the wakeup active state is
* just stored in a bitmask .
*/
int irq_gc_set_wake ( struct irq_data * d , unsigned int on )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
2013-05-06 18:30:22 +04:00
u32 mask = d - > mask ;
2011-04-03 13:42:53 +04:00
if ( ! ( mask & gc - > wake_enabled ) )
return - EINVAL ;
irq_gc_lock ( gc ) ;
if ( on )
gc - > wake_active | = mask ;
else
gc - > wake_active & = ~ mask ;
irq_gc_unlock ( gc ) ;
return 0 ;
}
2021-03-05 11:06:58 +03:00
EXPORT_SYMBOL_GPL ( irq_gc_set_wake ) ;
2011-04-03 13:42:53 +04:00
2014-11-07 09:44:19 +03:00
static u32 irq_readl_be ( void __iomem * addr )
{
return ioread32be ( addr ) ;
}
static void irq_writel_be ( u32 val , void __iomem * addr )
{
iowrite32be ( val , addr ) ;
}
2017-05-31 19:06:58 +03:00
void irq_init_generic_chip ( struct irq_chip_generic * gc , const char * name ,
int num_ct , unsigned int irq_base ,
void __iomem * reg_base , irq_flow_handler_t handler )
2013-05-06 18:30:25 +04:00
{
raw_spin_lock_init ( & gc - > lock ) ;
gc - > num_ct = num_ct ;
gc - > irq_base = irq_base ;
gc - > reg_base = reg_base ;
gc - > chip_types - > chip . name = name ;
gc - > chip_types - > handler = handler ;
}
2011-04-03 13:42:53 +04:00
/**
* irq_alloc_generic_chip - Allocate a generic chip and initialize it
* @ name : Name of the irq chip
* @ num_ct : Number of irq_chip_type instances associated with this
* @ irq_base : Interrupt base nr for this chip
* @ reg_base : Register base address ( virtual )
* @ handler : Default flow handler associated with this chip
*
* Returns an initialized irq_chip_generic structure . The chip defaults
* to the primary ( index 0 ) irq_chip_type and @ handler
*/
struct irq_chip_generic *
irq_alloc_generic_chip ( const char * name , int num_ct , unsigned int irq_base ,
void __iomem * reg_base , irq_flow_handler_t handler )
{
struct irq_chip_generic * gc ;
unsigned long sz = sizeof ( * gc ) + num_ct * sizeof ( struct irq_chip_type ) ;
gc = kzalloc ( sz , GFP_KERNEL ) ;
if ( gc ) {
2013-05-06 18:30:25 +04:00
irq_init_generic_chip ( gc , name , num_ct , irq_base , reg_base ,
handler ) ;
2011-04-03 13:42:53 +04:00
}
return gc ;
}
2011-10-17 06:08:46 +04:00
EXPORT_SYMBOL_GPL ( irq_alloc_generic_chip ) ;
2011-04-03 13:42:53 +04:00
2013-05-06 18:30:25 +04:00
static void
irq_gc_init_mask_cache ( struct irq_chip_generic * gc , enum irq_gc_flags flags )
{
struct irq_chip_type * ct = gc - > chip_types ;
u32 * mskptr = & gc - > mask_cache , mskreg = ct - > regs . mask ;
int i ;
for ( i = 0 ; i < gc - > num_ct ; i + + ) {
if ( flags & IRQ_GC_MASK_CACHE_PER_TYPE ) {
mskptr = & ct [ i ] . mask_cache_priv ;
mskreg = ct [ i ] . regs . mask ;
}
ct [ i ] . mask_cache = mskptr ;
if ( flags & IRQ_GC_INIT_MASK_CACHE )
2014-11-07 09:44:17 +03:00
* mskptr = irq_reg_readl ( gc , mskreg ) ;
2013-05-06 18:30:25 +04:00
}
}
2013-05-06 18:30:27 +04:00
/**
2020-11-16 13:18:15 +03:00
* __irq_alloc_domain_generic_chips - Allocate generic chips for an irq domain
2013-05-06 18:30:27 +04:00
* @ d : irq domain for which to allocate chips
2016-08-16 17:05:08 +03:00
* @ irqs_per_chip : Number of interrupts each chip handles ( max 32 )
2013-05-06 18:30:27 +04:00
* @ num_ct : Number of irq_chip_type instances associated with this
* @ name : Name of the irq chip
* @ handler : Default flow handler associated with these chips
* @ clr : IRQ_ * bits to clear in the mapping function
* @ set : IRQ_ * bits to set in the mapping function
2013-06-18 18:08:33 +04:00
* @ gcflags : Generic chip specific setup flags
2013-05-06 18:30:27 +04:00
*/
2016-08-16 17:05:08 +03:00
int __irq_alloc_domain_generic_chips ( struct irq_domain * d , int irqs_per_chip ,
int num_ct , const char * name ,
irq_flow_handler_t handler ,
unsigned int clr , unsigned int set ,
enum irq_gc_flags gcflags )
2013-05-06 18:30:27 +04:00
{
struct irq_domain_chip_generic * dgc ;
struct irq_chip_generic * gc ;
int numchips , sz , i ;
unsigned long flags ;
void * tmp ;
if ( d - > gc )
return - EBUSY ;
2013-07-14 02:37:30 +04:00
numchips = DIV_ROUND_UP ( d - > revmap_size , irqs_per_chip ) ;
2013-05-06 18:30:27 +04:00
if ( ! numchips )
return - EINVAL ;
/* Allocate a pointer, generic chip and chiptypes for each chip */
sz = sizeof ( * dgc ) + numchips * sizeof ( gc ) ;
sz + = numchips * ( sizeof ( * gc ) + num_ct * sizeof ( struct irq_chip_type ) ) ;
tmp = dgc = kzalloc ( sz , GFP_KERNEL ) ;
if ( ! dgc )
return - ENOMEM ;
dgc - > irqs_per_chip = irqs_per_chip ;
dgc - > num_chips = numchips ;
dgc - > irq_flags_to_set = set ;
dgc - > irq_flags_to_clear = clr ;
dgc - > gc_flags = gcflags ;
d - > gc = dgc ;
/* Calc pointer to the first generic chip */
tmp + = sizeof ( * dgc ) + numchips * sizeof ( gc ) ;
for ( i = 0 ; i < numchips ; i + + ) {
/* Store the pointer to the generic chip */
dgc - > gc [ i ] = gc = tmp ;
irq_init_generic_chip ( gc , name , num_ct , i * irqs_per_chip ,
NULL , handler ) ;
2014-11-07 09:44:19 +03:00
2013-05-06 18:30:27 +04:00
gc - > domain = d ;
2014-11-07 09:44:19 +03:00
if ( gcflags & IRQ_GC_BE_IO ) {
gc - > reg_readl = & irq_readl_be ;
gc - > reg_writel = & irq_writel_be ;
}
2013-05-06 18:30:27 +04:00
raw_spin_lock_irqsave ( & gc_lock , flags ) ;
list_add_tail ( & gc - > list , & gc_list ) ;
raw_spin_unlock_irqrestore ( & gc_lock , flags ) ;
/* Calc pointer to the next generic chip */
tmp + = sizeof ( * gc ) + num_ct * sizeof ( struct irq_chip_type ) ;
}
return 0 ;
}
2016-08-16 17:05:08 +03:00
EXPORT_SYMBOL_GPL ( __irq_alloc_domain_generic_chips ) ;
2013-05-06 18:30:27 +04:00
2016-08-01 17:27:53 +03:00
static struct irq_chip_generic *
__irq_get_domain_generic_chip ( struct irq_domain * d , unsigned int hw_irq )
{
struct irq_domain_chip_generic * dgc = d - > gc ;
int idx ;
if ( ! dgc )
return ERR_PTR ( - ENODEV ) ;
idx = hw_irq / dgc - > irqs_per_chip ;
if ( idx > = dgc - > num_chips )
return ERR_PTR ( - EINVAL ) ;
return dgc - > gc [ idx ] ;
}
2013-05-06 18:30:27 +04:00
/**
* irq_get_domain_generic_chip - Get a pointer to the generic chip of a hw_irq
* @ d : irq domain pointer
* @ hw_irq : Hardware interrupt number
*/
struct irq_chip_generic *
irq_get_domain_generic_chip ( struct irq_domain * d , unsigned int hw_irq )
{
2016-08-01 17:27:53 +03:00
struct irq_chip_generic * gc = __irq_get_domain_generic_chip ( d , hw_irq ) ;
2013-05-06 18:30:27 +04:00
2016-08-01 17:27:53 +03:00
return ! IS_ERR ( gc ) ? gc : NULL ;
2013-05-06 18:30:27 +04:00
}
EXPORT_SYMBOL_GPL ( irq_get_domain_generic_chip ) ;
2011-04-03 13:42:53 +04:00
/*
2017-12-02 20:11:04 +03:00
* Separate lockdep classes for interrupt chip which can nest irq_desc
* lock and request mutex .
2011-04-03 13:42:53 +04:00
*/
static struct lock_class_key irq_nested_lock_class ;
2017-12-02 20:11:04 +03:00
static struct lock_class_key irq_nested_request_class ;
2011-04-03 13:42:53 +04:00
2013-06-28 13:45:15 +04:00
/*
2013-05-06 18:30:27 +04:00
* irq_map_generic_chip - Map a generic chip for an irq domain
*/
2014-07-10 21:14:16 +04:00
int irq_map_generic_chip ( struct irq_domain * d , unsigned int virq ,
irq_hw_number_t hw_irq )
2013-05-06 18:30:27 +04:00
{
2015-05-16 12:44:15 +03:00
struct irq_data * data = irq_domain_get_irq_data ( d , virq ) ;
2013-05-06 18:30:27 +04:00
struct irq_domain_chip_generic * dgc = d - > gc ;
struct irq_chip_generic * gc ;
struct irq_chip_type * ct ;
struct irq_chip * chip ;
unsigned long flags ;
int idx ;
2016-08-01 17:27:53 +03:00
gc = __irq_get_domain_generic_chip ( d , hw_irq ) ;
if ( IS_ERR ( gc ) )
return PTR_ERR ( gc ) ;
2013-05-06 18:30:27 +04:00
idx = hw_irq % dgc - > irqs_per_chip ;
2013-05-29 06:10:52 +04:00
if ( test_bit ( idx , & gc - > unused ) )
return - ENOTSUPP ;
2013-05-06 18:30:27 +04:00
if ( test_bit ( idx , & gc - > installed ) )
return - EBUSY ;
ct = gc - > chip_types ;
chip = & ct - > chip ;
/* We only init the cache for the first mapping of a generic chip */
if ( ! gc - > installed ) {
raw_spin_lock_irqsave ( & gc - > lock , flags ) ;
irq_gc_init_mask_cache ( gc , dgc - > gc_flags ) ;
raw_spin_unlock_irqrestore ( & gc - > lock , flags ) ;
}
/* Mark the interrupt as installed */
set_bit ( idx , & gc - > installed ) ;
if ( dgc - > gc_flags & IRQ_GC_INIT_NESTED_LOCK )
2017-12-02 20:11:04 +03:00
irq_set_lockdep_class ( virq , & irq_nested_lock_class ,
& irq_nested_request_class ) ;
2013-05-06 18:30:27 +04:00
if ( chip - > irq_calc_mask )
chip - > irq_calc_mask ( data ) ;
else
data - > mask = 1 < < idx ;
2015-05-16 12:44:15 +03:00
irq_domain_set_info ( d , virq , hw_irq , chip , gc , ct - > handler , NULL , NULL ) ;
2013-05-06 18:30:27 +04:00
irq_modify_status ( virq , dgc - > irq_flags_to_clear , dgc - > irq_flags_to_set ) ;
return 0 ;
}
2016-08-01 17:27:38 +03:00
static void irq_unmap_generic_chip ( struct irq_domain * d , unsigned int virq )
{
struct irq_data * data = irq_domain_get_irq_data ( d , virq ) ;
struct irq_domain_chip_generic * dgc = d - > gc ;
unsigned int hw_irq = data - > hwirq ;
struct irq_chip_generic * gc ;
int irq_idx ;
gc = irq_get_domain_generic_chip ( d , hw_irq ) ;
if ( ! gc )
return ;
irq_idx = hw_irq % dgc - > irqs_per_chip ;
clear_bit ( irq_idx , & gc - > installed ) ;
irq_domain_set_info ( d , virq , hw_irq , & no_irq_chip , NULL , NULL , NULL ,
NULL ) ;
}
2013-05-06 18:30:27 +04:00
struct irq_domain_ops irq_generic_chip_ops = {
. map = irq_map_generic_chip ,
2016-08-01 17:27:38 +03:00
. unmap = irq_unmap_generic_chip ,
2013-05-06 18:30:27 +04:00
. xlate = irq_domain_xlate_onetwocell ,
} ;
EXPORT_SYMBOL_GPL ( irq_generic_chip_ops ) ;
2011-04-03 13:42:53 +04:00
/**
* irq_setup_generic_chip - Setup a range of interrupts with a generic chip
* @ gc : Generic irq chip holding all data
* @ msk : Bitmask holding the irqs to initialize relative to gc - > irq_base
* @ flags : Flags for initialization
* @ clr : IRQ_ * bits to clear
* @ set : IRQ_ * bits to set
*
* Set up max . 32 interrupts starting from gc - > irq_base . Note , this
* initializes all interrupts to the primary irq_chip_type and its
* associated handler .
*/
void irq_setup_generic_chip ( struct irq_chip_generic * gc , u32 msk ,
enum irq_gc_flags flags , unsigned int clr ,
unsigned int set )
{
struct irq_chip_type * ct = gc - > chip_types ;
2013-05-06 18:30:24 +04:00
struct irq_chip * chip = & ct - > chip ;
2011-04-03 13:42:53 +04:00
unsigned int i ;
2011-04-16 00:36:08 +04:00
raw_spin_lock ( & gc_lock ) ;
list_add_tail ( & gc - > list , & gc_list ) ;
raw_spin_unlock ( & gc_lock ) ;
2013-05-06 18:30:25 +04:00
irq_gc_init_mask_cache ( gc , flags ) ;
2013-05-06 18:30:19 +04:00
2011-04-03 13:42:53 +04:00
for ( i = gc - > irq_base ; msk ; msk > > = 1 , i + + ) {
2011-07-21 10:29:14 +04:00
if ( ! ( msk & 0x01 ) )
2011-04-03 13:42:53 +04:00
continue ;
if ( flags & IRQ_GC_INIT_NESTED_LOCK )
2017-12-02 20:11:04 +03:00
irq_set_lockdep_class ( i , & irq_nested_lock_class ,
& irq_nested_request_class ) ;
2011-04-03 13:42:53 +04:00
2013-05-06 18:30:22 +04:00
if ( ! ( flags & IRQ_GC_NO_MASK ) ) {
struct irq_data * d = irq_get_irq_data ( i ) ;
2013-05-06 18:30:24 +04:00
if ( chip - > irq_calc_mask )
chip - > irq_calc_mask ( d ) ;
else
d - > mask = 1 < < ( i - gc - > irq_base ) ;
2013-05-06 18:30:22 +04:00
}
2013-05-06 18:30:24 +04:00
irq_set_chip_and_handler ( i , chip , ct - > handler ) ;
2011-04-03 13:42:53 +04:00
irq_set_chip_data ( i , gc ) ;
irq_modify_status ( i , clr , set ) ;
}
gc - > irq_cnt = i - gc - > irq_base ;
}
2011-10-17 06:08:46 +04:00
EXPORT_SYMBOL_GPL ( irq_setup_generic_chip ) ;
2011-04-03 13:42:53 +04:00
/**
* irq_setup_alt_chip - Switch to alternative chip
* @ d : irq_data for this interrupt
2013-06-28 13:45:15 +04:00
* @ type : Flow type to be initialized
2011-04-03 13:42:53 +04:00
*
* Only to be called from chip - > irq_set_type ( ) callbacks .
*/
int irq_setup_alt_chip ( struct irq_data * d , unsigned int type )
{
struct irq_chip_generic * gc = irq_data_get_irq_chip_data ( d ) ;
struct irq_chip_type * ct = gc - > chip_types ;
unsigned int i ;
for ( i = 0 ; i < gc - > num_ct ; i + + , ct + + ) {
if ( ct - > type & type ) {
d - > chip = & ct - > chip ;
irq_data_to_desc ( d ) - > handle_irq = ct - > handler ;
return 0 ;
}
}
return - EINVAL ;
}
2011-10-17 06:08:46 +04:00
EXPORT_SYMBOL_GPL ( irq_setup_alt_chip ) ;
2011-04-16 00:36:08 +04:00
/**
* irq_remove_generic_chip - Remove a chip
* @ gc : Generic irq chip holding all data
* @ msk : Bitmask holding the irqs to initialize relative to gc - > irq_base
* @ clr : IRQ_ * bits to clear
* @ set : IRQ_ * bits to set
*
* Remove up to 32 interrupts starting from gc - > irq_base .
*/
void irq_remove_generic_chip ( struct irq_chip_generic * gc , u32 msk ,
unsigned int clr , unsigned int set )
{
unsigned int i = gc - > irq_base ;
raw_spin_lock ( & gc_lock ) ;
list_del ( & gc - > list ) ;
raw_spin_unlock ( & gc_lock ) ;
for ( ; msk ; msk > > = 1 , i + + ) {
2011-07-21 10:29:14 +04:00
if ( ! ( msk & 0x01 ) )
2011-04-16 00:36:08 +04:00
continue ;
/* Remove handler first. That will mask the irq line */
irq_set_handler ( i , NULL ) ;
irq_set_chip ( i , & no_irq_chip ) ;
irq_set_chip_data ( i , NULL ) ;
irq_modify_status ( i , clr , set ) ;
}
}
2011-10-17 06:08:46 +04:00
EXPORT_SYMBOL_GPL ( irq_remove_generic_chip ) ;
2011-04-16 00:36:08 +04:00
2013-05-06 18:30:27 +04:00
static struct irq_data * irq_gc_get_irq_data ( struct irq_chip_generic * gc )
{
unsigned int virq ;
if ( ! gc - > domain )
return irq_get_irq_data ( gc - > irq_base ) ;
/*
* We don ' t know which of the irqs has been actually
* installed . Use the first one .
*/
if ( ! gc - > installed )
return NULL ;
virq = irq_find_mapping ( gc - > domain , gc - > irq_base + __ffs ( gc - > installed ) ) ;
return virq ? irq_get_irq_data ( virq ) : NULL ;
}
2011-04-16 00:36:08 +04:00
# ifdef CONFIG_PM
static int irq_gc_suspend ( void )
{
struct irq_chip_generic * gc ;
list_for_each_entry ( gc , & gc_list , list ) {
struct irq_chip_type * ct = gc - > chip_types ;
2013-05-06 18:30:27 +04:00
if ( ct - > chip . irq_suspend ) {
struct irq_data * data = irq_gc_get_irq_data ( gc ) ;
if ( data )
ct - > chip . irq_suspend ( data ) ;
}
2015-07-23 02:21:39 +03:00
if ( gc - > suspend )
gc - > suspend ( gc ) ;
2011-04-16 00:36:08 +04:00
}
return 0 ;
}
static void irq_gc_resume ( void )
{
struct irq_chip_generic * gc ;
list_for_each_entry ( gc , & gc_list , list ) {
struct irq_chip_type * ct = gc - > chip_types ;
2015-07-23 02:21:39 +03:00
if ( gc - > resume )
gc - > resume ( gc ) ;
2013-05-06 18:30:27 +04:00
if ( ct - > chip . irq_resume ) {
struct irq_data * data = irq_gc_get_irq_data ( gc ) ;
if ( data )
ct - > chip . irq_resume ( data ) ;
}
2011-04-16 00:36:08 +04:00
}
}
# else
# define irq_gc_suspend NULL
# define irq_gc_resume NULL
# endif
static void irq_gc_shutdown ( void )
{
struct irq_chip_generic * gc ;
list_for_each_entry ( gc , & gc_list , list ) {
struct irq_chip_type * ct = gc - > chip_types ;
2013-05-06 18:30:27 +04:00
if ( ct - > chip . irq_pm_shutdown ) {
struct irq_data * data = irq_gc_get_irq_data ( gc ) ;
if ( data )
ct - > chip . irq_pm_shutdown ( data ) ;
}
2011-04-16 00:36:08 +04:00
}
}
static struct syscore_ops irq_gc_syscore_ops = {
. suspend = irq_gc_suspend ,
. resume = irq_gc_resume ,
. shutdown = irq_gc_shutdown ,
} ;
static int __init irq_gc_init_ops ( void )
{
register_syscore_ops ( & irq_gc_syscore_ops ) ;
return 0 ;
}
device_initcall ( irq_gc_init_ops ) ;