[LLVMbugs] [Bug 18244] New: Backend can miscompile functions that invoke setjmp()

bugzilla-daemon at llvm.org bugzilla-daemon at llvm.org
Fri Dec 13 11:39:52 PST 2013


http://llvm.org/bugs/show_bug.cgi?id=18244

            Bug ID: 18244
           Summary: Backend can miscompile functions that invoke setjmp()
           Product: tools
           Version: trunk
          Hardware: PC
                OS: Linux
            Status: ASSIGNED
          Severity: normal
          Priority: P
         Component: llc
          Assignee: mseaborn at chromium.org
          Reporter: mseaborn at chromium.org
                CC: llvmbugs at cs.uiuc.edu
    Classification: Unclassified

The backend needs to avoid reusing spill slots in functions that call setjmp().
 It does this correctly if setjmp() is called via a "call" instruction, but not
if it's called via an "invoke" instruction.

Here's an example of a program that is miscompiled as a result:

#include <stdio.h>

typedef char jmp_buf[1000];
extern "C" {
  // Declare setjmp() without "__attribute__((nothrow))":
  int setjmp(jmp_buf env);
  void longjmp(jmp_buf env, int val);
}

static int counter = 0;

__attribute__((noinline))
int f(void) {
  return ++counter;
}

__attribute__((noinline))
void g(int x, int expected) {
  if (x == expected) {
    printf("got %i (ok)\n", x);
  } else {
    printf("got %i but expected %i: ERROR\n", x, expected);
  }
}

// To test for the bug, this must be compiled with optimisation in
// order to run mem2reg and use spill slots.  But f() and g() must not
// be inlined.
int main() throw() {
  // Keep enough variables live across the setjmp() call that they
  // don't all fit in registers and the compiler allocates some spill
  // slots.
  int a1 = f();
  int a2 = f();
  int a3 = f();
  int a4 = f();
  int a5 = f();
  int a6 = f();
  int a7 = f();
  int a8 = f();
  jmp_buf buf;
  if (setjmp(buf)) {
    g(a1, 1);
    g(a2, 2);
    g(a3, 3);
    g(a4, 4);
    g(a5, 5);
    g(a6, 6);
    g(a7, 7);
    g(a8, 8);
    return 0;
  }
  // Again, keep enough variables live that they need spill slots.  A
  // correct compiler will realise that a1...aN are still live (via
  // setjmp()+longjmp()), and so not reuse the earlier spill slots.
  // An incorrect compiler will think that a1..aN are dead here and
  // wrongly reuse the earlier spill slots.
  int b1 = f();
  int b2 = f();
  int b3 = f();
  int b4 = f();
  int b5 = f();
  int b6 = f();
  int b7 = f();
  int b8 = f();
  int start = 8;
  g(b1, start + 1);
  g(b2, start + 2);
  g(b3, start + 3);
  g(b4, start + 4);
  g(b5, start + 5);
  g(b6, start + 6);
  g(b7, start + 7);
  g(b8, start + 8);
  longjmp(buf, 1);
}

On x86-64, when compiled with -O1 or -O2, this program prints:

...
got 14 but expected 6: ERROR
got 15 but expected 7: ERROR
got 16 but expected 8: ERROR

The bug is in Function::callsFunctionThatReturnsTwice(), which checks calls but
not invokes.  This is ultimately used by StackSlotColoring.cpp in its
exposesReturnsTwice() call.

This is similar to bug 18206.  An "invoke" of setjmp() can occur if:
 * setjmp() is called in an exception-handling context, e.g. in C++ a
destructor implicitly declared as "nothrow".
 * setjmp() is declared without "nothrow".  e.g. glibc's headers declare
setjmp() with "nothrow", but not all headers do.

I'll prepare a fix.

-- 
You are receiving this mail because:
You are on the CC list for the bug.
-------------- next part --------------
An HTML attachment was scrubbed...
URL: <http://lists.llvm.org/pipermail/llvm-bugs/attachments/20131213/38f7cb78/attachment.html>


More information about the llvm-bugs mailing list