blob: 2d36bbabcd02e7336f78957341b0439a85c24300 [file]
; Verify that aggregate PHI nodes in loops are correctly lowered to
; OpPhi with composite types, without introducing type mismatches
; between the PHI result and its incoming values.
; RUN: llc -verify-machineinstrs -O0 -mtriple=spirv64 %s -o - | FileCheck %s
; RUN: %if spirv-tools %{ llc -O0 -mtriple=spirv64 %s -o - -filetype=obj | spirv-val %}
; RUN: llc -verify-machineinstrs -O0 -mtriple=spirv-unknown-vulkan-compute %s -o - | FileCheck %s
; RUN: %if spirv-tools %{ llc -O0 -mtriple=spirv-unknown-vulkan-compute %s -o - -filetype=obj | spirv-val %}
; CHECK-DAG: %[[#ARRTY:]] = OpTypeArray
; CHECK-DAG: %[[#ARRVECTY:]] = OpTypeArray
; CHECK-DAG: %[[#STRUCTTY:]] = OpTypeStruct
; CHECK-DAG: %[[#PHI1:]] = OpPhi %[[#ARRTY]]
; CHECK-DAG: %[[#]] = OpPhi %[[#]]
; CHECK: %[[#]] = OpCompositeExtract %[[#]] %[[#PHI1]]
; CHECK: %[[#]] = OpFAdd
; CHECK: %[[#]] = OpCompositeInsert %[[#ARRTY]]
define void @loop_phi_single_element_array_scalar(ptr addrspace(1) %out) {
entry:
br label %loop
loop:
%i = phi i32 [ %next, %loop ], [ 0, %entry ]
%agg = phi [1 x float] [ %agg.new, %loop ], [ zeroinitializer, %entry ]
%prev = extractvalue [1 x float] %agg, 0
%sum = fadd float %prev, 1.0
%agg.new = insertvalue [1 x float] poison, float %sum, 0
%next = add i32 %i, 1
%cond = icmp slt i32 %next, 64
br i1 %cond, label %loop, label %exit
exit:
%final = extractvalue [1 x float] %agg, 0
store float %final, ptr addrspace(1) %out, align 4
ret void
}
; CHECK-DAG: %[[#PHI2:]] = OpPhi %[[#ARRVECTY]]
; CHECK-DAG: %[[#]] = OpPhi %[[#]]
; CHECK: %[[#]] = OpCompositeExtract %[[#]] %[[#PHI2]]
; CHECK: %[[#]] = OpFAdd
; CHECK: %[[#]] = OpCompositeInsert %[[#ARRVECTY]]
define void @loop_phi_single_element_array_vector(ptr addrspace(1) %out) {
entry:
br label %loop
loop:
%i = phi i32 [ %next, %loop ], [ 0, %entry ]
%agg = phi [1 x <4 x float>] [ %agg.new, %loop ], [ zeroinitializer, %entry ]
%prev = extractvalue [1 x <4 x float>] %agg, 0
%sum = fadd <4 x float> %prev, <float 1.0, float 1.0, float 1.0, float 1.0>
%agg.new = insertvalue [1 x <4 x float>] poison, <4 x float> %sum, 0
%next = add i32 %i, 1
%cond = icmp slt i32 %next, 64
br i1 %cond, label %loop, label %exit
exit:
%final = extractvalue [1 x <4 x float>] %agg, 0
store <4 x float> %final, ptr addrspace(1) %out, align 16
ret void
}
; CHECK-DAG: %[[#PHI3:]] = OpPhi %[[#ARRTY]]
; CHECK-DAG: %[[#]] = OpPhi %[[#]]
; CHECK: %[[#]] = OpCompositeExtract %[[#]] %[[#PHI3]]
; CHECK: %[[#]] = OpFAdd
; CHECK: %[[#]] = OpCompositeInsert %[[#ARRTY]]
define void @loop_phi_insert_into_agg(ptr addrspace(1) %out) {
entry:
br label %loop
loop:
%i = phi i32 [ %next, %loop ], [ 0, %entry ]
%agg = phi [1 x float] [ %agg.new, %loop ], [ zeroinitializer, %entry ]
%prev = extractvalue [1 x float] %agg, 0
%sum = fadd float %prev, 1.0
%agg.new = insertvalue [1 x float] %agg, float %sum, 0
%next = add i32 %i, 1
%cond = icmp slt i32 %next, 64
br i1 %cond, label %loop, label %exit
exit:
%final = extractvalue [1 x float] %agg, 0
store float %final, ptr addrspace(1) %out, align 4
ret void
}
; CHECK-DAG: %[[#PHI4:]] = OpPhi %[[#STRUCTTY]]
; CHECK-DAG: %[[#]] = OpPhi %[[#]]
; CHECK: %[[#]] = OpCompositeExtract %[[#]] %[[#PHI4]]
; CHECK: %[[#]] = OpFAdd
; CHECK: %[[#]] = OpCompositeInsert %[[#STRUCTTY]]
define void @loop_phi_struct(ptr addrspace(1) %out) {
entry:
br label %loop
loop:
%i = phi i32 [ %next, %loop ], [ 0, %entry ]
%agg = phi {float} [ %agg.new, %loop ], [ zeroinitializer, %entry ]
%prev = extractvalue {float} %agg, 0
%sum = fadd float %prev, 1.0
%agg.new = insertvalue {float} poison, float %sum, 0
%next = add i32 %i, 1
%cond = icmp slt i32 %next, 64
br i1 %cond, label %loop, label %exit
exit:
%final = extractvalue {float} %agg, 0
store float %final, ptr addrspace(1) %out, align 4
ret void
}