forked from mirrors/linux
		
	arm: omap: intc: switch over to linear irq domain
now that we don't need to support legacy board-files, we can completely switch over to a linear irq domain and make use of irq_alloc_domain_generic_chips() to allocate all generic irq chips for us. Signed-off-by: Felipe Balbi <balbi@ti.com> Signed-off-by: Tony Lindgren <tony@atomide.com>
This commit is contained in:
		
							parent
							
								
									d6a7c5c84f
								
							
						
					
					
						commit
						55601c9f24
					
				
					 1 changed files with 79 additions and 9 deletions
				
			
		| 
						 | 
					@ -188,14 +188,51 @@ void omap3_intc_suspend(void)
 | 
				
			||||||
	omap_ack_irq(NULL);
 | 
						omap_ack_irq(NULL);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static __init void
 | 
					static int __init omap_alloc_gc_of(struct irq_domain *d, void __iomem *base)
 | 
				
			||||||
omap_alloc_gc(void __iomem *base, unsigned int irq_start, unsigned int num)
 | 
					{
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
 | 
						int i;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = irq_alloc_domain_generic_chips(d, 32, 1, "INTC",
 | 
				
			||||||
 | 
								handle_level_irq, IRQ_NOREQUEST | IRQ_NOPROBE,
 | 
				
			||||||
 | 
								IRQ_LEVEL, 0);
 | 
				
			||||||
 | 
						if (ret) {
 | 
				
			||||||
 | 
							pr_warn("Failed to allocate irq chips\n");
 | 
				
			||||||
 | 
							return ret;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						for (i = 0; i < omap_nr_pending; i++) {
 | 
				
			||||||
 | 
							struct irq_chip_generic *gc;
 | 
				
			||||||
 | 
							struct irq_chip_type *ct;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							gc = irq_get_domain_generic_chip(d, 32 * i);
 | 
				
			||||||
 | 
							gc->reg_base = base;
 | 
				
			||||||
 | 
							ct = gc->chip_types;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ct->type = IRQ_TYPE_LEVEL_MASK;
 | 
				
			||||||
 | 
							ct->handler = handle_level_irq;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ct->chip.irq_ack = omap_mask_ack_irq;
 | 
				
			||||||
 | 
							ct->chip.irq_mask = irq_gc_mask_disable_reg;
 | 
				
			||||||
 | 
							ct->chip.irq_unmask = irq_gc_unmask_enable_reg;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ct->chip.flags |= IRQCHIP_SKIP_SET_WAKE;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							ct->regs.enable = INTC_MIR_CLEAR0 + 32 * i;
 | 
				
			||||||
 | 
							ct->regs.disable = INTC_MIR_SET0 + 32 * i;
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static void __init omap_alloc_gc_legacy(void __iomem *base,
 | 
				
			||||||
 | 
							unsigned int irq_start, unsigned int num)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct irq_chip_generic *gc;
 | 
						struct irq_chip_generic *gc;
 | 
				
			||||||
	struct irq_chip_type *ct;
 | 
						struct irq_chip_type *ct;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	gc = irq_alloc_generic_chip("INTC", 1, irq_start, base,
 | 
						gc = irq_alloc_generic_chip("INTC", 1, irq_start, base,
 | 
				
			||||||
					handle_level_irq);
 | 
								handle_level_irq);
 | 
				
			||||||
	ct = gc->chip_types;
 | 
						ct = gc->chip_types;
 | 
				
			||||||
	ct->chip.irq_ack = omap_mask_ack_irq;
 | 
						ct->chip.irq_ack = omap_mask_ack_irq;
 | 
				
			||||||
	ct->chip.irq_mask = irq_gc_mask_disable_reg;
 | 
						ct->chip.irq_mask = irq_gc_mask_disable_reg;
 | 
				
			||||||
| 
						 | 
					@ -205,16 +242,36 @@ omap_alloc_gc(void __iomem *base, unsigned int irq_start, unsigned int num)
 | 
				
			||||||
	ct->regs.enable = INTC_MIR_CLEAR0;
 | 
						ct->regs.enable = INTC_MIR_CLEAR0;
 | 
				
			||||||
	ct->regs.disable = INTC_MIR_SET0;
 | 
						ct->regs.disable = INTC_MIR_SET0;
 | 
				
			||||||
	irq_setup_generic_chip(gc, IRQ_MSK(num), IRQ_GC_INIT_MASK_CACHE,
 | 
						irq_setup_generic_chip(gc, IRQ_MSK(num), IRQ_GC_INIT_MASK_CACHE,
 | 
				
			||||||
				IRQ_NOREQUEST | IRQ_NOPROBE, 0);
 | 
								IRQ_NOREQUEST | IRQ_NOPROBE, 0);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static void __init omap_init_irq(u32 base, struct device_node *node)
 | 
					static int __init omap_init_irq_of(struct device_node *node)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						omap_irq_base = of_iomap(node, 0);
 | 
				
			||||||
 | 
						if (WARN_ON(!omap_irq_base))
 | 
				
			||||||
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						domain = irq_domain_add_linear(node, omap_nr_irqs,
 | 
				
			||||||
 | 
								&irq_generic_chip_ops, NULL);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						omap_irq_soft_reset();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						ret = omap_alloc_gc_of(domain, omap_irq_base);
 | 
				
			||||||
 | 
						if (ret < 0)
 | 
				
			||||||
 | 
							irq_domain_remove(domain);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return ret;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int __init omap_init_irq_legacy(u32 base)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	int j, irq_base;
 | 
						int j, irq_base;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	omap_irq_base = ioremap(base, SZ_4K);
 | 
						omap_irq_base = ioremap(base, SZ_4K);
 | 
				
			||||||
	if (WARN_ON(!omap_irq_base))
 | 
						if (WARN_ON(!omap_irq_base))
 | 
				
			||||||
		return;
 | 
							return -ENOMEM;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	irq_base = irq_alloc_descs(-1, 0, omap_nr_irqs, 0);
 | 
						irq_base = irq_alloc_descs(-1, 0, omap_nr_irqs, 0);
 | 
				
			||||||
	if (irq_base < 0) {
 | 
						if (irq_base < 0) {
 | 
				
			||||||
| 
						 | 
					@ -222,13 +279,23 @@ static void __init omap_init_irq(u32 base, struct device_node *node)
 | 
				
			||||||
		irq_base = 0;
 | 
							irq_base = 0;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	domain = irq_domain_add_legacy(node, omap_nr_irqs, irq_base, 0,
 | 
						domain = irq_domain_add_legacy(NULL, omap_nr_irqs, irq_base, 0,
 | 
				
			||||||
			&irq_domain_simple_ops, NULL);
 | 
								&irq_domain_simple_ops, NULL);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	omap_irq_soft_reset();
 | 
						omap_irq_soft_reset();
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	for (j = 0; j < omap_nr_irqs; j += 32)
 | 
						for (j = 0; j < omap_nr_irqs; j += 32)
 | 
				
			||||||
		omap_alloc_gc(omap_irq_base + j, j + irq_base, 32);
 | 
							omap_alloc_gc_legacy(omap_irq_base + j, j + irq_base, 32);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return 0;
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					static int __init omap_init_irq(u32 base, struct device_node *node)
 | 
				
			||||||
 | 
					{
 | 
				
			||||||
 | 
						if (node)
 | 
				
			||||||
 | 
							return omap_init_irq_of(node);
 | 
				
			||||||
 | 
						else
 | 
				
			||||||
 | 
							return omap_init_irq_legacy(base);
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
static asmlinkage void __exception_irq_entry
 | 
					static asmlinkage void __exception_irq_entry
 | 
				
			||||||
| 
						 | 
					@ -294,6 +361,7 @@ static int __init intc_of_init(struct device_node *node,
 | 
				
			||||||
			     struct device_node *parent)
 | 
								     struct device_node *parent)
 | 
				
			||||||
{
 | 
					{
 | 
				
			||||||
	struct resource res;
 | 
						struct resource res;
 | 
				
			||||||
 | 
						int ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	omap_nr_pending = 3;
 | 
						omap_nr_pending = 3;
 | 
				
			||||||
	omap_nr_irqs = 96;
 | 
						omap_nr_irqs = 96;
 | 
				
			||||||
| 
						 | 
					@ -311,7 +379,9 @@ static int __init intc_of_init(struct device_node *node,
 | 
				
			||||||
		omap_nr_pending = 4;
 | 
							omap_nr_pending = 4;
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	omap_init_irq(res.start, of_node_get(node));
 | 
						ret = omap_init_irq(-1, of_node_get(node));
 | 
				
			||||||
 | 
						if (ret < 0)
 | 
				
			||||||
 | 
							return ret;
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	set_handle_irq(omap_intc_handle_irq);
 | 
						set_handle_irq(omap_intc_handle_irq);
 | 
				
			||||||
 | 
					
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in a new issue