From ffdb454b89f5ef1fa1c6620119bcc6f750a9e233 Mon Sep 17 00:00:00 2001 From: Sven van Haastregt Date: Thu, 15 Feb 2024 14:00:43 +0100 Subject: [PATCH] [Backport to 18] Fix allowed types for OpConstantNull (#2361) The SPIR-V Specification allows `OpConstantNull` types to be scalar or vector booleans, integers, or floats. Update an assert for this and add a SPIR-V -> LLVM IR test. (cherry picked from commit 9ec969c1c379bde69522b3b9278e5f7aa1a2c9f9) --- lib/SPIRV/libSPIRV/SPIRVValue.h | 3 ++- test/OpConstantNull.spvasm | 46 +++++++++++++++++++++++++++++++++ 2 files changed, 48 insertions(+), 1 deletion(-) create mode 100644 test/OpConstantNull.spvasm diff --git a/lib/SPIRV/libSPIRV/SPIRVValue.h b/lib/SPIRV/libSPIRV/SPIRVValue.h index 9b5ef66de1..fb80bd5170 100644 --- a/lib/SPIRV/libSPIRV/SPIRVValue.h +++ b/lib/SPIRV/libSPIRV/SPIRVValue.h @@ -269,7 +269,8 @@ class SPIRVConstantNull : public SPIRVConstantEmpty { protected: void validate() const override { SPIRVConstantEmpty::validate(); - assert((Type->isTypeComposite() || Type->isTypeOpaque() || + assert((Type->isTypeBool() || Type->isTypeInt() || Type->isTypeFloat() || + Type->isTypeComposite() || Type->isTypeOpaque() || Type->isTypeEvent() || Type->isTypePointer() || Type->isTypeReserveId() || Type->isTypeDeviceEvent() || (Type->isTypeSubgroupAvcINTEL() && diff --git a/test/OpConstantNull.spvasm b/test/OpConstantNull.spvasm new file mode 100644 index 0000000000..64e8a3b521 --- /dev/null +++ b/test/OpConstantNull.spvasm @@ -0,0 +1,46 @@ +; REQUIRES: spirv-as +; RUN: spirv-as --target-env spv1.0 -o %t.spv %s +; RUN: spirv-val %t.spv +; RUN: llvm-spirv -r -o - %t.spv | llvm-dis | FileCheck %s + OpCapability Addresses + OpCapability Kernel + OpMemoryModel Physical32 OpenCL + OpEntryPoint Kernel %1 "testNullConstants" + %void = OpTypeVoid + %bool = OpTypeBool + %uint = OpTypeInt 32 0 + %float = OpTypeFloat 32 + %boolv3 = OpTypeVector %bool 3 + %uintv3 = OpTypeVector %uint 3 + %floatv3 = OpTypeVector %float 3 + %5 = OpTypeFunction %void %bool %boolv3 %uint %uintv3 %float %floatv3 + %bool0 = OpConstantNull %bool + %boolv30 = OpConstantNull %boolv3 + %uint0 = OpConstantNull %uint + %uintv30 = OpConstantNull %uintv3 + %float0 = OpConstantNull %float + %floatv30 = OpConstantNull %floatv3 + + %1 = OpFunction %void None %5 + %pb = OpFunctionParameter %bool + %pbv3 = OpFunctionParameter %boolv3 + %pu = OpFunctionParameter %uint + %puv3 = OpFunctionParameter %uintv3 + %pf = OpFunctionParameter %float + %pfv3 = OpFunctionParameter %floatv3 + %entry = OpLabel + %tb = OpLogicalEqual %bool %pb %bool0 + %tbv3 = OpLogicalEqual %boolv3 %pbv3 %boolv30 + %tu = OpIEqual %bool %pu %uint0 + %tuv3 = OpIEqual %boolv3 %puv3 %uintv30 + %tf = OpFOrdEqual %bool %pf %float0 + %tfv3 = OpFOrdEqual %boolv3 %pfv3 %floatv30 + OpReturn + OpFunctionEnd + +; CHECK: icmp eq i1 %[[#]], false +; CHECK: icmp eq <3 x i1> %[[#]], zeroinitializer +; CHECK: icmp eq i32 %[[#]], 0 +; CHECK: icmp eq <3 x i32> %[[#]], zeroinitializer +; CHECK: fcmp oeq float %[[#]], 0.000000e+00 +; CHECK: fcmp oeq <3 x float> %[[#]], zeroinitializer