Eric Christopher | cee313d | 2019-04-17 04:52:47 +0000 | [diff] [blame] | 1 | ; RUN: opt -codegenprepare -S < %s | FileCheck %s |
| 2 | |
| 3 | target datalayout = "e-i64:64-f80:128-n8:16:32:64-S128" |
| 4 | target triple = "x86_64-pc-linux-gnu" |
| 5 | |
| 6 | declare zeroext i1 @return_i1() |
| 7 | |
| 8 | define i32 @test_sor_basic(i32* %base) gc "statepoint-example" { |
| 9 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 10 | ; CHECK: getelementptr i32, i32* %base-new, i32 15 |
| 11 | entry: |
| 12 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 13 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr) |
| 14 | %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 15 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 16 | %ret = load i32, i32* %ptr-new |
| 17 | ret i32 %ret |
| 18 | } |
| 19 | |
| 20 | define i32 @test_sor_two_derived(i32* %base) gc "statepoint-example" { |
| 21 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 22 | ; CHECK: getelementptr i32, i32* %base, i32 12 |
| 23 | ; CHECK: getelementptr i32, i32* %base-new, i32 12 |
| 24 | ; CHECK: getelementptr i32, i32* %base-new, i32 15 |
| 25 | entry: |
| 26 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 27 | %ptr2 = getelementptr i32, i32* %base, i32 12 |
| 28 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr, i32* %ptr2) |
| 29 | %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 30 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 31 | %ptr2-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 9) |
| 32 | %ret = load i32, i32* %ptr-new |
| 33 | ret i32 %ret |
| 34 | } |
| 35 | |
| 36 | define i32 @test_sor_ooo(i32* %base) gc "statepoint-example" { |
| 37 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 38 | ; CHECK: getelementptr i32, i32* %base-new, i32 15 |
| 39 | entry: |
| 40 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 41 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr) |
| 42 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 43 | %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 44 | %ret = load i32, i32* %ptr-new |
| 45 | ret i32 %ret |
| 46 | } |
| 47 | |
| 48 | define i32 @test_sor_gep_smallint([3 x i32]* %base) gc "statepoint-example" { |
| 49 | ; CHECK: getelementptr [3 x i32], [3 x i32]* %base, i32 0, i32 2 |
| 50 | ; CHECK: getelementptr [3 x i32], [3 x i32]* %base-new, i32 0, i32 2 |
| 51 | entry: |
| 52 | %ptr = getelementptr [3 x i32], [3 x i32]* %base, i32 0, i32 2 |
| 53 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, [3 x i32]* %base, i32* %ptr) |
| 54 | %base-new = call [3 x i32]* @llvm.experimental.gc.relocate.p0a3i32(token %tok, i32 7, i32 7) |
| 55 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 56 | %ret = load i32, i32* %ptr-new |
| 57 | ret i32 %ret |
| 58 | } |
| 59 | |
| 60 | define i32 @test_sor_gep_largeint([3 x i32]* %base) gc "statepoint-example" { |
| 61 | ; CHECK: getelementptr [3 x i32], [3 x i32]* %base, i32 0, i32 21 |
| 62 | ; CHECK-NOT: getelementptr [3 x i32], [3 x i32]* %base-new, i32 0, i32 21 |
| 63 | entry: |
| 64 | %ptr = getelementptr [3 x i32], [3 x i32]* %base, i32 0, i32 21 |
| 65 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, [3 x i32]* %base, i32* %ptr) |
| 66 | %base-new = call [3 x i32]* @llvm.experimental.gc.relocate.p0a3i32(token %tok, i32 7, i32 7) |
| 67 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 68 | %ret = load i32, i32* %ptr-new |
| 69 | ret i32 %ret |
| 70 | } |
| 71 | |
| 72 | define i32 @test_sor_noop(i32* %base) gc "statepoint-example" { |
| 73 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 74 | ; CHECK: call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 75 | ; CHECK: call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 9) |
| 76 | entry: |
| 77 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 78 | %ptr2 = getelementptr i32, i32* %base, i32 12 |
| 79 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr, i32* %ptr2) |
| 80 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 81 | %ptr2-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 9) |
| 82 | %ret = load i32, i32* %ptr-new |
| 83 | ret i32 %ret |
| 84 | } |
| 85 | |
| 86 | define i32 @test_sor_basic_wrong_order(i32* %base) gc "statepoint-example" { |
| 87 | ; CHECK-LABEL: @test_sor_basic_wrong_order |
| 88 | ; Here we have base relocate inserted after derived. Make sure that we don't |
| 89 | ; produce uses of the relocated base pointer before it's definition. |
| 90 | entry: |
| 91 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 92 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 93 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr) |
| 94 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 95 | %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 96 | ; CHECK: %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 97 | ; CHECK-NEXT: getelementptr i32, i32* %base-new, i32 15 |
| 98 | %ret = load i32, i32* %ptr-new |
| 99 | ret i32 %ret |
| 100 | } |
| 101 | |
| 102 | define i32 @test_sor_noop_cross_bb(i1 %external-cond, i32* %base) gc "statepoint-example" { |
| 103 | ; CHECK-LABEL: @test_sor_noop_cross_bb |
| 104 | ; Here base relocate doesn't dominate derived relocate. Make sure that we don't |
| 105 | ; produce undefined use of the relocated base pointer. |
| 106 | entry: |
| 107 | %ptr = getelementptr i32, i32* %base, i32 15 |
| 108 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 109 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr) |
| 110 | br i1 %external-cond, label %left, label %right |
| 111 | |
| 112 | left: |
| 113 | %ptr-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 114 | ; CHECK: call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 115 | %ret-new = load i32, i32* %ptr-new |
| 116 | ret i32 %ret-new |
| 117 | |
| 118 | right: |
| 119 | %ptr-base = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 120 | ; CHECK: call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 121 | %ret-base = load i32, i32* %ptr-base |
| 122 | ret i32 %ret-base |
| 123 | } |
| 124 | |
| 125 | define i32 @test_sor_noop_same_bb(i1 %external-cond, i32* %base) gc "statepoint-example" { |
| 126 | ; CHECK-LABEL: @test_sor_noop_same_bb |
| 127 | ; Here base relocate doesn't dominate derived relocate. Make sure that we don't |
| 128 | ; produce undefined use of the relocated base pointer. |
| 129 | entry: |
| 130 | %ptr1 = getelementptr i32, i32* %base, i32 15 |
| 131 | ; CHECK: getelementptr i32, i32* %base, i32 15 |
| 132 | %ptr2 = getelementptr i32, i32* %base, i32 5 |
| 133 | ; CHECK: getelementptr i32, i32* %base, i32 5 |
| 134 | %tok = call token (i64, i32, i1 ()*, i32, i32, ...) @llvm.experimental.gc.statepoint.p0f_i1f(i64 0, i32 0, i1 ()* @return_i1, i32 0, i32 0, i32 0, i32 0, i32* %base, i32* %ptr1, i32* %ptr2) |
| 135 | ; CHECK: call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 136 | %ptr2-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 9) |
| 137 | %ret2-new = load i32, i32* %ptr2-new |
| 138 | ; CHECK: getelementptr i32, i32* %base-new, i32 5 |
| 139 | %ptr1-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 8) |
| 140 | %ret1-new = load i32, i32* %ptr1-new |
| 141 | ; CHECK: getelementptr i32, i32* %base-new, i32 15 |
| 142 | %base-new = call i32* @llvm.experimental.gc.relocate.p0i32(token %tok, i32 7, i32 7) |
| 143 | %ret-new = add i32 %ret2-new, %ret1-new |
| 144 | ret i32 %ret-new |
| 145 | } |
| 146 | |
| 147 | declare token @llvm.experimental.gc.statepoint.p0f_i1f(i64, i32, i1 ()*, i32, i32, ...) |
| 148 | declare i32* @llvm.experimental.gc.relocate.p0i32(token, i32, i32) |
| 149 | declare [3 x i32]* @llvm.experimental.gc.relocate.p0a3i32(token, i32, i32) |