; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 5 ; RUN: opt < %s -passes=instcombine -S | FileCheck %s define i1 @ceil_shift4(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP1]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %is_zero = icmp eq i32 %quot_or_rem, 0 ret i1 %is_zero } define i1 @ceil_shift4_add(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4_add( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %ceil = add i32 %quot, %zext_has_rem %res = icmp eq i32 %ceil, 0 ret i1 %res } define i1 @ceil_shift6(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift6( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP1]] ; %quot = lshr i32 %arg0, 6 %rem = and i32 %arg0, 63 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift6_ne(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift6_ne( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[RES:%.*]] = icmp ne i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[RES]] ; %quot = lshr i32 %arg0, 6 %rem = and i32 %arg0, 63 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp ne i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift11(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift11( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP1]] ; %quot = lshr i32 %arg0, 11 %rem = and i32 %arg0, 2047 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift11_ne(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift11_ne( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[RES:%.*]] = icmp ne i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[RES]] ; %quot = lshr i32 %arg0, 6 %rem = and i32 %arg0, 63 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp ne i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift0(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift0( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP1]] ; %quot = lshr i32 %arg0, 0 %rem = and i32 %arg0, 0 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift4_comm(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4_comm( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %zext_has_rem, %quot %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } declare void @use(i32) define i1 @ceil_shift4_used_1(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4_used_1( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = lshr i32 [[ARG0]], 4 ; CHECK-NEXT: call void @use(i32 [[TMP1]]) ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 4 call void @use(i32 %quot) %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift4_used_5(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4_used_5( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = lshr i32 [[ARG0]], 4 ; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[ARG0]], 15 ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP2]], 0 ; CHECK-NEXT: [[TMP4:%.*]] = zext i1 [[TMP3]] to i32 ; CHECK-NEXT: [[TMP5:%.*]] = or i32 [[TMP1]], [[TMP4]] ; CHECK-NEXT: call void @use(i32 [[TMP5]]) ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem call void @use(i32 %quot_or_rem) %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift4_used_add_nuw_nsw(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift4_used_add_nuw_nsw( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[QUOT:%.*]] = lshr i32 [[ARG0]], 4 ; CHECK-NEXT: [[REM:%.*]] = and i32 [[ARG0]], 15 ; CHECK-NEXT: [[HAS_REM:%.*]] = icmp ne i32 [[REM]], 0 ; CHECK-NEXT: [[ZEXT_HAS_REM:%.*]] = zext i1 [[HAS_REM]] to i32 ; CHECK-NEXT: [[CEIL:%.*]] = add nuw nsw i32 [[QUOT]], [[ZEXT_HAS_REM]] ; CHECK-NEXT: call void @use(i32 [[CEIL]]) ; CHECK-NEXT: [[RES:%.*]] = icmp eq i32 [[ARG0]], 0 ; CHECK-NEXT: ret i1 [[RES]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %ceil = add nuw nsw i32 %quot, %zext_has_rem call void @use(i32 %ceil) %res = icmp eq i32 %ceil, 0 ret i1 %res } define <4 x i1> @ceil_shift4_v4i32(<4 x i32> %arg0) { ; CHECK-LABEL: define <4 x i1> @ceil_shift4_v4i32( ; CHECK-SAME: <4 x i32> [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq <4 x i32> [[ARG0]], zeroinitializer ; CHECK-NEXT: ret <4 x i1> [[TMP1]] ; %quot = lshr <4 x i32> %arg0, splat (i32 16) %rem = and <4 x i32> %arg0, splat (i32 65535) %has_rem = icmp ne <4 x i32> %rem, zeroinitializer %zext_has_rem = zext <4 x i1> %has_rem to <4 x i32> %quot_or_rem = or <4 x i32> %quot, %zext_has_rem %res = icmp eq <4 x i32> %quot_or_rem, zeroinitializer ret <4 x i1> %res } define <8 x i1> @ceil_shift4_v8i16(<8 x i16> %arg0) { ; CHECK-LABEL: define <8 x i1> @ceil_shift4_v8i16( ; CHECK-SAME: <8 x i16> [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = icmp eq <8 x i16> [[ARG0]], zeroinitializer ; CHECK-NEXT: ret <8 x i1> [[TMP1]] ; %quot = lshr <8 x i16> %arg0, splat (i16 4) %rem = and <8 x i16> %arg0, splat (i16 15) %has_rem = icmp ne <8 x i16> %rem, zeroinitializer %zext_has_rem = zext <8 x i1> %has_rem to <8 x i16> %quot_or_rem = or <8 x i16> %quot, %zext_has_rem %res = icmp eq <8 x i16> %quot_or_rem, zeroinitializer ret <8 x i1> %res } ; negative tests define i1 @ceil_shift_not_mask_1(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift_not_mask_1( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = lshr i32 [[ARG0]], 4 ; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[ARG0]], 31 ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP2]], 0 ; CHECK-NEXT: [[TMP4:%.*]] = zext i1 [[TMP3]] to i32 ; CHECK-NEXT: [[TMP5:%.*]] = or i32 [[TMP1]], [[TMP4]] ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[TMP5]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 4 %rem = and i32 %arg0, 31 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift_not_mask_2(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift_not_mask_2( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[TMP1:%.*]] = lshr i32 [[ARG0]], 5 ; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[ARG0]], 15 ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP2]], 0 ; CHECK-NEXT: [[TMP4:%.*]] = zext i1 [[TMP3]] to i32 ; CHECK-NEXT: [[TMP5:%.*]] = or i32 [[TMP1]], [[TMP4]] ; CHECK-NEXT: [[TMP6:%.*]] = icmp eq i32 [[TMP5]], 0 ; CHECK-NEXT: ret i1 [[TMP6]] ; %quot = lshr i32 %arg0, 5 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_or_rem = or i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_or_rem, 0 ret i1 %res } define i1 @ceil_shift_not_add_or(i32 %arg0) { ; CHECK-LABEL: define i1 @ceil_shift_not_add_or( ; CHECK-SAME: i32 [[ARG0:%.*]]) { ; CHECK-NEXT: [[REM:%.*]] = and i32 [[ARG0]], 15 ; CHECK-NEXT: [[HAS_REM_NOT:%.*]] = icmp eq i32 [[REM]], 0 ; CHECK-NEXT: [[TMP1:%.*]] = and i32 [[ARG0]], 32 ; CHECK-NEXT: [[RES1:%.*]] = icmp eq i32 [[TMP1]], 0 ; CHECK-NEXT: [[RES:%.*]] = or i1 [[HAS_REM_NOT]], [[RES1]] ; CHECK-NEXT: ret i1 [[RES]] ; %quot = lshr i32 %arg0, 5 %rem = and i32 %arg0, 15 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %quot_and_rem = and i32 %quot, %zext_has_rem %res = icmp eq i32 %quot_and_rem, 0 ret i1 %res } define i32 @ceil_shift_should_infer_ge_zero(i32 %x) { ; CHECK-LABEL: define i32 @ceil_shift_should_infer_ge_zero( ; CHECK-SAME: i32 [[X:%.*]]) { ; CHECK-NEXT: [[COND_NOT:%.*]] = icmp eq i32 [[X]], 0 ; CHECK-NEXT: br i1 [[COND_NOT]], label %[[IF_ELSE:.*]], label %[[IF_THEN:.*]] ; CHECK: [[IF_THEN]]: ; CHECK-NEXT: [[TMP1:%.*]] = lshr i32 [[X]], 20 ; CHECK-NEXT: [[TMP2:%.*]] = and i32 [[X]], 1048575 ; CHECK-NEXT: [[TMP3:%.*]] = icmp ne i32 [[TMP2]], 0 ; CHECK-NEXT: [[TMP4:%.*]] = zext i1 [[TMP3]] to i32 ; CHECK-NEXT: [[TMP5:%.*]] = add nuw nsw i32 [[TMP1]], [[TMP4]] ; CHECK-NEXT: ret i32 [[TMP5]] ; CHECK: [[IF_ELSE]]: ; CHECK-NEXT: ret i32 0 ; %cond = icmp ne i32 %x, 0 br i1 %cond, label %if.then, label %if.else if.then: %quot = lshr i32 %x, 20 %rem = and i32 %x, 1048575 %has_rem = icmp ne i32 %rem, 0 %zext_has_rem = zext i1 %has_rem to i32 %ceil = add nuw nsw i32 %quot, %zext_has_rem %max = call i32 @llvm.umax.i32(i32 %ceil, i32 1) ret i32 %max if.else: ret i32 0 }