2 * SuperH process tracing
4 * Copyright (C) 1999, 2000 Kaz Kojima & Niibe Yutaka
5 * Copyright (C) 2002 - 2008 Paul Mundt
7 * Audit support by Yuichi Nakamura <ynakam@hitachisoft.jp>
9 * This file is subject to the terms and conditions of the GNU General Public
10 * License. See the file "COPYING" in the main directory of this archive
13 #include <linux/kernel.h>
14 #include <linux/sched.h>
16 #include <linux/smp.h>
17 #include <linux/errno.h>
18 #include <linux/ptrace.h>
19 #include <linux/user.h>
20 #include <linux/slab.h>
21 #include <linux/security.h>
22 #include <linux/signal.h>
24 #include <linux/audit.h>
25 #include <linux/seccomp.h>
26 #include <linux/tracehook.h>
27 #include <linux/elf.h>
28 #include <linux/regset.h>
29 #include <asm/uaccess.h>
30 #include <asm/pgtable.h>
31 #include <asm/system.h>
32 #include <asm/processor.h>
33 #include <asm/mmu_context.h>
34 #include <asm/syscalls.h>
38 * This routine will get a word off of the process kernel stack.
40 static inline int get_stack_long(struct task_struct
*task
, int offset
)
44 stack
= (unsigned char *)task_pt_regs(task
);
46 return (*((int *)stack
));
50 * This routine will put a word on the process kernel stack.
52 static inline int put_stack_long(struct task_struct
*task
, int offset
,
57 stack
= (unsigned char *)task_pt_regs(task
);
59 *(unsigned long *) stack
= data
;
63 void user_enable_single_step(struct task_struct
*child
)
65 /* Next scheduling will set up UBC */
66 if (child
->thread
.ubc_pc
== 0)
69 child
->thread
.ubc_pc
= get_stack_long(child
,
70 offsetof(struct pt_regs
, pc
));
72 set_tsk_thread_flag(child
, TIF_SINGLESTEP
);
75 void user_disable_single_step(struct task_struct
*child
)
77 clear_tsk_thread_flag(child
, TIF_SINGLESTEP
);
80 * Ensure the UBC is not programmed at the next context switch.
82 * Normally this is not needed but there are sequences such as
83 * singlestep, signal delivery, and continue that leave the
84 * ubc_pc non-zero leading to spurious SIGTRAPs.
86 if (child
->thread
.ubc_pc
!= 0) {
88 child
->thread
.ubc_pc
= 0;
93 * Called by kernel/ptrace.c when detaching..
95 * Make sure single step bits etc are not set.
97 void ptrace_disable(struct task_struct
*child
)
99 user_disable_single_step(child
);
102 static int genregs_get(struct task_struct
*target
,
103 const struct user_regset
*regset
,
104 unsigned int pos
, unsigned int count
,
105 void *kbuf
, void __user
*ubuf
)
107 const struct pt_regs
*regs
= task_pt_regs(target
);
110 ret
= user_regset_copyout(&pos
, &count
, &kbuf
, &ubuf
,
112 0, 16 * sizeof(unsigned long));
114 /* PC, PR, SR, GBR, MACH, MACL, TRA */
115 ret
= user_regset_copyout(&pos
, &count
, &kbuf
, &ubuf
,
117 offsetof(struct pt_regs
, pc
),
118 sizeof(struct pt_regs
));
120 ret
= user_regset_copyout_zero(&pos
, &count
, &kbuf
, &ubuf
,
121 sizeof(struct pt_regs
), -1);
126 static int genregs_set(struct task_struct
*target
,
127 const struct user_regset
*regset
,
128 unsigned int pos
, unsigned int count
,
129 const void *kbuf
, const void __user
*ubuf
)
131 struct pt_regs
*regs
= task_pt_regs(target
);
134 ret
= user_regset_copyin(&pos
, &count
, &kbuf
, &ubuf
,
136 0, 16 * sizeof(unsigned long));
137 if (!ret
&& count
> 0)
138 ret
= user_regset_copyin(&pos
, &count
, &kbuf
, &ubuf
,
140 offsetof(struct pt_regs
, pc
),
141 sizeof(struct pt_regs
));
143 ret
= user_regset_copyin_ignore(&pos
, &count
, &kbuf
, &ubuf
,
144 sizeof(struct pt_regs
), -1);
150 int fpregs_get(struct task_struct
*target
,
151 const struct user_regset
*regset
,
152 unsigned int pos
, unsigned int count
,
153 void *kbuf
, void __user
*ubuf
)
157 ret
= init_fpu(target
);
161 if ((boot_cpu_data
.flags
& CPU_HAS_FPU
))
162 return user_regset_copyout(&pos
, &count
, &kbuf
, &ubuf
,
163 &target
->thread
.fpu
.hard
, 0, -1);
165 return user_regset_copyout(&pos
, &count
, &kbuf
, &ubuf
,
166 &target
->thread
.fpu
.soft
, 0, -1);
169 static int fpregs_set(struct task_struct
*target
,
170 const struct user_regset
*regset
,
171 unsigned int pos
, unsigned int count
,
172 const void *kbuf
, const void __user
*ubuf
)
176 ret
= init_fpu(target
);
180 set_stopped_child_used_math(target
);
182 if ((boot_cpu_data
.flags
& CPU_HAS_FPU
))
183 return user_regset_copyin(&pos
, &count
, &kbuf
, &ubuf
,
184 &target
->thread
.fpu
.hard
, 0, -1);
186 return user_regset_copyin(&pos
, &count
, &kbuf
, &ubuf
,
187 &target
->thread
.fpu
.soft
, 0, -1);
190 static int fpregs_active(struct task_struct
*target
,
191 const struct user_regset
*regset
)
193 return tsk_used_math(target
) ? regset
->n
: 0;
198 static int dspregs_get(struct task_struct
*target
,
199 const struct user_regset
*regset
,
200 unsigned int pos
, unsigned int count
,
201 void *kbuf
, void __user
*ubuf
)
203 const struct pt_dspregs
*regs
= task_pt_dspregs(target
);
206 ret
= user_regset_copyout(&pos
, &count
, &kbuf
, &ubuf
, regs
,
207 0, sizeof(struct pt_dspregs
));
209 ret
= user_regset_copyout_zero(&pos
, &count
, &kbuf
, &ubuf
,
210 sizeof(struct pt_dspregs
), -1);
215 static int dspregs_set(struct task_struct
*target
,
216 const struct user_regset
*regset
,
217 unsigned int pos
, unsigned int count
,
218 const void *kbuf
, const void __user
*ubuf
)
220 struct pt_dspregs
*regs
= task_pt_dspregs(target
);
223 ret
= user_regset_copyin(&pos
, &count
, &kbuf
, &ubuf
, regs
,
224 0, sizeof(struct pt_dspregs
));
226 ret
= user_regset_copyin_ignore(&pos
, &count
, &kbuf
, &ubuf
,
227 sizeof(struct pt_dspregs
), -1);
232 static int dspregs_active(struct task_struct
*target
,
233 const struct user_regset
*regset
)
235 struct pt_regs
*regs
= task_pt_regs(target
);
237 return regs
->sr
& SR_DSP
? regset
->n
: 0;
242 * These are our native regset flavours.
254 static const struct user_regset sh_regsets
[] = {
258 * PC, PR, SR, GBR, MACH, MACL, TRA
261 .core_note_type
= NT_PRSTATUS
,
263 .size
= sizeof(long),
264 .align
= sizeof(long),
271 .core_note_type
= NT_PRFPREG
,
272 .n
= sizeof(struct user_fpu_struct
) / sizeof(long),
273 .size
= sizeof(long),
274 .align
= sizeof(long),
277 .active
= fpregs_active
,
283 .n
= sizeof(struct pt_dspregs
) / sizeof(long),
284 .size
= sizeof(long),
285 .align
= sizeof(long),
288 .active
= dspregs_active
,
293 static const struct user_regset_view user_sh_native_view
= {
296 .regsets
= sh_regsets
,
297 .n
= ARRAY_SIZE(sh_regsets
),
300 const struct user_regset_view
*task_user_regset_view(struct task_struct
*task
)
302 return &user_sh_native_view
;
305 long arch_ptrace(struct task_struct
*child
, long request
, long addr
, long data
)
307 struct user
* dummy
= NULL
;
308 unsigned long __user
*datap
= (unsigned long __user
*)data
;
312 /* read the word at location addr in the USER area. */
313 case PTRACE_PEEKUSR
: {
317 if ((addr
& 3) || addr
< 0 ||
318 addr
> sizeof(struct user
) - 3)
321 if (addr
< sizeof(struct pt_regs
))
322 tmp
= get_stack_long(child
, addr
);
323 else if (addr
>= (long) &dummy
->fpu
&&
324 addr
< (long) &dummy
->u_fpvalid
) {
325 if (!tsk_used_math(child
)) {
326 if (addr
== (long)&dummy
->fpu
.fpscr
)
331 tmp
= ((long *)&child
->thread
.fpu
)
332 [(addr
- (long)&dummy
->fpu
) >> 2];
333 } else if (addr
== (long) &dummy
->u_fpvalid
)
334 tmp
= !!tsk_used_math(child
);
337 ret
= put_user(tmp
, datap
);
341 case PTRACE_POKEUSR
: /* write the word at location addr in the USER area */
343 if ((addr
& 3) || addr
< 0 ||
344 addr
> sizeof(struct user
) - 3)
347 if (addr
< sizeof(struct pt_regs
))
348 ret
= put_stack_long(child
, addr
, data
);
349 else if (addr
>= (long) &dummy
->fpu
&&
350 addr
< (long) &dummy
->u_fpvalid
) {
351 set_stopped_child_used_math(child
);
352 ((long *)&child
->thread
.fpu
)
353 [(addr
- (long)&dummy
->fpu
) >> 2] = data
;
355 } else if (addr
== (long) &dummy
->u_fpvalid
) {
356 conditional_stopped_child_used_math(data
, child
);
362 return copy_regset_to_user(child
, &user_sh_native_view
,
364 0, sizeof(struct pt_regs
),
365 (void __user
*)data
);
367 return copy_regset_from_user(child
, &user_sh_native_view
,
369 0, sizeof(struct pt_regs
),
370 (const void __user
*)data
);
372 case PTRACE_GETFPREGS
:
373 return copy_regset_to_user(child
, &user_sh_native_view
,
375 0, sizeof(struct user_fpu_struct
),
376 (void __user
*)data
);
377 case PTRACE_SETFPREGS
:
378 return copy_regset_from_user(child
, &user_sh_native_view
,
380 0, sizeof(struct user_fpu_struct
),
381 (const void __user
*)data
);
384 case PTRACE_GETDSPREGS
:
385 return copy_regset_to_user(child
, &user_sh_native_view
,
387 0, sizeof(struct pt_dspregs
),
388 (void __user
*)data
);
389 case PTRACE_SETDSPREGS
:
390 return copy_regset_from_user(child
, &user_sh_native_view
,
392 0, sizeof(struct pt_dspregs
),
393 (const void __user
*)data
);
395 #ifdef CONFIG_BINFMT_ELF_FDPIC
396 case PTRACE_GETFDPIC
: {
397 unsigned long tmp
= 0;
400 case PTRACE_GETFDPIC_EXEC
:
401 tmp
= child
->mm
->context
.exec_fdpic_loadmap
;
403 case PTRACE_GETFDPIC_INTERP
:
404 tmp
= child
->mm
->context
.interp_fdpic_loadmap
;
411 if (put_user(tmp
, datap
)) {
419 ret
= ptrace_request(child
, request
, addr
, data
);
426 static inline int audit_arch(void)
430 #ifdef CONFIG_CPU_LITTLE_ENDIAN
431 arch
|= __AUDIT_ARCH_LE
;
437 asmlinkage
long do_syscall_trace_enter(struct pt_regs
*regs
)
441 secure_computing(regs
->regs
[0]);
443 if (test_thread_flag(TIF_SYSCALL_TRACE
) &&
444 tracehook_report_syscall_entry(regs
))
446 * Tracing decided this syscall should not happen.
447 * We'll return a bogus call number to get an ENOSYS
448 * error, but leave the original number in regs->regs[0].
452 if (unlikely(current
->audit_context
))
453 audit_syscall_entry(audit_arch(), regs
->regs
[3],
454 regs
->regs
[4], regs
->regs
[5],
455 regs
->regs
[6], regs
->regs
[7]);
457 return ret
?: regs
->regs
[0];
460 asmlinkage
void do_syscall_trace_leave(struct pt_regs
*regs
)
464 if (unlikely(current
->audit_context
))
465 audit_syscall_exit(AUDITSC_RESULT(regs
->regs
[0]),
468 step
= test_thread_flag(TIF_SINGLESTEP
);
469 if (step
|| test_thread_flag(TIF_SYSCALL_TRACE
))
470 tracehook_report_syscall_exit(regs
, step
);