Merge r202882 from amd64/i386:

For PT_TO_SCE stop that stops the ptraced process upon syscall entry,
syscall arguments are collected before ptracestop() is called. As a
consequence, debugger cannot modify syscall or its arguments.

In syscall(), reread syscall number and arguments after ptracestop(),
if debugger modified anything in the process environment. Since procfs
stopevent requires number of syscall arguments in p_xstat, this cannot
be solved by moving stop/trace point before argument fetching.

Move the code to read arguments into separate function
fetch_syscall_args() to avoid code duplication. Note that ktrace point
for modified syscall is intentionally recorded twice, once with original
arguments, and second time with the arguments set by debugger.

PT_TO_SCX stop is executed after cpu_syscall_set_retval() already.

Reviewed by:	kib
This commit is contained in:
Marius Strobl 2010-01-23 22:11:18 +00:00
parent ce3853023f
commit 29b6820f87

View File

@ -93,9 +93,18 @@ __FBSDID("$FreeBSD$");
#include <machine/tsb.h> #include <machine/tsb.h>
#include <machine/watch.h> #include <machine/watch.h>
struct syscall_args {
u_long code;
struct sysent *callp;
register_t args[8];
register_t *argp;
int narg;
};
void trap(struct trapframe *tf); void trap(struct trapframe *tf);
void syscall(struct trapframe *tf); void syscall(struct trapframe *tf);
static int fetch_syscall_args(struct thread *td, struct syscall_args *sa);
static int trap_pfault(struct thread *td, struct trapframe *tf); static int trap_pfault(struct thread *td, struct trapframe *tf);
extern char copy_fault[]; extern char copy_fault[];
@ -522,38 +531,93 @@ trap_pfault(struct thread *td, struct trapframe *tf)
/* Maximum number of arguments that can be passed via the out registers. */ /* Maximum number of arguments that can be passed via the out registers. */
#define REG_MAXARGS 6 #define REG_MAXARGS 6
static int
fetch_syscall_args(struct thread *td, struct syscall_args *sa)
{
struct trapframe *tf;
struct proc *p;
int reg;
int regcnt;
int error;
p = td->td_proc;
tf = td->td_frame;
reg = 0;
regcnt = REG_MAXARGS;
sa->code = tf->tf_global[1];
if (p->p_sysent->sv_prepsyscall) {
#if 0
(*p->p_sysent->sv_prepsyscall)(tf, sa->args, &sa->code,
&params);
#endif
} else if (sa->code == SYS_syscall || sa->code == SYS___syscall) {
sa->code = tf->tf_out[reg++];
regcnt--;
}
if (p->p_sysent->sv_mask)
sa->code &= p->p_sysent->sv_mask;
if (sa->code >= p->p_sysent->sv_size)
sa->callp = &p->p_sysent->sv_table[0];
else
sa->callp = &p->p_sysent->sv_table[sa->code];
sa->narg = sa->callp->sy_narg;
KASSERT(sa->narg <= sizeof(sa->args) / sizeof(sa->args[0]),
("Too many syscall arguments!"));
error = 0;
sa->argp = sa->args;
bcopy(&tf->tf_out[reg], sa->args, sizeof(sa->args[0]) * regcnt);
if (sa->narg > regcnt)
error = copyin((void *)(tf->tf_out[6] + SPOFF +
offsetof(struct frame, fr_pad[6])), &sa->args[regcnt],
(sa->narg - regcnt) * sizeof(sa->args[0]));
/*
* This may result in two records if debugger modified
* registers or memory during sleep at stop/ptrace point.
*/
#ifdef KTRACE
if (KTRPOINT(td, KTR_SYSCALL))
ktrsyscall(sa->code, sa->narg, sa->argp);
#endif
return (error);
}
/* /*
* Syscall handler. The arguments to the syscall are passed in the o registers * Syscall handler
* by the caller, and are saved in the trap frame. The syscall number is passed * The arguments to the syscall are passed in the out registers by the caller,
* in %g1 (and also saved in the trap frame). * and are saved in the trap frame. The syscall number is passed in %g1 (and
* also saved in the trap frame).
*/ */
void void
syscall(struct trapframe *tf) syscall(struct trapframe *tf)
{ {
struct sysent *callp; struct syscall_args sa;
struct thread *td; struct thread *td;
register_t args[8];
register_t *argp;
struct proc *p; struct proc *p;
u_long code;
int reg;
int regcnt;
int narg;
int error; int error;
td = curthread; td = curthread;
KASSERT(td != NULL, ("trap: curthread NULL")); KASSERT(td != NULL, ("trap: curthread NULL"));
KASSERT(td->td_proc != NULL, ("trap: curproc NULL")); KASSERT(td->td_proc != NULL, ("trap: curproc NULL"));
p = td->td_proc;
PCPU_INC(cnt.v_syscall); PCPU_INC(cnt.v_syscall);
p = td->td_proc;
td->td_syscalls++;
td->td_pticks = 0; td->td_pticks = 0;
td->td_frame = tf; td->td_frame = tf;
if (td->td_ucred != p->p_ucred) if (td->td_ucred != p->p_ucred)
cred_update_thread(td); cred_update_thread(td);
code = tf->tf_global[1]; if ((p->p_flag & P_TRACED) != 0) {
PROC_LOCK(p);
td->td_dbgflags &= ~TDB_USERWR;
PROC_UNLOCK(p);
}
/* /*
* For syscalls, we don't want to retry the faulting instruction * For syscalls, we don't want to retry the faulting instruction
@ -562,97 +626,68 @@ syscall(struct trapframe *tf)
td->td_pcb->pcb_tpc = tf->tf_tpc; td->td_pcb->pcb_tpc = tf->tf_tpc;
TF_DONE(tf); TF_DONE(tf);
reg = 0; error = fetch_syscall_args(td, &sa);
regcnt = REG_MAXARGS;
if (p->p_sysent->sv_prepsyscall) {
/*
* The prep code is MP aware.
*/
#if 0
(*p->p_sysent->sv_prepsyscall)(tf, args, &code, &params);
#endif
} else if (code == SYS_syscall || code == SYS___syscall) {
code = tf->tf_out[reg++];
regcnt--;
}
if (p->p_sysent->sv_mask)
code &= p->p_sysent->sv_mask;
if (code >= p->p_sysent->sv_size)
callp = &p->p_sysent->sv_table[0];
else
callp = &p->p_sysent->sv_table[code];
narg = callp->sy_narg;
KASSERT(narg <= sizeof(args) / sizeof(args[0]),
("Too many syscall arguments!"));
error = 0;
argp = args;
bcopy(&tf->tf_out[reg], args, sizeof(args[0]) * regcnt);
if (narg > regcnt)
error = copyin((void *)(tf->tf_out[6] + SPOFF +
offsetof(struct frame, fr_pad[6])),
&args[regcnt], (narg - regcnt) * sizeof(args[0]));
CTR5(KTR_SYSC, "syscall: td=%p %s(%#lx, %#lx, %#lx)", td, CTR5(KTR_SYSC, "syscall: td=%p %s(%#lx, %#lx, %#lx)", td,
syscallnames[code], argp[0], argp[1], argp[2]); syscallnames[sa.code], sa.argp[0], sa.argp[1], sa.argp[2]);
#ifdef KTRACE
if (KTRPOINT(td, KTR_SYSCALL))
ktrsyscall(code, narg, argp);
#endif
td->td_syscalls++;
if (error == 0) { if (error == 0) {
td->td_retval[0] = 0; td->td_retval[0] = 0;
td->td_retval[1] = 0; td->td_retval[1] = 0;
STOPEVENT(p, S_SCE, narg); /* MP aware */ STOPEVENT(p, S_SCE, sa.narg);
PTRACESTOP_SC(p, td, S_PT_SCE); PTRACESTOP_SC(p, td, S_PT_SCE);
if ((td->td_dbgflags & TDB_USERWR) != 0) {
/*
* Reread syscall number and arguments if
* debugger modified registers or memory.
*/
error = fetch_syscall_args(td, &sa);
if (error != 0)
goto retval;
td->td_retval[1] = 0;
}
AUDIT_SYSCALL_ENTER(code, td); AUDIT_SYSCALL_ENTER(sa.code, td);
error = (*callp->sy_call)(td, argp); error = (*sa.callp->sy_call)(td, sa.argp);
AUDIT_SYSCALL_EXIT(error, td); AUDIT_SYSCALL_EXIT(error, td);
CTR5(KTR_SYSC, "syscall: p=%p error=%d %s return %#lx %#lx ", p, CTR5(KTR_SYSC, "syscall: p=%p error=%d %s return %#lx %#lx",
error, syscallnames[code], td->td_retval[0], p, error, syscallnames[sa.code], td->td_retval[0],
td->td_retval[1]); td->td_retval[1]);
} }
retval:
cpu_set_syscall_retval(td, error); cpu_set_syscall_retval(td, error);
/* /*
* Check for misbehavior. * Check for misbehavior.
*/ */
WITNESS_WARN(WARN_PANIC, NULL, "System call %s returning", WITNESS_WARN(WARN_PANIC, NULL, "System call %s returning",
(code >= 0 && code < SYS_MAXSYSCALL) ? syscallnames[code] : "???"); (sa.code >= 0 && sa.code < SYS_MAXSYSCALL) ?
syscallnames[sa.code] : "???");
KASSERT(td->td_critnest == 0, KASSERT(td->td_critnest == 0,
("System call %s returning in a critical section", ("System call %s returning in a critical section",
(code >= 0 && code < SYS_MAXSYSCALL) ? syscallnames[code] : "???")); (sa.code >= 0 && sa.code < SYS_MAXSYSCALL) ?
syscallnames[sa.code] : "???"));
KASSERT(td->td_locks == 0, KASSERT(td->td_locks == 0,
("System call %s returning with %d locks held", ("System call %s returning with %d locks held",
(code >= 0 && code < SYS_MAXSYSCALL) ? syscallnames[code] : "???", (sa.code >= 0 && sa.code < SYS_MAXSYSCALL) ?
td->td_locks)); syscallnames[sa.code] : "???", td->td_locks));
/* /*
* Handle reschedule and other end-of-syscall issues * Handle reschedule and other end-of-syscall issues.
*/ */
userret(td, tf); userret(td, tf);
#ifdef KTRACE #ifdef KTRACE
if (KTRPOINT(td, KTR_SYSRET)) if (KTRPOINT(td, KTR_SYSRET))
ktrsysret(code, error, td->td_retval[0]); ktrsysret(sa.code, error, td->td_retval[0]);
#endif #endif
/* /*
* This works because errno is findable through the * This works because errno is findable through the
* register set. If we ever support an emulation where this * register set. If we ever support an emulation where this
* is not the case, this code will need to be revisited. * is not the case, this code will need to be revisited.
*/ */
STOPEVENT(p, S_SCX, code); STOPEVENT(p, S_SCX, sa.code);
PTRACESTOP_SC(p, td, S_PT_SCX); PTRACESTOP_SC(p, td, S_PT_SCX);
} }