forked from OSchip/llvm-project
[InstCombine] Do not fold 'and (sext (ashr X, Shift)), C' if Shift < 0
The 'and (sext (ashr X, ShiftC)), C' --> 'lshr (sext X), ShiftC' transformation would access out of bounds bits in APInt::getLowBitsSet if the shift count was larger than X's bit width or if it was negative. Fixes #56424
This commit is contained in:
parent
6656029a49
commit
ef7aed3e11
|
@ -1808,7 +1808,8 @@ Instruction *InstCombinerImpl::visitAnd(BinaryOperator &I) {
|
|||
|
||||
unsigned Width = Ty->getScalarSizeInBits();
|
||||
const APInt *ShiftC;
|
||||
if (match(Op0, m_OneUse(m_SExt(m_AShr(m_Value(X), m_APInt(ShiftC)))))) {
|
||||
if (match(Op0, m_OneUse(m_SExt(m_AShr(m_Value(X), m_APInt(ShiftC))))) &&
|
||||
ShiftC->ult(Width)) {
|
||||
if (*C == APInt::getLowBitsSet(Width, Width - ShiftC->getZExtValue())) {
|
||||
// We are clearing high bits that were potentially set by sext+ashr:
|
||||
// and (sext (ashr X, ShiftC)), C --> lshr (sext X), ShiftC
|
||||
|
|
|
@ -0,0 +1,27 @@
|
|||
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
|
||||
; RUN: opt < %s --passes=instcombine -S | FileCheck %s
|
||||
|
||||
; This would crash if we didn't check for a negative shift.
|
||||
; https://github.com/llvm/llvm-project/issues/56424
|
||||
define i64 @PR56424(i1 %cond, i32 %arg) {
|
||||
; CHECK-LABEL: @PR56424(
|
||||
; CHECK-NEXT: entry:
|
||||
; CHECK-NEXT: br i1 [[COND:%.*]], label [[IF_THEN:%.*]], label [[IF_END:%.*]]
|
||||
; CHECK: if.then:
|
||||
; CHECK-NEXT: br label [[IF_END]]
|
||||
; CHECK: if.end:
|
||||
; CHECK-NEXT: ret i64 0
|
||||
;
|
||||
entry:
|
||||
br i1 %cond, label %if.then, label %if.end
|
||||
|
||||
if.then:
|
||||
%shr = ashr i32 %arg, -2
|
||||
%sext = sext i32 %shr to i64
|
||||
br label %if.end
|
||||
|
||||
if.end:
|
||||
%val = phi i64 [ %sext, %if.then ], [ 0, %entry ]
|
||||
%and = and i64 -81, %val
|
||||
ret i64 %and
|
||||
}
|
Loading…
Reference in New Issue