This is the mail archive of the glibc-bugs@sourceware.org mailing list for the glibc project.


Index Nav: [Date Index] [Subject Index] [Author Index] [Thread Index]
Message Nav: [Date Prev] [Date Next] [Thread Prev] [Thread Next]
Other format: [Raw text]

[Bug dynamic-link/15786] New: ifunc resolver functions can smash function arguments


http://sourceware.org/bugzilla/show_bug.cgi?id=15786

            Bug ID: 15786
           Summary: ifunc resolver functions can smash function arguments
           Product: glibc
           Version: 2.17
            Status: NEW
          Severity: normal
          Priority: P2
         Component: dynamic-link
          Assignee: unassigned at sourceware dot org
          Reporter: cworth at cworth dot org

Created attachment 7118
  --> http://sourceware.org/bugzilla/attachment.cgi?id=7118&action=edit
Test case: library implementation

In using ifunc resolver functions, I've found that the arguments to
the underlying function are sometimes corrupted by the code in the
resolver functions itself.

By debugging, I've found that the problem is that the xmm0-xmm7
registers (on x864_64) are not being saved prior to the call of the
ifunc resolver, so that if the ifunc resolver writes to these
registers, (and the underlying function is expecting parameters in
these registers), then the parameter values are corrupted.

I will attach to this bug report a small test case demonstrating the
problem.

Can you please provide me some guidance on what I should be doing to
avoid this problem? I can imagine that guidance taking one of several
forms (ordered from least-convenient to most-convenient from my point
of view as a user of ifunc):

1. Don't do that!

   Perhaps I'm calling code that I just shouldn't be calling within an
   ifunc resolver. If there are limitations to what code can reliably
   be used within an ifunc resolver can you please point me to
   documentation describing those limitations?

2. You can do that, but be careful...

   Perhaps I should be manually saving xmm0-xmm7 if I happen to be
   using code that touches these registers? Is there any easy way to
   do this? Something like a gcc attribute that let me decorate my
   function would be really nice here.

3. This is a bug in glibc

   It would be particularly nice if glibc would take the extra care to
   save/restore these registers around calls to ifunc resolvers. I can
   imagine that the dl code in glibc is careful to never touch
   xmm0-xmm7 itself, so that in other cases there's no need to save
   these. But since ifunc resolvers effectively allow users to insert
   arbitrary code into the middle of the dl-resolution code path, it
   seems to me that glibc should be saving/restoring things completely
   here.

For the attached test case, I've written a tiny main program that
calls one function in a tiny library. That tiny library implements a
single function with an ifunc resolver. The resolver calls printf
which ends up smashing the function's argument in xmm0.

The test case includes a simple Makefile. An example session of
compiling the program, and running it to demonstrate the bug is below:

    $ make
    cc -g -Wall -Wextra -Wmissing-declarations -Werror=attributes -c
smash-xmm.c -o smash-xmm.o
    cc -g -Wall -Wextra -Wmissing-declarations -Werror=attributes -fPIC -shared
-Wl,-Bsymbolic -o libmylib.so mylib.c
    cc -g -Wall -Wextra -Wmissing-declarations -Werror=attributes smash-xmm.o
-L . -lmylib -o smash-xmm

    $ LD_LIBRARY_PATH=. ./smash-xmm
    Calling mylib_print_float with value: 42
    This printf (and underlying __strchrnul) smashes xmm registers
    mylib_print_float called with value: 0

The debugging I did to determine the problem is as follows.

My ifunc resolver is called by code from glibc's dl-trampoline.S as follows:

_dl_runtime_resolve:
    cfi_adjust_cfa_offset(16) # Incorporate PLT
    subq $56,%rsp
    cfi_adjust_cfa_offset(56)
    movq %rax,(%rsp)    # Preserve registers otherwise clobbered.
    movq %rcx, 8(%rsp)
    movq %rdx, 16(%rsp)
    movq %rsi, 24(%rsp)
    movq %rdi, 32(%rsp)
    movq %r8, 40(%rsp)
    movq %r9, 48(%rsp)
    movq 64(%rsp), %rsi    # Copy args pushed by PLT in register.
    movq 56(%rsp), %rdi    # %rdi: link_map, %rsi: reloc_index
    call _dl_fixup        # Call resolver.

Note that rcx, rdx, rsi, rdi, r8 and r9 are saved. I checked at
http://en.wikipedia.org/wiki/X86_calling_conventions that the System V
AMD64 ABI for x86-64 requires saving xmm0-xmm7 in addition to these
registers.

I also verified that in my original program, (from which my test case
was distilled), that my function's arguments exist in xmm0, xmm1, and
xmm2.

Finally, I stepped through the printf call in my ifunc resolver to
find the following code which smashes the xmm register values:

ENTRY (__strchrnul)
    movd    %esi, %xmm1
    movq    %rdi, %rcx
    punpcklbw %xmm1, %xmm1
    andq    $~15, %rdi
    pxor    %xmm2, %xmm2
    punpcklbw %xmm1, %xmm1
    orl    $0xffffffff, %esi
    movdqa    (%rdi), %xmm0
    pshufd    $0, %xmm1, %xmm1
    subq    %rdi, %rcx
    movdqa    %xmm0, %xmm3

Now, while I don't strictly need to call printf from within my ifunc
resolver, I was surprised to find my program mysteriously misbehaving
when I did so. And in the meantime, I'm left feeling quite uncertain
about what code I can safely call within the ifunc resolver without
causing similar problems.

Thanks in advance for your time.

-Carl

-- 
You are receiving this mail because:
You are on the CC list for the bug.


Index Nav: [Date Index] [Subject Index] [Author Index] [Thread Index]
Message Nav: [Date Prev] [Date Next] [Thread Prev] [Thread Next]