diff options
author | Jan Beulich <jbeulich@novell.com> | 2008-03-10 09:11:17 -0400 |
---|---|---|
committer | Ingo Molnar <mingo@elte.hu> | 2008-04-17 11:40:58 -0400 |
commit | 5b0e508415a83989fe704b4718a1a214bc333ca7 (patch) | |
tree | bf73f1e6b944c190441413cedfb49b05e04ffe6b /arch/x86 | |
parent | f694010185c429629ad5a65245da08103e611852 (diff) |
x86: prevent unconditional writes to DebugCtl MSR
Otherwise, enabling (or better, subsequent disabling) of single
stepping would cause a kernel oops on CPUs not having this MSR.
The patch could have been added a conditional to the MSR write in
user_disable_single_step(), but centralizing the updates seems safer
and (looking forward) better manageable.
Signed-off-by: Jan Beulich <jbeulich@novell.com>
Cc: Markus Metzger <markus.t.metzger@intel.com>
Signed-off-by: Ingo Molnar <mingo@elte.hu>
Diffstat (limited to 'arch/x86')
-rw-r--r-- | arch/x86/kernel/kprobes.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/process_32.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/process_64.c | 4 | ||||
-rw-r--r-- | arch/x86/kernel/step.c | 2 |
4 files changed, 7 insertions, 7 deletions
diff --git a/arch/x86/kernel/kprobes.c b/arch/x86/kernel/kprobes.c index 34a591283f5d..1e3de7db9ad5 100644 --- a/arch/x86/kernel/kprobes.c +++ b/arch/x86/kernel/kprobes.c | |||
@@ -410,13 +410,13 @@ static void __kprobes set_current_kprobe(struct kprobe *p, struct pt_regs *regs, | |||
410 | static void __kprobes clear_btf(void) | 410 | static void __kprobes clear_btf(void) |
411 | { | 411 | { |
412 | if (test_thread_flag(TIF_DEBUGCTLMSR)) | 412 | if (test_thread_flag(TIF_DEBUGCTLMSR)) |
413 | wrmsrl(MSR_IA32_DEBUGCTLMSR, 0); | 413 | update_debugctlmsr(0); |
414 | } | 414 | } |
415 | 415 | ||
416 | static void __kprobes restore_btf(void) | 416 | static void __kprobes restore_btf(void) |
417 | { | 417 | { |
418 | if (test_thread_flag(TIF_DEBUGCTLMSR)) | 418 | if (test_thread_flag(TIF_DEBUGCTLMSR)) |
419 | wrmsrl(MSR_IA32_DEBUGCTLMSR, current->thread.debugctlmsr); | 419 | update_debugctlmsr(current->thread.debugctlmsr); |
420 | } | 420 | } |
421 | 421 | ||
422 | static void __kprobes prepare_singlestep(struct kprobe *p, struct pt_regs *regs) | 422 | static void __kprobes prepare_singlestep(struct kprobe *p, struct pt_regs *regs) |
diff --git a/arch/x86/kernel/process_32.c b/arch/x86/kernel/process_32.c index 9230ce060d09..ec05fb750dfc 100644 --- a/arch/x86/kernel/process_32.c +++ b/arch/x86/kernel/process_32.c | |||
@@ -564,12 +564,12 @@ __switch_to_xtra(struct task_struct *prev_p, struct task_struct *next_p, | |||
564 | /* we clear debugctl to make sure DS | 564 | /* we clear debugctl to make sure DS |
565 | * is not in use when we change it */ | 565 | * is not in use when we change it */ |
566 | debugctl = 0; | 566 | debugctl = 0; |
567 | wrmsrl(MSR_IA32_DEBUGCTLMSR, 0); | 567 | update_debugctlmsr(0); |
568 | wrmsr(MSR_IA32_DS_AREA, next->ds_area_msr, 0); | 568 | wrmsr(MSR_IA32_DS_AREA, next->ds_area_msr, 0); |
569 | } | 569 | } |
570 | 570 | ||
571 | if (next->debugctlmsr != debugctl) | 571 | if (next->debugctlmsr != debugctl) |
572 | wrmsr(MSR_IA32_DEBUGCTLMSR, next->debugctlmsr, 0); | 572 | update_debugctlmsr(next->debugctlmsr); |
573 | 573 | ||
574 | if (test_tsk_thread_flag(next_p, TIF_DEBUG)) { | 574 | if (test_tsk_thread_flag(next_p, TIF_DEBUG)) { |
575 | set_debugreg(next->debugreg0, 0); | 575 | set_debugreg(next->debugreg0, 0); |
diff --git a/arch/x86/kernel/process_64.c b/arch/x86/kernel/process_64.c index 1ffce14cff6e..4f40272474dd 100644 --- a/arch/x86/kernel/process_64.c +++ b/arch/x86/kernel/process_64.c | |||
@@ -563,12 +563,12 @@ static inline void __switch_to_xtra(struct task_struct *prev_p, | |||
563 | /* we clear debugctl to make sure DS | 563 | /* we clear debugctl to make sure DS |
564 | * is not in use when we change it */ | 564 | * is not in use when we change it */ |
565 | debugctl = 0; | 565 | debugctl = 0; |
566 | wrmsrl(MSR_IA32_DEBUGCTLMSR, 0); | 566 | update_debugctlmsr(0); |
567 | wrmsrl(MSR_IA32_DS_AREA, next->ds_area_msr); | 567 | wrmsrl(MSR_IA32_DS_AREA, next->ds_area_msr); |
568 | } | 568 | } |
569 | 569 | ||
570 | if (next->debugctlmsr != debugctl) | 570 | if (next->debugctlmsr != debugctl) |
571 | wrmsrl(MSR_IA32_DEBUGCTLMSR, next->debugctlmsr); | 571 | update_debugctlmsr(next->debugctlmsr); |
572 | 572 | ||
573 | if (test_tsk_thread_flag(next_p, TIF_DEBUG)) { | 573 | if (test_tsk_thread_flag(next_p, TIF_DEBUG)) { |
574 | loaddebug(next, 0); | 574 | loaddebug(next, 0); |
diff --git a/arch/x86/kernel/step.c b/arch/x86/kernel/step.c index 071ff4798236..92c20fee6781 100644 --- a/arch/x86/kernel/step.c +++ b/arch/x86/kernel/step.c | |||
@@ -148,7 +148,7 @@ static void write_debugctlmsr(struct task_struct *child, unsigned long val) | |||
148 | if (child != current) | 148 | if (child != current) |
149 | return; | 149 | return; |
150 | 150 | ||
151 | wrmsrl(MSR_IA32_DEBUGCTLMSR, val); | 151 | update_debugctlmsr(val); |
152 | } | 152 | } |
153 | 153 | ||
154 | /* | 154 | /* |