llvm-project/llvm/test/CodeGen/SPIRV/logical-struct-access.ll
Steven Perron 35dfeb7b4d
[SPIRV] Enable DCE in instruction selection and update tests (#168428)
The instruction selection pass for SPIR-V now performs dead code
elimination (DCE).
This change removes unused instructions, leading to more optimized
SPIR-V output.

As a consequence of this, several tests were updated to ensure their
continued
correctness and to prevent previously tested code from being optimized
away.
Specifically:
- Many tests now store computed values into global variables to ensure
they are
  not eliminated by DCE, allowing their code generation to be verified.
- The test `keep-tracked-const.ll` was removed because it no longer
tested
its original intent. The check statements in this test were for
constants
generated when expanding a G_TRUNC instruction, which is now removed by
DCE
  instead of being expanded.
- A new test, `remove-dead-type-intrinsics.ll`, was added to confirm
that dead
  struct types are correctly removed by the compiler.

These updates improve the SPIR-V backends optimization capabilities and
maintain the robustness of the test suite.

---------

Co-authored-by: Nathan Gauër <github@keenuts.net>
2025-11-26 09:51:59 -05:00

110 lines
3.5 KiB
LLVM

; RUN: llc -O0 -mtriple=spirv-unknown-vulkan1.3-compute %s -o - | FileCheck %s
; CHECK-DAG: [[uint:%[0-9]+]] = OpTypeInt 32 0
%A = type {
i32,
i32
}
; CHECK-DAG: [[A:%[0-9]+]] = OpTypeStruct [[uint]] [[uint]]
%B = type {
%A,
i32,
%A
}
; CHECK-DAG: [[B:%[0-9]+]] = OpTypeStruct [[A]] [[uint]] [[A]]
; CHECK-DAG: [[uint_0:%[0-9]+]] = OpConstant [[uint]] 0
; CHECK-DAG: [[uint_1:%[0-9]+]] = OpConstant [[uint]] 1
; CHECK-DAG: [[uint_2:%[0-9]+]] = OpConstant [[uint]] 2
; CHECK-DAG: [[ptr_uint:%[0-9]+]] = OpTypePointer Function [[uint]]
; CHECK-DAG: [[ptr_A:%[0-9]+]] = OpTypePointer Function [[A]]
; CHECK-DAG: [[ptr_B:%[0-9]+]] = OpTypePointer Function [[B]]
define internal ptr @gep_B_0(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpAccessChain [[ptr_A]] [[tmp]] [[uint_0]]
%res = getelementptr %B, ptr %base, i32 0, i32 0
ret ptr %res
}
define internal ptr @gep_inbounds_B_0(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpInBoundsAccessChain [[ptr_A]] [[tmp]] [[uint_0]]
%res = getelementptr inbounds %B, ptr %base, i32 0, i32 0
ret ptr %res
}
define internal ptr @gep_B_1(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpAccessChain [[ptr_uint]] [[tmp]] [[uint_1]]
%res = getelementptr %B, ptr %base, i32 0, i32 1
ret ptr %res
}
define internal ptr @gep_inbounds_B_1(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpInBoundsAccessChain [[ptr_uint]] [[tmp]] [[uint_1]]
%res = getelementptr inbounds %B, ptr %base, i32 0, i32 1
ret ptr %res
}
define internal ptr @gep_B_2(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpAccessChain [[ptr_A]] [[tmp]] [[uint_2]]
%res = getelementptr %B, ptr %base, i32 0, i32 2
ret ptr %res
}
define internal ptr @gep_inbounds_B_2(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpInBoundsAccessChain [[ptr_A]] [[tmp]] [[uint_2]]
%res = getelementptr inbounds %B, ptr %base, i32 0, i32 2
ret ptr %res
}
define internal ptr @gep_B_2_1(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpAccessChain [[ptr_uint]] [[tmp]] [[uint_2]] [[uint_1]]
%res = getelementptr %B, ptr %base, i32 0, i32 2, i32 1
ret ptr %res
}
define internal ptr @gep_inbounds_B_2_1(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: {{%[0-9]+}} = OpInBoundsAccessChain [[ptr_uint]] [[tmp]] [[uint_2]] [[uint_1]]
%res = getelementptr inbounds %B, ptr %base, i32 0, i32 2, i32 1
ret ptr %res
}
define internal ptr @gep_B_2_A_1(ptr %base) {
; CHECK: [[tmp:%[0-9]+]] = OpFunctionParameter [[ptr_B]]
; CHECK: [[x:%[0-9]+]] = OpAccessChain [[ptr_A]] [[tmp]] [[uint_2]]
; CHECK: {{%[0-9]+}} = OpAccessChain [[ptr_uint]] [[x]] [[uint_1]]
%x = getelementptr %B, ptr %base, i32 0, i32 2
%res = getelementptr %A, ptr %x, i32 0, i32 1
ret ptr %res
}
define void @main() #1 {
entry:
%0 = alloca %B, align 4
; CHECK: [[tmp:%[0-9]+]] = OpVariable [[ptr_B]] Function
%1 = call ptr @gep_B_0(ptr %0)
%2 = call ptr @gep_inbounds_B_0(ptr %0)
%3 = call ptr @gep_B_1(ptr %0)
%4 = call ptr @gep_inbounds_B_1(ptr %0)
%5 = call ptr @gep_B_2(ptr %0)
%6 = call ptr @gep_inbounds_B_2(ptr %0)
%7 = call ptr @gep_B_2_1(ptr %0)
%8 = call ptr @gep_inbounds_B_2_1(ptr %0)
%10 = call ptr @gep_B_2_A_1(ptr %0)
ret void
}
attributes #1 = { "hlsl.numthreads"="4,8,16" "hlsl.shader"="compute" }