DiffuseProbeGridVisualizationAccelerationStructurePass.cpp 9.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200
  1. /*
  2. * Copyright (c) Contributors to the Open 3D Engine Project.
  3. * For complete copyright and license terms please see the LICENSE at the root of this distribution.
  4. *
  5. * SPDX-License-Identifier: Apache-2.0 OR MIT
  6. *
  7. */
  8. #include <Atom/Feature/RayTracing/RayTracingFeatureProcessorInterface.h>
  9. #include <Atom/RHI/FrameScheduler.h>
  10. #include <Atom/RHI/CommandList.h>
  11. #include <Atom/RHI/RHISystemInterface.h>
  12. #include <Atom/RPI.Public/RenderPipeline.h>
  13. #include <Atom/RPI.Public/Scene.h>
  14. #include <DiffuseProbeGrid_Traits_Platform.h>
  15. #include <Render/DiffuseProbeGridFeatureProcessor.h>
  16. #include <Render/DiffuseProbeGridVisualizationAccelerationStructurePass.h>
  17. namespace AZ
  18. {
  19. namespace Render
  20. {
  21. RPI::Ptr<DiffuseProbeGridVisualizationAccelerationStructurePass> DiffuseProbeGridVisualizationAccelerationStructurePass::Create(const RPI::PassDescriptor& descriptor)
  22. {
  23. RPI::Ptr<DiffuseProbeGridVisualizationAccelerationStructurePass> diffuseProbeGridVisualizationAccelerationStructurePass = aznew DiffuseProbeGridVisualizationAccelerationStructurePass(descriptor);
  24. return AZStd::move(diffuseProbeGridVisualizationAccelerationStructurePass);
  25. }
  26. DiffuseProbeGridVisualizationAccelerationStructurePass::DiffuseProbeGridVisualizationAccelerationStructurePass(const RPI::PassDescriptor& descriptor)
  27. : Pass(descriptor)
  28. {
  29. // disable this pass if we're on a platform that doesn't support raytracing
  30. if (RHI::RHISystemInterface::Get()->GetRayTracingSupport() == RHI::MultiDevice::NoDevices || !AZ_TRAIT_DIFFUSE_GI_PASSES_SUPPORTED)
  31. {
  32. SetEnabled(false);
  33. }
  34. }
  35. bool DiffuseProbeGridVisualizationAccelerationStructurePass::ShouldUpdate(const AZStd::shared_ptr<DiffuseProbeGrid>& diffuseProbeGrid) const
  36. {
  37. return (diffuseProbeGrid->GetVisualizationEnabled() && diffuseProbeGrid->GetVisualizationTlasUpdateRequired());
  38. }
  39. bool DiffuseProbeGridVisualizationAccelerationStructurePass::IsEnabled() const
  40. {
  41. if (!Pass::IsEnabled())
  42. {
  43. return false;
  44. }
  45. RPI::Scene* scene = m_pipeline->GetScene();
  46. if (!scene)
  47. {
  48. return false;
  49. }
  50. DiffuseProbeGridFeatureProcessor* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<DiffuseProbeGridFeatureProcessor>();
  51. if (diffuseProbeGridFeatureProcessor)
  52. {
  53. for (auto& diffuseProbeGrid : diffuseProbeGridFeatureProcessor->GetVisibleProbeGrids())
  54. {
  55. if (ShouldUpdate(diffuseProbeGrid))
  56. {
  57. return true;
  58. }
  59. }
  60. }
  61. return false;
  62. }
  63. void DiffuseProbeGridVisualizationAccelerationStructurePass::BuildInternal()
  64. {
  65. InitScope(RHI::ScopeId(GetPathName()));
  66. }
  67. void DiffuseProbeGridVisualizationAccelerationStructurePass::FrameBeginInternal(FramePrepareParams params)
  68. {
  69. params.m_frameGraphBuilder->ImportScopeProducer(*this);
  70. }
  71. void DiffuseProbeGridVisualizationAccelerationStructurePass::SetupFrameGraphDependencies(RHI::FrameGraphInterface frameGraph)
  72. {
  73. RPI::Scene* scene = m_pipeline->GetScene();
  74. DiffuseProbeGridFeatureProcessor* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<DiffuseProbeGridFeatureProcessor>();
  75. frameGraph.SetEstimatedItemCount(aznumeric_cast<uint32_t>(diffuseProbeGridFeatureProcessor->GetVisibleProbeGrids().size()));
  76. for (auto& diffuseProbeGrid : diffuseProbeGridFeatureProcessor->GetVisibleProbeGrids())
  77. {
  78. if (!ShouldUpdate(diffuseProbeGrid))
  79. {
  80. continue;
  81. }
  82. // import and attach the visualization TLAS buffers
  83. RHI::Ptr<RHI::RayTracingTlas>& visualizationTlas = diffuseProbeGrid->GetVisualizationTlas();
  84. const RHI::Ptr<RHI::Buffer>& tlasBuffer = visualizationTlas->GetTlasBuffer();
  85. const RHI::Ptr<RHI::Buffer>& tlasInstancesBuffer = visualizationTlas->GetTlasInstancesBuffer();
  86. if (tlasBuffer && tlasInstancesBuffer)
  87. {
  88. // TLAS buffer
  89. {
  90. AZ::RHI::AttachmentId attachmentId = diffuseProbeGrid->GetProbeVisualizationTlasAttachmentId();
  91. if (frameGraph.GetAttachmentDatabase().IsAttachmentValid(attachmentId) == false)
  92. {
  93. [[maybe_unused]] RHI::ResultCode result = frameGraph.GetAttachmentDatabase().ImportBuffer(attachmentId, tlasBuffer);
  94. AZ_Assert(result == RHI::ResultCode::Success, "Failed to import DiffuseProbeGrid visualization TLAS buffer with error %d", result);
  95. }
  96. uint32_t byteCount = aznumeric_cast<uint32_t>(tlasBuffer->GetDescriptor().m_byteCount);
  97. RHI::BufferViewDescriptor bufferViewDescriptor = RHI::BufferViewDescriptor::CreateRayTracingTLAS(byteCount);
  98. RHI::BufferScopeAttachmentDescriptor desc;
  99. desc.m_attachmentId = attachmentId;
  100. desc.m_bufferViewDescriptor = bufferViewDescriptor;
  101. desc.m_loadStoreAction.m_loadAction = AZ::RHI::AttachmentLoadAction::DontCare;
  102. frameGraph.UseShaderAttachment(
  103. desc, RHI::ScopeAttachmentAccess::Write, RHI::ScopeAttachmentStage::RayTracingShader);
  104. }
  105. // TLAS Instances buffer
  106. {
  107. AZ::RHI::AttachmentId attachmentId = diffuseProbeGrid->GetProbeVisualizationTlasInstancesAttachmentId();
  108. if (frameGraph.GetAttachmentDatabase().IsAttachmentValid(attachmentId) == false)
  109. {
  110. [[maybe_unused]] RHI::ResultCode result = frameGraph.GetAttachmentDatabase().ImportBuffer(attachmentId, tlasInstancesBuffer);
  111. AZ_Assert(result == RHI::ResultCode::Success, "Failed to import DiffuseProbeGrid visualization TLAS Instances buffer with error %d", result);
  112. }
  113. uint32_t byteCount = aznumeric_cast<uint32_t>(tlasInstancesBuffer->GetDescriptor().m_byteCount);
  114. RHI::BufferViewDescriptor bufferViewDescriptor = RHI::BufferViewDescriptor::CreateStructured(0, byteCount / RayTracingTlasInstanceElementSize, RayTracingTlasInstanceElementSize);
  115. RHI::BufferScopeAttachmentDescriptor desc;
  116. desc.m_attachmentId = attachmentId;
  117. desc.m_bufferViewDescriptor = bufferViewDescriptor;
  118. desc.m_loadStoreAction.m_loadAction = AZ::RHI::AttachmentLoadAction::Load;
  119. frameGraph.UseShaderAttachment(desc, RHI::ScopeAttachmentAccess::Read, RHI::ScopeAttachmentStage::RayTracingShader);
  120. }
  121. }
  122. }
  123. }
  124. void DiffuseProbeGridVisualizationAccelerationStructurePass::BuildCommandList(const RHI::FrameGraphExecuteContext& context)
  125. {
  126. RPI::Scene* scene = m_pipeline->GetScene();
  127. DiffuseProbeGridFeatureProcessor* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<DiffuseProbeGridFeatureProcessor>();
  128. // build the visualization BLAS from the DiffuseProbeGridFeatureProcessor
  129. // Note: the BLAS is used by all DiffuseProbeGrid visualization TLAS objects
  130. AZStd::vector<const RHI::DeviceRayTracingBlas*> changedBlasList;
  131. if (m_visualizationBlasBuilt == false)
  132. {
  133. context.GetCommandList()->BuildBottomLevelAccelerationStructure(*diffuseProbeGridFeatureProcessor->GetVisualizationBlas()->GetDeviceRayTracingBlas(context.GetDeviceIndex()));
  134. m_visualizationBlasBuilt = true;
  135. changedBlasList.push_back(diffuseProbeGridFeatureProcessor->GetVisualizationBlas()
  136. ->GetDeviceRayTracingBlas(context.GetDeviceIndex())
  137. .get());
  138. }
  139. // call BuildTopLevelAccelerationStructure for each DiffuseProbeGrid in this range
  140. for (uint32_t index = context.GetSubmitRange().m_startIndex; index < context.GetSubmitRange().m_endIndex; ++index)
  141. {
  142. AZStd::shared_ptr<DiffuseProbeGrid> diffuseProbeGrid = diffuseProbeGridFeatureProcessor->GetVisibleProbeGrids()[index];
  143. if (!ShouldUpdate(diffuseProbeGrid))
  144. {
  145. continue;
  146. }
  147. if (!diffuseProbeGrid->GetVisualizationTlas()->GetTlasBuffer())
  148. {
  149. continue;
  150. }
  151. // build the TLAS object
  152. context.GetCommandList()->BuildTopLevelAccelerationStructure(*diffuseProbeGrid->GetVisualizationTlas()->GetDeviceRayTracingTlas(context.GetDeviceIndex()), changedBlasList);
  153. }
  154. }
  155. void DiffuseProbeGridVisualizationAccelerationStructurePass::FrameEndInternal()
  156. {
  157. RPI::Scene* scene = m_pipeline->GetScene();
  158. DiffuseProbeGridFeatureProcessor* diffuseProbeGridFeatureProcessor = scene->GetFeatureProcessor<DiffuseProbeGridFeatureProcessor>();
  159. for (auto& diffuseProbeGrid : diffuseProbeGridFeatureProcessor->GetVisibleProbeGrids())
  160. {
  161. if (!ShouldUpdate(diffuseProbeGrid))
  162. {
  163. continue;
  164. }
  165. // TLAS is now updated
  166. diffuseProbeGrid->ResetVisualizationTlasUpdateRequired();
  167. }
  168. }
  169. } // namespace RPI
  170. } // namespace AZ