| //===- DXILResource.cpp - DXIL Resource helper objects --------------------===// |
| // |
| // Part of the LLVM Project, under the Apache License v2.0 with LLVM Exceptions. |
| // See https://llvm.org/LICENSE.txt for license information. |
| // SPDX-License-Identifier: Apache-2.0 WITH LLVM-exception |
| // |
| //===----------------------------------------------------------------------===// |
| /// |
| /// \file This file contains helper objects for working with DXIL Resources. |
| /// |
| //===----------------------------------------------------------------------===// |
| |
| #include "DXILResource.h" |
| #include "CBufferDataLayout.h" |
| #include "llvm/ADT/StringSwitch.h" |
| #include "llvm/IR/IRBuilder.h" |
| #include "llvm/IR/Metadata.h" |
| #include "llvm/IR/Module.h" |
| #include "llvm/Support/Debug.h" |
| #include "llvm/Support/Format.h" |
| |
| using namespace llvm; |
| using namespace llvm::dxil; |
| |
| template <typename T> void ResourceTable<T>::collect(Module &M) { |
| NamedMDNode *Entry = M.getNamedMetadata(MDName); |
| if (!Entry || Entry->getNumOperands() == 0) |
| return; |
| |
| uint32_t Counter = 0; |
| for (auto *Res : Entry->operands()) { |
| Data.push_back(T(Counter++, hlsl::FrontendResource(cast<MDNode>(Res)))); |
| } |
| } |
| |
| template <> void ResourceTable<ConstantBuffer>::collect(Module &M) { |
| NamedMDNode *Entry = M.getNamedMetadata(MDName); |
| if (!Entry || Entry->getNumOperands() == 0) |
| return; |
| |
| uint32_t Counter = 0; |
| for (auto *Res : Entry->operands()) { |
| Data.push_back( |
| ConstantBuffer(Counter++, hlsl::FrontendResource(cast<MDNode>(Res)))); |
| } |
| // FIXME: share CBufferDataLayout with CBuffer load lowering. |
| // See https://github.com/llvm/llvm-project/issues/58381 |
| CBufferDataLayout CBDL(M.getDataLayout(), /*IsLegacy*/ true); |
| for (auto &CB : Data) |
| CB.setSize(CBDL); |
| } |
| |
| void Resources::collect(Module &M) { |
| UAVs.collect(M); |
| CBuffers.collect(M); |
| } |
| |
| ResourceBase::ResourceBase(uint32_t I, hlsl::FrontendResource R) |
| : ID(I), GV(R.getGlobalVariable()), Name(""), Space(R.getSpace()), |
| LowerBound(R.getResourceIndex()), RangeSize(1) { |
| if (auto *ArrTy = dyn_cast<ArrayType>(GV->getValueType())) |
| RangeSize = ArrTy->getNumElements(); |
| } |
| |
| StringRef ResourceBase::getElementTypeName(ElementType ElTy) { |
| switch (ElTy) { |
| case ElementType::Invalid: |
| return "invalid"; |
| case ElementType::I1: |
| return "i1"; |
| case ElementType::I16: |
| return "i16"; |
| case ElementType::U16: |
| return "u16"; |
| case ElementType::I32: |
| return "i32"; |
| case ElementType::U32: |
| return "u32"; |
| case ElementType::I64: |
| return "i64"; |
| case ElementType::U64: |
| return "u64"; |
| case ElementType::F16: |
| return "f16"; |
| case ElementType::F32: |
| return "f32"; |
| case ElementType::F64: |
| return "f64"; |
| case ElementType::SNormF16: |
| return "snorm_f16"; |
| case ElementType::UNormF16: |
| return "unorm_f16"; |
| case ElementType::SNormF32: |
| return "snorm_f32"; |
| case ElementType::UNormF32: |
| return "unorm_f32"; |
| case ElementType::SNormF64: |
| return "snorm_f64"; |
| case ElementType::UNormF64: |
| return "unorm_f64"; |
| case ElementType::PackedS8x32: |
| return "p32i8"; |
| case ElementType::PackedU8x32: |
| return "p32u8"; |
| } |
| llvm_unreachable("All ElementType enums are handled in switch"); |
| } |
| |
| void ResourceBase::printElementType(ResourceKind Kind, ElementType ElTy, |
| unsigned Alignment, raw_ostream &OS) { |
| switch (Kind) { |
| default: |
| // TODO: add vector size. |
| OS << right_justify(getElementTypeName(ElTy), Alignment); |
| break; |
| case ResourceKind::RawBuffer: |
| OS << right_justify("byte", Alignment); |
| break; |
| case ResourceKind::StructuredBuffer: |
| OS << right_justify("struct", Alignment); |
| break; |
| case ResourceKind::CBuffer: |
| case ResourceKind::Sampler: |
| OS << right_justify("NA", Alignment); |
| break; |
| case ResourceKind::Invalid: |
| case ResourceKind::NumEntries: |
| break; |
| } |
| } |
| |
| StringRef ResourceBase::getKindName(ResourceKind Kind) { |
| switch (Kind) { |
| case ResourceKind::NumEntries: |
| case ResourceKind::Invalid: |
| return "invalid"; |
| case ResourceKind::Texture1D: |
| return "1d"; |
| case ResourceKind::Texture2D: |
| return "2d"; |
| case ResourceKind::Texture2DMS: |
| return "2dMS"; |
| case ResourceKind::Texture3D: |
| return "3d"; |
| case ResourceKind::TextureCube: |
| return "cube"; |
| case ResourceKind::Texture1DArray: |
| return "1darray"; |
| case ResourceKind::Texture2DArray: |
| return "2darray"; |
| case ResourceKind::Texture2DMSArray: |
| return "2darrayMS"; |
| case ResourceKind::TextureCubeArray: |
| return "cubearray"; |
| case ResourceKind::TypedBuffer: |
| return "buf"; |
| case ResourceKind::RawBuffer: |
| return "rawbuf"; |
| case ResourceKind::StructuredBuffer: |
| return "structbuf"; |
| case ResourceKind::CBuffer: |
| return "cbuffer"; |
| case ResourceKind::Sampler: |
| return "sampler"; |
| case ResourceKind::TBuffer: |
| return "tbuffer"; |
| case ResourceKind::RTAccelerationStructure: |
| return "ras"; |
| case ResourceKind::FeedbackTexture2D: |
| return "fbtex2d"; |
| case ResourceKind::FeedbackTexture2DArray: |
| return "fbtex2darray"; |
| } |
| llvm_unreachable("All ResourceKind enums are handled in switch"); |
| } |
| |
| void ResourceBase::printKind(ResourceKind Kind, unsigned Alignment, |
| raw_ostream &OS, bool SRV, bool HasCounter, |
| uint32_t SampleCount) { |
| switch (Kind) { |
| default: |
| OS << right_justify(getKindName(Kind), Alignment); |
| break; |
| |
| case ResourceKind::RawBuffer: |
| case ResourceKind::StructuredBuffer: |
| if (SRV) |
| OS << right_justify("r/o", Alignment); |
| else { |
| if (!HasCounter) |
| OS << right_justify("r/w", Alignment); |
| else |
| OS << right_justify("r/w+cnt", Alignment); |
| } |
| break; |
| case ResourceKind::TypedBuffer: |
| OS << right_justify("buf", Alignment); |
| break; |
| case ResourceKind::Texture2DMS: |
| case ResourceKind::Texture2DMSArray: { |
| std::string DimName = getKindName(Kind).str(); |
| if (SampleCount) |
| DimName += std::to_string(SampleCount); |
| OS << right_justify(DimName, Alignment); |
| } break; |
| case ResourceKind::CBuffer: |
| case ResourceKind::Sampler: |
| OS << right_justify("NA", Alignment); |
| break; |
| case ResourceKind::Invalid: |
| case ResourceKind::NumEntries: |
| break; |
| } |
| } |
| |
| void ResourceBase::print(raw_ostream &OS, StringRef IDPrefix, |
| StringRef BindingPrefix) const { |
| std::string ResID = IDPrefix.str(); |
| ResID += std::to_string(ID); |
| OS << right_justify(ResID, 8); |
| |
| std::string Bind = BindingPrefix.str(); |
| Bind += std::to_string(LowerBound); |
| if (Space) |
| Bind += ",space" + std::to_string(Space); |
| |
| OS << right_justify(Bind, 15); |
| if (RangeSize != UINT_MAX) |
| OS << right_justify(std::to_string(RangeSize), 6) << "\n"; |
| else |
| OS << right_justify("unbounded", 6) << "\n"; |
| } |
| |
| void UAVResource::print(raw_ostream &OS) const { |
| OS << "; " << left_justify(Name, 31); |
| |
| OS << right_justify("UAV", 10); |
| |
| printElementType(Shape, ExtProps.ElementType.value_or(ElementType::Invalid), |
| 8, OS); |
| |
| // FIXME: support SampleCount. |
| // See https://github.com/llvm/llvm-project/issues/58175 |
| printKind(Shape, 12, OS, /*SRV*/ false, HasCounter); |
| // Print the binding part. |
| ResourceBase::print(OS, "U", "u"); |
| } |
| |
| ConstantBuffer::ConstantBuffer(uint32_t I, hlsl::FrontendResource R) |
| : ResourceBase(I, R) {} |
| |
| void ConstantBuffer::setSize(CBufferDataLayout &DL) { |
| CBufferSizeInBytes = DL.getTypeAllocSizeInBytes(GV->getValueType()); |
| } |
| |
| void ConstantBuffer::print(raw_ostream &OS) const { |
| OS << "; " << left_justify(Name, 31); |
| |
| OS << right_justify("cbuffer", 10); |
| |
| printElementType(ResourceKind::CBuffer, ElementType::Invalid, 8, OS); |
| |
| printKind(ResourceKind::CBuffer, 12, OS, /*SRV*/ false, /*HasCounter*/ false); |
| // Print the binding part. |
| ResourceBase::print(OS, "CB", "cb"); |
| } |
| |
| template <typename T> void ResourceTable<T>::print(raw_ostream &OS) const { |
| for (auto &Res : Data) |
| Res.print(OS); |
| } |
| |
| MDNode *ResourceBase::ExtendedProperties::write(LLVMContext &Ctx) const { |
| IRBuilder<> B(Ctx); |
| SmallVector<Metadata *> Entries; |
| if (ElementType) { |
| Entries.emplace_back( |
| ConstantAsMetadata::get(B.getInt32(TypedBufferElementType))); |
| Entries.emplace_back(ConstantAsMetadata::get( |
| B.getInt32(static_cast<uint32_t>(*ElementType)))); |
| } |
| if (Entries.empty()) |
| return nullptr; |
| return MDNode::get(Ctx, Entries); |
| } |
| |
| void ResourceBase::write(LLVMContext &Ctx, |
| MutableArrayRef<Metadata *> Entries) const { |
| IRBuilder<> B(Ctx); |
| Entries[0] = ConstantAsMetadata::get(B.getInt32(ID)); |
| Entries[1] = ConstantAsMetadata::get(GV); |
| Entries[2] = MDString::get(Ctx, Name); |
| Entries[3] = ConstantAsMetadata::get(B.getInt32(Space)); |
| Entries[4] = ConstantAsMetadata::get(B.getInt32(LowerBound)); |
| Entries[5] = ConstantAsMetadata::get(B.getInt32(RangeSize)); |
| } |
| |
| MDNode *UAVResource::write() const { |
| auto &Ctx = GV->getContext(); |
| IRBuilder<> B(Ctx); |
| Metadata *Entries[11]; |
| ResourceBase::write(Ctx, Entries); |
| Entries[6] = |
| ConstantAsMetadata::get(B.getInt32(static_cast<uint32_t>(Shape))); |
| Entries[7] = ConstantAsMetadata::get(B.getInt1(GloballyCoherent)); |
| Entries[8] = ConstantAsMetadata::get(B.getInt1(HasCounter)); |
| Entries[9] = ConstantAsMetadata::get(B.getInt1(IsROV)); |
| Entries[10] = ExtProps.write(Ctx); |
| return MDNode::get(Ctx, Entries); |
| } |
| |
| MDNode *ConstantBuffer::write() const { |
| auto &Ctx = GV->getContext(); |
| IRBuilder<> B(Ctx); |
| Metadata *Entries[7]; |
| ResourceBase::write(Ctx, Entries); |
| |
| Entries[6] = ConstantAsMetadata::get(B.getInt32(CBufferSizeInBytes)); |
| return MDNode::get(Ctx, Entries); |
| } |
| |
| template <typename T> MDNode *ResourceTable<T>::write(Module &M) const { |
| if (Data.empty()) |
| return nullptr; |
| SmallVector<Metadata *> MDs; |
| for (auto &Res : Data) |
| MDs.emplace_back(Res.write()); |
| |
| NamedMDNode *Entry = M.getNamedMetadata(MDName); |
| if (Entry) |
| Entry->eraseFromParent(); |
| |
| return MDNode::get(M.getContext(), MDs); |
| } |
| |
| Metadata *Resources::writeUAVs(Module &M) const { return UAVs.write(M); } |
| void Resources::printUAVs(raw_ostream &OS) const { UAVs.print(OS); } |
| Metadata *Resources::writeCBuffers(Module &M) const { |
| return CBuffers.write(M); |
| } |
| void Resources::printCBuffers(raw_ostream &OS) const { CBuffers.print(OS); } |
| |
| void Resources::dump() const { |
| printCBuffers(dbgs()); |
| printUAVs(dbgs()); |
| } |