DxilContainer.cpp 4.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125
  1. ///////////////////////////////////////////////////////////////////////////////
  2. // //
  3. // DxilContainer.cpp //
  4. // Copyright (C) Microsoft Corporation. All rights reserved. //
  5. // This file is distributed under the University of Illinois Open Source //
  6. // License. See LICENSE.TXT for details. //
  7. // //
  8. // Provides support for manipulating DXIL container structures. //
  9. // //
  10. ///////////////////////////////////////////////////////////////////////////////
  11. #include "dxc/HLSL/DxilContainer.h"
  12. #include <algorithm>
  13. namespace hlsl {
  14. DxilPartIterator begin(const DxilContainerHeader *pHeader) {
  15. return{ pHeader, 0 };
  16. }
  17. DxilPartIterator end(const DxilContainerHeader *pHeader) {
  18. return{ pHeader, pHeader->PartCount };
  19. }
  20. void InitDxilContainer(_Out_ DxilContainerHeader *pHeader, uint32_t partCount,
  21. uint32_t containerSizeInBytes) {
  22. memset(pHeader, 0, sizeof(*pHeader));
  23. pHeader->HeaderFourCC = DFCC_Container;
  24. pHeader->Version.Major = DxilContainerVersionMajor;
  25. pHeader->Version.Minor = DxilContainerVersionMinor;
  26. pHeader->PartCount = partCount;
  27. pHeader->ContainerSizeInBytes = containerSizeInBytes;
  28. }
  29. const DxilContainerHeader *IsDxilContainerLike(const void *ptr, size_t length) {
  30. if (ptr == nullptr || length < 4)
  31. return nullptr;
  32. if (DFCC_Container != *reinterpret_cast<const uint32_t *>(ptr))
  33. return nullptr;
  34. return reinterpret_cast<const DxilContainerHeader *>(ptr);
  35. }
  36. bool IsValidDxilContainer(const DxilContainerHeader *pHeader, size_t length) {
  37. // Validate that the header is where it's supposed to be.
  38. if (pHeader == nullptr) return false;
  39. if (length < sizeof(DxilContainerHeader)) return false;
  40. // Validate the header values.
  41. if (pHeader->HeaderFourCC != DFCC_Container) return false;
  42. if (pHeader->Version.Major != DxilContainerVersionMajor) return false;
  43. if (pHeader->ContainerSizeInBytes > length) return false;
  44. if (pHeader->ContainerSizeInBytes > DxilContainerMaxSize) return false;
  45. // Make sure that the count of offsets fits.
  46. size_t partOffsetTableBytes = sizeof(uint32_t) * pHeader->PartCount;
  47. if (partOffsetTableBytes + sizeof(DxilContainerHeader) >
  48. pHeader->ContainerSizeInBytes)
  49. return false;
  50. // Make sure that each part is within the bounds.
  51. const uint8_t *pLinearContainer = reinterpret_cast<const uint8_t *>(pHeader);
  52. const uint32_t *pPartOffsetTable =
  53. reinterpret_cast<const uint32_t *>(pHeader + 1);
  54. for (uint32_t i = 0; i < pHeader->PartCount; ++i) {
  55. // The part header should fit.
  56. if (pPartOffsetTable[i] >
  57. (pHeader->ContainerSizeInBytes - sizeof(DxilPartHeader)))
  58. return false;
  59. // The contents of the part should fit.
  60. const DxilPartHeader *pPartHeader =
  61. reinterpret_cast<const DxilPartHeader *>(pLinearContainer +
  62. pPartOffsetTable[i]);
  63. if (pPartOffsetTable[i] + sizeof(DxilPartHeader) + pPartHeader->PartSize >
  64. pHeader->ContainerSizeInBytes) {
  65. return false;
  66. }
  67. }
  68. // Note: the container parts may overlap and there may be holes
  69. // based on this validation
  70. return true;
  71. }
  72. const DxilPartHeader *GetDxilPartByType(const DxilContainerHeader *pHeader, DxilFourCC fourCC) {
  73. if (!IsDxilContainerLike(pHeader, pHeader->ContainerSizeInBytes)) {
  74. return nullptr;
  75. }
  76. const DxilPartIterator partIter =
  77. find_if(begin(pHeader), end(pHeader), DxilPartIsType(fourCC));
  78. if (partIter == end(pHeader)) {
  79. return nullptr;
  80. }
  81. return *partIter;
  82. }
  83. DxilPartHeader *GetDxilPartByType(DxilContainerHeader *pHeader,
  84. DxilFourCC fourCC) {
  85. return const_cast<DxilPartHeader *>(GetDxilPartByType(
  86. static_cast<const DxilContainerHeader *>(pHeader), fourCC));
  87. }
  88. const DxilProgramHeader *GetDxilProgramHeader(const DxilContainerHeader *pHeader, DxilFourCC fourCC) {
  89. if (!IsDxilContainerLike(pHeader, pHeader->ContainerSizeInBytes)) {
  90. return nullptr;
  91. }
  92. const DxilPartHeader *PartHeader = GetDxilPartByType(pHeader, fourCC);
  93. if (!PartHeader) {
  94. return nullptr;
  95. }
  96. const DxilProgramHeader *ProgramHeader =
  97. reinterpret_cast<const DxilProgramHeader *>(GetDxilPartData(PartHeader));
  98. return IsValidDxilProgramHeader(ProgramHeader,
  99. ProgramHeader->SizeInUint32 * 4)
  100. ? ProgramHeader
  101. : nullptr;
  102. }
  103. DxilProgramHeader *GetDxilProgramHeader(DxilContainerHeader *pHeader, DxilFourCC fourCC) {
  104. return const_cast<DxilProgramHeader *>(
  105. GetDxilProgramHeader(static_cast<const DxilContainerHeader *>(pHeader), fourCC));
  106. }
  107. } // namespace hlsl