diff --git a/llvm/lib/Analysis/ValueTracking.cpp b/llvm/lib/Analysis/ValueTracking.cpp --- a/llvm/lib/Analysis/ValueTracking.cpp +++ b/llvm/lib/Analysis/ValueTracking.cpp @@ -2924,6 +2924,12 @@ isKnownNonZero(II->getArgOperand(1), DemandedElts, Depth, Q)) return true; break; + case Intrinsic::cttz: + return computeKnownBits(II->getArgOperand(0), DemandedElts, Depth, Q) + .Zero[0]; + case Intrinsic::ctlz: + return computeKnownBits(II->getArgOperand(0), DemandedElts, Depth, Q) + .isNonNegative(); case Intrinsic::fshr: case Intrinsic::fshl: // If Op0 == Op1, this is a rotate. rotate(x, y) != 0 iff x != 0. diff --git a/llvm/test/Analysis/ValueTracking/known-non-zero.ll b/llvm/test/Analysis/ValueTracking/known-non-zero.ll --- a/llvm/test/Analysis/ValueTracking/known-non-zero.ll +++ b/llvm/test/Analysis/ValueTracking/known-non-zero.ll @@ -678,11 +678,7 @@ define i1 @ctlz_true_nonzero(i8 %xx, i8 %ind) { ; CHECK-LABEL: @ctlz_true_nonzero( -; CHECK-NEXT: [[XS:%.*]] = lshr i8 [[XX:%.*]], 1 -; CHECK-NEXT: [[X:%.*]] = call i8 @llvm.ctlz.i8(i8 [[XS]], i1 true) -; CHECK-NEXT: [[Z:%.*]] = or i8 [[X]], [[IND:%.*]] -; CHECK-NEXT: [[R:%.*]] = icmp eq i8 [[Z]], 0 -; CHECK-NEXT: ret i1 [[R]] +; CHECK-NEXT: ret i1 false ; %xs = lshr i8 %xx, 1 %x = call i8 @llvm.ctlz.i8(i8 %xs, i1 true) @@ -693,11 +689,7 @@ define i1 @ctlz_false_nonzero(i8 %xx, i8 %ind) { ; CHECK-LABEL: @ctlz_false_nonzero( -; CHECK-NEXT: [[XA:%.*]] = and i8 [[XX:%.*]], 127 -; CHECK-NEXT: [[X:%.*]] = call i8 @llvm.ctlz.i8(i8 [[XA]], i1 true) -; CHECK-NEXT: [[Z:%.*]] = or i8 [[X]], [[IND:%.*]] -; CHECK-NEXT: [[R:%.*]] = icmp eq i8 [[Z]], 0 -; CHECK-NEXT: ret i1 [[R]] +; CHECK-NEXT: ret i1 false ; %xa = and i8 %xx, 127 %x = call i8 @llvm.ctlz.i8(i8 %xa, i1 true) @@ -723,11 +715,7 @@ define i1 @cttz_true_nonzero(i8 %xx, i8 %ind) { ; CHECK-LABEL: @cttz_true_nonzero( -; CHECK-NEXT: [[XS:%.*]] = shl i8 [[XX:%.*]], 1 -; CHECK-NEXT: [[X:%.*]] = call i8 @llvm.cttz.i8(i8 [[XS]], i1 true) -; CHECK-NEXT: [[Z:%.*]] = or i8 [[X]], [[IND:%.*]] -; CHECK-NEXT: [[R:%.*]] = icmp eq i8 [[Z]], 0 -; CHECK-NEXT: ret i1 [[R]] +; CHECK-NEXT: ret i1 false ; %xs = shl i8 %xx, 1 %x = call i8 @llvm.cttz.i8(i8 %xs, i1 true) @@ -738,11 +726,7 @@ define i1 @cttz_false_nonzero(i8 %xx, i8 %ind) { ; CHECK-LABEL: @cttz_false_nonzero( -; CHECK-NEXT: [[XA:%.*]] = and i8 [[XX:%.*]], -2 -; CHECK-NEXT: [[X:%.*]] = call i8 @llvm.cttz.i8(i8 [[XA]], i1 true) -; CHECK-NEXT: [[Z:%.*]] = or i8 [[X]], [[IND:%.*]] -; CHECK-NEXT: [[R:%.*]] = icmp eq i8 [[Z]], 0 -; CHECK-NEXT: ret i1 [[R]] +; CHECK-NEXT: ret i1 false ; %xa = and i8 %xx, -2 %x = call i8 @llvm.cttz.i8(i8 %xa, i1 true)