diff options
| author | David Majnemer <david.majnemer@gmail.com> | 2016-06-25 07:37:27 +0000 |
|---|---|---|
| committer | David Majnemer <david.majnemer@gmail.com> | 2016-06-25 07:37:27 +0000 |
| commit | 1fea77c6fc6ab0d5557078656be6c54eba769b42 (patch) | |
| tree | 018f2212532c40b25e7f9547fc881c10faf96285 | |
| parent | 50e3dfe9d0a79d799e246884768f49c3d96746be (diff) | |
| download | bcm5719-llvm-1fea77c6fc6ab0d5557078656be6c54eba769b42.tar.gz bcm5719-llvm-1fea77c6fc6ab0d5557078656be6c54eba769b42.zip | |
[SimplifyCFG] Replace calls to null/undef with unreachable
Calling null is undefined behavior, a call to undef can be trivially
treated as a call to null.
llvm-svn: 273776
| -rw-r--r-- | llvm/lib/Transforms/Utils/Local.cpp | 6 | ||||
| -rw-r--r-- | llvm/lib/Transforms/Utils/SimplifyCFG.cpp | 6 | ||||
| -rw-r--r-- | llvm/test/Transforms/SimplifyCFG/UnreachableEliminate.ll | 31 |
3 files changed, 42 insertions, 1 deletions
diff --git a/llvm/lib/Transforms/Utils/Local.cpp b/llvm/lib/Transforms/Utils/Local.cpp index 4d76aae2971..edacfd20f99 100644 --- a/llvm/lib/Transforms/Utils/Local.cpp +++ b/llvm/lib/Transforms/Utils/Local.cpp @@ -1414,6 +1414,12 @@ static bool markAliveBlocks(Function &F, } if (CallInst *CI = dyn_cast<CallInst>(BBI)) { + Value *Callee = CI->getCalledValue(); + if (isa<ConstantPointerNull>(Callee) || isa<UndefValue>(Callee)) { + changeToUnreachable(CI, /*UseLLVMTrap=*/false); + Changed = true; + break; + } if (CI->doesNotReturn()) { // If we found a call to a no-return function, insert an unreachable // instruction after it. Make sure there isn't *already* one there diff --git a/llvm/lib/Transforms/Utils/SimplifyCFG.cpp b/llvm/lib/Transforms/Utils/SimplifyCFG.cpp index d22f5c63dee..7924c33038a 100644 --- a/llvm/lib/Transforms/Utils/SimplifyCFG.cpp +++ b/llvm/lib/Transforms/Utils/SimplifyCFG.cpp @@ -5388,7 +5388,7 @@ static bool passingValueIsAlwaysUndefined(Value *V, Instruction *I) { if (I->use_empty()) return false; - if (C->isNullValue()) { + if (C->isNullValue() || isa<UndefValue>(C)) { // Only look at the first use, avoid hurting compile time with long uselists User *Use = *I->user_begin(); @@ -5417,6 +5417,10 @@ static bool passingValueIsAlwaysUndefined(Value *V, Instruction *I) { if (!SI->isVolatile()) return SI->getPointerAddressSpace() == 0 && SI->getPointerOperand() == I; + + // A call to null is undefined. + if (auto CS = CallSite(Use)) + return CS.getCalledValue() == I; } return false; } diff --git a/llvm/test/Transforms/SimplifyCFG/UnreachableEliminate.ll b/llvm/test/Transforms/SimplifyCFG/UnreachableEliminate.ll index 87872a6a8a1..be612b288b7 100644 --- a/llvm/test/Transforms/SimplifyCFG/UnreachableEliminate.ll +++ b/llvm/test/Transforms/SimplifyCFG/UnreachableEliminate.ll @@ -96,3 +96,34 @@ bb2: store i8 2, i8* %ptr.2, align 8 ret void } + +define i32 @test7(i1 %X) { +entry: + br i1 %X, label %if, label %else + +if: + call void undef() + br label %else + +else: + %phi = phi i32 [ 0, %entry ], [ 1, %if ] + ret i32 %phi +} +; CHECK-LABEL: define i32 @test7( +; CHECK-NOT: call +; CHECK: ret i32 0 + +define void @test8(i1 %X, void ()* %Y) { +entry: + br i1 %X, label %if, label %else + +if: + br label %else + +else: + %phi = phi void ()* [ %Y, %entry ], [ null, %if ] + call void %phi() + ret void +} +; CHECK-LABEL: define void @test8( +; CHECK: call void %Y( |

