ptrace.h 6.9 KB
Newer Older
H
H. Peter Anvin 已提交
1 2
#ifndef _ASM_X86_PTRACE_H
#define _ASM_X86_PTRACE_H
T
Thomas Gleixner 已提交
3 4 5

#include <linux/compiler.h>	/* For __user */
#include <asm/ptrace-abi.h>
6
#include <asm/processor-flags.h>
T
Thomas Gleixner 已提交
7

8 9
#ifdef __KERNEL__
#include <asm/segment.h>
10
#include <asm/page_types.h>
11
#endif
12

T
Thomas Gleixner 已提交
13 14 15 16 17 18
#ifndef __ASSEMBLY__

#ifdef __i386__
/* this struct defines the way the registers are stored on the
   stack during a system call. */

19 20
#ifndef __KERNEL__

T
Thomas Gleixner 已提交
21 22 23 24 25 26 27 28 29 30 31
struct pt_regs {
	long ebx;
	long ecx;
	long edx;
	long esi;
	long edi;
	long ebp;
	long eax;
	int  xds;
	int  xes;
	int  xfs;
32
	int  xgs;
T
Thomas Gleixner 已提交
33 34 35 36 37 38 39 40
	long orig_eax;
	long eip;
	int  xcs;
	long eflags;
	long esp;
	int  xss;
};

41 42 43
#else /* __KERNEL__ */

struct pt_regs {
44 45 46 47 48 49
	unsigned long bx;
	unsigned long cx;
	unsigned long dx;
	unsigned long si;
	unsigned long di;
	unsigned long bp;
50
	unsigned long ax;
51 52 53
	unsigned long ds;
	unsigned long es;
	unsigned long fs;
54
	unsigned long gs;
55
	unsigned long orig_ax;
56 57 58 59 60
	unsigned long ip;
	unsigned long cs;
	unsigned long flags;
	unsigned long sp;
	unsigned long ss;
61
};
T
Thomas Gleixner 已提交
62 63 64 65 66

#endif /* __KERNEL__ */

#else /* __i386__ */

67 68
#ifndef __KERNEL__

T
Thomas Gleixner 已提交
69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96
struct pt_regs {
	unsigned long r15;
	unsigned long r14;
	unsigned long r13;
	unsigned long r12;
	unsigned long rbp;
	unsigned long rbx;
/* arguments: non interrupts/non tracing syscalls only save upto here*/
	unsigned long r11;
	unsigned long r10;
	unsigned long r9;
	unsigned long r8;
	unsigned long rax;
	unsigned long rcx;
	unsigned long rdx;
	unsigned long rsi;
	unsigned long rdi;
	unsigned long orig_rax;
/* end of arguments */
/* cpu exception frame or undefined */
	unsigned long rip;
	unsigned long cs;
	unsigned long eflags;
	unsigned long rsp;
	unsigned long ss;
/* top of stack page */
};

97 98 99 100 101 102 103 104 105 106 107 108 109 110 111 112 113 114 115 116 117 118 119 120 121 122 123 124 125
#else /* __KERNEL__ */

struct pt_regs {
	unsigned long r15;
	unsigned long r14;
	unsigned long r13;
	unsigned long r12;
	unsigned long bp;
	unsigned long bx;
/* arguments: non interrupts/non tracing syscalls only save upto here*/
	unsigned long r11;
	unsigned long r10;
	unsigned long r9;
	unsigned long r8;
	unsigned long ax;
	unsigned long cx;
	unsigned long dx;
	unsigned long si;
	unsigned long di;
	unsigned long orig_ax;
/* end of arguments */
/* cpu exception frame or undefined */
	unsigned long ip;
	unsigned long cs;
	unsigned long flags;
	unsigned long sp;
	unsigned long ss;
/* top of stack page */
};
T
Thomas Gleixner 已提交
126

H
Harvey Harrison 已提交
127 128
#endif /* __KERNEL__ */
#endif /* !__i386__ */
T
Thomas Gleixner 已提交
129

M
Markus Metzger 已提交
130

H
Harvey Harrison 已提交
131 132
#ifdef __KERNEL__

M
Markus Metzger 已提交
133
#include <linux/init.h>
T
Thomas Gleixner 已提交
134

M
Markus Metzger 已提交
135
struct cpuinfo_x86;
T
Thomas Gleixner 已提交
136 137
struct task_struct;

H
Harvey Harrison 已提交
138 139
extern unsigned long profile_pc(struct pt_regs *regs);

T
Thomas Gleixner 已提交
140
extern unsigned long
H
Harvey Harrison 已提交
141
convert_ip_to_linear(struct task_struct *child, struct pt_regs *regs);
142
extern void send_sigtrap(struct task_struct *tsk, struct pt_regs *regs,
143
			 int error_code, int si_code);
144 145
void signal_fault(struct pt_regs *regs, void __user *frame, char *where);

146 147 148
extern long syscall_trace_enter(struct pt_regs *);
extern void syscall_trace_leave(struct pt_regs *);

149 150 151 152
static inline unsigned long regs_return_value(struct pt_regs *regs)
{
	return regs->ax;
}
R
Roland McGrath 已提交
153

154 155 156 157 158 159 160 161 162 163 164 165 166 167 168 169 170 171 172
/*
 * user_mode_vm(regs) determines whether a register set came from user mode.
 * This is true if V8086 mode was enabled OR if the register set was from
 * protected mode with RPL-3 CS value.  This tricky test checks that with
 * one comparison.  Many places in the kernel can bypass this full check
 * if they have already ruled out V8086 mode, so user_mode(regs) can be used.
 */
