diff options
author | Mark Rutland <mark.rutland@arm.com> | 2013-06-03 16:33:53 -0400 |
---|---|---|
committer | Thomas Gleixner <tglx@linutronix.de> | 2013-06-24 16:27:37 -0400 |
commit | 064706514ec3fea740c2656e03c4f01f6a551ac4 (patch) | |
tree | 84b20f418dc6939aed90184fbeaafa02a3c147b5 | |
parent | 70e5975d3a04be5479a28eec4a2fb10f98ad2785 (diff) |
clocksource: Add generic dummy timer driver
Several architectures have a dummy timer driver tightly coupled with
their broadcast code to support machines without cpu-local timers (or
where there is a lack of driver support).
Since 12ad100046: "clockevents: Add generic timer broadcast function"
it's been possible to write broadcast-capable timer drivers decoupled
from the broadcast mechanism. We can use this functionality to implement
a generic dummy timer driver that can be shared by all architectures
with generic tick broadcast (ARCH_HAS_TICK_BROADCAST).
This patch implements a generic dummy timer using this facility.
[sboyd: Make percpu data static, use __this_cpu_ptr(), move to
early_initcall to properly register on each CPU, only
register if more than one CPU possible]
Signed-off-by: Mark Rutland <mark.rutland@arm.com>
Signed-off-by: Stephen Boyd <sboyd@codeaurora.org>
Acked-by: Marc Zyngier <marc.zyngier@arm.com>,
Cc: John Stultz <john.stultz@linaro.org>
Cc: Daniel Lezcano <daniel.lezcano@linaro.org>
Cc: linux-arm-kernel@lists.infradead.org
Link: http://lkml.kernel.org/r/1370291642-13259-3-git-send-email-sboyd@codeaurora.org
Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
-rw-r--r-- | drivers/clocksource/Makefile | 1 | ||||
-rw-r--r-- | drivers/clocksource/dummy_timer.c | 69 |
2 files changed, 70 insertions, 0 deletions
diff --git a/drivers/clocksource/Makefile b/drivers/clocksource/Makefile index 4853ea0f8fd5..9ba8b4d867e3 100644 --- a/drivers/clocksource/Makefile +++ b/drivers/clocksource/Makefile | |||
@@ -31,3 +31,4 @@ obj-$(CONFIG_VF_PIT_TIMER) += vf_pit_timer.o | |||
31 | 31 | ||
32 | obj-$(CONFIG_ARM_ARCH_TIMER) += arm_arch_timer.o | 32 | obj-$(CONFIG_ARM_ARCH_TIMER) += arm_arch_timer.o |
33 | obj-$(CONFIG_CLKSRC_METAG_GENERIC) += metag_generic.o | 33 | obj-$(CONFIG_CLKSRC_METAG_GENERIC) += metag_generic.o |
34 | obj-$(CONFIG_ARCH_HAS_TICK_BROADCAST) += dummy_timer.o | ||
diff --git a/drivers/clocksource/dummy_timer.c b/drivers/clocksource/dummy_timer.c new file mode 100644 index 000000000000..1f55f9620338 --- /dev/null +++ b/drivers/clocksource/dummy_timer.c | |||
@@ -0,0 +1,69 @@ | |||
1 | /* | ||
2 | * linux/drivers/clocksource/dummy_timer.c | ||
3 | * | ||
4 | * Copyright (C) 2013 ARM Ltd. | ||
5 | * All Rights Reserved | ||
6 | * | ||
7 | * This program is free software; you can redistribute it and/or modify | ||
8 | * it under the terms of the GNU General Public License version 2 as | ||
9 | * published by the Free Software Foundation. | ||
10 | */ | ||
11 | #include <linux/clockchips.h> | ||
12 | #include <linux/cpu.h> | ||
13 | #include <linux/init.h> | ||
14 | #include <linux/percpu.h> | ||
15 | #include <linux/cpumask.h> | ||
16 | |||
17 | static DEFINE_PER_CPU(struct clock_event_device, dummy_timer_evt); | ||
18 | |||
19 | static void dummy_timer_set_mode(enum clock_event_mode mode, | ||
20 | struct clock_event_device *evt) | ||
21 | { | ||
22 | /* | ||
23 | * Core clockevents code will call this when exchanging timer devices. | ||
24 | * We don't need to do anything here. | ||
25 | */ | ||
26 | } | ||
27 | |||
28 | static void __cpuinit dummy_timer_setup(void) | ||
29 | { | ||
30 | int cpu = smp_processor_id(); | ||
31 | struct clock_event_device *evt = __this_cpu_ptr(&dummy_timer_evt); | ||
32 | |||
33 | evt->name = "dummy_timer"; | ||
34 | evt->features = CLOCK_EVT_FEAT_PERIODIC | | ||
35 | CLOCK_EVT_FEAT_ONESHOT | | ||
36 | CLOCK_EVT_FEAT_DUMMY; | ||
37 | evt->rating = 100; | ||
38 | evt->set_mode = dummy_timer_set_mode; | ||
39 | evt->cpumask = cpumask_of(cpu); | ||
40 | |||
41 | clockevents_register_device(evt); | ||
42 | } | ||
43 | |||
44 | static int __cpuinit dummy_timer_cpu_notify(struct notifier_block *self, | ||
45 | unsigned long action, void *hcpu) | ||
46 | { | ||
47 | if ((action & ~CPU_TASKS_FROZEN) == CPU_STARTING) | ||
48 | dummy_timer_setup(); | ||
49 | |||
50 | return NOTIFY_OK; | ||
51 | } | ||
52 | |||
53 | static struct notifier_block dummy_timer_cpu_nb __cpuinitdata = { | ||
54 | .notifier_call = dummy_timer_cpu_notify, | ||
55 | }; | ||
56 | |||
57 | static int __init dummy_timer_register(void) | ||
58 | { | ||
59 | int err = register_cpu_notifier(&dummy_timer_cpu_nb); | ||
60 | if (err) | ||
61 | return err; | ||
62 | |||
63 | /* We won't get a call on the boot CPU, so register immediately */ | ||
64 | if (num_possible_cpus() > 1) | ||
65 | dummy_timer_setup(); | ||
66 | |||
67 | return 0; | ||
68 | } | ||
69 | early_initcall(dummy_timer_register); | ||