From: Sanjay Patel Date: Fri, 2 Nov 2018 15:51:47 +0000 (+0000) Subject: [ValueTracking] allow non-canonical shuffles when computing signbits X-Git-Tag: android-x86-9.0-r1~11080 X-Git-Url: http://git.osdn.net/view?a=commitdiff_plain;h=6709348e9f65d8fbead4f2d25f9cc174edc075d9;p=android-x86%2Fexternal-llvm.git [ValueTracking] allow non-canonical shuffles when computing signbits This possibility is noted in D53987 for a different case, so we need to adjust the existing code. git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@345988 91177308-0d34-0410-b5e6-96231b3b80d8 --- diff --git a/lib/Analysis/ValueTracking.cpp b/lib/Analysis/ValueTracking.cpp index 3cef373f324..89a621576ec 100644 --- a/lib/Analysis/ValueTracking.cpp +++ b/lib/Analysis/ValueTracking.cpp @@ -2511,27 +2511,29 @@ static unsigned ComputeNumSignBitsImpl(const Value *V, unsigned Depth, // extended, shifted, etc). return ComputeNumSignBits(U->getOperand(0), Depth + 1, Q); - case Instruction::ShuffleVector: + case Instruction::ShuffleVector: { // If the shuffle mask contains any undefined elements, that element of the // result is undefined. Propagating information from a source operand may // not be correct in that case, so just bail out. if (cast(U)->getMask()->containsUndefElement()) break; - assert((!isa(U->getOperand(0)) || - !isa(U->getOperand(1))) - && "Should have simplified shuffle with 2 undef inputs"); + // If everything is undef, we can't say anything. This should be simplified. + Value *Op0 = U->getOperand(0), *Op1 = U->getOperand(1); + if (isa(Op0) && isa(Op1)) + break; // Look through shuffle of 1 source vector. - if (isa(U->getOperand(0))) - return ComputeNumSignBits(U->getOperand(1), Depth + 1, Q); - if (isa(U->getOperand(1))) - return ComputeNumSignBits(U->getOperand(0), Depth + 1, Q); + if (isa(Op0)) + return ComputeNumSignBits(Op1, Depth + 1, Q); + if (isa(Op1)) + return ComputeNumSignBits(Op0, Depth + 1, Q); // TODO: We can look through shuffles of 2 sources by computing the minimum // sign bits for each operand (similar to what we do for binops). break; } + } // Finally, if we can prove that the top bits of the result are 0's or 1's, // use this information. diff --git a/unittests/Analysis/ValueTrackingTest.cpp b/unittests/Analysis/ValueTrackingTest.cpp index ccae9d19ebb..f7d715c6447 100644 --- a/unittests/Analysis/ValueTrackingTest.cpp +++ b/unittests/Analysis/ValueTrackingTest.cpp @@ -494,6 +494,26 @@ TEST(ValueTracking, ComputeNumSignBits_PR32045) { EXPECT_EQ(ComputeNumSignBits(RVal, M->getDataLayout()), 1u); } +// No guarantees for canonical IR in this analysis, so this just bails out. +TEST(ValueTracking, ComputeNumSignBits_Shuffle) { + StringRef Assembly = "define <2 x i32> @f() { " + " %val = shufflevector <2 x i32> undef, <2 x i32> undef, <2 x i32> " + " ret <2 x i32> %val " + "} "; + + LLVMContext Context; + SMDiagnostic Error; + auto M = parseAssemblyString(Assembly, Error, Context); + assert(M && "Bad assembly?"); + + auto *F = M->getFunction("f"); + assert(F && "Bad assembly?"); + + auto *RVal = + cast(F->getEntryBlock().getTerminator())->getOperand(0); + EXPECT_EQ(ComputeNumSignBits(RVal, M->getDataLayout()), 1u); +} + TEST(ValueTracking, ComputeKnownBits) { StringRef Assembly = "define i32 @f(i32 %a, i32 %b) { " " %ash = mul i32 %a, 8 "