diff --git a/lib/SPIRV/SPIRVReader.cpp b/lib/SPIRV/SPIRVReader.cpp index 6bc007e7a..914906f52 100644 --- a/lib/SPIRV/SPIRVReader.cpp +++ b/lib/SPIRV/SPIRVReader.cpp @@ -4506,6 +4506,25 @@ bool SPIRVToLLVM::transMetadata() { ConstantAsMetadata::get(getUInt32(M, EM->getLiterals()[0])); F->setMetadata(kSPIR2MD::SubgroupSize, MDNode::get(*Context, SizeMD)); } + // Generate metadata for SubgroupsPerWorkgroup/SubgroupsPerWorkgroupId. + auto EmitSubgroupsPerWorkgroupMD = [this, F](SPIRVExecutionModeKind EMK, + uint64_t Value) { + NamedMDNode *ExecModeMD = + M->getOrInsertNamedMetadata(kSPIRVMD::ExecutionMode); + SmallVector OperandVec; + OperandVec.push_back(ConstantAsMetadata::get(F)); + OperandVec.push_back(ConstantAsMetadata::get(getUInt32(M, EMK))); + OperandVec.push_back(ConstantAsMetadata::get(getUInt32(M, Value))); + ExecModeMD->addOperand(MDNode::get(*Context, OperandVec)); + }; + if (auto *EM = BF->getExecutionMode(ExecutionModeSubgroupsPerWorkgroup)) { + EmitSubgroupsPerWorkgroupMD(EM->getExecutionMode(), EM->getLiterals()[0]); + } else if (auto *EM = BF->getExecutionModeId( + ExecutionModeSubgroupsPerWorkgroupId)) { + if (auto Val = transIdAsConstant(EM->getLiterals()[0])) { + EmitSubgroupsPerWorkgroupMD(EM->getExecutionMode(), *Val); + } + } // Generate metadata for max_work_group_size if (auto *EM = BF->getExecutionMode(ExecutionModeMaxWorkgroupSizeINTEL)) { F->setMetadata(kSPIR2MD::MaxWGSize, diff --git a/lib/SPIRV/libSPIRV/SPIRVEntry.cpp b/lib/SPIRV/libSPIRV/SPIRVEntry.cpp index 90ea1e4f4..8a7fd81ac 100644 --- a/lib/SPIRV/libSPIRV/SPIRVEntry.cpp +++ b/lib/SPIRV/libSPIRV/SPIRVEntry.cpp @@ -667,6 +667,8 @@ void SPIRVExecutionMode::decode(std::istream &I) { case ExecutionModeSharedLocalMemorySizeINTEL: case ExecutionModeNamedBarrierCountINTEL: case ExecutionModeSubgroupSize: + case ExecutionModeSubgroupsPerWorkgroup: + case ExecutionModeSubgroupsPerWorkgroupId: case ExecutionModeMaxWorkDimINTEL: case ExecutionModeNumSIMDWorkitemsINTEL: case ExecutionModeSchedulerTargetFmaxMhzINTEL: diff --git a/test/SubgroupsPerWorkgroup.spvasm b/test/SubgroupsPerWorkgroup.spvasm new file mode 100644 index 000000000..0d7e84336 --- /dev/null +++ b/test/SubgroupsPerWorkgroup.spvasm @@ -0,0 +1,24 @@ +; REQUIRES: spirv-as + +; RUN: spirv-as %s --target-env spv1.2 -o %t.spv +; RUN: spirv-val %t.spv +; RUN: llvm-spirv -r -o %t.rev.bc %t.spv +; RUN: llvm-dis %t.rev.bc -o - | FileCheck %s + + OpCapability Addresses + OpCapability Linkage + OpCapability Kernel + OpCapability SubgroupDispatch + OpMemoryModel Physical64 OpenCL + OpEntryPoint Kernel %fn "testSubgroupsPerWorkgroup" + OpExecutionMode %fn SubgroupsPerWorkgroup 8 + %void = OpTypeVoid + %fnTy = OpTypeFunction %void + +; CHECK: !spirv.ExecutionMode = !{![[MD:[0-9]+]]} +; CHECK: ![[MD]] = !{ptr @testSubgroupsPerWorkgroup, i32 36, i32 8} + + %fn = OpFunction %void None %fnTy + %entry = OpLabel + OpReturn + OpFunctionEnd diff --git a/test/SubgroupsPerWorkgroupId.spvasm b/test/SubgroupsPerWorkgroupId.spvasm new file mode 100644 index 000000000..5b9d2a0a7 --- /dev/null +++ b/test/SubgroupsPerWorkgroupId.spvasm @@ -0,0 +1,27 @@ +; REQUIRES: spirv-as + +; RUN: spirv-as %s --target-env spv1.2 -o %t.spv +; RUN: spirv-val %t.spv +; RUN: llvm-spirv -r -o %t.rev.bc %t.spv +; RUN: llvm-dis %t.rev.bc -o - | FileCheck %s + + OpCapability Addresses + OpCapability Linkage + OpCapability Kernel + OpCapability SubgroupDispatch + OpMemoryModel Physical64 OpenCL + OpEntryPoint Kernel %fn "testSubgroupsPerWorkgroupId" + OpExecutionModeId %fn SubgroupsPerWorkgroupId %uint_8 + %void = OpTypeVoid + %uint = OpTypeInt 32 0 + %uint_4 = OpConstant %uint 4 + %uint_8 = OpSpecConstantOp %uint IAdd %uint_4 %uint_4 + %fnTy = OpTypeFunction %void + +; CHECK: !spirv.ExecutionMode = !{![[MD:[0-9]+]]} +; CHECK: ![[MD]] = !{ptr @testSubgroupsPerWorkgroupId, i32 37, i32 8} + + %fn = OpFunction %void None %fnTy + %entry = OpLabel + OpReturn + OpFunctionEnd