ComputeBufferVK.cpp 4.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140
  1. // Jolt Physics Library (https://github.com/jrouwe/JoltPhysics)
  2. // SPDX-FileCopyrightText: 2025 Jorrit Rouwe
  3. // SPDX-License-Identifier: MIT
  4. #include <Jolt/Jolt.h>
  5. #ifdef JPH_USE_VK
  6. #include <Jolt/Compute/VK/ComputeBufferVK.h>
  7. #include <Jolt/Compute/VK/ComputeSystemVK.h>
  8. JPH_NAMESPACE_BEGIN
  9. ComputeBufferVK::ComputeBufferVK(ComputeSystemVK *inComputeSystem, EType inType, uint64 inSize, uint inStride) :
  10. ComputeBuffer(inType, inSize, inStride),
  11. mComputeSystem(inComputeSystem)
  12. {
  13. }
  14. bool ComputeBufferVK::Initialize(const void *inData)
  15. {
  16. VkDeviceSize buffer_size = VkDeviceSize(mSize * mStride);
  17. switch (mType)
  18. {
  19. case EType::Buffer:
  20. JPH_ASSERT(inData != nullptr);
  21. [[fallthrough]];
  22. case EType::UploadBuffer:
  23. case EType::RWBuffer:
  24. if (!mComputeSystem->CreateBuffer(buffer_size, VK_BUFFER_USAGE_TRANSFER_SRC_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT | VK_MEMORY_PROPERTY_HOST_CACHED_BIT, mBufferCPU))
  25. return false;
  26. if (!mComputeSystem->CreateBuffer(buffer_size, VK_BUFFER_USAGE_STORAGE_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_SRC_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT, VK_MEMORY_PROPERTY_DEVICE_LOCAL_BIT, mBufferGPU))
  27. return false;
  28. if (inData != nullptr)
  29. {
  30. void *data = mComputeSystem->MapBuffer(mBufferCPU);
  31. memcpy(data, inData, size_t(buffer_size));
  32. mComputeSystem->UnmapBuffer(mBufferCPU);
  33. mNeedsSync = true;
  34. }
  35. break;
  36. case EType::ConstantBuffer:
  37. if (!mComputeSystem->CreateBuffer(buffer_size, VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT | VK_MEMORY_PROPERTY_HOST_CACHED_BIT, mBufferCPU))
  38. return false;
  39. if (inData != nullptr)
  40. {
  41. void* data = mComputeSystem->MapBuffer(mBufferCPU);
  42. memcpy(data, inData, size_t(buffer_size));
  43. mComputeSystem->UnmapBuffer(mBufferCPU);
  44. }
  45. break;
  46. case EType::ReadbackBuffer:
  47. JPH_ASSERT(inData == nullptr, "Can't upload data to a readback buffer");
  48. if (!mComputeSystem->CreateBuffer(buffer_size, VK_BUFFER_USAGE_TRANSFER_DST_BIT, VK_MEMORY_PROPERTY_HOST_VISIBLE_BIT | VK_MEMORY_PROPERTY_HOST_COHERENT_BIT | VK_MEMORY_PROPERTY_HOST_CACHED_BIT, mBufferCPU))
  49. return false;
  50. break;
  51. }
  52. return true;
  53. }
  54. ComputeBufferVK::~ComputeBufferVK()
  55. {
  56. mComputeSystem->FreeBuffer(mBufferGPU);
  57. mComputeSystem->FreeBuffer(mBufferCPU);
  58. }
  59. void ComputeBufferVK::Barrier(VkCommandBuffer inCommandBuffer, VkPipelineStageFlags inToStage, VkAccessFlagBits inToFlags, bool inForce) const
  60. {
  61. if (mAccessStage == inToStage && mAccessFlagBits == inToFlags && !inForce)
  62. return;
  63. VkBufferMemoryBarrier b = {};
  64. b.sType = VK_STRUCTURE_TYPE_BUFFER_MEMORY_BARRIER;
  65. b.srcAccessMask = mAccessFlagBits;
  66. b.dstAccessMask = inToFlags;
  67. b.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
  68. b.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED;
  69. b.buffer = mBufferGPU.mBuffer != VK_NULL_HANDLE? mBufferGPU.mBuffer : mBufferCPU.mBuffer;
  70. b.offset = 0;
  71. b.size = VK_WHOLE_SIZE;
  72. vkCmdPipelineBarrier(inCommandBuffer, mAccessStage, inToStage, 0, 0, nullptr, 1, &b, 0, nullptr);
  73. mAccessStage = inToStage;
  74. mAccessFlagBits = inToFlags;
  75. }
  76. bool ComputeBufferVK::SyncCPUToGPU(VkCommandBuffer inCommandBuffer) const
  77. {
  78. if (!mNeedsSync)
  79. return false;
  80. // Barrier before write
  81. Barrier(inCommandBuffer, VK_PIPELINE_STAGE_TRANSFER_BIT, VK_ACCESS_TRANSFER_WRITE_BIT, false);
  82. // Copy from CPU to GPU
  83. VkBufferCopy copy = {};
  84. copy.srcOffset = 0;
  85. copy.dstOffset = 0;
  86. copy.size = GetSize() * GetStride();
  87. vkCmdCopyBuffer(inCommandBuffer, mBufferCPU.mBuffer, mBufferGPU.mBuffer, 1, &copy);
  88. mNeedsSync = false;
  89. return true;
  90. }
  91. void *ComputeBufferVK::MapInternal(EMode inMode)
  92. {
  93. switch (inMode)
  94. {
  95. case EMode::Read:
  96. JPH_ASSERT(mType == EType::ReadbackBuffer);
  97. break;
  98. case EMode::Write:
  99. JPH_ASSERT(mType == EType::UploadBuffer || mType == EType::ConstantBuffer);
  100. mNeedsSync = true;
  101. break;
  102. }
  103. return mComputeSystem->MapBuffer(mBufferCPU);
  104. }
  105. void ComputeBufferVK::UnmapInternal()
  106. {
  107. mComputeSystem->UnmapBuffer(mBufferCPU);
  108. }
  109. ComputeBufferResult ComputeBufferVK::CreateReadBackBuffer() const
  110. {
  111. return mComputeSystem->CreateComputeBuffer(ComputeBuffer::EType::ReadbackBuffer, mSize, mStride);
  112. }
  113. JPH_NAMESPACE_END
  114. #endif // JPH_USE_VK