Index: llvm/lib/Analysis/ValueTracking.cpp =================================================================== --- llvm/lib/Analysis/ValueTracking.cpp +++ llvm/lib/Analysis/ValueTracking.cpp @@ -5082,6 +5082,37 @@ Known, Depth + 1, Q, TLI); break; } + case Instruction::PHI: { + const PHINode *P = cast(Op); + // Unreachable blocks may have zero-operand PHI nodes. + if (P->getNumIncomingValues() == 0) + break; + + // Otherwise take the unions of the known bit sets of the operands, + // taking conservative care to avoid excessive recursion. + if (Depth < MaxAnalysisRecursionDepth - 1) { + // Skip if every incoming value references to ourself. + if (isa_and_nonnull(P->hasConstantValue())) + break; + + for (Value *IncValue : P->incoming_values()) { + // Skip direct self references. + if (IncValue == P) + continue; + + KnownFPClass KnownSrc; + // Recurse, but cap the recursion to one level, because we don't want to + // waste time spinning around in loops. + computeKnownFPClass(IncValue, DemandedElts, InterestedClasses, KnownSrc, + MaxAnalysisRecursionDepth - 1, Q, TLI); + Known |= KnownSrc; + if (Known.KnownFPClasses == fcAllFlags) + break; + } + } + + break; + } default: break; } Index: llvm/unittests/Analysis/ValueTrackingTest.cpp =================================================================== --- llvm/unittests/Analysis/ValueTrackingTest.cpp +++ llvm/unittests/Analysis/ValueTrackingTest.cpp @@ -1630,6 +1630,30 @@ expectKnownFPClass(fcAllFlags, std::nullopt, A7); } +TEST_F(ComputeKnownFPClassTest, UnreachablePhi) { + parseAssembly( + "define float @test(float %arg) {\n" + "entry:\n" + " ret float 0.0\n" + "unreachable:\n" + " %A = phi float\n" + " ret float %A\n" + "}\n"); + expectKnownFPClass(fcAllFlags, std::nullopt); +} + +TEST_F(ComputeKnownFPClassTest, SelfPhiOnly) { + parseAssembly( + "define float @test(float %arg) {\n" + "entry:\n" + " ret float 0.0\n" + "loop:\n" + " %A = phi float [ %A, %loop ]\n" + " br label %loop\n" + "}\n"); + expectKnownFPClass(fcAllFlags, std::nullopt); +} + TEST_F(ComputeKnownFPClassTest, CannotBeOrderedLessThanZero) { parseAssembly("define float @test(float %arg) {\n" " %A = fmul float %arg, %arg"