Don't drop function/call return attributes like 'nounwind'.
authorDuncan Sands <baldrick@free.fr>
Fri, 1 Feb 2008 20:37:16 +0000 (20:37 +0000)
committerDuncan Sands <baldrick@free.fr>
Fri, 1 Feb 2008 20:37:16 +0000 (20:37 +0000)
git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@46645 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Transforms/IPO/ArgumentPromotion.cpp
test/Transforms/ArgumentPromotion/2008-02-01-ReturnAttrs.ll [new file with mode: 0644]

index 3ac1c7bb2b23af4c457aa67c99892bdbeb65747b..4486677e6307e560b464b0110e769b36fbe53e45 100644 (file)
@@ -404,6 +404,10 @@ Function *ArgPromotion::DoPromotion(Function *F,
   ParamAttrsVector ParamAttrsVec;
   const ParamAttrsList *PAL = F->getParamAttrs();
 
+  // Add any return attributes.
+  if (unsigned attrs = PAL ? PAL->getParamAttrs(0) : 0)
+    ParamAttrsVec.push_back(ParamAttrsWithIndex::get(0, attrs));
+
   unsigned ArgIndex = 1;
   for (Function::arg_iterator I = F->arg_begin(), E = F->arg_end(); I != E;
        ++I, ++ArgIndex) {
@@ -491,6 +495,10 @@ Function *ArgPromotion::DoPromotion(Function *F,
     Instruction *Call = CS.getInstruction();
     PAL = CS.getParamAttrs();
     
+    // Add any return attributes.
+    if (unsigned attrs = PAL ? PAL->getParamAttrs(0) : 0)
+      ParamAttrsVec.push_back(ParamAttrsWithIndex::get(0, attrs));
+
     // Loop over the operands, inserting GEP and loads in the caller as
     // appropriate.
     CallSite::arg_iterator AI = CS.arg_begin();
diff --git a/test/Transforms/ArgumentPromotion/2008-02-01-ReturnAttrs.ll b/test/Transforms/ArgumentPromotion/2008-02-01-ReturnAttrs.ll
new file mode 100644 (file)
index 0000000..67c061f
--- /dev/null
@@ -0,0 +1,15 @@
+; RUN: llvm-as < %s | opt -argpromotion | llvm-dis | grep nounwind | count 2
+
+define internal i32 @deref(i32* %x) nounwind {
+entry:
+       %tmp2 = load i32* %x, align 4           ; <i32> [#uses=1]
+       ret i32 %tmp2
+}
+
+define i32 @f(i32 %x) {
+entry:
+       %x_addr = alloca i32            ; <i32*> [#uses=2]
+       store i32 %x, i32* %x_addr, align 4
+       %tmp1 = call i32 @deref( i32* %x_addr ) nounwind                ; <i32> [#uses=1]
+       ret i32 %tmp1
+}