Add support for setjmp/longjmp primitives
Patch checked in for Bill Wendling :)


git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@6241 91177308-0d34-0410-b5e6-96231b3b80d8
diff --git a/include/llvm/Intrinsics.h b/include/llvm/Intrinsics.h
index c07d9c1..a1deb5f 100644
--- a/include/llvm/Intrinsics.h
+++ b/include/llvm/Intrinsics.h
@@ -12,9 +12,13 @@
 namespace LLVMIntrinsic {
   enum ID {
     not_intrinsic = 0,   // Must be zero
+
     va_start,            // Used to represent a va_start call in C
     va_end,              // Used to represent a va_end call in C
     va_copy,             // Used to represent a va_copy call in C
+
+    setjmp,              // Used to represent a setjmp call in C
+    longjmp,             // Used to represent a longjmp call in C
   };
 }
 
diff --git a/lib/Target/CBackend/CBackend.cpp b/lib/Target/CBackend/CBackend.cpp
index 577054e..bfd6bc9 100644
--- a/lib/Target/CBackend/CBackend.cpp
+++ b/lib/Target/CBackend/CBackend.cpp
@@ -543,6 +543,7 @@
   Out << "/* Provide Declarations */\n";
   generateAllocaDecl(Out);
   Out << "#include <stdarg.h>\n";
+  Out << "#include <setjmp.h>\n";
   
   // Provide a definition for null if one does not already exist,
   // and for `bool' if not compiling with a C++ compiler.
@@ -1022,7 +1023,6 @@
         writeOperand(&I.getParent()->getParent()->aback());
         Out << ")";
         return;
-
       case LLVMIntrinsic::va_end:
         Out << "va_end((va_list)*";
         writeOperand(I.getOperand(1));
@@ -1035,6 +1035,19 @@
         writeOperand(I.getOperand(2));
         Out << ")";
         return;
+        
+      case LLVMIntrinsic::setjmp:
+        Out << "setjmp((jmp_buf)";
+        writeOperand(I.getOperand(1));
+        Out << ")";
+        return;
+      case LLVMIntrinsic::longjmp:
+        Out << "longjmp((jmp_buf)";
+        writeOperand(I.getOperand(1));
+        Out << ", ";
+        writeOperand(I.getOperand(2));
+        Out << ")";
+        return;
       }
     }
 
diff --git a/lib/Target/CBackend/Writer.cpp b/lib/Target/CBackend/Writer.cpp
index 577054e..bfd6bc9 100644
--- a/lib/Target/CBackend/Writer.cpp
+++ b/lib/Target/CBackend/Writer.cpp
@@ -543,6 +543,7 @@
   Out << "/* Provide Declarations */\n";
   generateAllocaDecl(Out);
   Out << "#include <stdarg.h>\n";
+  Out << "#include <setjmp.h>\n";
   
   // Provide a definition for null if one does not already exist,
   // and for `bool' if not compiling with a C++ compiler.
@@ -1022,7 +1023,6 @@
         writeOperand(&I.getParent()->getParent()->aback());
         Out << ")";
         return;
-
       case LLVMIntrinsic::va_end:
         Out << "va_end((va_list)*";
         writeOperand(I.getOperand(1));
@@ -1035,6 +1035,19 @@
         writeOperand(I.getOperand(2));
         Out << ")";
         return;
+        
+      case LLVMIntrinsic::setjmp:
+        Out << "setjmp((jmp_buf)";
+        writeOperand(I.getOperand(1));
+        Out << ")";
+        return;
+      case LLVMIntrinsic::longjmp:
+        Out << "longjmp((jmp_buf)";
+        writeOperand(I.getOperand(1));
+        Out << ", ";
+        writeOperand(I.getOperand(2));
+        Out << ")";
+        return;
       }
     }
 
diff --git a/lib/VMCore/Function.cpp b/lib/VMCore/Function.cpp
index 0d5f90b..89195a3 100644
--- a/lib/VMCore/Function.cpp
+++ b/lib/VMCore/Function.cpp
@@ -164,20 +164,16 @@
     return 0;  // All intrinsics start with 'llvm.'
   
   switch (getName()[5]) {
+  case 'l':
+    if (getName() == "llvm.longjmp")  return LLVMIntrinsic::longjmp;
+    break;
+  case 's':
+    if (getName() == "llvm.setjmp")   return LLVMIntrinsic::setjmp;
+    break;
   case 'v':
-    if (getName().size() >= 9) {
-      switch (getName()[8]) {
-      case 's':
-        if (getName() == "llvm.va_start") return LLVMIntrinsic::va_start;
-        break;
-      case 'e':
-        if (getName() == "llvm.va_end") return LLVMIntrinsic::va_end;
-        break;
-      case 'c':
-        if (getName() == "llvm.va_copy") return LLVMIntrinsic::va_copy;
-        break;
-      }
-    }
+    if (getName() == "llvm.va_copy")  return LLVMIntrinsic::va_copy;
+    if (getName() == "llvm.va_end")   return LLVMIntrinsic::va_end;
+    if (getName() == "llvm.va_start") return LLVMIntrinsic::va_start;
     break;
   }
   // The "llvm." namespace is reserved!
diff --git a/lib/VMCore/Verifier.cpp b/lib/VMCore/Verifier.cpp
index 7dc870f..54f8f05 100644
--- a/lib/VMCore/Verifier.cpp
+++ b/lib/VMCore/Verifier.cpp
@@ -517,8 +517,10 @@
             " args!", &CI);
     NumArgs = 1;
     break;
-  case LLVMIntrinsic::va_end: NumArgs = 1; break;
+  case LLVMIntrinsic::va_end:  NumArgs = 1; break;
   case LLVMIntrinsic::va_copy: NumArgs = 2; break;
+  case LLVMIntrinsic::setjmp:  NumArgs = 1; break;
+  case LLVMIntrinsic::longjmp: NumArgs = 2; break;
   case LLVMIntrinsic::not_intrinsic: 
     assert(0 && "Invalid intrinsic!"); NumArgs = 0; break;
   }