DescriptorSetTemplate.cs 7.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210
  1. using Ryujinx.Graphics.GAL;
  2. using Silk.NET.Vulkan;
  3. using System;
  4. using System.Numerics;
  5. using System.Runtime.CompilerServices;
  6. namespace Ryujinx.Graphics.Vulkan
  7. {
  8. class DescriptorSetTemplate : IDisposable
  9. {
  10. /// <summary>
  11. /// Renderdoc seems to crash when doing a templated uniform update with count > 1 on a push descriptor.
  12. /// When this is true, consecutive buffers are always updated individually.
  13. /// </summary>
  14. private const bool RenderdocPushCountBug = true;
  15. private readonly VulkanRenderer _gd;
  16. private readonly Device _device;
  17. public readonly DescriptorUpdateTemplate Template;
  18. public readonly int Size;
  19. public unsafe DescriptorSetTemplate(
  20. VulkanRenderer gd,
  21. Device device,
  22. ResourceBindingSegment[] segments,
  23. PipelineLayoutCacheEntry plce,
  24. PipelineBindPoint pbp,
  25. int setIndex)
  26. {
  27. _gd = gd;
  28. _device = device;
  29. // Create a template from the set usages. Assumes the descriptor set is updated in segment order then binding order.
  30. DescriptorUpdateTemplateEntry* entries = stackalloc DescriptorUpdateTemplateEntry[segments.Length];
  31. nuint structureOffset = 0;
  32. for (int seg = 0; seg < segments.Length; seg++)
  33. {
  34. ResourceBindingSegment segment = segments[seg];
  35. int binding = segment.Binding;
  36. int count = segment.Count;
  37. if (IsBufferType(segment.Type))
  38. {
  39. entries[seg] = new DescriptorUpdateTemplateEntry()
  40. {
  41. DescriptorType = segment.Type.Convert(),
  42. DstBinding = (uint)binding,
  43. DescriptorCount = (uint)count,
  44. Offset = structureOffset,
  45. Stride = (nuint)Unsafe.SizeOf<DescriptorBufferInfo>()
  46. };
  47. structureOffset += (nuint)(Unsafe.SizeOf<DescriptorBufferInfo>() * count);
  48. }
  49. else if (IsBufferTextureType(segment.Type))
  50. {
  51. entries[seg] = new DescriptorUpdateTemplateEntry()
  52. {
  53. DescriptorType = segment.Type.Convert(),
  54. DstBinding = (uint)binding,
  55. DescriptorCount = (uint)count,
  56. Offset = structureOffset,
  57. Stride = (nuint)Unsafe.SizeOf<BufferView>()
  58. };
  59. structureOffset += (nuint)(Unsafe.SizeOf<BufferView>() * count);
  60. }
  61. else
  62. {
  63. entries[seg] = new DescriptorUpdateTemplateEntry()
  64. {
  65. DescriptorType = segment.Type.Convert(),
  66. DstBinding = (uint)binding,
  67. DescriptorCount = (uint)count,
  68. Offset = structureOffset,
  69. Stride = (nuint)Unsafe.SizeOf<DescriptorImageInfo>()
  70. };
  71. structureOffset += (nuint)(Unsafe.SizeOf<DescriptorImageInfo>() * count);
  72. }
  73. }
  74. Size = (int)structureOffset;
  75. var info = new DescriptorUpdateTemplateCreateInfo()
  76. {
  77. SType = StructureType.DescriptorUpdateTemplateCreateInfo,
  78. DescriptorUpdateEntryCount = (uint)segments.Length,
  79. PDescriptorUpdateEntries = entries,
  80. TemplateType = DescriptorUpdateTemplateType.DescriptorSet,
  81. DescriptorSetLayout = plce.DescriptorSetLayouts[setIndex],
  82. PipelineBindPoint = pbp,
  83. PipelineLayout = plce.PipelineLayout,
  84. Set = (uint)setIndex,
  85. };
  86. DescriptorUpdateTemplate result;
  87. gd.Api.CreateDescriptorUpdateTemplate(device, &info, null, &result).ThrowOnError();
  88. Template = result;
  89. }
  90. public unsafe DescriptorSetTemplate(
  91. VulkanRenderer gd,
  92. Device device,
  93. ResourceDescriptorCollection descriptors,
  94. long updateMask,
  95. PipelineLayoutCacheEntry plce,
  96. PipelineBindPoint pbp,
  97. int setIndex)
  98. {
  99. _gd = gd;
  100. _device = device;
  101. // Create a template from the set usages. Assumes the descriptor set is updated in segment order then binding order.
  102. int segmentCount = BitOperations.PopCount((ulong)updateMask);
  103. DescriptorUpdateTemplateEntry* entries = stackalloc DescriptorUpdateTemplateEntry[segmentCount];
  104. int entry = 0;
  105. nuint structureOffset = 0;
  106. void AddBinding(int binding, int count)
  107. {
  108. entries[entry++] = new DescriptorUpdateTemplateEntry()
  109. {
  110. DescriptorType = DescriptorType.UniformBuffer,
  111. DstBinding = (uint)binding,
  112. DescriptorCount = (uint)count,
  113. Offset = structureOffset,
  114. Stride = (nuint)Unsafe.SizeOf<DescriptorBufferInfo>()
  115. };
  116. structureOffset += (nuint)(Unsafe.SizeOf<DescriptorBufferInfo>() * count);
  117. }
  118. int startBinding = 0;
  119. int bindingCount = 0;
  120. foreach (ResourceDescriptor descriptor in descriptors.Descriptors)
  121. {
  122. for (int i = 0; i < descriptor.Count; i++)
  123. {
  124. int binding = descriptor.Binding + i;
  125. if ((updateMask & (1L << binding)) != 0)
  126. {
  127. if (bindingCount > 0 && (RenderdocPushCountBug || startBinding + bindingCount != binding))
  128. {
  129. AddBinding(startBinding, bindingCount);
  130. bindingCount = 0;
  131. }
  132. if (bindingCount == 0)
  133. {
  134. startBinding = binding;
  135. }
  136. bindingCount++;
  137. }
  138. }
  139. }
  140. if (bindingCount > 0)
  141. {
  142. AddBinding(startBinding, bindingCount);
  143. }
  144. Size = (int)structureOffset;
  145. var info = new DescriptorUpdateTemplateCreateInfo()
  146. {
  147. SType = StructureType.DescriptorUpdateTemplateCreateInfo,
  148. DescriptorUpdateEntryCount = (uint)entry,
  149. PDescriptorUpdateEntries = entries,
  150. TemplateType = DescriptorUpdateTemplateType.PushDescriptorsKhr,
  151. DescriptorSetLayout = plce.DescriptorSetLayouts[setIndex],
  152. PipelineBindPoint = pbp,
  153. PipelineLayout = plce.PipelineLayout,
  154. Set = (uint)setIndex,
  155. };
  156. DescriptorUpdateTemplate result;
  157. gd.Api.CreateDescriptorUpdateTemplate(device, &info, null, &result).ThrowOnError();
  158. Template = result;
  159. }
  160. private static bool IsBufferType(ResourceType type)
  161. {
  162. return type == ResourceType.UniformBuffer || type == ResourceType.StorageBuffer;
  163. }
  164. private static bool IsBufferTextureType(ResourceType type)
  165. {
  166. return type == ResourceType.BufferTexture || type == ResourceType.BufferImage;
  167. }
  168. public unsafe void Dispose()
  169. {
  170. _gd.Api.DestroyDescriptorUpdateTemplate(_device, Template, null);
  171. }
  172. }
  173. }