static inline int user_mode(struct pt_regs *regs)
{
#ifdef CONFIG_X86_32
	return (regs->cs & SEGMENT_RPL_MASK) == USER_RPL;
#else
	return !!(regs->cs & 3);
#endif
}

static inline int user_mode_vm(struct pt_regs *regs)
{
#ifdef CONFIG_X86_32
173
	return ((regs->cs & SEGMENT_RPL_MASK) | (regs->flags & X86_VM_MASK)) >=
174
		USER_RPL;
175 176 177 178 179 180 181 182
#else
	return user_mode(regs);
#endif
}

static inline int v8086_mode(struct pt_regs *regs)
{
#ifdef CONFIG_X86_32
183
	return (regs->flags & X86_VM_MASK);
184 185 186 187 188
#else
	return 0;	/* No V86 mode support in long mode */
#endif
}

189 190
/*
 * X86_32 CPUs don't save ss and esp if the CPU is already in kernel mode
191 192
 * when it traps.  The previous stack will be directly underneath the saved
 * registers, and 'sp/ss' won't even have been saved. Thus the '&regs->sp'.
193 194 195
 *
 * This is valid only for kernel mode traps.
 */
196
static inline unsigned long kernel_stack_pointer(struct pt_regs *regs)
197 198
{
#ifdef CONFIG_X86_32
199
	return (unsigned long)(&regs->sp);
200 201 202 203 204 205 206 207 208 209 210 211 212 213 214
#else
	return regs->sp;
#endif
}

static inline unsigned long instruction_pointer(struct pt_regs *regs)
{
	return regs->ip;
}

static inline unsigned long frame_pointer(struct pt_regs *regs)
{
	return regs->bp;
}

R
Roland McGrath 已提交
215 216 217 218 219
static inline unsigned long user_stack_pointer(struct pt_regs *regs)
{
	return regs->sp;
}

220 221 222 223 224 225 226 227 228 229
/* Query offset/name of register from its name/offset */
extern int regs_query_register_offset(const char *name);
extern const char *regs_query_register_name(unsigned int offset);
#define MAX_REG_OFFSET (offsetof(struct pt_regs, ss))

/**
 * regs_get_register() - get register value from its offset
 * @regs:	pt_regs from which register value is gotten.
 * @offset:	offset number of the register.
 *
230 231
 * regs_get_register returns the value of a register. The @offset is the
 * offset of the register in struct pt_regs address which specified by @regs.
232 233 234 235 236 237 238 239 240 241 242 243 244 245 246
 * If @offset is bigger than MAX_REG_OFFSET, this returns 0.
 */
static inline unsigned long regs_get_register(struct pt_regs *regs,
					      unsigned int offset)
{
	if (unlikely(offset > MAX_REG_OFFSET))
		return 0;
	return *(unsigned long *)((unsigned long)regs + offset);
}

/**
 * regs_within_kernel_stack() - check the address in the stack
 * @regs:	pt_regs which contains kernel stack pointer.
 * @addr:	address which is checked.
 *
247
 * regs_within_kernel_stack() checks @addr is within the kernel stack page(s).
248 249 250 251 252 253 254 255 256 257 258 259 260 261 262
 * If @addr is within the kernel stack, it returns true. If not, returns false.
 */
static inline int regs_within_kernel_stack(struct pt_regs *regs,
					   unsigned long addr)
{
	return ((addr & ~(THREAD_SIZE - 1))  ==
		(kernel_stack_pointer(regs) & ~(THREAD_SIZE - 1)));
}

/**
 * regs_get_kernel_stack_nth() - get Nth entry of the stack
 * @regs:	pt_regs which contains kernel stack pointer.
 * @n:		stack entry number.
 *
 * regs_get_kernel_stack_nth() returns @n th entry of the kernel stack which
263
 * is specified by @regs. If the @n th entry is NOT in the kernel stack,
264 265 266 267 268 269 270 271 272 273 274 275 276
 * this returns 0.
 */
static inline unsigned long regs_get_kernel_stack_nth(struct pt_regs *regs,
						      unsigned int n)
{
	unsigned long *addr = (unsigned long *)kernel_stack_pointer(regs);
	addr += n;
	if (regs_within_kernel_stack(regs, (unsigned long)addr))
		return *addr;
	else
		return 0;
}

R
Roland McGrath 已提交
277
#define arch_has_single_step()	(1)
278 279 280 281 282 283
#ifdef CONFIG_X86_DEBUGCTLMSR
#define arch_has_block_step()	(1)
#else
#define arch_has_block_step()	(boot_cpu_data.x86 >= 6)
#endif

284 285
#define ARCH_HAS_USER_SINGLE_STEP_INFO

R
Roland McGrath 已提交
286 287 288 289 290 291 292 293
struct user_desc;
extern int do_get_thread_area(struct task_struct *p, int idx,
			      struct user_desc __user *info);
extern int do_set_thread_area(struct task_struct *p, int idx,
			      struct user_desc __user *info, int can_allocate);

#endif /* __KERNEL__ */

T
Thomas Gleixner 已提交
294 295
#endif /* !__ASSEMBLY__ */

H
H. Peter Anvin 已提交
296
#endif /* _ASM_X86_PTRACE_H */