blob: 1bdd436f01d021f699f72bc1bfa64520767bf435 [file] [log] [blame]
; NOTE: Assertions have been autogenerated by utils/update_test_checks.py UTC_ARGS: --version 6
; RUN: opt -passes=simplifycfg -S < %s | FileCheck %s
target datalayout = "pu1:64:64-pe2:64:64:64:32"
;; TODO: it would probably be better to just emit a pointer compare against null.
define void @test_default_null_base(ptr addrspace(0) align 8 %ptr) {
; CHECK-LABEL: define void @test_default_null_base(
; CHECK-SAME: ptr align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 0
; CHECK-NEXT: br i1 [[COND]], label %[[TRUE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr [[PTR]], align 8
; CHECK-NEXT: store i64 3, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET:.*]]
; CHECK: [[COMMON_RET]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(0) %ptr, null
%cond2 = icmp eq ptr addrspace(0) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(0) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(0) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(0) %ptr, align 8
ret void
}
;; We should not introduce ptrtoint instructions with unstable pointers
define void @test_default_inttoptr_base(ptr addrspace(0) align 8 %ptr) {
; CHECK-LABEL: define void @test_default_inttoptr_base(
; CHECK-SAME: ptr align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 4
; CHECK-NEXT: br i1 [[COND]], label %[[TRUE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr [[PTR]], align 8
; CHECK-NEXT: store i64 3, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET:.*]]
; CHECK: [[COMMON_RET]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(0) %ptr, inttoptr (i32 4 to ptr addrspace(0))
%cond2 = icmp eq ptr addrspace(0) %ptr, inttoptr (i32 4 to ptr addrspace(0))
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(0) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(0) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(0) %ptr, align 8
ret void
}
;; We should not introduce ptrtoint instructions with unstable pointers
define void @test_default_mixed_base(ptr addrspace(0) align 8 %ptr) {
; CHECK-LABEL: define void @test_default_mixed_base(
; CHECK-SAME: ptr align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[COND2:%.*]] = icmp eq ptr [[PTR]], null
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 4
; CHECK-NEXT: br i1 [[COND]], label %[[FALSE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr [[PTR]], align 8
; CHECK-NEXT: br i1 [[COND2]], label %[[TRUE2:.*]], label %[[FALSE2]]
; CHECK: [[COMMON_RET:.*]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
; CHECK: [[FALSE2]]:
; CHECK-NEXT: store i64 3, ptr [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(0) %ptr, inttoptr (i32 4 to ptr addrspace(0))
%cond2 = icmp eq ptr addrspace(0) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(0) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(0) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(0) %ptr, align 8
ret void
}
;; We should not introduce ptrtoint instructions with unstable pointers
define void @test_unstable_null_base(ptr addrspace(1) align 8 %ptr) {
; CHECK-LABEL: define void @test_unstable_null_base(
; CHECK-SAME: ptr addrspace(1) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[COND1:%.*]] = icmp eq ptr addrspace(1) [[PTR]], null
; CHECK-NEXT: [[COND2:%.*]] = icmp eq ptr addrspace(1) [[PTR]], null
; CHECK-NEXT: br i1 [[COND1]], label %[[TRUE1:.*]], label %[[FALSE1:.*]]
; CHECK: [[TRUE1]]:
; CHECK-NEXT: br i1 [[COND2]], label %[[TRUE2:.*]], label %[[FALSE2:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[TRUE1]]
; CHECK: [[COMMON_RET:.*]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
; CHECK: [[FALSE2]]:
; CHECK-NEXT: store i64 3, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(1) %ptr, null
%cond2 = icmp eq ptr addrspace(1) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(1) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(1) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(1) %ptr, align 8
ret void
}
;; We should not introduce ptrtoint instructions with unstable pointers
define void @test_unstable_inttoptr_base(ptr addrspace(1) align 8 %ptr) {
; CHECK-LABEL: define void @test_unstable_inttoptr_base(
; CHECK-SAME: ptr addrspace(1) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[COND1:%.*]] = icmp eq ptr addrspace(1) [[PTR]], inttoptr (i32 4 to ptr addrspace(1))
; CHECK-NEXT: [[COND2:%.*]] = icmp eq ptr addrspace(1) [[PTR]], inttoptr (i32 4 to ptr addrspace(1))
; CHECK-NEXT: br i1 [[COND1]], label %[[TRUE1:.*]], label %[[FALSE1:.*]]
; CHECK: [[TRUE1]]:
; CHECK-NEXT: br i1 [[COND2]], label %[[TRUE2:.*]], label %[[FALSE2:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[TRUE1]]
; CHECK: [[COMMON_RET:.*]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
; CHECK: [[FALSE2]]:
; CHECK-NEXT: store i64 3, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(1) %ptr, inttoptr (i32 4 to ptr addrspace(1))
%cond2 = icmp eq ptr addrspace(1) %ptr, inttoptr (i32 4 to ptr addrspace(1))
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(1) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(1) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(1) %ptr, align 8
ret void
}
;; We should not introduce ptrtoint instructions with unstable pointers
define void @test_unstable_mixed_base(ptr addrspace(1) align 8 %ptr) {
; CHECK-LABEL: define void @test_unstable_mixed_base(
; CHECK-SAME: ptr addrspace(1) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[COND1:%.*]] = icmp eq ptr addrspace(1) [[PTR]], inttoptr (i32 4 to ptr addrspace(1))
; CHECK-NEXT: [[COND2:%.*]] = icmp eq ptr addrspace(1) [[PTR]], null
; CHECK-NEXT: br i1 [[COND1]], label %[[TRUE1:.*]], label %[[FALSE1:.*]]
; CHECK: [[TRUE1]]:
; CHECK-NEXT: br i1 [[COND2]], label %[[TRUE2:.*]], label %[[FALSE2:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[TRUE1]]
; CHECK: [[COMMON_RET:.*]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
; CHECK: [[FALSE2]]:
; CHECK-NEXT: store i64 3, ptr addrspace(1) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(1) %ptr, inttoptr (i32 4 to ptr addrspace(1))
%cond2 = icmp eq ptr addrspace(1) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(1) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(1) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(1) %ptr, align 8
ret void
}
;; This transformation is fine for pointers with external state.
;; TODO: it would probably be better to just emit a pointer compare against null.
define void @test_external_null_base(ptr addrspace(2) align 8 %ptr) {
; CHECK-LABEL: define void @test_external_null_base(
; CHECK-SAME: ptr addrspace(2) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr addrspace(2) [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 0
; CHECK-NEXT: br i1 [[COND]], label %[[TRUE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: store i64 3, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET:.*]]
; CHECK: [[COMMON_RET]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(2) %ptr, null
%cond2 = icmp eq ptr addrspace(2) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(2) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(2) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(2) %ptr, align 8
ret void
}
;; This transformation is fine for pointers with external state (even with inttoptr).
define void @test_external_inttoptr_base(ptr addrspace(2) align 8 %ptr) {
; CHECK-LABEL: define void @test_external_inttoptr_base(
; CHECK-SAME: ptr addrspace(2) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr addrspace(2) [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 4
; CHECK-NEXT: br i1 [[COND]], label %[[TRUE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: store i64 3, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET:.*]]
; CHECK: [[COMMON_RET]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(2) %ptr, inttoptr (i32 4 to ptr addrspace(2))
%cond2 = icmp eq ptr addrspace(2) %ptr, inttoptr (i32 4 to ptr addrspace(2))
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(2) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(2) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(2) %ptr, align 8
ret void
}
;; This transformation is fine for pointers with external state (even with inttoptr).
define void @test_external_mixed_base(ptr addrspace(2) align 8 %ptr) {
; CHECK-LABEL: define void @test_external_mixed_base(
; CHECK-SAME: ptr addrspace(2) align 8 [[PTR:%.*]]) {
; CHECK-NEXT: [[COND2:%.*]] = icmp eq ptr addrspace(2) [[PTR]], null
; CHECK-NEXT: [[MAGICPTR:%.*]] = ptrtoint ptr addrspace(2) [[PTR]] to i64
; CHECK-NEXT: [[COND:%.*]] = icmp eq i64 [[MAGICPTR]], 4
; CHECK-NEXT: br i1 [[COND]], label %[[FALSE2:.*]], label %[[FALSE1:.*]]
; CHECK: [[FALSE1]]:
; CHECK-NEXT: store i64 1, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br i1 [[COND2]], label %[[TRUE2:.*]], label %[[FALSE2]]
; CHECK: [[COMMON_RET:.*]]:
; CHECK-NEXT: ret void
; CHECK: [[TRUE2]]:
; CHECK-NEXT: store i64 2, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
; CHECK: [[FALSE2]]:
; CHECK-NEXT: store i64 3, ptr addrspace(2) [[PTR]], align 8
; CHECK-NEXT: br label %[[COMMON_RET]]
;
%cond1 = icmp eq ptr addrspace(2) %ptr, inttoptr (i32 4 to ptr addrspace(2))
%cond2 = icmp eq ptr addrspace(2) %ptr, null
br i1 %cond1, label %true1, label %false1
true1:
br i1 %cond2, label %true2, label %false2
false1:
store i64 1, ptr addrspace(2) %ptr, align 8
br label %true1
true2:
store i64 2, ptr addrspace(2) %ptr, align 8
ret void
false2:
store i64 3, ptr addrspace(2) %ptr, align 8
ret void
}