diff options
Diffstat (limited to 'arch/mips/alchemy/pb1200/irqmap.c')
-rw-r--r-- | arch/mips/alchemy/pb1200/irqmap.c | 160 |
1 files changed, 160 insertions, 0 deletions
diff --git a/arch/mips/alchemy/pb1200/irqmap.c b/arch/mips/alchemy/pb1200/irqmap.c new file mode 100644 index 000000000000..2a505ad8715b --- /dev/null +++ b/arch/mips/alchemy/pb1200/irqmap.c | |||
@@ -0,0 +1,160 @@ | |||
1 | /* | ||
2 | * BRIEF MODULE DESCRIPTION | ||
3 | * Au1xxx irq map table | ||
4 | * | ||
5 | * This program is free software; you can redistribute it and/or modify it | ||
6 | * under the terms of the GNU General Public License as published by the | ||
7 | * Free Software Foundation; either version 2 of the License, or (at your | ||
8 | * option) any later version. | ||
9 | * | ||
10 | * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESS OR IMPLIED | ||
11 | * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF | ||
12 | * MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN | ||
13 | * NO EVENT SHALL THE AUTHOR BE LIABLE FOR ANY DIRECT, INDIRECT, | ||
14 | * INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT | ||
15 | * NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF | ||
16 | * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON | ||
17 | * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT | ||
18 | * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF | ||
19 | * THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. | ||
20 | * | ||
21 | * You should have received a copy of the GNU General Public License along | ||
22 | * with this program; if not, write to the Free Software Foundation, Inc., | ||
23 | * 675 Mass Ave, Cambridge, MA 02139, USA. | ||
24 | */ | ||
25 | |||
26 | #include <linux/init.h> | ||
27 | #include <linux/interrupt.h> | ||
28 | |||
29 | #include <asm/mach-au1x00/au1000.h> | ||
30 | |||
31 | #ifdef CONFIG_MIPS_PB1200 | ||
32 | #include <asm/mach-pb1x00/pb1200.h> | ||
33 | #endif | ||
34 | |||
35 | #ifdef CONFIG_MIPS_DB1200 | ||
36 | #include <asm/mach-db1x00/db1200.h> | ||
37 | #define PB1200_INT_BEGIN DB1200_INT_BEGIN | ||
38 | #define PB1200_INT_END DB1200_INT_END | ||
39 | #endif | ||
40 | |||
41 | struct au1xxx_irqmap __initdata au1xxx_irq_map[] = { | ||
42 | /* This is external interrupt cascade */ | ||
43 | { AU1000_GPIO_7, INTC_INT_LOW_LEVEL, 0 }, | ||
44 | }; | ||
45 | |||
46 | int __initdata au1xxx_nr_irqs = ARRAY_SIZE(au1xxx_irq_map); | ||
47 | |||
48 | /* | ||
49 | * Support for External interrupts on the Pb1200 Development platform. | ||
50 | */ | ||
51 | static volatile int pb1200_cascade_en; | ||
52 | |||
53 | irqreturn_t pb1200_cascade_handler(int irq, void *dev_id) | ||
54 | { | ||
55 | unsigned short bisr = bcsr->int_status; | ||
56 | int extirq_nr = 0; | ||
57 | |||
58 | /* Clear all the edge interrupts. This has no effect on level. */ | ||
59 | bcsr->int_status = bisr; | ||
60 | for ( ; bisr; bisr &= bisr - 1) { | ||
61 | extirq_nr = PB1200_INT_BEGIN + __ffs(bisr); | ||
62 | /* Ack and dispatch IRQ */ | ||
63 | do_IRQ(extirq_nr); | ||
64 | } | ||
65 | |||
66 | return IRQ_RETVAL(1); | ||
67 | } | ||
68 | |||
69 | inline void pb1200_enable_irq(unsigned int irq_nr) | ||
70 | { | ||
71 | bcsr->intset_mask = 1 << (irq_nr - PB1200_INT_BEGIN); | ||
72 | bcsr->intset = 1 << (irq_nr - PB1200_INT_BEGIN); | ||
73 | } | ||
74 | |||
75 | inline void pb1200_disable_irq(unsigned int irq_nr) | ||
76 | { | ||
77 | bcsr->intclr_mask = 1 << (irq_nr - PB1200_INT_BEGIN); | ||
78 | bcsr->intclr = 1 << (irq_nr - PB1200_INT_BEGIN); | ||
79 | } | ||
80 | |||
81 | static unsigned int pb1200_setup_cascade(void) | ||
82 | { | ||
83 | return request_irq(AU1000_GPIO_7, &pb1200_cascade_handler, | ||
84 | 0, "Pb1200 Cascade", &pb1200_cascade_handler); | ||
85 | } | ||
86 | |||
87 | static unsigned int pb1200_startup_irq(unsigned int irq) | ||
88 | { | ||
89 | if (++pb1200_cascade_en == 1) { | ||
90 | int res; | ||
91 | |||
92 | res = pb1200_setup_cascade(); | ||
93 | if (res) | ||
94 | return res; | ||
95 | } | ||
96 | |||
97 | pb1200_enable_irq(irq); | ||
98 | |||
99 | return 0; | ||
100 | } | ||
101 | |||
102 | static void pb1200_shutdown_irq(unsigned int irq) | ||
103 | { | ||
104 | pb1200_disable_irq(irq); | ||
105 | if (--pb1200_cascade_en == 0) | ||
106 | free_irq(AU1000_GPIO_7, &pb1200_cascade_handler); | ||
107 | } | ||
108 | |||
109 | static struct irq_chip external_irq_type = { | ||
110 | #ifdef CONFIG_MIPS_PB1200 | ||
111 | .name = "Pb1200 Ext", | ||
112 | #endif | ||
113 | #ifdef CONFIG_MIPS_DB1200 | ||
114 | .name = "Db1200 Ext", | ||
115 | #endif | ||
116 | .startup = pb1200_startup_irq, | ||
117 | .shutdown = pb1200_shutdown_irq, | ||
118 | .ack = pb1200_disable_irq, | ||
119 | .mask = pb1200_disable_irq, | ||
120 | .mask_ack = pb1200_disable_irq, | ||
121 | .unmask = pb1200_enable_irq, | ||
122 | }; | ||
123 | |||
124 | void _board_init_irq(void) | ||
125 | { | ||
126 | unsigned int irq; | ||
127 | |||
128 | #ifdef CONFIG_MIPS_PB1200 | ||
129 | /* We have a problem with CPLD rev 3. */ | ||
130 | if (((bcsr->whoami & BCSR_WHOAMI_CPLD) >> 4) <= 3) { | ||
131 | printk(KERN_ERR "WARNING!!!\n"); | ||
132 | printk(KERN_ERR "WARNING!!!\n"); | ||
133 | printk(KERN_ERR "WARNING!!!\n"); | ||
134 | printk(KERN_ERR "WARNING!!!\n"); | ||
135 | printk(KERN_ERR "WARNING!!!\n"); | ||
136 | printk(KERN_ERR "WARNING!!!\n"); | ||
137 | printk(KERN_ERR "Pb1200 must be at CPLD rev 4. Please have Pb1200\n"); | ||
138 | printk(KERN_ERR "updated to latest revision. This software will\n"); | ||
139 | printk(KERN_ERR "not work on anything less than CPLD rev 4.\n"); | ||
140 | printk(KERN_ERR "WARNING!!!\n"); | ||
141 | printk(KERN_ERR "WARNING!!!\n"); | ||
142 | printk(KERN_ERR "WARNING!!!\n"); | ||
143 | printk(KERN_ERR "WARNING!!!\n"); | ||
144 | printk(KERN_ERR "WARNING!!!\n"); | ||
145 | printk(KERN_ERR "WARNING!!!\n"); | ||
146 | panic("Game over. Your score is 0."); | ||
147 | } | ||
148 | #endif | ||
149 | |||
150 | for (irq = PB1200_INT_BEGIN; irq <= PB1200_INT_END; irq++) { | ||
151 | set_irq_chip_and_handler(irq, &external_irq_type, | ||
152 | handle_level_irq); | ||
153 | pb1200_disable_irq(irq); | ||
154 | } | ||
155 | |||
156 | /* | ||
157 | * GPIO_7 can not be hooked here, so it is hooked upon first | ||
158 | * request of any source attached to the cascade. | ||
159 | */ | ||
160 | } | ||