Blame view

arch/arm/plat-versatile/fpga-irq.c 1.56 KB
c41b16f8c   Russell King   ARM: integrator/v...
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
  /*
   *  Support for Versatile FPGA-based IRQ controllers
   */
  #include <linux/irq.h>
  #include <linux/io.h>
  
  #include <asm/mach/irq.h>
  #include <plat/fpga-irq.h>
  
  #define IRQ_STATUS		0x00
  #define IRQ_RAW_STATUS		0x04
  #define IRQ_ENABLE_SET		0x08
  #define IRQ_ENABLE_CLEAR	0x0c
  
  static void fpga_irq_mask(struct irq_data *d)
  {
  	struct fpga_irq_data *f = irq_data_get_irq_chip_data(d);
  	u32 mask = 1 << (d->irq - f->irq_start);
  
  	writel(mask, f->base + IRQ_ENABLE_CLEAR);
  }
  
  static void fpga_irq_unmask(struct irq_data *d)
  {
  	struct fpga_irq_data *f = irq_data_get_irq_chip_data(d);
  	u32 mask = 1 << (d->irq - f->irq_start);
  
  	writel(mask, f->base + IRQ_ENABLE_SET);
  }
  
  static void fpga_irq_handle(unsigned int irq, struct irq_desc *desc)
  {
6845664a6   Thomas Gleixner   arm: Cleanup the ...
33
  	struct fpga_irq_data *f = irq_desc_get_handler_data(desc);
c41b16f8c   Russell King   ARM: integrator/v...
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
  	u32 status = readl(f->base + IRQ_STATUS);
  
  	if (status == 0) {
  		do_bad_IRQ(irq, desc);
  		return;
  	}
  
  	do {
  		irq = ffs(status) - 1;
  		status &= ~(1 << irq);
  
  		generic_handle_irq(irq + f->irq_start);
  	} while (status);
  }
  
  void __init fpga_irq_init(int parent_irq, u32 valid, struct fpga_irq_data *f)
  {
  	unsigned int i;
  
  	f->chip.irq_ack = fpga_irq_mask;
  	f->chip.irq_mask = fpga_irq_mask;
  	f->chip.irq_unmask = fpga_irq_unmask;
  
  	if (parent_irq != -1) {
6845664a6   Thomas Gleixner   arm: Cleanup the ...
58
59
  		irq_set_handler_data(parent_irq, f);
  		irq_set_chained_handler(parent_irq, fpga_irq_handle);
c41b16f8c   Russell King   ARM: integrator/v...
60
61
62
63
64
  	}
  
  	for (i = 0; i < 32; i++) {
  		if (valid & (1 << i)) {
  			unsigned int irq = f->irq_start + i;
6845664a6   Thomas Gleixner   arm: Cleanup the ...
65
  			irq_set_chip_data(irq, f);
f38c02f3b   Thomas Gleixner   arm: Fold irq_set...
66
67
  			irq_set_chip_and_handler(irq, &f->chip,
  						 handle_level_irq);
c41b16f8c   Russell King   ARM: integrator/v...
68
69
70
71
  			set_irq_flags(irq, IRQF_VALID | IRQF_PROBE);
  		}
  	}
  }