Re: [PATCH] clear the frame pointer in copy_thread

From: Yanmin Zhang
Date: Tue Jun 19 2012 - 21:26:48 EST


On Tue, 2012-06-19 at 06:57 -0700, H. Peter Anvin wrote:
> It isn't clear to me this is the right thing to do this in the kernel. A frame pointer isn't obligatory and doing this in user space seems more appropriate.
Peter,

Thanks for the kind comments. You are right.

Yanmin

>
> "Su, Xuemin" <xuemin.su@xxxxxxxxx> wrote:
>
> >From: xueminsu <xuemin.su@xxxxxxxxx>
> >
> >When creating a userspace thread, kernel copies all parent
> >threadâs registers to the child thread, including bp register
> >which is used to create stack frame. At this point, the bp
> >register is pointing to the parent threadâs last stack frame.
> >
> >So, after the child thread is created, its bp register points
> >to the parent threadâs stack area, and the value would be pushed
> >to its own stack to form its first stack frame when it calls its
> >first function.
> >
> >This would not be a problem if the child thread and the parent
> >thread do not share the VM or if they share the VM and have the
> >same stack. But if they share the VM and have different stacks,
> >this would cause an issue: when we try to print the child threadâs
> >stack trace, we would eventually walk into the parent threadâs
> >stack area, and sometimes we will occasionally prints out the
> >parent threadâs stack trace.
> >
> >Currently some version of libc clears the frame pointer just after
> >the thread is created, but this is not guaranteed.
> >
> >Signed-off-by: xueminsu <xuemin.su@xxxxxxxxx>
> >---
> > arch/x86/kernel/process_32.c | 9 +++++++++
> > arch/x86/kernel/process_64.c | 9 +++++++++
> > 2 files changed, 18 insertions(+), 0 deletions(-)
> >
> >diff --git a/arch/x86/kernel/process_32.c
> >b/arch/x86/kernel/process_32.c
> >index 516fa18..b1b6d42 100644
> >--- a/arch/x86/kernel/process_32.c
> >+++ b/arch/x86/kernel/process_32.c
> >@@ -139,6 +139,15 @@ int copy_thread(unsigned long clone_flags,
> >unsigned long sp,
> > childregs->ax = 0;
> > childregs->sp = sp;
> >
> >+ /*
> >+ * If the child thread and the parent thread share the VM
> >+ * but have different stack, the child thread's bp register
> >+ * should be set to null, preventing it from pointing to
> >+ * the parent thread's stack area.
> >+ */
> >+ if ((sp != regs->sp) && (clone_flags & CLONE_VM))
> >+ childregs->bp = 0;
> >+
> > p->thread.sp = (unsigned long) childregs;
> > p->thread.sp0 = (unsigned long) (childregs+1);
> >
> >diff --git a/arch/x86/kernel/process_64.c
> >b/arch/x86/kernel/process_64.c
> >index 61cdf7f..5acff83 100644
> >--- a/arch/x86/kernel/process_64.c
> >+++ b/arch/x86/kernel/process_64.c
> >@@ -163,6 +163,15 @@ int copy_thread(unsigned long clone_flags,
> >unsigned long sp,
> > else
> > childregs->sp = (unsigned long)childregs;
> >
> >+ /*
> >+ * If the child thread and the parent thread share the VM
> >+ * but have different stack, the child thread's bp register
> >+ * should be set to null, preventing it from pointing to
> >+ * the parent thread's stack area.
> >+ */
> >+ if ((sp != regs->sp) && (clone_flags & CLONE_VM))
> >+ childregs->bp = 0;
> >+
> > p->thread.sp = (unsigned long) childregs;
> > p->thread.sp0 = (unsigned long) (childregs+1);
> > p->thread.usersp = me->thread.usersp;
> >--
> >1.7.6
>


--
To unsubscribe from this list: send the line "unsubscribe linux-kernel" in
the body of a message to majordomo@xxxxxxxxxxxxxxx
More majordomo info at http://vger.kernel.org/majordomo-info.html
Please read the FAQ at http://www.tux.org/lkml/