diff options
| author | Russell King <rmk+kernel@arm.linux.org.uk> | 2013-02-04 09:51:49 -0500 |
|---|---|---|
| committer | Russell King <rmk+kernel@arm.linux.org.uk> | 2013-02-04 09:51:49 -0500 |
| commit | 2fb20e2e35cd1455d022ab7a553d7f1663ffbdeb (patch) | |
| tree | 4b3bb0f76c15264fe7d4cced91127f3d96ff31f5 /kernel/time | |
| parent | 9cb543124a2d31af42ce61a4c30765ecc8e5f1fa (diff) | |
| parent | 5b91ab0abc957145c3ff6be03eb9a3901797019f (diff) | |
Merge branch 'for-rmk/broadcast' of git://git.kernel.org/pub/scm/linux/kernel/git/will/linux into devel-stable
Diffstat (limited to 'kernel/time')
| -rw-r--r-- | kernel/time/Kconfig | 4 | ||||
| -rw-r--r-- | kernel/time/tick-broadcast.c | 30 |
2 files changed, 34 insertions, 0 deletions
diff --git a/kernel/time/Kconfig b/kernel/time/Kconfig index 8601f0db1261..b69692250af4 100644 --- a/kernel/time/Kconfig +++ b/kernel/time/Kconfig | |||
| @@ -38,6 +38,10 @@ config GENERIC_CLOCKEVENTS_BUILD | |||
| 38 | default y | 38 | default y |
| 39 | depends on GENERIC_CLOCKEVENTS | 39 | depends on GENERIC_CLOCKEVENTS |
| 40 | 40 | ||
| 41 | # Architecture can handle broadcast in a driver-agnostic way | ||
| 42 | config ARCH_HAS_TICK_BROADCAST | ||
| 43 | bool | ||
| 44 | |||
| 41 | # Clockevents broadcasting infrastructure | 45 | # Clockevents broadcasting infrastructure |
| 42 | config GENERIC_CLOCKEVENTS_BROADCAST | 46 | config GENERIC_CLOCKEVENTS_BROADCAST |
| 43 | bool | 47 | bool |
diff --git a/kernel/time/tick-broadcast.c b/kernel/time/tick-broadcast.c index f113755695e2..f726537d24eb 100644 --- a/kernel/time/tick-broadcast.c +++ b/kernel/time/tick-broadcast.c | |||
| @@ -18,6 +18,7 @@ | |||
| 18 | #include <linux/percpu.h> | 18 | #include <linux/percpu.h> |
| 19 | #include <linux/profile.h> | 19 | #include <linux/profile.h> |
| 20 | #include <linux/sched.h> | 20 | #include <linux/sched.h> |
| 21 | #include <linux/smp.h> | ||
| 21 | 22 | ||
| 22 | #include "tick-internal.h" | 23 | #include "tick-internal.h" |
| 23 | 24 | ||
| @@ -86,6 +87,11 @@ int tick_is_broadcast_device(struct clock_event_device *dev) | |||
| 86 | return (dev && tick_broadcast_device.evtdev == dev); | 87 | return (dev && tick_broadcast_device.evtdev == dev); |
| 87 | } | 88 | } |
| 88 | 89 | ||
| 90 | static void err_broadcast(const struct cpumask *mask) | ||
| 91 | { | ||
| 92 | pr_crit_once("Failed to broadcast timer tick. Some CPUs may be unresponsive.\n"); | ||
| 93 | } | ||
| 94 | |||
| 89 | /* | 95 | /* |
| 90 | * Check, if the device is disfunctional and a place holder, which | 96 | * Check, if the device is disfunctional and a place holder, which |
| 91 | * needs to be handled by the broadcast device. | 97 | * needs to be handled by the broadcast device. |
| @@ -105,6 +111,13 @@ int tick_device_uses_broadcast(struct clock_event_device *dev, int cpu) | |||
| 105 | */ | 111 | */ |
| 106 | if (!tick_device_is_functional(dev)) { | 112 | if (!tick_device_is_functional(dev)) { |
| 107 | dev->event_handler = tick_handle_periodic; | 113 | dev->event_handler = tick_handle_periodic; |
| 114 | if (!dev->broadcast) | ||
| 115 | dev->broadcast = tick_broadcast; | ||
| 116 | if (!dev->broadcast) { | ||
| 117 | pr_warn_once("%s depends on broadcast, but no broadcast function available\n", | ||
| 118 | dev->name); | ||
| 119 | dev->broadcast = err_broadcast; | ||
| 120 | } | ||
| 108 | cpumask_set_cpu(cpu, tick_get_broadcast_mask()); | 121 | cpumask_set_cpu(cpu, tick_get_broadcast_mask()); |
| 109 | tick_broadcast_start_periodic(tick_broadcast_device.evtdev); | 122 | tick_broadcast_start_periodic(tick_broadcast_device.evtdev); |
| 110 | ret = 1; | 123 | ret = 1; |
| @@ -125,6 +138,23 @@ int tick_device_uses_broadcast(struct clock_event_device *dev, int cpu) | |||
| 125 | return ret; | 138 | return ret; |
| 126 | } | 139 | } |
| 127 | 140 | ||
| 141 | #ifdef CONFIG_GENERIC_CLOCKEVENTS_BROADCAST | ||
| 142 | int tick_receive_broadcast(void) | ||
| 143 | { | ||
| 144 | struct tick_device *td = this_cpu_ptr(&tick_cpu_device); | ||
| 145 | struct clock_event_device *evt = td->evtdev; | ||
| 146 | |||
| 147 | if (!evt) | ||
| 148 | return -ENODEV; | ||
| 149 | |||
| 150 | if (!evt->event_handler) | ||
| 151 | return -EINVAL; | ||
| 152 | |||
| 153 | evt->event_handler(evt); | ||
| 154 | return 0; | ||
| 155 | } | ||
| 156 | #endif | ||
| 157 | |||
| 128 | /* | 158 | /* |
| 129 | * Broadcast the event to the cpus, which are set in the mask (mangled). | 159 | * Broadcast the event to the cpus, which are set in the mask (mangled). |
| 130 | */ | 160 | */ |
