diff options
Diffstat (limited to 'kernel/irq/devres.c')
| -rw-r--r-- | kernel/irq/devres.c | 86 | 
1 files changed, 86 insertions, 0 deletions
| diff --git a/kernel/irq/devres.c b/kernel/irq/devres.c index 1613bfd48365..194c506d9d20 100644 --- a/kernel/irq/devres.c +++ b/kernel/irq/devres.c @@ -4,6 +4,8 @@  #include <linux/gfp.h>  #include <linux/irq.h> +#include "internals.h" +  /*   * Device resource management aware IRQ request/free implementation.   */ @@ -198,3 +200,87 @@ int __devm_irq_alloc_descs(struct device *dev, int irq, unsigned int from,  	return base;  }  EXPORT_SYMBOL_GPL(__devm_irq_alloc_descs); + +#ifdef CONFIG_GENERIC_IRQ_CHIP +/** + * devm_irq_alloc_generic_chip - Allocate and initialize a generic chip + *                               for a managed device + * @dev:	Device to allocate the generic chip for + * @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 * +devm_irq_alloc_generic_chip(struct device *dev, 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 = devm_kzalloc(dev, sz, GFP_KERNEL); +	if (gc) +		irq_init_generic_chip(gc, name, num_ct, +				      irq_base, reg_base, handler); + +	return gc; +} +EXPORT_SYMBOL_GPL(devm_irq_alloc_generic_chip); + +struct irq_generic_chip_devres { +	struct irq_chip_generic *gc; +	u32 msk; +	unsigned int clr; +	unsigned int set; +}; + +static void devm_irq_remove_generic_chip(struct device *dev, void *res) +{ +	struct irq_generic_chip_devres *this = res; + +	irq_remove_generic_chip(this->gc, this->msk, this->clr, this->set); +} + +/** + * devm_irq_setup_generic_chip - Setup a range of interrupts with a generic + *                               chip for a managed device + * + * @dev:	Device to setup the generic chip for + * @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. + */ +int devm_irq_setup_generic_chip(struct device *dev, struct irq_chip_generic *gc, +				u32 msk, enum irq_gc_flags flags, +				unsigned int clr, unsigned int set) +{ +	struct irq_generic_chip_devres *dr; + +	dr = devres_alloc(devm_irq_remove_generic_chip, +			  sizeof(*dr), GFP_KERNEL); +	if (!dr) +		return -ENOMEM; + +	irq_setup_generic_chip(gc, msk, flags, clr, set); + +	dr->gc = gc; +	dr->msk = msk; +	dr->clr = clr; +	dr->set = set; +	devres_add(dev, dr); + +	return 0; +} +EXPORT_SYMBOL_GPL(devm_irq_setup_generic_chip); +#endif /* CONFIG_GENERIC_IRQ_CHIP */ |