From: Bill Wendling Date: Wed, 14 Aug 2019 16:44:07 +0000 (+0000) Subject: Ignore indirect branches from callbr. X-Git-Url: https://granicus.if.org/sourcecode?a=commitdiff_plain;h=55ac3ac76a8688623d3f033ffe38b99b7fd42661;p=llvm Ignore indirect branches from callbr. Summary: We can't speculate around indirect branches: indirectbr and invoke. The callbr instruction needs to be included here. Reviewers: nickdesaulniers, manojgupta, chandlerc Reviewed By: chandlerc Subscribers: llvm-commits Tags: #llvm Differential Revision: https://reviews.llvm.org/D66200 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@368873 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Transforms/Scalar/SpeculateAroundPHIs.cpp b/lib/Transforms/Scalar/SpeculateAroundPHIs.cpp index c13fb3e0451..e6db11f47ea 100644 --- a/lib/Transforms/Scalar/SpeculateAroundPHIs.cpp +++ b/lib/Transforms/Scalar/SpeculateAroundPHIs.cpp @@ -777,8 +777,10 @@ static bool tryToSpeculatePHIs(SmallVectorImpl &PNs, // speculation if the predecessor is an invoke. This doesn't seem // fundamental and we should probably be splitting critical edges // differently. - if (isa(PredBB->getTerminator()) || - isa(PredBB->getTerminator())) { + const auto *TermInst = PredBB->getTerminator(); + if (isa(TermInst) || + isa(TermInst) || + isa(TermInst)) { LLVM_DEBUG(dbgs() << " Invalid: predecessor terminator: " << PredBB->getName() << "\n"); return false; diff --git a/test/Transforms/SpeculateAroundPHIs/pr42991.ll b/test/Transforms/SpeculateAroundPHIs/pr42991.ll new file mode 100644 index 00000000000..4aef689f53d --- /dev/null +++ b/test/Transforms/SpeculateAroundPHIs/pr42991.ll @@ -0,0 +1,44 @@ +; RUN: opt -S -passes=spec-phis %s + +; This testcase crashes during the speculate around PHIs pass. The pass however +; results in no changes. + +define i32 @test1() { +entry: + callbr void asm sideeffect "", "X,X,~{dirflag},~{fpsr},~{flags}"(i8* blockaddress(@test1, %return), i8* blockaddress(@test1, %f)) + to label %asm.fallthrough [label %return, label %f] + +asm.fallthrough: + br label %return + +f: + br label %return + +return: + %retval.0 = phi i32 [ 0, %f ], [ 1, %asm.fallthrough ], [ 1, %entry ] + ret i32 %retval.0 +} + +define void @test2() { +entry: + br label %tailrecurse + +tailrecurse: + %call = tail call i32 @test3() + %tobool1 = icmp eq i32 %call, 0 + callbr void asm sideeffect "", "X,X,~{dirflag},~{fpsr},~{flags}"(i8* blockaddress(@test2, %test1.exit), i8* blockaddress(@test2, %f.i)) + to label %if.end6 [label %test1.exit, label %f.i] + +f.i: + br label %test1.exit + +test1.exit: + %retval.0.i = phi i1 [ false, %f.i ], [ true, %tailrecurse ] + %brmerge = or i1 %tobool1, %retval.0.i + br i1 %brmerge, label %if.end6, label %tailrecurse + +if.end6: + ret void +} + +declare i32 @test3()