Manman Ren | 3a16331 | 2013-04-05 17:11:45 +0000 | [diff] [blame] | 1 | // RUN: %clang_cc1 -triple x86_64-apple-darwin -O1 -disable-llvm-optzns %s -emit-llvm -o - | FileCheck %s |
| 2 | // RUN: %clang_cc1 -triple x86_64-apple-darwin -O1 -struct-path-tbaa -disable-llvm-optzns %s -emit-llvm -o - | FileCheck %s -check-prefix=PATH |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 3 | // Test TBAA metadata generated by front-end. |
| 4 | |
Manman Ren | c80e031 | 2013-04-05 19:06:10 +0000 | [diff] [blame] | 5 | typedef unsigned char uint8_t; |
| 6 | typedef unsigned short uint16_t; |
| 7 | typedef unsigned int uint32_t; |
| 8 | typedef unsigned long long uint64_t; |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 9 | typedef struct |
| 10 | { |
| 11 | uint16_t f16; |
| 12 | uint32_t f32; |
| 13 | uint16_t f16_2; |
| 14 | uint32_t f32_2; |
| 15 | } StructA; |
| 16 | typedef struct |
| 17 | { |
| 18 | uint16_t f16; |
| 19 | StructA a; |
| 20 | uint32_t f32; |
| 21 | } StructB; |
| 22 | typedef struct |
| 23 | { |
| 24 | uint16_t f16; |
| 25 | StructB b; |
| 26 | uint32_t f32; |
| 27 | } StructC; |
| 28 | typedef struct |
| 29 | { |
| 30 | uint16_t f16; |
| 31 | StructB b; |
| 32 | uint32_t f32; |
| 33 | uint8_t f8; |
| 34 | } StructD; |
| 35 | |
| 36 | typedef struct |
| 37 | { |
| 38 | uint16_t f16; |
| 39 | uint32_t f32; |
| 40 | } StructS; |
| 41 | typedef struct |
| 42 | { |
| 43 | uint16_t f16; |
| 44 | uint32_t f32; |
| 45 | } StructS2; |
| 46 | |
| 47 | uint32_t g(uint32_t *s, StructA *A, uint64_t count) { |
| 48 | // CHECK: define i32 @{{.*}}( |
| 49 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 50 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 51 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 52 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_i32:!.*]] |
| 53 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 54 | *s = 1; |
| 55 | A->f32 = 4; |
| 56 | return *s; |
| 57 | } |
| 58 | |
| 59 | uint32_t g2(uint32_t *s, StructA *A, uint64_t count) { |
| 60 | // CHECK: define i32 @{{.*}}( |
| 61 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 62 | // CHECK: store i16 4, i16* %{{.*}}, align 2, !tbaa !5 |
| 63 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 64 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_i32]] |
| 65 | // PATH: store i16 4, i16* %{{.*}}, align 2, !tbaa [[TAG_A_f16:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 66 | *s = 1; |
| 67 | A->f16 = 4; |
| 68 | return *s; |
| 69 | } |
| 70 | |
| 71 | uint32_t g3(StructA *A, StructB *B, uint64_t count) { |
| 72 | // CHECK: define i32 @{{.*}}( |
| 73 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 74 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 75 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 76 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 77 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_B_a_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 78 | A->f32 = 1; |
| 79 | B->a.f32 = 4; |
| 80 | return A->f32; |
| 81 | } |
| 82 | |
| 83 | uint32_t g4(StructA *A, StructB *B, uint64_t count) { |
| 84 | // CHECK: define i32 @{{.*}}( |
| 85 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 86 | // CHECK: store i16 4, i16* %{{.*}}, align 2, !tbaa !5 |
| 87 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 88 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 89 | // PATH: store i16 4, i16* %{{.*}}, align 2, !tbaa [[TAG_B_a_f16:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 90 | A->f32 = 1; |
| 91 | B->a.f16 = 4; |
| 92 | return A->f32; |
| 93 | } |
| 94 | |
| 95 | uint32_t g5(StructA *A, StructB *B, uint64_t count) { |
| 96 | // CHECK: define i32 @{{.*}}( |
| 97 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 98 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 99 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 100 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 101 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_B_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 102 | A->f32 = 1; |
| 103 | B->f32 = 4; |
| 104 | return A->f32; |
| 105 | } |
| 106 | |
| 107 | uint32_t g6(StructA *A, StructB *B, uint64_t count) { |
| 108 | // CHECK: define i32 @{{.*}}( |
| 109 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 110 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 111 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 112 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 113 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_B_a_f32_2:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 114 | A->f32 = 1; |
| 115 | B->a.f32_2 = 4; |
| 116 | return A->f32; |
| 117 | } |
| 118 | |
| 119 | uint32_t g7(StructA *A, StructS *S, uint64_t count) { |
| 120 | // CHECK: define i32 @{{.*}}( |
| 121 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 122 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 123 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 124 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 125 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_S_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 126 | A->f32 = 1; |
| 127 | S->f32 = 4; |
| 128 | return A->f32; |
| 129 | } |
| 130 | |
| 131 | uint32_t g8(StructA *A, StructS *S, uint64_t count) { |
| 132 | // CHECK: define i32 @{{.*}}( |
| 133 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 134 | // CHECK: store i16 4, i16* %{{.*}}, align 2, !tbaa !5 |
| 135 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 136 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_A_f32]] |
| 137 | // PATH: store i16 4, i16* %{{.*}}, align 2, !tbaa [[TAG_S_f16:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 138 | A->f32 = 1; |
| 139 | S->f16 = 4; |
| 140 | return A->f32; |
| 141 | } |
| 142 | |
| 143 | uint32_t g9(StructS *S, StructS2 *S2, uint64_t count) { |
| 144 | // CHECK: define i32 @{{.*}}( |
| 145 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 146 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 147 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 148 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_S_f32]] |
| 149 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_S2_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 150 | S->f32 = 1; |
| 151 | S2->f32 = 4; |
| 152 | return S->f32; |
| 153 | } |
| 154 | |
| 155 | uint32_t g10(StructS *S, StructS2 *S2, uint64_t count) { |
| 156 | // CHECK: define i32 @{{.*}}( |
| 157 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 158 | // CHECK: store i16 4, i16* %{{.*}}, align 2, !tbaa !5 |
| 159 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 160 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_S_f32]] |
| 161 | // PATH: store i16 4, i16* %{{.*}}, align 2, !tbaa [[TAG_S2_f16:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 162 | S->f32 = 1; |
| 163 | S2->f16 = 4; |
| 164 | return S->f32; |
| 165 | } |
| 166 | |
| 167 | uint32_t g11(StructC *C, StructD *D, uint64_t count) { |
| 168 | // CHECK: define i32 @{{.*}}( |
| 169 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 170 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 171 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 172 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_C_b_a_f32:!.*]] |
| 173 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_D_b_a_f32:!.*]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 174 | C->b.a.f32 = 1; |
| 175 | D->b.a.f32 = 4; |
| 176 | return C->b.a.f32; |
| 177 | } |
| 178 | |
| 179 | uint32_t g12(StructC *C, StructD *D, uint64_t count) { |
| 180 | // CHECK: define i32 @{{.*}}( |
| 181 | // CHECK: store i32 1, i32* %{{.*}}, align 4, !tbaa !4 |
| 182 | // CHECK: store i32 4, i32* %{{.*}}, align 4, !tbaa !4 |
| 183 | // TODO: differentiate the two accesses. |
| 184 | // PATH: define i32 @{{.*}}( |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 185 | // PATH: store i32 1, i32* %{{.*}}, align 4, !tbaa [[TAG_B_a_f32]] |
| 186 | // PATH: store i32 4, i32* %{{.*}}, align 4, !tbaa [[TAG_B_a_f32]] |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 187 | StructB *b1 = &(C->b); |
| 188 | StructB *b2 = &(D->b); |
| 189 | // b1, b2 have different context. |
| 190 | b1->a.f32 = 1; |
| 191 | b2->a.f32 = 4; |
| 192 | return b1->a.f32; |
| 193 | } |
| 194 | |
Manman Ren | c7f2bfb | 2013-04-22 19:50:07 +0000 | [diff] [blame] | 195 | // Make sure that zero-length bitfield works. |
| 196 | #define ATTR __attribute__ ((ms_struct)) |
| 197 | struct five { |
| 198 | char a; |
| 199 | int :0; /* ignored; prior field is not a bitfield. */ |
| 200 | char b; |
| 201 | char c; |
| 202 | } ATTR; |
| 203 | char g13(struct five *a, struct five *b) { |
| 204 | return a->b; |
| 205 | // CHECK: define signext i8 @{{.*}}( |
| 206 | // CHECK: load i8* %{{.*}}, align 1, !tbaa !1 |
| 207 | // PATH: define signext i8 @{{.*}}( |
| 208 | // PATH: load i8* %{{.*}}, align 1, !tbaa [[TAG_five_b:!.*]] |
| 209 | } |
| 210 | |
| 211 | struct six { |
| 212 | char a; |
| 213 | int :0; |
| 214 | char b; |
| 215 | char c; |
| 216 | }; |
| 217 | char g14(struct six *a, struct six *b) { |
| 218 | // CHECK: define signext i8 @{{.*}}( |
| 219 | // CHECK: load i8* %{{.*}}, align 1, !tbaa !1 |
| 220 | // PATH: define signext i8 @{{.*}}( |
| 221 | // PATH: load i8* %{{.*}}, align 1, !tbaa [[TAG_six_b:!.*]] |
| 222 | return a->b; |
| 223 | } |
| 224 | |
Manman Ren | b3cced9 | 2013-04-05 00:05:09 +0000 | [diff] [blame] | 225 | // CHECK: !1 = metadata !{metadata !"omnipotent char", metadata !2} |
| 226 | // CHECK: !2 = metadata !{metadata !"Simple C/C++ TBAA"} |
| 227 | // CHECK: !4 = metadata !{metadata !"int", metadata !1} |
| 228 | // CHECK: !5 = metadata !{metadata !"short", metadata !1} |
| 229 | |
Manman Ren | ca83518 | 2013-04-11 23:02:56 +0000 | [diff] [blame] | 230 | // PATH: [[TYPE_CHAR:!.*]] = metadata !{metadata !"omnipotent char", i64 0, metadata !3} |
| 231 | // PATH: [[TAG_i32]] = metadata !{metadata [[TYPE_INT:!.*]], metadata [[TYPE_INT]], i64 0} |
| 232 | // PATH: [[TYPE_INT]] = metadata !{metadata !"int", i64 0, metadata [[TYPE_CHAR]]} |
| 233 | // PATH: [[TAG_A_f32]] = metadata !{metadata [[TYPE_A:!.*]], metadata [[TYPE_INT]], i64 4} |
| 234 | // PATH: [[TYPE_A]] = metadata !{metadata !"_ZTS7StructA", i64 0, metadata [[TYPE_SHORT:!.*]], i64 4, metadata [[TYPE_INT]], i64 8, metadata [[TYPE_SHORT]], i64 12, metadata [[TYPE_INT]]} |
| 235 | // PATH: [[TYPE_SHORT:!.*]] = metadata !{metadata !"short", i64 0, metadata [[TYPE_CHAR]]} |
| 236 | // PATH: [[TAG_A_f16]] = metadata !{metadata [[TYPE_A]], metadata [[TYPE_SHORT]], i64 0} |
| 237 | // PATH: [[TAG_B_a_f32]] = metadata !{metadata [[TYPE_B:!.*]], metadata [[TYPE_INT]], i64 8} |
| 238 | // PATH: [[TYPE_B]] = metadata !{metadata !"_ZTS7StructB", i64 0, metadata [[TYPE_SHORT]], i64 4, metadata [[TYPE_A]], i64 20, metadata [[TYPE_INT]]} |
| 239 | // PATH: [[TAG_B_a_f16]] = metadata !{metadata [[TYPE_B]], metadata [[TYPE_SHORT]], i64 4} |
| 240 | // PATH: [[TAG_B_f32]] = metadata !{metadata [[TYPE_B]], metadata [[TYPE_INT]], i64 20} |
| 241 | // PATH: [[TAG_B_a_f32_2]] = metadata !{metadata [[TYPE_B]], metadata [[TYPE_INT]], i64 16} |
| 242 | // PATH: [[TAG_S_f32]] = metadata !{metadata [[TYPE_S:!.*]], metadata [[TYPE_INT]], i64 4} |
| 243 | // PATH: [[TYPE_S]] = metadata !{metadata !"_ZTS7StructS", i64 0, metadata [[TYPE_SHORT]], i64 4, metadata [[TYPE_INT]]} |
| 244 | // PATH: [[TAG_S_f16]] = metadata !{metadata [[TYPE_S]], metadata [[TYPE_SHORT]], i64 0} |
| 245 | // PATH: [[TAG_S2_f32]] = metadata !{metadata [[TYPE_S2:!.*]], metadata [[TYPE_INT]], i64 4} |
| 246 | // PATH: [[TYPE_S2]] = metadata !{metadata !"_ZTS8StructS2", i64 0, metadata [[TYPE_SHORT]], i64 4, metadata [[TYPE_INT]]} |
| 247 | // PATH: [[TAG_S2_f16]] = metadata !{metadata [[TYPE_S2]], metadata [[TYPE_SHORT]], i64 0} |
| 248 | // PATH: [[TAG_C_b_a_f32]] = metadata !{metadata [[TYPE_C:!.*]], metadata [[TYPE_INT]], i64 12} |
| 249 | // PATH: [[TYPE_C]] = metadata !{metadata !"_ZTS7StructC", i64 0, metadata [[TYPE_SHORT]], i64 4, metadata [[TYPE_B]], i64 28, metadata [[TYPE_INT]]} |
| 250 | // PATH: [[TAG_D_b_a_f32]] = metadata !{metadata [[TYPE_D:!.*]], metadata [[TYPE_INT]], i64 12} |
| 251 | // PATH: [[TYPE_D]] = metadata !{metadata !"_ZTS7StructD", i64 0, metadata [[TYPE_SHORT]], i64 4, metadata [[TYPE_B]], i64 28, metadata [[TYPE_INT]], i64 32, metadata [[TYPE_CHAR]]} |
Manman Ren | c7f2bfb | 2013-04-22 19:50:07 +0000 | [diff] [blame] | 252 | // PATH: [[TAG_five_b]] = metadata !{metadata [[TYPE_five:!.*]], metadata [[TYPE_CHAR]], i64 1} |
| 253 | // PATH: [[TYPE_five]] = metadata !{metadata !"_ZTS4five", i64 0, metadata [[TYPE_CHAR]], i64 1, metadata [[TYPE_CHAR]], i64 2, metadata [[TYPE_CHAR]]} |
| 254 | // PATH: [[TAG_six_b]] = metadata !{metadata [[TYPE_six:!.*]], metadata [[TYPE_CHAR]], i64 4} |
| 255 | // PATH: [[TYPE_six]] = metadata !{metadata !"_ZTS3six", i64 0, metadata [[TYPE_CHAR]], i64 4, metadata [[TYPE_INT]], i64 4, metadata [[TYPE_CHAR]], i64 5, metadata [[TYPE_CHAR]]} |