inalloca: Don't remove dead arguments in the presence of inalloca args
authorReid Kleckner <reid@kleckner.net>
Mon, 3 Feb 2014 20:42:49 +0000 (20:42 +0000)
committerReid Kleckner <reid@kleckner.net>
Mon, 3 Feb 2014 20:42:49 +0000 (20:42 +0000)
It disturbs the layout of the parameters in memory and registers,
leading to problems in the backend.

The plan for optimizing internal inalloca functions going forward is to
essentially SROA the argument memory and demote any captured arguments
(things that aren't trivially written by a load or store) to an indirect
pointer to a static alloca.

git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@200717 91177308-0d34-0410-b5e6-96231b3b80d8

lib/Transforms/IPO/DeadArgumentElimination.cpp
test/Transforms/DeadArgElim/keepalive.ll

index 81dbd329c4b6a1288ab7dc68ae5e2bcd52342f45..d372ba22d6e3b26689fba8336da06780fd5b9de1 100644 (file)
@@ -531,6 +531,13 @@ DAE::Liveness DAE::SurveyUses(const Value *V, UseVector &MaybeLiveUses) {
 // well as arguments to functions which have their "address taken".
 //
 void DAE::SurveyFunction(const Function &F) {
+  // Functions with inalloca parameters are expecting args in a particular
+  // register and memory layout.
+  if (F.getAttributes().hasAttrSomewhere(Attribute::InAlloca)) {
+    MarkLive(F);
+    return;
+  }
+
   unsigned RetCount = NumRetVals(&F);
   // Assume all return values are dead
   typedef SmallVector<Liveness, 5> RetVals;
index 82e01f225843a0330910a2f752128c30b860f032..b66df792ca84d4e2bb39a5d80ea8cd7c3b535ee1 100644 (file)
@@ -28,4 +28,20 @@ define void @caller() {
         ret void
 }
 
+; We can't remove 'this' here, as that would put argmem in ecx instead of
+; memory.
+define internal x86_thiscallcc i32 @unused_this(i32* %this, i32* inalloca %argmem) {
+       %v = load i32* %argmem
+       ret i32 %v
+}
+; CHECK-LABEL: define internal x86_thiscallcc i32 @unused_this(i32* %this, i32* inalloca %argmem)
+
+define i32 @caller2() {
+       %t = alloca i32
+       %m = alloca i32, inalloca
+       store i32 42, i32* %m
+       %v = call x86_thiscallcc i32 @unused_this(i32* %t, i32* inalloca %m)
+       ret i32 %v
+}
+
 ; CHECK: attributes #0 = { nounwind }