aboutsummaryrefslogtreecommitdiffstats
path: root/arch/x86/kernel/step.c
diff options
context:
space:
mode:
authorRoland McGrath <roland@redhat.com>2008-01-30 07:30:54 -0500
committerIngo Molnar <mingo@elte.hu>2008-01-30 07:30:54 -0500
commit10faa81e102e2b7695f386812055cd2ef9e44b4c (patch)
treebd21d2094eacca5b07d5ee5977cb0ad57ebc5c02 /arch/x86/kernel/step.c
parent7e9916040b3020d0f36d68bb7512e3b80b623097 (diff)
x86: debugctlmsr arch_has_block_step
This implements user-mode step-until-branch on x86 using the BTF bit in MSR_IA32_DEBUGCTLMSR. It's just like single-step, only less so. Signed-off-by: Roland McGrath <roland@redhat.com> Signed-off-by: Ingo Molnar <mingo@elte.hu> Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
Diffstat (limited to 'arch/x86/kernel/step.c')
-rw-r--r--arch/x86/kernel/step.c64
1 files changed, 61 insertions, 3 deletions
diff --git a/arch/x86/kernel/step.c b/arch/x86/kernel/step.c
index 243bff650ca5..cf4b9dac4a05 100644
--- a/arch/x86/kernel/step.c
+++ b/arch/x86/kernel/step.c
@@ -107,7 +107,10 @@ static int is_setting_trap_flag(struct task_struct *child, struct pt_regs *regs)
107 return 0; 107 return 0;
108} 108}
109 109
110void user_enable_single_step(struct task_struct *child) 110/*
111 * Enable single-stepping. Return nonzero if user mode is not using TF itself.
112 */
113static int enable_single_step(struct task_struct *child)
111{ 114{
112 struct pt_regs *regs = task_pt_regs(child); 115 struct pt_regs *regs = task_pt_regs(child);
113 116
@@ -122,7 +125,7 @@ void user_enable_single_step(struct task_struct *child)
122 * If TF was already set, don't do anything else 125 * If TF was already set, don't do anything else
123 */ 126 */
124 if (regs->eflags & X86_EFLAGS_TF) 127 if (regs->eflags & X86_EFLAGS_TF)
125 return; 128 return 0;
126 129
127 /* Set TF on the kernel stack.. */ 130 /* Set TF on the kernel stack.. */
128 regs->eflags |= X86_EFLAGS_TF; 131 regs->eflags |= X86_EFLAGS_TF;
@@ -133,13 +136,68 @@ void user_enable_single_step(struct task_struct *child)
133 * won't clear it by hand later. 136 * won't clear it by hand later.
134 */ 137 */
135 if (is_setting_trap_flag(child, regs)) 138 if (is_setting_trap_flag(child, regs))
136 return; 139 return 0;
137 140
138 set_tsk_thread_flag(child, TIF_FORCED_TF); 141 set_tsk_thread_flag(child, TIF_FORCED_TF);
142
143 return 1;
144}
145
146/*
147 * Install this value in MSR_IA32_DEBUGCTLMSR whenever child is running.
148 */
149static void write_debugctlmsr(struct task_struct *child, unsigned long val)
150{
151 child->thread.debugctlmsr = val;
152
153 if (child != current)
154 return;
155
156#ifdef CONFIG_X86_64
157 wrmsrl(MSR_IA32_DEBUGCTLMSR, val);
158#else
159 wrmsr(MSR_IA32_DEBUGCTLMSR, val, 0);
160#endif
161}
162
163/*
164 * Enable single or block step.
165 */
166static void enable_step(struct task_struct *child, bool block)
167{
168 /*
169 * Make sure block stepping (BTF) is not enabled unless it should be.
170 * Note that we don't try to worry about any is_setting_trap_flag()
171 * instructions after the first when using block stepping.
172 * So noone should try to use debugger block stepping in a program
173 * that uses user-mode single stepping itself.
174 */
175 if (enable_single_step(child) && block) {
176 set_tsk_thread_flag(child, TIF_DEBUGCTLMSR);
177 write_debugctlmsr(child, DEBUGCTLMSR_BTF);
178 } else if (test_and_clear_tsk_thread_flag(child, TIF_DEBUGCTLMSR)) {
179 write_debugctlmsr(child, 0);
180 }
181}
182
183void user_enable_single_step(struct task_struct *child)
184{
185 enable_step(child, 0);
186}
187
188void user_enable_block_step(struct task_struct *child)
189{
190 enable_step(child, 1);
139} 191}
140 192
141void user_disable_single_step(struct task_struct *child) 193void user_disable_single_step(struct task_struct *child)
142{ 194{
195 /*
196 * Make sure block stepping (BTF) is disabled.
197 */
198 if (test_and_clear_tsk_thread_flag(child, TIF_DEBUGCTLMSR))
199 write_debugctlmsr(child, 0);
200
143 /* Always clear TIF_SINGLESTEP... */ 201 /* Always clear TIF_SINGLESTEP... */
144 clear_tsk_thread_flag(child, TIF_SINGLESTEP); 202 clear_tsk_thread_flag(child, TIF_SINGLESTEP);
145 203