Home | History | Annotate | Download | only in AMDGPU
      1 ; RUN: opt -mtriple=amdgcn-amd-amdhsa -load-store-vectorizer -S -o - %s | FileCheck %s
      2 ; Copy of test/CodeGen/AMDGPU/merge-stores.ll with some additions
      3 
      4 target datalayout = "e-p:32:32-p1:64:64-p2:64:64-p3:32:32-p4:64:64-p5:32:32-p24:64:64-i64:64-v16:16-v24:32-v32:32-v48:64-v96:128-v192:256-v256:256-v512:512-v1024:1024-v2048:2048-n32:64"
      5 
      6 ; TODO: Vector element tests
      7 ; TODO: Non-zero base offset for load and store combinations
      8 ; TODO: Same base addrspacecasted
      9 
     10 
     11 ; CHECK-LABEL: @merge_global_store_2_constants_i8(
     12 ; CHECK: store <2 x i8> <i8 -56, i8 123>, <2 x i8> addrspace(1)* %{{[0-9]+}}, align 2
     13 define void @merge_global_store_2_constants_i8(i8 addrspace(1)* %out) #0 {
     14   %out.gep.1 = getelementptr i8, i8 addrspace(1)* %out, i32 1
     15 
     16   store i8 123, i8 addrspace(1)* %out.gep.1
     17   store i8 456, i8 addrspace(1)* %out, align 2
     18   ret void
     19 }
     20 
     21 ; CHECK-LABEL: @merge_global_store_2_constants_i8_natural_align
     22 ; CHECK: store <2 x i8>
     23 define void @merge_global_store_2_constants_i8_natural_align(i8 addrspace(1)* %out) #0 {
     24   %out.gep.1 = getelementptr i8, i8 addrspace(1)* %out, i32 1
     25 
     26   store i8 123, i8 addrspace(1)* %out.gep.1
     27   store i8 456, i8 addrspace(1)* %out
     28   ret void
     29 }
     30 
     31 ; CHECK-LABEL: @merge_global_store_2_constants_i16
     32 ; CHECK: store <2 x i16> <i16 456, i16 123>, <2 x i16> addrspace(1)* %{{[0-9]+}}, align 4
     33 define void @merge_global_store_2_constants_i16(i16 addrspace(1)* %out) #0 {
     34   %out.gep.1 = getelementptr i16, i16 addrspace(1)* %out, i32 1
     35 
     36   store i16 123, i16 addrspace(1)* %out.gep.1
     37   store i16 456, i16 addrspace(1)* %out, align 4
     38   ret void
     39 }
     40 
     41 ; CHECK-LABEL: @merge_global_store_2_constants_0_i16
     42 ; CHECK: store <2 x i16> zeroinitializer, <2 x i16> addrspace(1)* %{{[0-9]+}}, align 4
     43 define void @merge_global_store_2_constants_0_i16(i16 addrspace(1)* %out) #0 {
     44   %out.gep.1 = getelementptr i16, i16 addrspace(1)* %out, i32 1
     45 
     46   store i16 0, i16 addrspace(1)* %out.gep.1
     47   store i16 0, i16 addrspace(1)* %out, align 4
     48   ret void
     49 }
     50 
     51 ; CHECK-LABEL: @merge_global_store_2_constants_i16_natural_align
     52 ; CHECK: store <2 x i16>
     53 define void @merge_global_store_2_constants_i16_natural_align(i16 addrspace(1)* %out) #0 {
     54   %out.gep.1 = getelementptr i16, i16 addrspace(1)* %out, i32 1
     55 
     56   store i16 123, i16 addrspace(1)* %out.gep.1
     57   store i16 456, i16 addrspace(1)* %out
     58   ret void
     59 }
     60 
     61 ; CHECK-LABEL: @merge_global_store_2_constants_half_natural_align
     62 ; CHECK: store <2 x half>
     63 define void @merge_global_store_2_constants_half_natural_align(half addrspace(1)* %out) #0 {
     64   %out.gep.1 = getelementptr half, half addrspace(1)* %out, i32 1
     65 
     66   store half 2.0, half addrspace(1)* %out.gep.1
     67   store half 1.0, half addrspace(1)* %out
     68   ret void
     69 }
     70 
     71 ; CHECK-LABEL: @merge_global_store_2_constants_i32
     72 ; CHECK: store <2 x i32> <i32 456, i32 123>, <2 x i32> addrspace(1)* %{{[0-9]+}}, align 4
     73 define void @merge_global_store_2_constants_i32(i32 addrspace(1)* %out) #0 {
     74   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
     75 
     76   store i32 123, i32 addrspace(1)* %out.gep.1
     77   store i32 456, i32 addrspace(1)* %out
     78   ret void
     79 }
     80 
     81 ; CHECK-LABEL: @merge_global_store_2_constants_i32_f32
     82 ; CHECK: store <2 x i32> <i32 456, i32 1065353216>, <2 x i32> addrspace(1)* %{{[0-9]+}}, align 4
     83 define void @merge_global_store_2_constants_i32_f32(i32 addrspace(1)* %out) #0 {
     84   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
     85   %out.gep.1.bc = bitcast i32 addrspace(1)* %out.gep.1 to float addrspace(1)*
     86   store float 1.0, float addrspace(1)* %out.gep.1.bc
     87   store i32 456, i32 addrspace(1)* %out
     88   ret void
     89 }
     90 
     91 ; CHECK-LABEL: @merge_global_store_2_constants_f32_i32
     92 ; CHECK  store <2 x float> <float 4.000000e+00, float 0x370EC00000000000>, <2 x float> addrspace(1)* %{{[0-9]+$}}
     93 define void @merge_global_store_2_constants_f32_i32(float addrspace(1)* %out) #0 {
     94   %out.gep.1 = getelementptr float, float addrspace(1)* %out, i32 1
     95   %out.gep.1.bc = bitcast float addrspace(1)* %out.gep.1 to i32 addrspace(1)*
     96   store i32 123, i32 addrspace(1)* %out.gep.1.bc
     97   store float 4.0, float addrspace(1)* %out
     98   ret void
     99 }
    100 
    101 ; CHECK-LABEL: @merge_global_store_4_constants_i32
    102 ; CHECK: store <4 x i32> <i32 1234, i32 123, i32 456, i32 333>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    103 define void @merge_global_store_4_constants_i32(i32 addrspace(1)* %out) #0 {
    104   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    105   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    106   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    107 
    108   store i32 123, i32 addrspace(1)* %out.gep.1
    109   store i32 456, i32 addrspace(1)* %out.gep.2
    110   store i32 333, i32 addrspace(1)* %out.gep.3
    111   store i32 1234, i32 addrspace(1)* %out
    112   ret void
    113 }
    114 
    115 ; CHECK-LABEL: @merge_global_store_4_constants_f32_order
    116 ; CHECK: store <4 x float> <float 8.000000e+00, float 1.000000e+00, float 2.000000e+00, float 4.000000e+00>, <4 x float> addrspace(1)* %{{[0-9]+}}
    117 define void @merge_global_store_4_constants_f32_order(float addrspace(1)* %out) #0 {
    118   %out.gep.1 = getelementptr float, float addrspace(1)* %out, i32 1
    119   %out.gep.2 = getelementptr float, float addrspace(1)* %out, i32 2
    120   %out.gep.3 = getelementptr float, float addrspace(1)* %out, i32 3
    121 
    122   store float 8.0, float addrspace(1)* %out
    123   store float 1.0, float addrspace(1)* %out.gep.1
    124   store float 2.0, float addrspace(1)* %out.gep.2
    125   store float 4.0, float addrspace(1)* %out.gep.3
    126   ret void
    127 }
    128 
    129 ; First store is out of order.
    130 ; CHECK-LABEL: @merge_global_store_4_constants_f32
    131 ; CHECK: store <4 x float> <float 8.000000e+00, float 1.000000e+00, float 2.000000e+00, float 4.000000e+00>, <4 x float> addrspace(1)* %{{[0-9]+}}, align 4
    132 define void @merge_global_store_4_constants_f32(float addrspace(1)* %out) #0 {
    133   %out.gep.1 = getelementptr float, float addrspace(1)* %out, i32 1
    134   %out.gep.2 = getelementptr float, float addrspace(1)* %out, i32 2
    135   %out.gep.3 = getelementptr float, float addrspace(1)* %out, i32 3
    136 
    137   store float 1.0, float addrspace(1)* %out.gep.1
    138   store float 2.0, float addrspace(1)* %out.gep.2
    139   store float 4.0, float addrspace(1)* %out.gep.3
    140   store float 8.0, float addrspace(1)* %out
    141   ret void
    142 }
    143 
    144 ; CHECK-LABEL: @merge_global_store_4_constants_mixed_i32_f32
    145 ; CHECK: store <4 x i32> <i32 1090519040, i32 11, i32 1073741824, i32 17>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    146 define void @merge_global_store_4_constants_mixed_i32_f32(float addrspace(1)* %out) #0 {
    147   %out.gep.1 = getelementptr float, float addrspace(1)* %out, i32 1
    148   %out.gep.2 = getelementptr float, float addrspace(1)* %out, i32 2
    149   %out.gep.3 = getelementptr float, float addrspace(1)* %out, i32 3
    150 
    151   %out.gep.1.bc = bitcast float addrspace(1)* %out.gep.1 to i32 addrspace(1)*
    152   %out.gep.3.bc = bitcast float addrspace(1)* %out.gep.3 to i32 addrspace(1)*
    153 
    154   store i32 11, i32 addrspace(1)* %out.gep.1.bc
    155   store float 2.0, float addrspace(1)* %out.gep.2
    156   store i32 17, i32 addrspace(1)* %out.gep.3.bc
    157   store float 8.0, float addrspace(1)* %out
    158   ret void
    159 }
    160 
    161 ; CHECK-LABEL: @merge_global_store_3_constants_i32
    162 ; CHECK: store <3 x i32> <i32 1234, i32 123, i32 456>, <3 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    163 define void @merge_global_store_3_constants_i32(i32 addrspace(1)* %out) #0 {
    164   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    165   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    166 
    167   store i32 123, i32 addrspace(1)* %out.gep.1
    168   store i32 456, i32 addrspace(1)* %out.gep.2
    169   store i32 1234, i32 addrspace(1)* %out
    170   ret void
    171 }
    172 
    173 ; CHECK-LABEL: @merge_global_store_2_constants_i64
    174 ; CHECK: store <2 x i64> <i64 456, i64 123>, <2 x i64> addrspace(1)* %{{[0-9]+}}, align 8
    175 define void @merge_global_store_2_constants_i64(i64 addrspace(1)* %out) #0 {
    176   %out.gep.1 = getelementptr i64, i64 addrspace(1)* %out, i64 1
    177 
    178   store i64 123, i64 addrspace(1)* %out.gep.1
    179   store i64 456, i64 addrspace(1)* %out
    180   ret void
    181 }
    182 
    183 ; CHECK-LABEL: @merge_global_store_4_constants_i64
    184 ; CHECK: store <2 x i64> <i64 456, i64 333>, <2 x i64> addrspace(1)* %{{[0-9]+}}, align 8
    185 ; CHECK: store <2 x i64> <i64 1234, i64 123>, <2 x i64> addrspace(1)* %{{[0-9]+}}, align 8
    186 define void @merge_global_store_4_constants_i64(i64 addrspace(1)* %out) #0 {
    187   %out.gep.1 = getelementptr i64, i64 addrspace(1)* %out, i64 1
    188   %out.gep.2 = getelementptr i64, i64 addrspace(1)* %out, i64 2
    189   %out.gep.3 = getelementptr i64, i64 addrspace(1)* %out, i64 3
    190 
    191   store i64 123, i64 addrspace(1)* %out.gep.1
    192   store i64 456, i64 addrspace(1)* %out.gep.2
    193   store i64 333, i64 addrspace(1)* %out.gep.3
    194   store i64 1234, i64 addrspace(1)* %out
    195   ret void
    196 }
    197 
    198 ; CHECK-LABEL: @merge_global_store_2_adjacent_loads_i32
    199 ; CHECK: [[LOAD:%[0-9]+]] = load <2 x i32>
    200 ; CHECK: [[ELT0:%[0-9]+]] = extractelement <2 x i32> [[LOAD]], i32 0
    201 ; CHECK: [[ELT1:%[0-9]+]] = extractelement <2 x i32> [[LOAD]], i32 1
    202 ; CHECK: [[INSERT0:%[0-9]+]] = insertelement <2 x i32> undef, i32 [[ELT0]], i32 0
    203 ; CHECK: [[INSERT1:%[0-9]+]] = insertelement <2 x i32> [[INSERT0]], i32 [[ELT1]], i32 1
    204 ; CHECK: store <2 x i32> [[INSERT1]]
    205 define void @merge_global_store_2_adjacent_loads_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    206   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    207   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    208 
    209   %lo = load i32, i32 addrspace(1)* %in
    210   %hi = load i32, i32 addrspace(1)* %in.gep.1
    211 
    212   store i32 %lo, i32 addrspace(1)* %out
    213   store i32 %hi, i32 addrspace(1)* %out.gep.1
    214   ret void
    215 }
    216 
    217 ; CHECK-LABEL: @merge_global_store_2_adjacent_loads_i32_nonzero_base
    218 ; CHECK: extractelement
    219 ; CHECK: extractelement
    220 ; CHECK: insertelement
    221 ; CHECK: insertelement
    222 ; CHECK: store <2 x i32>
    223 define void @merge_global_store_2_adjacent_loads_i32_nonzero_base(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    224   %in.gep.0 = getelementptr i32, i32 addrspace(1)* %in, i32 2
    225   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 3
    226 
    227   %out.gep.0 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    228   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    229   %lo = load i32, i32 addrspace(1)* %in.gep.0
    230   %hi = load i32, i32 addrspace(1)* %in.gep.1
    231 
    232   store i32 %lo, i32 addrspace(1)* %out.gep.0
    233   store i32 %hi, i32 addrspace(1)* %out.gep.1
    234   ret void
    235 }
    236 
    237 ; CHECK-LABEL: @merge_global_store_2_adjacent_loads_shuffle_i32
    238 ; CHECK: [[LOAD:%[0-9]+]] = load <2 x i32>
    239 ; CHECK: [[ELT0:%[0-9]+]] = extractelement <2 x i32> [[LOAD]], i32 0
    240 ; CHECK: [[ELT1:%[0-9]+]] = extractelement <2 x i32> [[LOAD]], i32 1
    241 ; CHECK: [[INSERT0:%[0-9]+]] = insertelement <2 x i32> undef, i32 [[ELT1]], i32 0
    242 ; CHECK: [[INSERT1:%[0-9]+]] = insertelement <2 x i32> [[INSERT0]], i32 [[ELT0]], i32 1
    243 ; CHECK: store <2 x i32> [[INSERT1]]
    244 define void @merge_global_store_2_adjacent_loads_shuffle_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    245   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    246   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    247 
    248   %lo = load i32, i32 addrspace(1)* %in
    249   %hi = load i32, i32 addrspace(1)* %in.gep.1
    250 
    251   store i32 %hi, i32 addrspace(1)* %out
    252   store i32 %lo, i32 addrspace(1)* %out.gep.1
    253   ret void
    254 }
    255 
    256 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_i32
    257 ; CHECK: load <4 x i32>
    258 ; CHECK: store <4 x i32>
    259 define void @merge_global_store_4_adjacent_loads_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    260   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    261   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    262   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    263   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    264   %in.gep.2 = getelementptr i32, i32 addrspace(1)* %in, i32 2
    265   %in.gep.3 = getelementptr i32, i32 addrspace(1)* %in, i32 3
    266 
    267   %x = load i32, i32 addrspace(1)* %in
    268   %y = load i32, i32 addrspace(1)* %in.gep.1
    269   %z = load i32, i32 addrspace(1)* %in.gep.2
    270   %w = load i32, i32 addrspace(1)* %in.gep.3
    271 
    272   store i32 %x, i32 addrspace(1)* %out
    273   store i32 %y, i32 addrspace(1)* %out.gep.1
    274   store i32 %z, i32 addrspace(1)* %out.gep.2
    275   store i32 %w, i32 addrspace(1)* %out.gep.3
    276   ret void
    277 }
    278 
    279 ; CHECK-LABEL: @merge_global_store_3_adjacent_loads_i32
    280 ; CHECK: load <3 x i32>
    281 ; CHECK: store <3 x i32>
    282 define void @merge_global_store_3_adjacent_loads_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    283   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    284   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    285   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    286   %in.gep.2 = getelementptr i32, i32 addrspace(1)* %in, i32 2
    287 
    288   %x = load i32, i32 addrspace(1)* %in
    289   %y = load i32, i32 addrspace(1)* %in.gep.1
    290   %z = load i32, i32 addrspace(1)* %in.gep.2
    291 
    292   store i32 %x, i32 addrspace(1)* %out
    293   store i32 %y, i32 addrspace(1)* %out.gep.1
    294   store i32 %z, i32 addrspace(1)* %out.gep.2
    295   ret void
    296 }
    297 
    298 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_f32
    299 ; CHECK: load <4 x float>
    300 ; CHECK: store <4 x float>
    301 define void @merge_global_store_4_adjacent_loads_f32(float addrspace(1)* %out, float addrspace(1)* %in) #0 {
    302   %out.gep.1 = getelementptr float, float addrspace(1)* %out, i32 1
    303   %out.gep.2 = getelementptr float, float addrspace(1)* %out, i32 2
    304   %out.gep.3 = getelementptr float, float addrspace(1)* %out, i32 3
    305   %in.gep.1 = getelementptr float, float addrspace(1)* %in, i32 1
    306   %in.gep.2 = getelementptr float, float addrspace(1)* %in, i32 2
    307   %in.gep.3 = getelementptr float, float addrspace(1)* %in, i32 3
    308 
    309   %x = load float, float addrspace(1)* %in
    310   %y = load float, float addrspace(1)* %in.gep.1
    311   %z = load float, float addrspace(1)* %in.gep.2
    312   %w = load float, float addrspace(1)* %in.gep.3
    313 
    314   store float %x, float addrspace(1)* %out
    315   store float %y, float addrspace(1)* %out.gep.1
    316   store float %z, float addrspace(1)* %out.gep.2
    317   store float %w, float addrspace(1)* %out.gep.3
    318   ret void
    319 }
    320 
    321 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_i32_nonzero_base
    322 ; CHECK: load <4 x i32>
    323 ; CHECK: store <4 x i32>
    324 define void @merge_global_store_4_adjacent_loads_i32_nonzero_base(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    325   %in.gep.0 = getelementptr i32, i32 addrspace(1)* %in, i32 11
    326   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 12
    327   %in.gep.2 = getelementptr i32, i32 addrspace(1)* %in, i32 13
    328   %in.gep.3 = getelementptr i32, i32 addrspace(1)* %in, i32 14
    329   %out.gep.0 = getelementptr i32, i32 addrspace(1)* %out, i32 7
    330   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 8
    331   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 9
    332   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 10
    333 
    334   %x = load i32, i32 addrspace(1)* %in.gep.0
    335   %y = load i32, i32 addrspace(1)* %in.gep.1
    336   %z = load i32, i32 addrspace(1)* %in.gep.2
    337   %w = load i32, i32 addrspace(1)* %in.gep.3
    338 
    339   store i32 %x, i32 addrspace(1)* %out.gep.0
    340   store i32 %y, i32 addrspace(1)* %out.gep.1
    341   store i32 %z, i32 addrspace(1)* %out.gep.2
    342   store i32 %w, i32 addrspace(1)* %out.gep.3
    343   ret void
    344 }
    345 
    346 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_inverse_i32
    347 ; CHECK: load <4 x i32>
    348 ; CHECK: store <4 x i32>
    349 define void @merge_global_store_4_adjacent_loads_inverse_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    350   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    351   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    352   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    353   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    354   %in.gep.2 = getelementptr i32, i32 addrspace(1)* %in, i32 2
    355   %in.gep.3 = getelementptr i32, i32 addrspace(1)* %in, i32 3
    356 
    357   %x = load i32, i32 addrspace(1)* %in
    358   %y = load i32, i32 addrspace(1)* %in.gep.1
    359   %z = load i32, i32 addrspace(1)* %in.gep.2
    360   %w = load i32, i32 addrspace(1)* %in.gep.3
    361 
    362   ; Make sure the barrier doesn't stop this
    363   tail call void @llvm.amdgcn.s.barrier() #1
    364 
    365   store i32 %w, i32 addrspace(1)* %out.gep.3
    366   store i32 %z, i32 addrspace(1)* %out.gep.2
    367   store i32 %y, i32 addrspace(1)* %out.gep.1
    368   store i32 %x, i32 addrspace(1)* %out
    369 
    370   ret void
    371 }
    372 
    373 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_shuffle_i32
    374 ; CHECK: load <4 x i32>
    375 ; CHECK: store <4 x i32>
    376 define void @merge_global_store_4_adjacent_loads_shuffle_i32(i32 addrspace(1)* %out, i32 addrspace(1)* %in) #0 {
    377   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    378   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    379   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    380   %in.gep.1 = getelementptr i32, i32 addrspace(1)* %in, i32 1
    381   %in.gep.2 = getelementptr i32, i32 addrspace(1)* %in, i32 2
    382   %in.gep.3 = getelementptr i32, i32 addrspace(1)* %in, i32 3
    383 
    384   %x = load i32, i32 addrspace(1)* %in
    385   %y = load i32, i32 addrspace(1)* %in.gep.1
    386   %z = load i32, i32 addrspace(1)* %in.gep.2
    387   %w = load i32, i32 addrspace(1)* %in.gep.3
    388 
    389   ; Make sure the barrier doesn't stop this
    390   tail call void @llvm.amdgcn.s.barrier() #1
    391 
    392   store i32 %w, i32 addrspace(1)* %out
    393   store i32 %z, i32 addrspace(1)* %out.gep.1
    394   store i32 %y, i32 addrspace(1)* %out.gep.2
    395   store i32 %x, i32 addrspace(1)* %out.gep.3
    396 
    397   ret void
    398 }
    399 
    400 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_i8
    401 ; CHECK: load <4 x i8>
    402 ; CHECK: extractelement <4 x i8>
    403 ; CHECK: extractelement <4 x i8>
    404 ; CHECK: extractelement <4 x i8>
    405 ; CHECK: extractelement <4 x i8>
    406 ; CHECK: insertelement <4 x i8>
    407 ; CHECK: insertelement <4 x i8>
    408 ; CHECK: insertelement <4 x i8>
    409 ; CHECK: insertelement <4 x i8>
    410 ; CHECK: store <4 x i8>
    411 define void @merge_global_store_4_adjacent_loads_i8(i8 addrspace(1)* %out, i8 addrspace(1)* %in) #0 {
    412   %out.gep.1 = getelementptr i8, i8 addrspace(1)* %out, i8 1
    413   %out.gep.2 = getelementptr i8, i8 addrspace(1)* %out, i8 2
    414   %out.gep.3 = getelementptr i8, i8 addrspace(1)* %out, i8 3
    415   %in.gep.1 = getelementptr i8, i8 addrspace(1)* %in, i8 1
    416   %in.gep.2 = getelementptr i8, i8 addrspace(1)* %in, i8 2
    417   %in.gep.3 = getelementptr i8, i8 addrspace(1)* %in, i8 3
    418 
    419   %x = load i8, i8 addrspace(1)* %in, align 4
    420   %y = load i8, i8 addrspace(1)* %in.gep.1
    421   %z = load i8, i8 addrspace(1)* %in.gep.2
    422   %w = load i8, i8 addrspace(1)* %in.gep.3
    423 
    424   store i8 %x, i8 addrspace(1)* %out, align 4
    425   store i8 %y, i8 addrspace(1)* %out.gep.1
    426   store i8 %z, i8 addrspace(1)* %out.gep.2
    427   store i8 %w, i8 addrspace(1)* %out.gep.3
    428   ret void
    429 }
    430 
    431 ; CHECK-LABEL: @merge_global_store_4_adjacent_loads_i8_natural_align
    432 ; CHECK: load <4 x i8>
    433 ; CHECK: store <4 x i8>
    434 define void @merge_global_store_4_adjacent_loads_i8_natural_align(i8 addrspace(1)* %out, i8 addrspace(1)* %in) #0 {
    435   %out.gep.1 = getelementptr i8, i8 addrspace(1)* %out, i8 1
    436   %out.gep.2 = getelementptr i8, i8 addrspace(1)* %out, i8 2
    437   %out.gep.3 = getelementptr i8, i8 addrspace(1)* %out, i8 3
    438   %in.gep.1 = getelementptr i8, i8 addrspace(1)* %in, i8 1
    439   %in.gep.2 = getelementptr i8, i8 addrspace(1)* %in, i8 2
    440   %in.gep.3 = getelementptr i8, i8 addrspace(1)* %in, i8 3
    441 
    442   %x = load i8, i8 addrspace(1)* %in
    443   %y = load i8, i8 addrspace(1)* %in.gep.1
    444   %z = load i8, i8 addrspace(1)* %in.gep.2
    445   %w = load i8, i8 addrspace(1)* %in.gep.3
    446 
    447   store i8 %x, i8 addrspace(1)* %out
    448   store i8 %y, i8 addrspace(1)* %out.gep.1
    449   store i8 %z, i8 addrspace(1)* %out.gep.2
    450   store i8 %w, i8 addrspace(1)* %out.gep.3
    451   ret void
    452 }
    453 
    454 ; CHECK-LABEL: @merge_global_store_4_vector_elts_loads_v4i32
    455 ; CHECK: load <4 x i32>
    456 ; CHECK: store <4 x i32>
    457 define void @merge_global_store_4_vector_elts_loads_v4i32(i32 addrspace(1)* %out, <4 x i32> addrspace(1)* %in) #0 {
    458   %out.gep.1 = getelementptr i32, i32 addrspace(1)* %out, i32 1
    459   %out.gep.2 = getelementptr i32, i32 addrspace(1)* %out, i32 2
    460   %out.gep.3 = getelementptr i32, i32 addrspace(1)* %out, i32 3
    461   %vec = load <4 x i32>, <4 x i32> addrspace(1)* %in
    462 
    463   %x = extractelement <4 x i32> %vec, i32 0
    464   %y = extractelement <4 x i32> %vec, i32 1
    465   %z = extractelement <4 x i32> %vec, i32 2
    466   %w = extractelement <4 x i32> %vec, i32 3
    467 
    468   store i32 %x, i32 addrspace(1)* %out
    469   store i32 %y, i32 addrspace(1)* %out.gep.1
    470   store i32 %z, i32 addrspace(1)* %out.gep.2
    471   store i32 %w, i32 addrspace(1)* %out.gep.3
    472   ret void
    473 }
    474 
    475 ; CHECK-LABEL: @merge_local_store_2_constants_i8
    476 ; CHECK: store <2 x i8> <i8 -56, i8 123>, <2 x i8> addrspace(3)* %{{[0-9]+}}, align 2
    477 define void @merge_local_store_2_constants_i8(i8 addrspace(3)* %out) #0 {
    478   %out.gep.1 = getelementptr i8, i8 addrspace(3)* %out, i32 1
    479 
    480   store i8 123, i8 addrspace(3)* %out.gep.1
    481   store i8 456, i8 addrspace(3)* %out, align 2
    482   ret void
    483 }
    484 
    485 ; CHECK-LABEL: @merge_local_store_2_constants_i32
    486 ; CHECK: store <2 x i32> <i32 456, i32 123>, <2 x i32> addrspace(3)* %{{[0-9]+}}, align 4
    487 define void @merge_local_store_2_constants_i32(i32 addrspace(3)* %out) #0 {
    488   %out.gep.1 = getelementptr i32, i32 addrspace(3)* %out, i32 1
    489 
    490   store i32 123, i32 addrspace(3)* %out.gep.1
    491   store i32 456, i32 addrspace(3)* %out
    492   ret void
    493 }
    494 
    495 ; CHECK-LABEL: @merge_local_store_2_constants_i32_align_2
    496 ; CHECK: store i32
    497 ; CHECK: store i32
    498 define void @merge_local_store_2_constants_i32_align_2(i32 addrspace(3)* %out) #0 {
    499   %out.gep.1 = getelementptr i32, i32 addrspace(3)* %out, i32 1
    500 
    501   store i32 123, i32 addrspace(3)* %out.gep.1, align 2
    502   store i32 456, i32 addrspace(3)* %out, align 2
    503   ret void
    504 }
    505 
    506 ; CHECK-LABEL: @merge_local_store_4_constants_i32
    507 ; CHECK: store <2 x i32> <i32 456, i32 333>, <2 x i32> addrspace(3)*
    508 ; CHECK: store <2 x i32> <i32 1234, i32 123>, <2 x i32> addrspace(3)*
    509 define void @merge_local_store_4_constants_i32(i32 addrspace(3)* %out) #0 {
    510   %out.gep.1 = getelementptr i32, i32 addrspace(3)* %out, i32 1
    511   %out.gep.2 = getelementptr i32, i32 addrspace(3)* %out, i32 2
    512   %out.gep.3 = getelementptr i32, i32 addrspace(3)* %out, i32 3
    513 
    514   store i32 123, i32 addrspace(3)* %out.gep.1
    515   store i32 456, i32 addrspace(3)* %out.gep.2
    516   store i32 333, i32 addrspace(3)* %out.gep.3
    517   store i32 1234, i32 addrspace(3)* %out
    518   ret void
    519 }
    520 
    521 ; CHECK-LABEL: @merge_global_store_5_constants_i32
    522 ; CHECK: store <4 x i32> <i32 9, i32 12, i32 16, i32 -12>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    523 ; CHECK: store i32
    524 define void @merge_global_store_5_constants_i32(i32 addrspace(1)* %out) {
    525   store i32 9, i32 addrspace(1)* %out, align 4
    526   %idx1 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 1
    527   store i32 12, i32 addrspace(1)* %idx1, align 4
    528   %idx2 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 2
    529   store i32 16, i32 addrspace(1)* %idx2, align 4
    530   %idx3 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 3
    531   store i32 -12, i32 addrspace(1)* %idx3, align 4
    532   %idx4 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 4
    533   store i32 11, i32 addrspace(1)* %idx4, align 4
    534   ret void
    535 }
    536 
    537 ; CHECK-LABEL: @merge_global_store_6_constants_i32
    538 ; CHECK: store <4 x i32> <i32 13, i32 15, i32 62, i32 63>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    539 ; CHECK: store <2 x i32> <i32 11, i32 123>, <2 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    540 define void @merge_global_store_6_constants_i32(i32 addrspace(1)* %out) {
    541   store i32 13, i32 addrspace(1)* %out, align 4
    542   %idx1 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 1
    543   store i32 15, i32 addrspace(1)* %idx1, align 4
    544   %idx2 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 2
    545   store i32 62, i32 addrspace(1)* %idx2, align 4
    546   %idx3 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 3
    547   store i32 63, i32 addrspace(1)* %idx3, align 4
    548   %idx4 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 4
    549   store i32 11, i32 addrspace(1)* %idx4, align 4
    550   %idx5 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 5
    551   store i32 123, i32 addrspace(1)* %idx5, align 4
    552   ret void
    553 }
    554 
    555 ; CHECK-LABEL: @merge_global_store_7_constants_i32
    556 ; CHECK: store <4 x i32> <i32 34, i32 999, i32 65, i32 33>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    557 ; CHECK: store <3 x i32> <i32 98, i32 91, i32 212>, <3 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    558 define void @merge_global_store_7_constants_i32(i32 addrspace(1)* %out) {
    559   store i32 34, i32 addrspace(1)* %out, align 4
    560   %idx1 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 1
    561   store i32 999, i32 addrspace(1)* %idx1, align 4
    562   %idx2 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 2
    563   store i32 65, i32 addrspace(1)* %idx2, align 4
    564   %idx3 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 3
    565   store i32 33, i32 addrspace(1)* %idx3, align 4
    566   %idx4 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 4
    567   store i32 98, i32 addrspace(1)* %idx4, align 4
    568   %idx5 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 5
    569   store i32 91, i32 addrspace(1)* %idx5, align 4
    570   %idx6 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 6
    571   store i32 212, i32 addrspace(1)* %idx6, align 4
    572   ret void
    573 }
    574 
    575 ; CHECK-LABEL: @merge_global_store_8_constants_i32
    576 ; CHECK: store <4 x i32> <i32 34, i32 999, i32 65, i32 33>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    577 ; CHECK: store <4 x i32> <i32 98, i32 91, i32 212, i32 999>, <4 x i32> addrspace(1)* %{{[0-9]+}}, align 4
    578 define void @merge_global_store_8_constants_i32(i32 addrspace(1)* %out) {
    579   store i32 34, i32 addrspace(1)* %out, align 4
    580   %idx1 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 1
    581   store i32 999, i32 addrspace(1)* %idx1, align 4
    582   %idx2 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 2
    583   store i32 65, i32 addrspace(1)* %idx2, align 4
    584   %idx3 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 3
    585   store i32 33, i32 addrspace(1)* %idx3, align 4
    586   %idx4 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 4
    587   store i32 98, i32 addrspace(1)* %idx4, align 4
    588   %idx5 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 5
    589   store i32 91, i32 addrspace(1)* %idx5, align 4
    590   %idx6 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 6
    591   store i32 212, i32 addrspace(1)* %idx6, align 4
    592   %idx7 = getelementptr inbounds i32, i32 addrspace(1)* %out, i64 7
    593   store i32 999, i32 addrspace(1)* %idx7, align 4
    594   ret void
    595 }
    596 
    597 ; CHECK-LABEL: @copy_v3i32_align4
    598 ; CHECK: %vec = load <3 x i32>, <3 x i32> addrspace(1)* %in, align 4
    599 ; CHECK: store <3 x i32> %vec, <3 x i32> addrspace(1)* %out
    600 define void @copy_v3i32_align4(<3 x i32> addrspace(1)* noalias %out, <3 x i32> addrspace(1)* noalias %in) #0 {
    601   %vec = load <3 x i32>, <3 x i32> addrspace(1)* %in, align 4
    602   store <3 x i32> %vec, <3 x i32> addrspace(1)* %out
    603   ret void
    604 }
    605 
    606 ; CHECK-LABEL: @copy_v3i64_align4
    607 ; CHECK: %vec = load <3 x i64>, <3 x i64> addrspace(1)* %in, align 4
    608 ; CHECK: store <3 x i64> %vec, <3 x i64> addrspace(1)* %out
    609 define void @copy_v3i64_align4(<3 x i64> addrspace(1)* noalias %out, <3 x i64> addrspace(1)* noalias %in) #0 {
    610   %vec = load <3 x i64>, <3 x i64> addrspace(1)* %in, align 4
    611   store <3 x i64> %vec, <3 x i64> addrspace(1)* %out
    612   ret void
    613 }
    614 
    615 ; CHECK-LABEL: @copy_v3f32_align4
    616 ; CHECK: %vec = load <3 x float>, <3 x float> addrspace(1)* %in, align 4
    617 ; CHECK: store <3 x float>
    618 define void @copy_v3f32_align4(<3 x float> addrspace(1)* noalias %out, <3 x float> addrspace(1)* noalias %in) #0 {
    619   %vec = load <3 x float>, <3 x float> addrspace(1)* %in, align 4
    620   %fadd = fadd <3 x float> %vec, <float 1.0, float 2.0, float 4.0>
    621   store <3 x float> %fadd, <3 x float> addrspace(1)* %out
    622   ret void
    623 }
    624 
    625 ; CHECK-LABEL: @copy_v3f64_align4
    626 ; CHECK: %vec = load <3 x double>, <3 x double> addrspace(1)* %in, align 4
    627 ; CHECK: store <3 x double> %fadd, <3 x double> addrspace(1)* %out
    628 define void @copy_v3f64_align4(<3 x double> addrspace(1)* noalias %out, <3 x double> addrspace(1)* noalias %in) #0 {
    629   %vec = load <3 x double>, <3 x double> addrspace(1)* %in, align 4
    630   %fadd = fadd <3 x double> %vec, <double 1.0, double 2.0, double 4.0>
    631   store <3 x double> %fadd, <3 x double> addrspace(1)* %out
    632   ret void
    633 }
    634 
    635 declare void @llvm.amdgcn.s.barrier() #1
    636 
    637 attributes #0 = { nounwind }
    638 attributes #1 = { convergent nounwind }
    639