diff options
author | Olof Johansson <olof@lixom.net> | 2013-04-14 23:49:21 -0400 |
---|---|---|
committer | Olof Johansson <olof@lixom.net> | 2013-04-14 23:49:28 -0400 |
commit | b56a7f9206efa2982140837e36aca88cfedd66bc (patch) | |
tree | a1270fbf47cbaf6e056b2a06ede71cd622df987a /arch/arm/mach-msm | |
parent | 56c5c13f7080f9299a92b3fb6a1bf22689d607cc (diff) | |
parent | e25e3d1fef2c57e49aef64535341c15fe2b29b4a (diff) |
Merge tag 'msm-core-3.10' of git://git.kernel.org/pub/scm/linux/kernel/git/davidb/linux-msm into next/soc
From David Brown:
Patches for MSM core
These patches are changes to the MSM timer code that will be for
upcoming targets, including a generalization of the binding and
preventing a missing timer interrupt.
* tag 'msm-core-3.10' of git://git.kernel.org/pub/scm/linux/kernel/git/davidb/linux-msm:
ARM: msm: Wait for timer clear to complete
ARM: msm: Rework timer binding to be more general
Signed-off-by: Olof Johansson <olof@lixom.net>
Diffstat (limited to 'arch/arm/mach-msm')
-rw-r--r-- | arch/arm/mach-msm/timer.c | 110 |
1 files changed, 51 insertions, 59 deletions
diff --git a/arch/arm/mach-msm/timer.c b/arch/arm/mach-msm/timer.c index f9fd77e8f1f5..284313f3e02c 100644 --- a/arch/arm/mach-msm/timer.c +++ b/arch/arm/mach-msm/timer.c | |||
@@ -30,19 +30,22 @@ | |||
30 | 30 | ||
31 | #include "common.h" | 31 | #include "common.h" |
32 | 32 | ||
33 | #define TIMER_MATCH_VAL 0x0000 | 33 | #define TIMER_MATCH_VAL 0x0000 |
34 | #define TIMER_COUNT_VAL 0x0004 | 34 | #define TIMER_COUNT_VAL 0x0004 |
35 | #define TIMER_ENABLE 0x0008 | 35 | #define TIMER_ENABLE 0x0008 |
36 | #define TIMER_ENABLE_CLR_ON_MATCH_EN BIT(1) | 36 | #define TIMER_ENABLE_CLR_ON_MATCH_EN BIT(1) |
37 | #define TIMER_ENABLE_EN BIT(0) | 37 | #define TIMER_ENABLE_EN BIT(0) |
38 | #define TIMER_CLEAR 0x000C | 38 | #define TIMER_CLEAR 0x000C |
39 | #define DGT_CLK_CTL_DIV_4 0x3 | 39 | #define DGT_CLK_CTL 0x10 |
40 | #define DGT_CLK_CTL_DIV_4 0x3 | ||
41 | #define TIMER_STS_GPT0_CLR_PEND BIT(10) | ||
40 | 42 | ||
41 | #define GPT_HZ 32768 | 43 | #define GPT_HZ 32768 |
42 | 44 | ||
43 | #define MSM_DGT_SHIFT 5 | 45 | #define MSM_DGT_SHIFT 5 |
44 | 46 | ||
45 | static void __iomem *event_base; | 47 | static void __iomem *event_base; |
48 | static void __iomem *sts_base; | ||
46 | 49 | ||
47 | static irqreturn_t msm_timer_interrupt(int irq, void *dev_id) | 50 | static irqreturn_t msm_timer_interrupt(int irq, void *dev_id) |
48 | { | 51 | { |
@@ -67,6 +70,11 @@ static int msm_timer_set_next_event(unsigned long cycles, | |||
67 | 70 | ||
68 | writel_relaxed(ctrl, event_base + TIMER_CLEAR); | 71 | writel_relaxed(ctrl, event_base + TIMER_CLEAR); |
69 | writel_relaxed(cycles, event_base + TIMER_MATCH_VAL); | 72 | writel_relaxed(cycles, event_base + TIMER_MATCH_VAL); |
73 | |||
74 | if (sts_base) | ||
75 | while (readl_relaxed(sts_base) & TIMER_STS_GPT0_CLR_PEND) | ||
76 | cpu_relax(); | ||
77 | |||
70 | writel_relaxed(ctrl | TIMER_ENABLE_EN, event_base + TIMER_ENABLE); | 78 | writel_relaxed(ctrl | TIMER_ENABLE_EN, event_base + TIMER_ENABLE); |
71 | return 0; | 79 | return 0; |
72 | } | 80 | } |
@@ -137,9 +145,6 @@ static int __cpuinit msm_local_timer_setup(struct clock_event_device *evt) | |||
137 | if (!smp_processor_id()) | 145 | if (!smp_processor_id()) |
138 | return 0; | 146 | return 0; |
139 | 147 | ||
140 | writel_relaxed(0, event_base + TIMER_ENABLE); | ||
141 | writel_relaxed(0, event_base + TIMER_CLEAR); | ||
142 | writel_relaxed(~0, event_base + TIMER_MATCH_VAL); | ||
143 | evt->irq = msm_clockevent.irq; | 148 | evt->irq = msm_clockevent.irq; |
144 | evt->name = "local_timer"; | 149 | evt->name = "local_timer"; |
145 | evt->features = msm_clockevent.features; | 150 | evt->features = msm_clockevent.features; |
@@ -177,9 +182,6 @@ static void __init msm_timer_init(u32 dgt_hz, int sched_bits, int irq, | |||
177 | struct clocksource *cs = &msm_clocksource; | 182 | struct clocksource *cs = &msm_clocksource; |
178 | int res; | 183 | int res; |
179 | 184 | ||
180 | writel_relaxed(0, event_base + TIMER_ENABLE); | ||
181 | writel_relaxed(0, event_base + TIMER_CLEAR); | ||
182 | writel_relaxed(~0, event_base + TIMER_MATCH_VAL); | ||
183 | ce->cpumask = cpumask_of(0); | 185 | ce->cpumask = cpumask_of(0); |
184 | ce->irq = irq; | 186 | ce->irq = irq; |
185 | 187 | ||
@@ -217,13 +219,9 @@ err: | |||
217 | } | 219 | } |
218 | 220 | ||
219 | #ifdef CONFIG_OF | 221 | #ifdef CONFIG_OF |
220 | static const struct of_device_id msm_dgt_match[] __initconst = { | 222 | static const struct of_device_id msm_timer_match[] __initconst = { |
221 | { .compatible = "qcom,msm-dgt" }, | 223 | { .compatible = "qcom,kpss-timer" }, |
222 | { }, | 224 | { .compatible = "qcom,scss-timer" }, |
223 | }; | ||
224 | |||
225 | static const struct of_device_id msm_gpt_match[] __initconst = { | ||
226 | { .compatible = "qcom,msm-gpt" }, | ||
227 | { }, | 225 | { }, |
228 | }; | 226 | }; |
229 | 227 | ||
@@ -234,33 +232,29 @@ void __init msm_dt_timer_init(void) | |||
234 | int irq; | 232 | int irq; |
235 | struct resource res; | 233 | struct resource res; |
236 | u32 percpu_offset; | 234 | u32 percpu_offset; |
237 | void __iomem *dgt_clk_ctl; | 235 | void __iomem *base; |
236 | void __iomem *cpu0_base; | ||
238 | 237 | ||
239 | np = of_find_matching_node(NULL, msm_gpt_match); | 238 | np = of_find_matching_node(NULL, msm_timer_match); |
240 | if (!np) { | 239 | if (!np) { |
241 | pr_err("Can't find GPT DT node\n"); | 240 | pr_err("Can't find msm timer DT node\n"); |
242 | return; | 241 | return; |
243 | } | 242 | } |
244 | 243 | ||
245 | event_base = of_iomap(np, 0); | 244 | base = of_iomap(np, 0); |
246 | if (!event_base) { | 245 | if (!base) { |
247 | pr_err("Failed to map event base\n"); | 246 | pr_err("Failed to map event base\n"); |
248 | return; | 247 | return; |
249 | } | 248 | } |
250 | 249 | ||
251 | irq = irq_of_parse_and_map(np, 0); | 250 | /* We use GPT0 for the clockevent */ |
251 | irq = irq_of_parse_and_map(np, 1); | ||
252 | if (irq <= 0) { | 252 | if (irq <= 0) { |
253 | pr_err("Can't get irq\n"); | 253 | pr_err("Can't get irq\n"); |
254 | return; | 254 | return; |
255 | } | 255 | } |
256 | of_node_put(np); | ||
257 | |||
258 | np = of_find_matching_node(NULL, msm_dgt_match); | ||
259 | if (!np) { | ||
260 | pr_err("Can't find DGT DT node\n"); | ||
261 | return; | ||
262 | } | ||
263 | 256 | ||
257 | /* We use CPU0's DGT for the clocksource */ | ||
264 | if (of_property_read_u32(np, "cpu-offset", &percpu_offset)) | 258 | if (of_property_read_u32(np, "cpu-offset", &percpu_offset)) |
265 | percpu_offset = 0; | 259 | percpu_offset = 0; |
266 | 260 | ||
@@ -269,45 +263,43 @@ void __init msm_dt_timer_init(void) | |||
269 | return; | 263 | return; |
270 | } | 264 | } |
271 | 265 | ||
272 | source_base = ioremap(res.start + percpu_offset, resource_size(&res)); | 266 | cpu0_base = ioremap(res.start + percpu_offset, resource_size(&res)); |
273 | if (!source_base) { | 267 | if (!cpu0_base) { |
274 | pr_err("Failed to map source base\n"); | 268 | pr_err("Failed to map source base\n"); |
275 | return; | 269 | return; |
276 | } | 270 | } |
277 | 271 | ||
278 | if (!of_address_to_resource(np, 1, &res)) { | ||
279 | dgt_clk_ctl = ioremap(res.start + percpu_offset, | ||
280 | resource_size(&res)); | ||
281 | if (!dgt_clk_ctl) { | ||
282 | pr_err("Failed to map DGT control base\n"); | ||
283 | return; | ||
284 | } | ||
285 | writel_relaxed(DGT_CLK_CTL_DIV_4, dgt_clk_ctl); | ||
286 | iounmap(dgt_clk_ctl); | ||
287 | } | ||
288 | |||
289 | if (of_property_read_u32(np, "clock-frequency", &freq)) { | 272 | if (of_property_read_u32(np, "clock-frequency", &freq)) { |
290 | pr_err("Unknown frequency\n"); | 273 | pr_err("Unknown frequency\n"); |
291 | return; | 274 | return; |
292 | } | 275 | } |
293 | of_node_put(np); | 276 | of_node_put(np); |
294 | 277 | ||
278 | event_base = base + 0x4; | ||
279 | sts_base = base + 0x88; | ||
280 | source_base = cpu0_base + 0x24; | ||
281 | freq /= 4; | ||
282 | writel_relaxed(DGT_CLK_CTL_DIV_4, source_base + DGT_CLK_CTL); | ||
283 | |||
295 | msm_timer_init(freq, 32, irq, !!percpu_offset); | 284 | msm_timer_init(freq, 32, irq, !!percpu_offset); |
296 | } | 285 | } |
297 | #endif | 286 | #endif |
298 | 287 | ||
299 | static int __init msm_timer_map(phys_addr_t event, phys_addr_t source) | 288 | static int __init msm_timer_map(phys_addr_t addr, u32 event, u32 source, |
289 | u32 sts) | ||
300 | { | 290 | { |
301 | event_base = ioremap(event, SZ_64); | 291 | void __iomem *base; |
302 | if (!event_base) { | 292 | |
303 | pr_err("Failed to map event base\n"); | 293 | base = ioremap(addr, SZ_256); |
304 | return 1; | 294 | if (!base) { |
305 | } | 295 | pr_err("Failed to map timer base\n"); |
306 | source_base = ioremap(source, SZ_64); | 296 | return -ENOMEM; |
307 | if (!source_base) { | ||
308 | pr_err("Failed to map source base\n"); | ||
309 | return 1; | ||
310 | } | 297 | } |
298 | event_base = base + event; | ||
299 | source_base = base + source; | ||
300 | if (sts) | ||
301 | sts_base = base + sts; | ||
302 | |||
311 | return 0; | 303 | return 0; |
312 | } | 304 | } |
313 | 305 | ||
@@ -315,7 +307,7 @@ void __init msm7x01_timer_init(void) | |||
315 | { | 307 | { |
316 | struct clocksource *cs = &msm_clocksource; | 308 | struct clocksource *cs = &msm_clocksource; |
317 | 309 | ||
318 | if (msm_timer_map(0xc0100000, 0xc0100010)) | 310 | if (msm_timer_map(0xc0100000, 0x0, 0x10, 0x0)) |
319 | return; | 311 | return; |
320 | cs->read = msm_read_timer_count_shift; | 312 | cs->read = msm_read_timer_count_shift; |
321 | cs->mask = CLOCKSOURCE_MASK((32 - MSM_DGT_SHIFT)); | 313 | cs->mask = CLOCKSOURCE_MASK((32 - MSM_DGT_SHIFT)); |
@@ -326,14 +318,14 @@ void __init msm7x01_timer_init(void) | |||
326 | 318 | ||
327 | void __init msm7x30_timer_init(void) | 319 | void __init msm7x30_timer_init(void) |
328 | { | 320 | { |
329 | if (msm_timer_map(0xc0100004, 0xc0100024)) | 321 | if (msm_timer_map(0xc0100000, 0x4, 0x24, 0x80)) |
330 | return; | 322 | return; |
331 | msm_timer_init(24576000 / 4, 32, 1, false); | 323 | msm_timer_init(24576000 / 4, 32, 1, false); |
332 | } | 324 | } |
333 | 325 | ||
334 | void __init qsd8x50_timer_init(void) | 326 | void __init qsd8x50_timer_init(void) |
335 | { | 327 | { |
336 | if (msm_timer_map(0xAC100000, 0xAC100010)) | 328 | if (msm_timer_map(0xAC100000, 0x0, 0x10, 0x34)) |
337 | return; | 329 | return; |
338 | msm_timer_init(19200000 / 4, 32, 7, false); | 330 | msm_timer_init(19200000 / 4, 32, 7, false); |
339 | } | 331 | } |