x86 - 32-bit ptrace emulation mishandles 6th arg
[ jdike - Pushing Chuck's patch - see http://lkml.org/lkml/2005/9/16/261 for some history and a test program. UML is also broken without this patch - its processes get SIGBUS from the corrupt 6th argument to mmap being interpretted as a file offset ] When the 32-bit vDSO is used to make a system call, the %ebp register for the 6th syscall arg has to be loaded from the user stack (where it's pushed by the vDSO user code). The native i386 kernel always does this before stopping for syscall tracing, so %ebp can be seen and modified via ptrace to access the 6th syscall argument. The x86-64 kernel fails to do this, presenting the stack address to ptrace instead. This makes the %rbp value seen by 64-bit ptrace of a 32-bit process, and the %ebp value seen by a 32-bit caller of ptrace, both differ from the native i386 behavior. This patch fixes the problem by putting the word loaded from the user stack into %rbp before calling syscall_trace_enter, and reloading the 6th syscall argument from there afterwards (so ptrace can change it). This makes the behavior match that of i386 kernels. Original-Patch-By: Roland McGrath <roland@redhat.com> Signed-off-by: Chuck Ebbert <76306.1226@compuserve.com> Signed-off-by: Jeff Dike <jdike@linux.intel.com> Signed-off-by: Thomas Gleixner <tglx@linutronix.de>
This commit is contained in:
parent
fd181c72a3
commit
ecd744eec3
1 changed files with 6 additions and 13 deletions
|
@ -159,20 +159,16 @@ sysenter_do_call:
|
||||||
|
|
||||||
sysenter_tracesys:
|
sysenter_tracesys:
|
||||||
CFI_RESTORE_STATE
|
CFI_RESTORE_STATE
|
||||||
|
xchgl %r9d,%ebp
|
||||||
SAVE_REST
|
SAVE_REST
|
||||||
CLEAR_RREGS
|
CLEAR_RREGS
|
||||||
|
movq %r9,R9(%rsp)
|
||||||
movq $-ENOSYS,RAX(%rsp) /* really needed? */
|
movq $-ENOSYS,RAX(%rsp) /* really needed? */
|
||||||
movq %rsp,%rdi /* &pt_regs -> arg1 */
|
movq %rsp,%rdi /* &pt_regs -> arg1 */
|
||||||
call syscall_trace_enter
|
call syscall_trace_enter
|
||||||
LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
|
LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
|
||||||
RESTORE_REST
|
RESTORE_REST
|
||||||
movl %ebp, %ebp
|
xchgl %ebp,%r9d
|
||||||
/* no need to do an access_ok check here because rbp has been
|
|
||||||
32bit zero extended */
|
|
||||||
1: movl (%rbp),%r9d
|
|
||||||
.section __ex_table,"a"
|
|
||||||
.quad 1b,ia32_badarg
|
|
||||||
.previous
|
|
||||||
jmp sysenter_do_call
|
jmp sysenter_do_call
|
||||||
CFI_ENDPROC
|
CFI_ENDPROC
|
||||||
ENDPROC(ia32_sysenter_target)
|
ENDPROC(ia32_sysenter_target)
|
||||||
|
@ -262,20 +258,17 @@ cstar_do_call:
|
||||||
|
|
||||||
cstar_tracesys:
|
cstar_tracesys:
|
||||||
CFI_RESTORE_STATE
|
CFI_RESTORE_STATE
|
||||||
|
xchgl %r9d,%ebp
|
||||||
SAVE_REST
|
SAVE_REST
|
||||||
CLEAR_RREGS
|
CLEAR_RREGS
|
||||||
|
movq %r9,R9(%rsp)
|
||||||
movq $-ENOSYS,RAX(%rsp) /* really needed? */
|
movq $-ENOSYS,RAX(%rsp) /* really needed? */
|
||||||
movq %rsp,%rdi /* &pt_regs -> arg1 */
|
movq %rsp,%rdi /* &pt_regs -> arg1 */
|
||||||
call syscall_trace_enter
|
call syscall_trace_enter
|
||||||
LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
|
LOAD_ARGS32 ARGOFFSET /* reload args from stack in case ptrace changed it */
|
||||||
RESTORE_REST
|
RESTORE_REST
|
||||||
|
xchgl %ebp,%r9d
|
||||||
movl RSP-ARGOFFSET(%rsp), %r8d
|
movl RSP-ARGOFFSET(%rsp), %r8d
|
||||||
/* no need to do an access_ok check here because r8 has been
|
|
||||||
32bit zero extended */
|
|
||||||
1: movl (%r8),%r9d
|
|
||||||
.section __ex_table,"a"
|
|
||||||
.quad 1b,ia32_badarg
|
|
||||||
.previous
|
|
||||||
jmp cstar_do_call
|
jmp cstar_do_call
|
||||||
END(ia32_cstar_target)
|
END(ia32_cstar_target)
|
||||||
|
|
||||||
|
|
Loading…
Reference in a new issue