mirror of
https://github.com/RPCS3/llvm.git
synced 2025-04-14 03:50:48 +00:00
[ValueTracking] fix bug computing isKnownToBeAPowerOfTwo() with arithmetic shift right (PR25900)
This is a fix for: https://llvm.org/bugs/show_bug.cgi?id=25900 If we think that an arithmetic right shift of a power of two is always a power of two, an sdiv gets wrongly converted to udiv. Differential Revision: http://reviews.llvm.org/D15827 git-svn-id: https://llvm.org/svn/llvm-project/llvm/trunk@256655 91177308-0d34-0410-b5e6-96231b3b80d8
This commit is contained in:
parent
04e678255f
commit
08e3bf995e
@ -1743,9 +1743,10 @@ bool isKnownToBeAPowerOfTwo(Value *V, bool OrZero, unsigned Depth,
|
|||||||
return false;
|
return false;
|
||||||
|
|
||||||
Value *X = nullptr, *Y = nullptr;
|
Value *X = nullptr, *Y = nullptr;
|
||||||
// A shift of a power of two is a power of two or zero.
|
// A shift left or a logical shift right of a power of two is a power of two
|
||||||
|
// or zero.
|
||||||
if (OrZero && (match(V, m_Shl(m_Value(X), m_Value())) ||
|
if (OrZero && (match(V, m_Shl(m_Value(X), m_Value())) ||
|
||||||
match(V, m_Shr(m_Value(X), m_Value()))))
|
match(V, m_LShr(m_Value(X), m_Value()))))
|
||||||
return isKnownToBeAPowerOfTwo(X, /*OrZero*/ true, Depth, Q, DL);
|
return isKnownToBeAPowerOfTwo(X, /*OrZero*/ true, Depth, Q, DL);
|
||||||
|
|
||||||
if (ZExtInst *ZI = dyn_cast<ZExtInst>(V))
|
if (ZExtInst *ZI = dyn_cast<ZExtInst>(V))
|
||||||
|
20
test/Analysis/ValueTracking/known-power-of-two.ll
Normal file
20
test/Analysis/ValueTracking/known-power-of-two.ll
Normal file
@ -0,0 +1,20 @@
|
|||||||
|
; RUN: opt -S -instcombine < %s | FileCheck %s
|
||||||
|
|
||||||
|
; https://llvm.org/bugs/show_bug.cgi?id=25900
|
||||||
|
; An arithmetic shift right of a power of two is not a power
|
||||||
|
; of two if the original value is the sign bit. Therefore,
|
||||||
|
; we can't transform the sdiv into a udiv.
|
||||||
|
|
||||||
|
define i32 @pr25900(i32 %d) {
|
||||||
|
%and = and i32 %d, -2147483648
|
||||||
|
; The next 3 lines prevent another fold from masking the bug.
|
||||||
|
%ext = zext i32 %and to i64
|
||||||
|
%or = or i64 %ext, 4294967296
|
||||||
|
%trunc = trunc i64 %or to i32
|
||||||
|
%ashr = ashr exact i32 %trunc, 31
|
||||||
|
%div = sdiv i32 4, %ashr
|
||||||
|
ret i32 %div
|
||||||
|
|
||||||
|
; CHECK: sdiv
|
||||||
|
}
|
||||||
|
|
Loading…
x
Reference in New Issue
Block a user