提交 4ac3dbec 编写于 作者: S Santosh Shilimkar 提交者: Robert Richter

oprofile: Fix the hang while taking the cpu offline

The kernel build with CONFIG_OPROFILE and CPU_HOTPLUG enabled.
The oprofile is initialised using system timer in absence of hardware
counters supports. Oprofile isn't started from userland.

In this setup while doing a CPU offline the kernel hangs in infinite
for loop inside lock_hrtimer_base() function

This happens because as part of oprofile_cpu_notify(, it tries to
stop an hrtimer which was never started. These per-cpu hrtimers
are started when the oprfile is started.
	echo 1	> /dev/oprofile/enable

This problem also existwhen the cpu is booted with maxcpus parameter
set. When bringing the remaining cpus online the timers are started
even if oprofile is not yet enabled.

This patch fix this issue by adding a state variable so that
these hrtimer start/stop is only attempted when oprofile is
started

For stable kernels v2.6.35.y and v2.6.36.y.
Reported-by: NJan Sebastien <s-jan@ti.com>
Tested-by: Nsricharan <r.sricharan@ti.com>
Signed-off-by: NSantosh Shilimkar <santosh.shilimkar@ti.com>
Cc: stable@kernel.org
Signed-off-by: NRobert Richter <robert.richter@amd.com>
上级 0b849ee8
...@@ -21,6 +21,7 @@ ...@@ -21,6 +21,7 @@
#include "oprof.h" #include "oprof.h"
static DEFINE_PER_CPU(struct hrtimer, oprofile_hrtimer); static DEFINE_PER_CPU(struct hrtimer, oprofile_hrtimer);
static int ctr_running;
static enum hrtimer_restart oprofile_hrtimer_notify(struct hrtimer *hrtimer) static enum hrtimer_restart oprofile_hrtimer_notify(struct hrtimer *hrtimer)
{ {
...@@ -33,6 +34,9 @@ static void __oprofile_hrtimer_start(void *unused) ...@@ -33,6 +34,9 @@ static void __oprofile_hrtimer_start(void *unused)
{ {
struct hrtimer *hrtimer = &__get_cpu_var(oprofile_hrtimer); struct hrtimer *hrtimer = &__get_cpu_var(oprofile_hrtimer);
if (!ctr_running)
return;
hrtimer_init(hrtimer, CLOCK_MONOTONIC, HRTIMER_MODE_REL); hrtimer_init(hrtimer, CLOCK_MONOTONIC, HRTIMER_MODE_REL);
hrtimer->function = oprofile_hrtimer_notify; hrtimer->function = oprofile_hrtimer_notify;
...@@ -42,7 +46,10 @@ static void __oprofile_hrtimer_start(void *unused) ...@@ -42,7 +46,10 @@ static void __oprofile_hrtimer_start(void *unused)
static int oprofile_hrtimer_start(void) static int oprofile_hrtimer_start(void)
{ {
get_online_cpus();
ctr_running = 1;
on_each_cpu(__oprofile_hrtimer_start, NULL, 1); on_each_cpu(__oprofile_hrtimer_start, NULL, 1);
put_online_cpus();
return 0; return 0;
} }
...@@ -50,6 +57,9 @@ static void __oprofile_hrtimer_stop(int cpu) ...@@ -50,6 +57,9 @@ static void __oprofile_hrtimer_stop(int cpu)
{ {
struct hrtimer *hrtimer = &per_cpu(oprofile_hrtimer, cpu); struct hrtimer *hrtimer = &per_cpu(oprofile_hrtimer, cpu);
if (!ctr_running)
return;
hrtimer_cancel(hrtimer); hrtimer_cancel(hrtimer);
} }
...@@ -57,8 +67,11 @@ static void oprofile_hrtimer_stop(void) ...@@ -57,8 +67,11 @@ static void oprofile_hrtimer_stop(void)
{ {
int cpu; int cpu;
get_online_cpus();
for_each_online_cpu(cpu) for_each_online_cpu(cpu)
__oprofile_hrtimer_stop(cpu); __oprofile_hrtimer_stop(cpu);
ctr_running = 0;
put_online_cpus();
} }
static int __cpuinit oprofile_cpu_notify(struct notifier_block *self, static int __cpuinit oprofile_cpu_notify(struct notifier_block *self,
......
Markdown is supported
0% .
You are about to add 0 people to the discussion. Proceed with caution.
先完成此消息的编辑!
想要评论请 注册