diff --git a/llvm/lib/Analysis/InstructionSimplify.cpp b/llvm/lib/Analysis/InstructionSimplify.cpp index da1a67c73388..b810f32643b6 100644 --- a/llvm/lib/Analysis/InstructionSimplify.cpp +++ b/llvm/lib/Analysis/InstructionSimplify.cpp @@ -4846,8 +4846,8 @@ static Value *simplifyPHINode(PHINode *PN, ArrayRef IncomingValues, if (HasUndefInput) { // We cannot start executing a trapping constant expression on more control // flow paths. - auto *CE = dyn_cast(CommonValue); - if (CE && CE->canTrap()) + auto *C = dyn_cast(CommonValue); + if (C && C->canTrap()) return nullptr; // If we have a PHI node like phi(X, undef, X), where X is defined by some diff --git a/llvm/lib/IR/Constants.cpp b/llvm/lib/IR/Constants.cpp index 965012e7f5ed..139ff5325899 100644 --- a/llvm/lib/IR/Constants.cpp +++ b/llvm/lib/IR/Constants.cpp @@ -564,21 +564,25 @@ void llvm::deleteConstant(Constant *C) { } static bool canTrapImpl(const Constant *C, - SmallPtrSetImpl &NonTrappingOps) { - assert(C->getType()->isFirstClassType() && "Cannot evaluate aggregate vals!"); - // The only thing that could possibly trap are constant exprs. + SmallPtrSetImpl &NonTrappingOps) { + assert(C->getType()->isFirstClassType() && + "Cannot evaluate non-first-class types!"); + // ConstantExpr or ConstantAggregate trap if any operands can trap. + if (isa(C) || isa(C)) { + for (unsigned i = 0, e = C->getNumOperands(); i != e; ++i) { + const Constant *Op = cast(C->getOperand(i)); + if (isa(Op) || isa(Op)) { + if (NonTrappingOps.insert(Op).second && canTrapImpl(Op, NonTrappingOps)) + return true; + } + } + } + + // The only leafs that can trap are constant expressions. const ConstantExpr *CE = dyn_cast(C); if (!CE) return false; - // ConstantExpr traps if any operands can trap. - for (unsigned i = 0, e = C->getNumOperands(); i != e; ++i) { - if (ConstantExpr *Op = dyn_cast(CE->getOperand(i))) { - if (NonTrappingOps.insert(Op).second && canTrapImpl(Op, NonTrappingOps)) - return true; - } - } - // Otherwise, only specific operations can trap. switch (CE->getOpcode()) { default: @@ -595,7 +599,7 @@ static bool canTrapImpl(const Constant *C, } bool Constant::canTrap() const { - SmallPtrSet NonTrappingOps; + SmallPtrSet NonTrappingOps; return canTrapImpl(this, NonTrappingOps); } diff --git a/llvm/test/Transforms/InstSimplify/phi.ll b/llvm/test/Transforms/InstSimplify/phi.ll index 8565883b0492..822023edd525 100644 --- a/llvm/test/Transforms/InstSimplify/phi.ll +++ b/llvm/test/Transforms/InstSimplify/phi.ll @@ -203,7 +203,8 @@ define <1 x i64> @pr49839_vector(i1 %c) { ; CHECK: if: ; CHECK-NEXT: br label [[JOIN]] ; CHECK: join: -; CHECK-NEXT: ret <1 x i64> +; CHECK-NEXT: [[PHI:%.*]] = phi <1 x i64> [ poison, [[IF]] ], [ , [[ENTRY:%.*]] ] +; CHECK-NEXT: ret <1 x i64> [[PHI]] ; entry: br i1 %c, label %if, label %join