; 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 }