; 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, %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 }