diff options
author | Linus Torvalds <torvalds@linux-foundation.org> | 2012-03-27 19:03:32 -0400 |
---|---|---|
committer | Linus Torvalds <torvalds@linux-foundation.org> | 2012-03-27 19:03:32 -0400 |
commit | d61b7a572b292e2be409e13b4b3adf475f18fb29 (patch) | |
tree | e9d30390860147136c05e66abf1edda1bc5b0562 /arch/arm/mach-s3c24xx/irq-s3c2412.c | |
parent | 18d9946bc7e2252fe3c0f2f609ac383c627edefd (diff) | |
parent | f4e2467bad53023589cbff18dd1ab6e0aa3f004c (diff) |
Merge tag 'cleanup' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc
Pull "ARM: global cleanups" from Arnd Bergmann:
"Quite a bit of code gets removed, and some stuff moved around, mostly
the old samsung s3c24xx stuff. There should be no functional changes
in this series otherwise. Some cleanups have dependencies on other
arm-soc branches and will be sent in the second round.
Signed-off-by: Arnd Bergmann <arnd@arndb.de>"
Fixed up trivial conflicts mainly due to #include's being changes on
both sides.
* tag 'cleanup' of git://git.kernel.org/pub/scm/linux/kernel/git/arm/arm-soc: (121 commits)
ep93xx: Remove unnecessary includes of ep93xx-regs.h
ep93xx: Move EP93XX_SYSCON defines to SoC private header
ep93xx: Move crunch code to mach-ep93xx directory
ep93xx: Make syscon access functions private to SoC
ep93xx: Configure GPIO ports in core code
ep93xx: Move peripheral defines to local SoC header
ep93xx: Convert the watchdog driver into a platform device.
ep93xx: Use ioremap for backlight driver
ep93xx: Move GPIO defines to gpio-ep93xx.h
ep93xx: Don't use system controller defines in audio drivers
ep93xx: Move PHYS_BASE defines to local SoC header file
ARM: EXYNOS: Add clock register addresses for EXYNOS4X12 bus devfreq driver
ARM: EXYNOS: add clock registers for exynos4x12-cpufreq
PM / devfreq: update the name of EXYNOS clock registers that were omitted
PM / devfreq: update the name of EXYNOS clock register
ARM: EXYNOS: change the prefix S5P_ to EXYNOS4_ for clock
ARM: EXYNOS: use static declaration on regarding clock
ARM: EXYNOS: replace clock.c for other new EXYNOS SoCs
ARM: OMAP2+: Fix build error after merge
ARM: S3C24XX: remove call to s3c24xx_setup_clocks
...
Diffstat (limited to 'arch/arm/mach-s3c24xx/irq-s3c2412.c')
-rw-r--r-- | arch/arm/mach-s3c24xx/irq-s3c2412.c | 214 |
1 files changed, 214 insertions, 0 deletions
diff --git a/arch/arm/mach-s3c24xx/irq-s3c2412.c b/arch/arm/mach-s3c24xx/irq-s3c2412.c new file mode 100644 index 000000000000..e65619ddbccc --- /dev/null +++ b/arch/arm/mach-s3c24xx/irq-s3c2412.c | |||
@@ -0,0 +1,214 @@ | |||
1 | /* linux/arch/arm/mach-s3c2412/irq.c | ||
2 | * | ||
3 | * Copyright (c) 2006 Simtec Electronics | ||
4 | * Ben Dooks <ben@simtec.co.uk> | ||
5 | * | ||
6 | * This program is free software; you can redistribute it and/or modify | ||
7 | * it under the terms of the GNU General Public License as published by | ||
8 | * the Free Software Foundation; either version 2 of the License, or | ||
9 | * (at your option) any later version. | ||
10 | * | ||
11 | * This program is distributed in the hope that it will be useful, | ||
12 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | ||
13 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | ||
14 | * GNU General Public License for more details. | ||
15 | * | ||
16 | * You should have received a copy of the GNU General Public License | ||
17 | * along with this program; if not, write to the Free Software | ||
18 | * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA 02111-1307 USA | ||
19 | * | ||
20 | */ | ||
21 | |||
22 | #include <linux/init.h> | ||
23 | #include <linux/module.h> | ||
24 | #include <linux/interrupt.h> | ||
25 | #include <linux/ioport.h> | ||
26 | #include <linux/device.h> | ||
27 | #include <linux/io.h> | ||
28 | |||
29 | #include <mach/hardware.h> | ||
30 | #include <asm/irq.h> | ||
31 | |||
32 | #include <asm/mach/irq.h> | ||
33 | |||
34 | #include <mach/regs-irq.h> | ||
35 | #include <mach/regs-gpio.h> | ||
36 | #include <mach/regs-power.h> | ||
37 | |||
38 | #include <plat/cpu.h> | ||
39 | #include <plat/irq.h> | ||
40 | #include <plat/pm.h> | ||
41 | |||
42 | #define INTMSK(start, end) ((1 << ((end) + 1 - (start))) - 1) | ||
43 | #define INTMSK_SUB(start, end) (INTMSK(start, end) << ((start - S3C2410_IRQSUB(0)))) | ||
44 | |||
45 | /* the s3c2412 changes the behaviour of IRQ_EINT0 through IRQ_EINT3 by | ||
46 | * having them turn up in both the INT* and the EINT* registers. Whilst | ||
47 | * both show the status, they both now need to be acked when the IRQs | ||
48 | * go off. | ||
49 | */ | ||
50 | |||
51 | static void | ||
52 | s3c2412_irq_mask(struct irq_data *data) | ||
53 | { | ||
54 | unsigned long bitval = 1UL << (data->irq - IRQ_EINT0); | ||
55 | unsigned long mask; | ||
56 | |||
57 | mask = __raw_readl(S3C2410_INTMSK); | ||
58 | __raw_writel(mask | bitval, S3C2410_INTMSK); | ||
59 | |||
60 | mask = __raw_readl(S3C2412_EINTMASK); | ||
61 | __raw_writel(mask | bitval, S3C2412_EINTMASK); | ||
62 | } | ||
63 | |||
64 | static inline void | ||
65 | s3c2412_irq_ack(struct irq_data *data) | ||
66 | { | ||
67 | unsigned long bitval = 1UL << (data->irq - IRQ_EINT0); | ||
68 | |||
69 | __raw_writel(bitval, S3C2412_EINTPEND); | ||
70 | __raw_writel(bitval, S3C2410_SRCPND); | ||
71 | __raw_writel(bitval, S3C2410_INTPND); | ||
72 | } | ||
73 | |||
74 | static inline void | ||
75 | s3c2412_irq_maskack(struct irq_data *data) | ||
76 | { | ||
77 | unsigned long bitval = 1UL << (data->irq - IRQ_EINT0); | ||
78 | unsigned long mask; | ||
79 | |||
80 | mask = __raw_readl(S3C2410_INTMSK); | ||
81 | __raw_writel(mask|bitval, S3C2410_INTMSK); | ||
82 | |||
83 | mask = __raw_readl(S3C2412_EINTMASK); | ||
84 | __raw_writel(mask | bitval, S3C2412_EINTMASK); | ||
85 | |||
86 | __raw_writel(bitval, S3C2412_EINTPEND); | ||
87 | __raw_writel(bitval, S3C2410_SRCPND); | ||
88 | __raw_writel(bitval, S3C2410_INTPND); | ||
89 | } | ||
90 | |||
91 | static void | ||
92 | s3c2412_irq_unmask(struct irq_data *data) | ||
93 | { | ||
94 | unsigned long bitval = 1UL << (data->irq - IRQ_EINT0); | ||
95 | unsigned long mask; | ||
96 | |||
97 | mask = __raw_readl(S3C2412_EINTMASK); | ||
98 | __raw_writel(mask & ~bitval, S3C2412_EINTMASK); | ||
99 | |||
100 | mask = __raw_readl(S3C2410_INTMSK); | ||
101 | __raw_writel(mask & ~bitval, S3C2410_INTMSK); | ||
102 | } | ||
103 | |||
104 | static struct irq_chip s3c2412_irq_eint0t4 = { | ||
105 | .irq_ack = s3c2412_irq_ack, | ||
106 | .irq_mask = s3c2412_irq_mask, | ||
107 | .irq_unmask = s3c2412_irq_unmask, | ||
108 | .irq_set_wake = s3c_irq_wake, | ||
109 | .irq_set_type = s3c_irqext_type, | ||
110 | }; | ||
111 | |||
112 | #define INTBIT(x) (1 << ((x) - S3C2410_IRQSUB(0))) | ||
113 | |||
114 | /* CF and SDI sub interrupts */ | ||
115 | |||
116 | static void s3c2412_irq_demux_cfsdi(unsigned int irq, struct irq_desc *desc) | ||
117 | { | ||
118 | unsigned int subsrc, submsk; | ||
119 | |||
120 | subsrc = __raw_readl(S3C2410_SUBSRCPND); | ||
121 | submsk = __raw_readl(S3C2410_INTSUBMSK); | ||
122 | |||
123 | subsrc &= ~submsk; | ||
124 | |||
125 | if (subsrc & INTBIT(IRQ_S3C2412_SDI)) | ||
126 | generic_handle_irq(IRQ_S3C2412_SDI); | ||
127 | |||
128 | if (subsrc & INTBIT(IRQ_S3C2412_CF)) | ||
129 | generic_handle_irq(IRQ_S3C2412_CF); | ||
130 | } | ||
131 | |||
132 | #define INTMSK_CFSDI (1UL << (IRQ_S3C2412_CFSDI - IRQ_EINT0)) | ||
133 | #define SUBMSK_CFSDI INTMSK_SUB(IRQ_S3C2412_SDI, IRQ_S3C2412_CF) | ||
134 | |||
135 | static void s3c2412_irq_cfsdi_mask(struct irq_data *data) | ||
136 | { | ||
137 | s3c_irqsub_mask(data->irq, INTMSK_CFSDI, SUBMSK_CFSDI); | ||
138 | } | ||
139 | |||
140 | static void s3c2412_irq_cfsdi_unmask(struct irq_data *data) | ||
141 | { | ||
142 | s3c_irqsub_unmask(data->irq, INTMSK_CFSDI); | ||
143 | } | ||
144 | |||
145 | static void s3c2412_irq_cfsdi_ack(struct irq_data *data) | ||
146 | { | ||
147 | s3c_irqsub_maskack(data->irq, INTMSK_CFSDI, SUBMSK_CFSDI); | ||
148 | } | ||
149 | |||
150 | static struct irq_chip s3c2412_irq_cfsdi = { | ||
151 | .name = "s3c2412-cfsdi", | ||
152 | .irq_ack = s3c2412_irq_cfsdi_ack, | ||
153 | .irq_mask = s3c2412_irq_cfsdi_mask, | ||
154 | .irq_unmask = s3c2412_irq_cfsdi_unmask, | ||
155 | }; | ||
156 | |||
157 | static int s3c2412_irq_rtc_wake(struct irq_data *data, unsigned int state) | ||
158 | { | ||
159 | unsigned long pwrcfg; | ||
160 | |||
161 | pwrcfg = __raw_readl(S3C2412_PWRCFG); | ||
162 | if (state) | ||
163 | pwrcfg &= ~S3C2412_PWRCFG_RTC_MASKIRQ; | ||
164 | else | ||
165 | pwrcfg |= S3C2412_PWRCFG_RTC_MASKIRQ; | ||
166 | __raw_writel(pwrcfg, S3C2412_PWRCFG); | ||
167 | |||
168 | return s3c_irq_chip.irq_set_wake(data, state); | ||
169 | } | ||
170 | |||
171 | static struct irq_chip s3c2412_irq_rtc_chip; | ||
172 | |||
173 | static int s3c2412_irq_add(struct device *dev, struct subsys_interface *sif) | ||
174 | { | ||
175 | unsigned int irqno; | ||
176 | |||
177 | for (irqno = IRQ_EINT0; irqno <= IRQ_EINT3; irqno++) { | ||
178 | irq_set_chip_and_handler(irqno, &s3c2412_irq_eint0t4, | ||
179 | handle_edge_irq); | ||
180 | set_irq_flags(irqno, IRQF_VALID); | ||
181 | } | ||
182 | |||
183 | /* add demux support for CF/SDI */ | ||
184 | |||
185 | irq_set_chained_handler(IRQ_S3C2412_CFSDI, s3c2412_irq_demux_cfsdi); | ||
186 | |||
187 | for (irqno = IRQ_S3C2412_SDI; irqno <= IRQ_S3C2412_CF; irqno++) { | ||
188 | irq_set_chip_and_handler(irqno, &s3c2412_irq_cfsdi, | ||
189 | handle_level_irq); | ||
190 | set_irq_flags(irqno, IRQF_VALID); | ||
191 | } | ||
192 | |||
193 | /* change RTC IRQ's set wake method */ | ||
194 | |||
195 | s3c2412_irq_rtc_chip = s3c_irq_chip; | ||
196 | s3c2412_irq_rtc_chip.irq_set_wake = s3c2412_irq_rtc_wake; | ||
197 | |||
198 | irq_set_chip(IRQ_RTC, &s3c2412_irq_rtc_chip); | ||
199 | |||
200 | return 0; | ||
201 | } | ||
202 | |||
203 | static struct subsys_interface s3c2412_irq_interface = { | ||
204 | .name = "s3c2412_irq", | ||
205 | .subsys = &s3c2412_subsys, | ||
206 | .add_dev = s3c2412_irq_add, | ||
207 | }; | ||
208 | |||
209 | static int s3c2412_irq_init(void) | ||
210 | { | ||
211 | return subsys_interface_register(&s3c2412_irq_interface); | ||
212 | } | ||
213 | |||
214 | arch_initcall(s3c2412_irq_init); | ||