llvm/llvm/test/Transforms/InstCombine/minmax-demandbits.ll

; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt -S -passes=instcombine < %s | FileCheck %s


define i32 @and_umax_less(i32 %A) {
; CHECK-LABEL: @and_umax_less(
; CHECK-NEXT:    [[X:%.*]] = and i32 [[A:%.*]], -32
; CHECK-NEXT:    ret i32 [[X]]
;
  %l0 = icmp ugt i32 31, %A
  %l1 = select i1 %l0, i32 31, i32 %A
  %x = and i32 %l1, -32
  ret i32 %x
}

define i32 @and_umax_muchless(i32 %A) {
; CHECK-LABEL: @and_umax_muchless(
; CHECK-NEXT:    [[X:%.*]] = and i32 [[A:%.*]], -32
; CHECK-NEXT:    ret i32 [[X]]
;
  %l0 = icmp ugt i32 12, %A
  %l1 = select i1 %l0, i32 12, i32 %A
  %x = and i32 %l1, -32
  ret i32 %x
}

define i32 @and_umax_more(i32 %A) {
; CHECK-LABEL: @and_umax_more(
; CHECK-NEXT:    [[L1:%.*]] = call i32 @llvm.umax.i32(i32 [[A:%.*]], i32 32)
; CHECK-NEXT:    [[X:%.*]] = and i32 [[L1]], -32
; CHECK-NEXT:    ret i32 [[X]]
;
  %l0 = icmp ugt i32 32, %A
  %l1 = select i1 %l0, i32 32, i32 %A
  %x = and i32 %l1, -32
  ret i32 %x
}

define i32 @shr_umax(i32 %A) {
; CHECK-LABEL: @shr_umax(
; CHECK-NEXT:    [[X:%.*]] = lshr i32 [[A:%.*]], 4
; CHECK-NEXT:    ret i32 [[X]]
;
  %l0 = icmp ugt i32 15, %A
  %l1 = select i1 %l0, i32 15, i32 %A
  %x = lshr i32 %l1, 4
  ret i32 %x
}

; Various constants for C2 & umax(A, C1)

define i8 @t_0_1(i8 %A) {
; CHECK-LABEL: @t_0_1(
; CHECK-NEXT:    [[X:%.*]] = and i8 [[A:%.*]], 1
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 0
  %l1 = select i1 %l2, i8 %A, i8 0
  %x = and i8 %l1, 1
  ret i8 %x
}

define i8 @t_0_10(i8 %A) {
; CHECK-LABEL: @t_0_10(
; CHECK-NEXT:    [[X:%.*]] = and i8 [[A:%.*]], 10
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 0
  %l1 = select i1 %l2, i8 %A, i8 0
  %x = and i8 %l1, 10
  ret i8 %x
}

define i8 @t_1_10(i8 %A) {
; CHECK-LABEL: @t_1_10(
; CHECK-NEXT:    [[X:%.*]] = and i8 [[A:%.*]], 10
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 1
  %l1 = select i1 %l2, i8 %A, i8 1
  %x = and i8 %l1, 10
  ret i8 %x
}

define i8 @t_2_4(i8 %A) {
; CHECK-LABEL: @t_2_4(
; CHECK-NEXT:    [[X:%.*]] = and i8 [[A:%.*]], 4
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 2
  %l1 = select i1 %l2, i8 %A, i8 2
  %x = and i8 %l1, 4
  ret i8 %x
}

define i8 @t_2_192(i8 %A) {
; CHECK-LABEL: @t_2_192(
; CHECK-NEXT:    [[X:%.*]] = and i8 [[A:%.*]], -64
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 2
  %l1 = select i1 %l2, i8 %A, i8 2
  %x = and i8 %l1, -64
  ret i8 %x
}

define i8 @t_2_63_or(i8 %A) {
; CHECK-LABEL: @t_2_63_or(
; CHECK-NEXT:    [[X:%.*]] = or i8 [[A:%.*]], 63
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 2
  %l1 = select i1 %l2, i8 %A, i8 2
  %x = or i8 %l1, 63
  ret i8 %x
}

define i8 @f_1_1(i8 %A) {
; CHECK-LABEL: @f_1_1(
; CHECK-NEXT:    [[L1:%.*]] = call i8 @llvm.umax.i8(i8 [[A:%.*]], i8 1)
; CHECK-NEXT:    [[X:%.*]] = and i8 [[L1]], 1
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 1
  %l1 = select i1 %l2, i8 %A, i8 1
  %x = and i8 %l1, 1
  ret i8 %x
}

define i8 @f_32_32(i8 %A) {
; CHECK-LABEL: @f_32_32(
; CHECK-NEXT:    [[L1:%.*]] = call i8 @llvm.umax.i8(i8 [[A:%.*]], i8 32)
; CHECK-NEXT:    [[X:%.*]] = and i8 [[L1]], -32
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 32
  %l1 = select i1 %l2, i8 %A, i8 32
  %x = and i8 %l1, -32
  ret i8 %x
}

define i8 @f_191_192(i8 %A) {
; CHECK-LABEL: @f_191_192(
; CHECK-NEXT:    [[L1:%.*]] = call i8 @llvm.umax.i8(i8 [[A:%.*]], i8 -65)
; CHECK-NEXT:    [[X:%.*]] = and i8 [[L1]], -64
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 191
  %l1 = select i1 %l2, i8 %A, i8 191
  %x = and i8 %l1, 192
  ret i8 %x
}

define i8 @f_10_1(i8 %A) {
; CHECK-LABEL: @f_10_1(
; CHECK-NEXT:    [[L1:%.*]] = call i8 @llvm.umax.i8(i8 [[A:%.*]], i8 10)
; CHECK-NEXT:    [[X:%.*]] = and i8 [[L1]], 1
; CHECK-NEXT:    ret i8 [[X]]
;
  %l2 = icmp ugt i8 %A, 10
  %l1 = select i1 %l2, i8 %A, i8 10
  %x = and i8 %l1, 1
  ret i8 %x
}

define i32 @and_umin(i32 %A) {
; CHECK-LABEL: @and_umin(
; CHECK-NEXT:    ret i32 0
;
  %l0 = icmp ult i32 15, %A
  %l1 = select i1 %l0, i32 15, i32 %A
  %x = and i32 %l1, -32
  ret i32 %x
}

define i32 @or_umin(i32 %A) {
; CHECK-LABEL: @or_umin(
; CHECK-NEXT:    ret i32 31
;
  %l0 = icmp ult i32 15, %A
  %l1 = select i1 %l0, i32 15, i32 %A
  %x = or i32 %l1, 31
  ret i32 %x
}

define i8 @or_min_31_30(i8 %A) {
; CHECK-LABEL: @or_min_31_30(
; CHECK-NEXT:    [[R:%.*]] = or i8 [[A:%.*]], 31
; CHECK-NEXT:    ret i8 [[R]]
;
  %cmp = icmp ult i8 %A, -30
  %min = select i1 %cmp, i8 %A, i8 -30
  %r = or i8 %min, 31
  ret i8 %r
}

define i8 @and_min_7_7(i8 %A) {
; CHECK-LABEL: @and_min_7_7(
; CHECK-NEXT:    [[R:%.*]] = and i8 [[A:%.*]], -8
; CHECK-NEXT:    ret i8 [[R]]
;
  %l2 = icmp ult i8 %A, -7
  %min = select i1 %l2, i8 %A, i8 -7
  %r = and i8 %min, -8
  ret i8 %r
}

define i8 @and_min_7_8(i8 %A) {
; CHECK-LABEL: @and_min_7_8(
; CHECK-NEXT:    [[R:%.*]] = and i8 [[A:%.*]], -8
; CHECK-NEXT:    ret i8 [[R]]
;
  %l2 = icmp ult i8 %A, -8
  %min = select i1 %l2, i8 %A, i8 -8
  %r = and i8 %min, -8
  ret i8 %r
}

define i8 @and_min_7_9(i8 %A) {
; CHECK-LABEL: @and_min_7_9(
; CHECK-NEXT:    [[MIN:%.*]] = call i8 @llvm.umin.i8(i8 [[A:%.*]], i8 -9)
; CHECK-NEXT:    [[R:%.*]] = and i8 [[MIN]], -8
; CHECK-NEXT:    ret i8 [[R]]
;
  %l2 = icmp ult i8 %A, -9
  %min = select i1 %l2, i8 %A, i8 -9
  %r = and i8 %min, -8
  ret i8 %r
}