[InstCombine] Signed saturation patterns
[llvm-complete.git] / test / Transforms / LoadStoreVectorizer / AMDGPU / extended-index.ll
blobb8e95a6793e8ccfd1531c25636dfd5b5b3bfd6e2
1 ; RUN: opt -mtriple=amdgcn-amd-amdhsa -basicaa -load-store-vectorizer -S -o - %s | FileCheck %s
2 ; RUN: opt -mtriple=amdgcn-amd-amdhsa -aa-pipeline=basic-aa -passes='function(load-store-vectorizer)' -S -o - %s | FileCheck %s
4 target datalayout = "e-p:64:64-p1:64:64-p2:32:32-p3:32:32-p4:64:64-p5:32:32-p6:32:32-i64:64-v16:16-v24:32-v32:32-v48:64-v96:128-v192:256-v256:256-v512:512-v1024:1024-v2048:2048-n32:64-S32-A5"
6 declare i32 @llvm.amdgcn.workitem.id.x() #1
8 ; CHECK-LABEL: @basic_merge_sext_index(
9 ; CHECK: sext i32 %id.x to i64
10 ; CHECK: load <2 x float>
11 ; CHECK: store <2 x float> zeroinitializer
12 define amdgpu_kernel void @basic_merge_sext_index(float addrspace(1)* nocapture %a, float addrspace(1)* nocapture %b, float addrspace(1)* nocapture readonly %c) #0 {
13 entry:
14   %id.x = call i32 @llvm.amdgcn.workitem.id.x()
15   %sext.id.x = sext i32 %id.x to i64
16   %a.idx.x = getelementptr inbounds float, float addrspace(1)* %a, i64 %sext.id.x
17   %c.idx.x = getelementptr inbounds float, float addrspace(1)* %c, i64 %sext.id.x
18   %a.idx.x.1 = getelementptr inbounds float, float addrspace(1)* %a.idx.x, i64 1
19   %c.idx.x.1 = getelementptr inbounds float, float addrspace(1)* %c.idx.x, i64 1
21   %ld.c = load float, float addrspace(1)* %c.idx.x, align 4
22   %ld.c.idx.1 = load float, float addrspace(1)* %c.idx.x.1, align 4
24   store float 0.0, float addrspace(1)* %a.idx.x, align 4
25   store float 0.0, float addrspace(1)* %a.idx.x.1, align 4
27   %add = fadd float %ld.c, %ld.c.idx.1
28   store float %add, float addrspace(1)* %b, align 4
29   ret void
32 ; CHECK-LABEL: @basic_merge_zext_index(
33 ; CHECK: zext i32 %id.x to i64
34 ; CHECK: load <2 x float>
35 ; CHECK: store <2 x float>
36 define amdgpu_kernel void @basic_merge_zext_index(float addrspace(1)* nocapture %a, float addrspace(1)* nocapture %b, float addrspace(1)* nocapture readonly %c) #0 {
37 entry:
38   %id.x = call i32 @llvm.amdgcn.workitem.id.x()
39   %zext.id.x = zext i32 %id.x to i64
40   %a.idx.x = getelementptr inbounds float, float addrspace(1)* %a, i64 %zext.id.x
41   %c.idx.x = getelementptr inbounds float, float addrspace(1)* %c, i64 %zext.id.x
42   %a.idx.x.1 = getelementptr inbounds float, float addrspace(1)* %a.idx.x, i64 1
43   %c.idx.x.1 = getelementptr inbounds float, float addrspace(1)* %c.idx.x, i64 1
45   %ld.c = load float, float addrspace(1)* %c.idx.x, align 4
46   %ld.c.idx.1 = load float, float addrspace(1)* %c.idx.x.1, align 4
47   store float 0.0, float addrspace(1)* %a.idx.x, align 4
48   store float 0.0, float addrspace(1)* %a.idx.x.1, align 4
50   %add = fadd float %ld.c, %ld.c.idx.1
51   store float %add, float addrspace(1)* %b, align 4
52   ret void
55 ; CHECK-LABEL: @merge_op_zext_index(
56 ; CHECK: load <2 x float>
57 ; CHECK: store <2 x float>
58 define amdgpu_kernel void @merge_op_zext_index(float addrspace(1)* nocapture noalias %a, float addrspace(1)* nocapture noalias %b, float addrspace(1)* nocapture readonly noalias %c) #0 {
59 entry:
60   %id.x = call i32 @llvm.amdgcn.workitem.id.x()
61   %shl = shl i32 %id.x, 2
62   %zext.id.x = zext i32 %shl to i64
63   %a.0 = getelementptr inbounds float, float addrspace(1)* %a, i64 %zext.id.x
64   %c.0 = getelementptr inbounds float, float addrspace(1)* %c, i64 %zext.id.x
66   %id.x.1 = or i32 %shl, 1
67   %id.x.1.ext = zext i32 %id.x.1 to i64
69   %a.1 = getelementptr inbounds float, float addrspace(1)* %a, i64 %id.x.1.ext
70   %c.1 = getelementptr inbounds float, float addrspace(1)* %c, i64 %id.x.1.ext
72   %ld.c.0 = load float, float addrspace(1)* %c.0, align 4
73   store float 0.0, float addrspace(1)* %a.0, align 4
74   %ld.c.1 = load float, float addrspace(1)* %c.1, align 4
75   store float 0.0, float addrspace(1)* %a.1, align 4
77   %add = fadd float %ld.c.0, %ld.c.1
78   store float %add, float addrspace(1)* %b, align 4
79   ret void
82 ; CHECK-LABEL: @merge_op_sext_index(
83 ; CHECK: load <2 x float>
84 ; CHECK: store <2 x float>
85 define amdgpu_kernel void @merge_op_sext_index(float addrspace(1)* nocapture noalias %a, float addrspace(1)* nocapture noalias %b, float addrspace(1)* nocapture readonly noalias %c) #0 {
86 entry:
87   %id.x = call i32 @llvm.amdgcn.workitem.id.x()
88   %shl = shl i32 %id.x, 2
89   %zext.id.x = sext i32 %shl to i64
90   %a.0 = getelementptr inbounds float, float addrspace(1)* %a, i64 %zext.id.x
91   %c.0 = getelementptr inbounds float, float addrspace(1)* %c, i64 %zext.id.x
93   %id.x.1 = or i32 %shl, 1
94   %id.x.1.ext = sext i32 %id.x.1 to i64
96   %a.1 = getelementptr inbounds float, float addrspace(1)* %a, i64 %id.x.1.ext
97   %c.1 = getelementptr inbounds float, float addrspace(1)* %c, i64 %id.x.1.ext
99   %ld.c.0 = load float, float addrspace(1)* %c.0, align 4
100   store float 0.0, float addrspace(1)* %a.0, align 4
101   %ld.c.1 = load float, float addrspace(1)* %c.1, align 4
102   store float 0.0, float addrspace(1)* %a.1, align 4
104   %add = fadd float %ld.c.0, %ld.c.1
105   store float %add, float addrspace(1)* %b, align 4
106   ret void
109 ; This case fails to vectorize if not using the extra extension
110 ; handling in isConsecutiveAccess.
112 ; CHECK-LABEL: @zext_trunc_phi_1(
113 ; CHECK: loop:
114 ; CHECK: load <2 x i32>
115 ; CHECK: store <2 x i32>
116 define amdgpu_kernel void @zext_trunc_phi_1(i32 addrspace(1)* nocapture noalias %a, i32 addrspace(1)* nocapture noalias %b, i32 addrspace(1)* nocapture readonly noalias %c, i32 %n, i64 %arst, i64 %aoeu) #0 {
117 entry:
118   %cmp0 = icmp eq i32 %n, 0
119   br i1 %cmp0, label %exit, label %loop
121 loop:
122   %indvars.iv = phi i64 [ %indvars.iv.next, %loop ], [ 0, %entry ]
123   %trunc.iv = trunc i64 %indvars.iv to i32
124   %idx = shl i32 %trunc.iv, 4
126   %idx.ext = zext i32 %idx to i64
127   %c.0 = getelementptr inbounds i32, i32 addrspace(1)* %c, i64 %idx.ext
128   %a.0 = getelementptr inbounds i32, i32 addrspace(1)* %a, i64 %idx.ext
130   %idx.1 = or i32 %idx, 1
131   %idx.1.ext = zext i32 %idx.1 to i64
132   %c.1 = getelementptr inbounds i32, i32 addrspace(1)* %c, i64 %idx.1.ext
133   %a.1 = getelementptr inbounds i32, i32 addrspace(1)* %a, i64 %idx.1.ext
135   %ld.c.0 = load i32, i32 addrspace(1)* %c.0, align 4
136   store i32 %ld.c.0, i32 addrspace(1)* %a.0, align 4
137   %ld.c.1 = load i32, i32 addrspace(1)* %c.1, align 4
138   store i32 %ld.c.1, i32 addrspace(1)* %a.1, align 4
140   %indvars.iv.next = add i64 %indvars.iv, 1
141   %lftr.wideiv = trunc i64 %indvars.iv.next to i32
143   %exitcond = icmp eq i32 %lftr.wideiv, %n
144   br i1 %exitcond, label %exit, label %loop
146 exit:
147   ret void
150 attributes #0 = { nounwind }
151 attributes #1 = { nounwind readnone }