llvm/llvm/test/Transforms/SCCP/range-attribute.ll

; NOTE: Assertions have been autogenerated by utils/update_test_checks.py
; RUN: opt < %s -passes=ipsccp -S | FileCheck %s --check-prefixes=CHECK,IPSCCP
; RUN: opt < %s -passes=sccp -S | FileCheck %s --check-prefixes=CHECK,SCCP

declare void @use(i1)
declare i32 @get_i32()

define void @range_attribute(i32 range(i32 0, 10) %v) {
; CHECK-LABEL: @range_attribute(
; CHECK-NEXT:    call void @use(i1 true)
; CHECK-NEXT:    [[C2:%.*]] = icmp ult i32 [[V:%.*]], 9
; CHECK-NEXT:    call void @use(i1 [[C2]])
; CHECK-NEXT:    call void @use(i1 false)
; CHECK-NEXT:    [[C4:%.*]] = icmp ugt i32 [[V]], 8
; CHECK-NEXT:    call void @use(i1 [[C4]])
; CHECK-NEXT:    ret void
;
  %c1 = icmp ult i32 %v, 10
  call void @use(i1 %c1)
  %c2 = icmp ult i32 %v, 9
  call void @use(i1 %c2)
  %c3 = icmp ugt i32 %v, 9
  call void @use(i1 %c3)
  %c4 = icmp ugt i32 %v, 8
  call void @use(i1 %c4)
  ret void
}

define i32 @range_attribute_single(i32 range(i32 0, 1) %v) {
; IPSCCP-LABEL: @range_attribute_single(
; IPSCCP-NEXT:    ret i32 0
;
; SCCP-LABEL: @range_attribute_single(
; SCCP-NEXT:    ret i32 [[V:%.*]]
;
  ret i32 %v
}

define void @call_range_attribute() {
; CHECK-LABEL: @call_range_attribute(
; CHECK-NEXT:    [[V:%.*]] = call range(i32 0, 10) i32 @get_i32()
; CHECK-NEXT:    call void @use(i1 true)
; CHECK-NEXT:    [[C2:%.*]] = icmp ult i32 [[V]], 9
; CHECK-NEXT:    call void @use(i1 [[C2]])
; CHECK-NEXT:    call void @use(i1 false)
; CHECK-NEXT:    [[C4:%.*]] = icmp ugt i32 [[V]], 8
; CHECK-NEXT:    call void @use(i1 [[C4]])
; CHECK-NEXT:    ret void
;
  %v = call range(i32 0, 10) i32 @get_i32()
  %c1 = icmp ult i32 %v, 10
  call void @use(i1 %c1)
  %c2 = icmp ult i32 %v, 9
  call void @use(i1 %c2)
  %c3 = icmp ugt i32 %v, 9
  call void @use(i1 %c3)
  %c4 = icmp ugt i32 %v, 8
  call void @use(i1 %c4)
  ret void
}


declare range(i32 0, 10) i32 @get_i32_in_range()

define void @call_range_result() {
; CHECK-LABEL: @call_range_result(
; CHECK-NEXT:    [[V:%.*]] = call i32 @get_i32_in_range()
; CHECK-NEXT:    call void @use(i1 true)
; CHECK-NEXT:    [[C2:%.*]] = icmp ult i32 [[V]], 9
; CHECK-NEXT:    call void @use(i1 [[C2]])
; CHECK-NEXT:    call void @use(i1 false)
; CHECK-NEXT:    [[C4:%.*]] = icmp ugt i32 [[V]], 8
; CHECK-NEXT:    call void @use(i1 [[C4]])
; CHECK-NEXT:    ret void
;
  %v = call i32 @get_i32_in_range()
  %c1 = icmp ult i32 %v, 10
  call void @use(i1 %c1)
  %c2 = icmp ult i32 %v, 9
  call void @use(i1 %c2)
  %c3 = icmp ugt i32 %v, 9
  call void @use(i1 %c3)
  %c4 = icmp ugt i32 %v, 8
  call void @use(i1 %c4)
  ret void
}

define internal i1 @ip_cmp_range_attribute(i32 %v) {
; IPSCCP-LABEL: @ip_cmp_range_attribute(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_range_attribute(
; SCCP-NEXT:    [[C:%.*]] = icmp ult i32 [[V:%.*]], 10
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = icmp ult i32 %v, 10
  ret i1 %c
}

define i1 @ip_range_attribute(i32 range(i32 0, 10) %v) {
; IPSCCP-LABEL: @ip_range_attribute(
; IPSCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_attribute(i32 [[V:%.*]])
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_range_attribute(
; SCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_attribute(i32 [[V:%.*]])
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = call i1 @ip_cmp_range_attribute(i32 %v)
  ret i1 %c
}

define internal i1 @ip_cmp_range_call(i32 %v) {
; IPSCCP-LABEL: @ip_cmp_range_call(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_range_call(
; SCCP-NEXT:    [[C:%.*]] = icmp ult i32 [[V:%.*]], 10
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = icmp ult i32 %v, 10
  ret i1 %c
}

define i1 @ip_range_call() {
; IPSCCP-LABEL: @ip_range_call(
; IPSCCP-NEXT:    [[V:%.*]] = call range(i32 0, 10) i32 @get_i32()
; IPSCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_call(i32 [[V]])
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_range_call(
; SCCP-NEXT:    [[V:%.*]] = call range(i32 0, 10) i32 @get_i32()
; SCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_call(i32 [[V]])
; SCCP-NEXT:    ret i1 [[C]]
;
  %v = call range(i32 0, 10) i32 @get_i32()
  %c = call i1 @ip_cmp_range_call(i32 %v)
  ret i1 %c
}

define internal i1 @ip_cmp_range_result(i32 %v) {
; IPSCCP-LABEL: @ip_cmp_range_result(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_range_result(
; SCCP-NEXT:    [[C:%.*]] = icmp ult i32 [[V:%.*]], 10
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = icmp ult i32 %v, 10
  ret i1 %c
}

define i1 @ip_range_result() {
; IPSCCP-LABEL: @ip_range_result(
; IPSCCP-NEXT:    [[V:%.*]] = call range(i32 0, 10) i32 @get_i32()
; IPSCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_result(i32 [[V]])
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_range_result(
; SCCP-NEXT:    [[V:%.*]] = call range(i32 0, 10) i32 @get_i32()
; SCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_range_result(i32 [[V]])
; SCCP-NEXT:    ret i1 [[C]]
;
  %v = call range(i32 0, 10) i32 @get_i32()
  %c = call i1 @ip_cmp_range_result(i32 %v)
  ret i1 %c
}

define internal i1 @ip_cmp_with_range_attribute(i32 range(i32 0, 10) %v) {
; IPSCCP-LABEL: @ip_cmp_with_range_attribute(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_with_range_attribute(
; SCCP-NEXT:    [[C:%.*]] = icmp eq i32 [[V:%.*]], 5
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = icmp eq i32 %v, 5
  ret i1 %c
}

define i1 @ip_range_attribute_constant() {
; IPSCCP-LABEL: @ip_range_attribute_constant(
; IPSCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_with_range_attribute(i32 5)
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_range_attribute_constant(
; SCCP-NEXT:    [[C:%.*]] = call i1 @ip_cmp_with_range_attribute(i32 5)
; SCCP-NEXT:    ret i1 [[C]]
;
  %c = call i1 @ip_cmp_with_range_attribute(i32 5)
  ret i1 %c
}

define internal i1 @ip_cmp_attribute_overdefined_callee(i32 range(i32 0, 10) %x) {
; IPSCCP-LABEL: @ip_cmp_attribute_overdefined_callee(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_attribute_overdefined_callee(
; SCCP-NEXT:    ret i1 true
;
  %cmp = icmp ult i32 %x, 10
  ret i1 %cmp
}

define i1 @ip_cmp_attribute_overdefined_caller(i32 %x) {
; IPSCCP-LABEL: @ip_cmp_attribute_overdefined_caller(
; IPSCCP-NEXT:    [[RES:%.*]] = call i1 @ip_cmp_attribute_overdefined_callee(i32 [[X:%.*]])
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_cmp_attribute_overdefined_caller(
; SCCP-NEXT:    [[RES:%.*]] = call i1 @ip_cmp_attribute_overdefined_callee(i32 [[X:%.*]])
; SCCP-NEXT:    ret i1 [[RES]]
;
  %res = call i1 @ip_cmp_attribute_overdefined_callee(i32 %x)
  ret i1 %res
}

define internal i1 @ip_cmp_attribute_intersect_callee(i32 range(i32 0, 10) %x) {
; IPSCCP-LABEL: @ip_cmp_attribute_intersect_callee(
; IPSCCP-NEXT:    ret i1 poison
;
; SCCP-LABEL: @ip_cmp_attribute_intersect_callee(
; SCCP-NEXT:    [[CMP2:%.*]] = icmp uge i32 [[X:%.*]], 5
; SCCP-NEXT:    [[AND:%.*]] = and i1 true, [[CMP2]]
; SCCP-NEXT:    ret i1 [[AND]]
;
  %cmp1 = icmp ult i32 %x, 10
  %cmp2 = icmp uge i32 %x, 5
  %and = and i1 %cmp1, %cmp2
  ret i1 %and
}

define i1 @ip_cmp_attribute_intersect_caller(i32 range(i32 5, 15) %x) {
; IPSCCP-LABEL: @ip_cmp_attribute_intersect_caller(
; IPSCCP-NEXT:    [[RES:%.*]] = call i1 @ip_cmp_attribute_intersect_callee(i32 [[X:%.*]])
; IPSCCP-NEXT:    ret i1 true
;
; SCCP-LABEL: @ip_cmp_attribute_intersect_caller(
; SCCP-NEXT:    [[RES:%.*]] = call i1 @ip_cmp_attribute_intersect_callee(i32 [[X:%.*]])
; SCCP-NEXT:    ret i1 [[RES]]
;
  %res = call i1 @ip_cmp_attribute_intersect_callee(i32 %x)
  ret i1 %res
}