
IPSCCP can currently return worse results than SCCP for arguments that are tracked interprocedurally, because information from attributes is not used for them. Fix this by intersecting in the attribute information when propagating lattice values from calls.
245 lines
7.2 KiB
LLVM
245 lines
7.2 KiB
LLVM
; 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
|
|
}
|