|
|
|
@ -62,15 +62,21 @@ Application::Application() { |
|
|
|
|
swapchain = make_unique<Swapchain>(); |
|
|
|
|
Instance::instance->initImGui(*swapchain); |
|
|
|
|
descriptorPool = make_unique<DescriptorPool>(); |
|
|
|
|
graphicsPipeline = unique_ptr<GraphicsPipeline>(new GraphicsPipeline("shaders/vert.spv", "shaders/frag.spv", |
|
|
|
|
graphicsPipeline = unique_ptr<GraphicsPipeline>( |
|
|
|
|
new GraphicsPipeline( |
|
|
|
|
"shaders/vert.spv", "shaders/frag.spv", |
|
|
|
|
swapchain->renderPass, |
|
|
|
|
{descriptorPool->layouts[DescriptorSet::WORLD]})); |
|
|
|
|
{descriptorPool->layouts[DescriptorSet::WORLD]} |
|
|
|
|
) |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
cameraUniformBuffer = make_unique<Buffer>(sizeof(CameraUniformData), |
|
|
|
|
cameraUniformBuffer = make_unique<Buffer>( |
|
|
|
|
sizeof(CameraUniformData), |
|
|
|
|
VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT, |
|
|
|
|
VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE, |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT); |
|
|
|
|
cameraUniformBuffer->setName("Camera"); |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT, |
|
|
|
|
"Camera" |
|
|
|
|
); |
|
|
|
|
camera = make_unique<Camera>(swapchain->extent); |
|
|
|
|
updateCameraBuffer(); |
|
|
|
|
|
|
|
|
@ -81,10 +87,11 @@ Application::Application() { |
|
|
|
|
bool grabbing; |
|
|
|
|
} initialGrabInformation {}; |
|
|
|
|
initialGrabInformation.distanceToFace = 1e20; |
|
|
|
|
grabBuffer = make_unique<Buffer>(sizeof(GrabInformation), |
|
|
|
|
grabBuffer = make_unique<Buffer>( |
|
|
|
|
sizeof(GrabInformation), |
|
|
|
|
VK_BUFFER_USAGE_STORAGE_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT, |
|
|
|
|
VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE, 0); |
|
|
|
|
grabBuffer->setName("Grab"); |
|
|
|
|
VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE, 0, "Grab" |
|
|
|
|
); |
|
|
|
|
grabBuffer->setData(&initialGrabInformation, 0, sizeof(initialGrabInformation)); |
|
|
|
|
grabber = make_unique<Grabber>(); |
|
|
|
|
|
|
|
|
@ -92,8 +99,9 @@ Application::Application() { |
|
|
|
|
sizeof(SizesUniformData), |
|
|
|
|
VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT, |
|
|
|
|
VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE, |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT); |
|
|
|
|
sizeInformationBuffer->setName("Sizes"); |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT, |
|
|
|
|
"Sizes" |
|
|
|
|
); |
|
|
|
|
sizeInformation = &sizeInformationBuffer->access<SizesUniformData>(); |
|
|
|
|
*sizeInformation = {}; |
|
|
|
|
|
|
|
|
@ -109,15 +117,19 @@ Application::Application() { |
|
|
|
|
sizeof(SimulationUniformData), |
|
|
|
|
VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT, |
|
|
|
|
VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE, |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT); |
|
|
|
|
simulationPropertiesBuffer->setName("Simulation properties"); |
|
|
|
|
VMA_ALLOCATION_CREATE_HOST_ACCESS_SEQUENTIAL_WRITE_BIT | VMA_ALLOCATION_CREATE_MAPPED_BIT, |
|
|
|
|
"Simulation properties" |
|
|
|
|
); |
|
|
|
|
simulationPropertiesBuffer->access<SimulationUniformData>() = simulationUniformData; |
|
|
|
|
|
|
|
|
|
{ |
|
|
|
|
int w, h; |
|
|
|
|
stbi_uc *data = stbi_load("textures/rocky_diff.jpg", &w, &h, nullptr, STBI_rgb_alpha); |
|
|
|
|
VkDeviceSize imageSize = w * h * 4; |
|
|
|
|
firstImage = make_unique<Image>(data, imageSize, w, h, VK_FORMAT_R8G8B8A8_SRGB, VK_IMAGE_TILING_OPTIMAL, VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT); |
|
|
|
|
firstImage = make_unique<Image>( |
|
|
|
|
data, imageSize, w, h, VK_FORMAT_R8G8B8A8_SRGB, VK_IMAGE_TILING_OPTIMAL, |
|
|
|
|
VK_IMAGE_USAGE_SAMPLED_BIT | VK_IMAGE_USAGE_TRANSFER_DST_BIT |
|
|
|
|
); |
|
|
|
|
firstImage->createView(VK_IMAGE_ASPECT_COLOR_BIT); |
|
|
|
|
firstImage->createSampler(); |
|
|
|
|
} |
|
|
|
@ -125,7 +137,9 @@ Application::Application() { |
|
|
|
|
descriptorPool->bindBuffer(*cameraUniformBuffer, VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, DescriptorSet::WORLD, 0); |
|
|
|
|
descriptorPool->bindImage(*firstImage, VK_DESCRIPTOR_TYPE_COMBINED_IMAGE_SAMPLER, DescriptorSet::WORLD, 2); |
|
|
|
|
|
|
|
|
|
descriptorPool->bindBuffer(*simulationPropertiesBuffer, VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, DescriptorSet::SIMULATION, 0); |
|
|
|
|
descriptorPool->bindBuffer( |
|
|
|
|
*simulationPropertiesBuffer, VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, DescriptorSet::SIMULATION, 0 |
|
|
|
|
); |
|
|
|
|
descriptorPool->bindBuffer(*grabBuffer, VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::SIMULATION, 1); |
|
|
|
|
|
|
|
|
|
createComputePipelines(); |
|
|
|
@ -135,7 +149,8 @@ Application::Application() { |
|
|
|
|
|
|
|
|
|
void Application::mainLoop() { |
|
|
|
|
vkDeviceWaitIdle(Instance::GetDevice()); |
|
|
|
|
std::future compute = std::async(std::launch::async, [this](){ |
|
|
|
|
std::future compute = std::async( |
|
|
|
|
std::launch::async, [this]() { |
|
|
|
|
while (!glfwWindowShouldClose(Instance::instance->window)) { |
|
|
|
|
auto t1 = system_clock::now(); |
|
|
|
|
update(); |
|
|
|
@ -143,13 +158,16 @@ void Application::mainLoop() { |
|
|
|
|
|
|
|
|
|
auto measuredUpdateDuration = duration<float>(t2 - t1); |
|
|
|
|
auto requestedUpdateDuration = duration<float>( |
|
|
|
|
simulationPropertiesBuffer->access<SimulationUniformData>().dt); |
|
|
|
|
simulationPropertiesBuffer->access<SimulationUniformData>().dt |
|
|
|
|
); |
|
|
|
|
std::this_thread::sleep_for(requestedUpdateDuration - measuredUpdateDuration); |
|
|
|
|
|
|
|
|
|
performanceInformation.updateDuration = measuredUpdateDuration.count(); |
|
|
|
|
performanceInformation.recentTotalUpdateDurations.push(std::max(requestedUpdateDuration, measuredUpdateDuration).count()); |
|
|
|
|
performanceInformation.recentTotalUpdateDurations.push( |
|
|
|
|
std::max(requestedUpdateDuration, measuredUpdateDuration).count()); |
|
|
|
|
} |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
auto t1 = system_clock::now(); |
|
|
|
|
while (!glfwWindowShouldClose(Instance::instance->window)) { |
|
|
|
@ -247,27 +265,37 @@ void Application::addSoftBody(const std::string &modelFile, size_t count) { |
|
|
|
|
|
|
|
|
|
auto commandBuffer = Instance::instance->renderingCommandPool->beginSingleTimeCommandBuffer(); |
|
|
|
|
|
|
|
|
|
VkBufferUsageFlags bufferUsageFlags = VK_BUFFER_USAGE_STORAGE_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_TRANSFER_SRC_BIT; |
|
|
|
|
VkBufferUsageFlags bufferUsageFlags = |
|
|
|
|
VK_BUFFER_USAGE_STORAGE_BUFFER_BIT | VK_BUFFER_USAGE_TRANSFER_DST_BIT | VK_BUFFER_USAGE_TRANSFER_SRC_BIT; |
|
|
|
|
VmaMemoryUsage memoryUsage = VMA_MEMORY_USAGE_AUTO_PREFER_DEVICE; |
|
|
|
|
|
|
|
|
|
if (vertexBuffers[0] == nullptr) { |
|
|
|
|
newVertexBuffers[0] = make_unique<Buffer>(newVertices.size() * sizeof(Vertex), |
|
|
|
|
newVertexBuffers[0] = make_unique<Buffer>( |
|
|
|
|
newVertices.size() * sizeof(Vertex), |
|
|
|
|
bufferUsageFlags | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, |
|
|
|
|
memoryUsage, 0); |
|
|
|
|
newVertexBuffers[1] = make_unique<Buffer>(newVertices.size() * sizeof(Vertex), |
|
|
|
|
memoryUsage, 0, "Vertices 0" |
|
|
|
|
); |
|
|
|
|
newVertexBuffers[1] = make_unique<Buffer>( |
|
|
|
|
newVertices.size() * sizeof(Vertex), |
|
|
|
|
bufferUsageFlags | VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, |
|
|
|
|
memoryUsage, 0); |
|
|
|
|
newFaceBuffer = make_unique<Buffer>(newFaces.size() * sizeof(Face), |
|
|
|
|
memoryUsage, 0, "Vertices 1" |
|
|
|
|
); |
|
|
|
|
newFaceBuffer = make_unique<Buffer>( |
|
|
|
|
newFaces.size() * sizeof(Face), |
|
|
|
|
bufferUsageFlags | VK_BUFFER_USAGE_INDEX_BUFFER_BIT, |
|
|
|
|
memoryUsage, 0); |
|
|
|
|
newEdgeBuffer = make_unique<Buffer>(constraintData.edges.size() * sizeof(Edge), bufferUsageFlags, memoryUsage, 0); |
|
|
|
|
newTetrahedronBuffer = make_unique<Buffer>(constraintData.tetrahedra.size() * sizeof(Tetrahedron), bufferUsageFlags, memoryUsage, 0); |
|
|
|
|
|
|
|
|
|
newVertexBuffers[0]->setName("Vertices 0"); |
|
|
|
|
newVertexBuffers[1]->setName("Vertices 1"); |
|
|
|
|
newFaceBuffer->setName("Faces"); |
|
|
|
|
newEdgeBuffer->setName("Edges"); |
|
|
|
|
newTetrahedronBuffer->setName("Tetrahedra"); |
|
|
|
|
memoryUsage, 0, "Faces" |
|
|
|
|
); |
|
|
|
|
newEdgeBuffer = make_unique<Buffer>( |
|
|
|
|
constraintData.edges.size() * sizeof(Edge), |
|
|
|
|
bufferUsageFlags, |
|
|
|
|
memoryUsage, |
|
|
|
|
0, |
|
|
|
|
"Edges" |
|
|
|
|
); |
|
|
|
|
newTetrahedronBuffer = make_unique<Buffer>( |
|
|
|
|
constraintData.tetrahedra.size() * sizeof(Tetrahedron), bufferUsageFlags, memoryUsage, 0, |
|
|
|
|
"Tetrahedra" |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
newVertexBuffers[0]->setData(newVertices.data(), 0, newVertexBuffers[0]->size, commandBuffer); |
|
|
|
|
newVertexBuffers[1]->setData(newVertices.data(), 0, newVertexBuffers[1]->size, commandBuffer); |
|
|
|
@ -275,11 +303,19 @@ void Application::addSoftBody(const std::string &modelFile, size_t count) { |
|
|
|
|
newEdgeBuffer->setData(constraintData.edges.data(), 0, newEdgeBuffer->size, commandBuffer); |
|
|
|
|
newTetrahedronBuffer->setData(constraintData.tetrahedra.data(), 0, newTetrahedronBuffer->size, commandBuffer); |
|
|
|
|
} else { |
|
|
|
|
newVertexBuffers[0] = vertexBuffers[0]->appended(newVertices.data(), newVertices.size() * sizeof(Vertex), commandBuffer); |
|
|
|
|
newVertexBuffers[1] = vertexBuffers[1]->appended(newVertices.data(), newVertices.size() * sizeof(Vertex), commandBuffer); |
|
|
|
|
newVertexBuffers[0] = vertexBuffers[0]->appended( |
|
|
|
|
newVertices.data(), newVertices.size() * sizeof(Vertex), commandBuffer |
|
|
|
|
); |
|
|
|
|
newVertexBuffers[1] = vertexBuffers[1]->appended( |
|
|
|
|
newVertices.data(), newVertices.size() * sizeof(Vertex), commandBuffer |
|
|
|
|
); |
|
|
|
|
newFaceBuffer = faceBuffer->appended(newFaces.data(), newFaces.size() * sizeof(Face), commandBuffer); |
|
|
|
|
newEdgeBuffer = edgeBuffer->replaced(constraintData.edges.data(), constraintData.edges.size() * sizeof(Edge), commandBuffer); |
|
|
|
|
newTetrahedronBuffer = tetrahedronBuffer->replaced(constraintData.tetrahedra.data(), constraintData.tetrahedra.size() * sizeof(Tetrahedron), commandBuffer); |
|
|
|
|
newEdgeBuffer = edgeBuffer->replaced( |
|
|
|
|
constraintData.edges.data(), constraintData.edges.size() * sizeof(Edge), commandBuffer |
|
|
|
|
); |
|
|
|
|
newTetrahedronBuffer = tetrahedronBuffer->replaced( |
|
|
|
|
constraintData.tetrahedra.data(), constraintData.tetrahedra.size() * sizeof(Tetrahedron), commandBuffer |
|
|
|
|
); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
VkQueue queue = Instance::instance->graphicsAndPresentQueue; |
|
|
|
@ -291,7 +327,9 @@ void Application::addSoftBody(const std::string &modelFile, size_t count) { |
|
|
|
|
edgeBuffer = std::move(newEdgeBuffer); |
|
|
|
|
tetrahedronBuffer = std::move(newTetrahedronBuffer); |
|
|
|
|
|
|
|
|
|
descriptorPool->bindBuffer(*vertexBuffers[1 - currentDrawVertexBuffer], VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::MESH, 0); |
|
|
|
|
descriptorPool->bindBuffer( |
|
|
|
|
*vertexBuffers[1 - currentDrawVertexBuffer], VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::MESH, 0 |
|
|
|
|
); |
|
|
|
|
descriptorPool->bindBuffer(*faceBuffer, VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::MESH, 1); |
|
|
|
|
descriptorPool->bindBuffer(*edgeBuffer, VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::MESH, 2); |
|
|
|
|
descriptorPool->bindBuffer(*tetrahedronBuffer, VK_DESCRIPTOR_TYPE_STORAGE_BUFFER, DescriptorSet::MESH, 4); |
|
|
|
@ -330,11 +368,13 @@ void Application::createComputePipelines() { |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::MESH]); |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::WORLD]); |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::SIMULATION]); |
|
|
|
|
pushRanges.push_back({ |
|
|
|
|
pushRanges.push_back( |
|
|
|
|
{ |
|
|
|
|
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT, |
|
|
|
|
.offset = 0, |
|
|
|
|
.size = sizeof(GrabPushData) |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
); |
|
|
|
|
grabPipeline = unique_ptr<ComputePipeline>(new ComputePipeline("shaders/grab.spv", layouts, pushRanges)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -344,11 +384,13 @@ void Application::createComputePipelines() { |
|
|
|
|
{ |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::MESH]); |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::SIMULATION]); |
|
|
|
|
pushRanges.push_back({ |
|
|
|
|
pushRanges.push_back( |
|
|
|
|
{ |
|
|
|
|
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT, |
|
|
|
|
.offset = 0, |
|
|
|
|
.size = sizeof(PBDPushData) |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
); |
|
|
|
|
pbdPipeline = unique_ptr<ComputePipeline>(new ComputePipeline("shaders/pbd.spv", layouts, pushRanges)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -357,11 +399,13 @@ void Application::createComputePipelines() { |
|
|
|
|
|
|
|
|
|
{ |
|
|
|
|
layouts.push_back(descriptorPool->layouts[DescriptorSet::MESH]); |
|
|
|
|
pushRanges.push_back({ |
|
|
|
|
pushRanges.push_back( |
|
|
|
|
{ |
|
|
|
|
.stageFlags = VK_SHADER_STAGE_COMPUTE_BIT, |
|
|
|
|
.offset = 0, |
|
|
|
|
.size = sizeof(uint32_t) |
|
|
|
|
}); |
|
|
|
|
} |
|
|
|
|
); |
|
|
|
|
normalPipeline = unique_ptr<ComputePipeline>(new ComputePipeline("shaders/normal.spv", layouts, pushRanges)); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -379,7 +423,9 @@ void Application::drawFrame(float dt) { |
|
|
|
|
vkWaitForFences(Instance::GetDevice(), 1, &renderFence->handle, VK_TRUE, UINT64_MAX); |
|
|
|
|
|
|
|
|
|
uint32_t imageIndex; |
|
|
|
|
VkResult result = vkAcquireNextImageKHR(Instance::GetDevice(), swapchain->handle, UINT64_MAX, imageAvailable->handle, VK_NULL_HANDLE, &imageIndex); |
|
|
|
|
VkResult result = vkAcquireNextImageKHR( |
|
|
|
|
Instance::GetDevice(), swapchain->handle, UINT64_MAX, imageAvailable->handle, VK_NULL_HANDLE, &imageIndex |
|
|
|
|
); |
|
|
|
|
if (result == VK_ERROR_OUT_OF_DATE_KHR) { |
|
|
|
|
submitMutex.lock(); |
|
|
|
|
swapchain->recreateSwapchain(); |
|
|
|
@ -414,8 +460,10 @@ void Application::drawFrame(float dt) { |
|
|
|
|
barrier.srcAccessMask = VK_ACCESS_SHADER_WRITE_BIT; |
|
|
|
|
barrier.dstAccessMask = VK_ACCESS_VERTEX_ATTRIBUTE_READ_BIT; |
|
|
|
|
|
|
|
|
|
vkCmdPipelineBarrier(cmdBuffer, VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT, VK_PIPELINE_STAGE_VERTEX_INPUT_BIT, 0, 1, |
|
|
|
|
&barrier, 1, &vertexBufferBarrier, 0, nullptr); |
|
|
|
|
vkCmdPipelineBarrier( |
|
|
|
|
cmdBuffer, VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT, VK_PIPELINE_STAGE_VERTEX_INPUT_BIT, 0, 1, |
|
|
|
|
&barrier, 1, &vertexBufferBarrier, 0, nullptr |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
VkRenderPassBeginInfo renderPassInfo {}; |
|
|
|
|
renderPassInfo.sType = VK_STRUCTURE_TYPE_RENDER_PASS_BEGIN_INFO; |
|
|
|
@ -504,7 +552,10 @@ void Application::recordDrawCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
VkDeviceSize offsets[] = {0}; |
|
|
|
|
vkCmdBindVertexBuffers(commandBuffer, 0, 1, buffers, offsets); |
|
|
|
|
vkCmdBindIndexBuffer(commandBuffer, faceBuffer->handle, 0, VK_INDEX_TYPE_UINT32); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, graphicsPipeline->layout, 0, 1, &descriptorPool->sets[DescriptorSet::WORLD], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_GRAPHICS, graphicsPipeline->layout, 0, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::WORLD], 0, nullptr |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
for (const auto &softBody: softBodies) { |
|
|
|
|
vkCmdDrawIndexed(commandBuffer, softBody->faces.size() * 3, 1, softBody->firstIndex, 0, 0); |
|
|
|
@ -551,7 +602,10 @@ void Application::update() { |
|
|
|
|
copyRegion.srcOffset = 0; |
|
|
|
|
copyRegion.dstOffset = 0; |
|
|
|
|
|
|
|
|
|
vkCmdCopyBuffer(cmdBuffer, vertexBuffers[1 - currentDrawVertexBuffer]->handle, vertexBuffers[currentDrawVertexBuffer]->handle, 1, ©Region); |
|
|
|
|
vkCmdCopyBuffer( |
|
|
|
|
cmdBuffer, vertexBuffers[1 - currentDrawVertexBuffer]->handle, |
|
|
|
|
vertexBuffers[currentDrawVertexBuffer]->handle, 1, ©Region |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
vkEndCommandBuffer(cmdBuffer); |
|
|
|
|
|
|
|
|
@ -592,16 +646,28 @@ void Application::recordGrabCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
// TODO maybe add buffermemorybarrier for camera uniform, because it can be changed from main drawing thread
|
|
|
|
|
|
|
|
|
|
vkCmdBindPipeline(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->handle); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 0, 1, &descriptorPool->sets[DescriptorSet::MESH], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 1, 1, &descriptorPool->sets[DescriptorSet::WORLD], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 2, 1, &descriptorPool->sets[DescriptorSet::SIMULATION], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 0, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::MESH], 0, nullptr |
|
|
|
|
); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 1, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::WORLD], 0, nullptr |
|
|
|
|
); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, grabPipeline->layout, 2, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::SIMULATION], 0, nullptr |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
GrabPushData pushConstants {}; |
|
|
|
|
|
|
|
|
|
if (grabber->started()) { |
|
|
|
|
pushConstants.state = 0; |
|
|
|
|
pushConstants.screenPosition = grabber->previousCursorPosition; |
|
|
|
|
vkCmdPushConstants(commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), &pushConstants); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), |
|
|
|
|
&pushConstants |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
uint32_t faceInvocations = GetGroupCount(sizeInformation->faceCount, BLOCK_SIZE_GRAB); |
|
|
|
|
|
|
|
|
@ -609,7 +675,10 @@ void Application::recordGrabCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
computePipelineBarrier(commandBuffer); |
|
|
|
|
|
|
|
|
|
pushConstants.state = 1; |
|
|
|
|
vkCmdPushConstants(commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), &pushConstants); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), |
|
|
|
|
&pushConstants |
|
|
|
|
); |
|
|
|
|
vkCmdDispatch(commandBuffer, 1, 1, 1); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -619,7 +688,10 @@ void Application::recordGrabCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
if (grabber->moved(screenDelta)) { |
|
|
|
|
pushConstants.state = 2; |
|
|
|
|
pushConstants.screenDelta = screenDelta; |
|
|
|
|
vkCmdPushConstants(commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), &pushConstants); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), |
|
|
|
|
&pushConstants |
|
|
|
|
); |
|
|
|
|
vkCmdDispatch(commandBuffer, 1, 1, 1); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -627,7 +699,10 @@ void Application::recordGrabCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
|
|
|
|
|
if (grabber->stopped()) { |
|
|
|
|
pushConstants.state = 3; |
|
|
|
|
vkCmdPushConstants(commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), &pushConstants); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, grabPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(GrabPushData), |
|
|
|
|
&pushConstants |
|
|
|
|
); |
|
|
|
|
vkCmdDispatch(commandBuffer, 1, 1, 1); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
@ -638,8 +713,14 @@ void Application::recordPBDCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
uint32_t vertexGroupCount = GetGroupCount(sizeInformation->vertexCount, BLOCK_SIZE_PBD); |
|
|
|
|
|
|
|
|
|
vkCmdBindPipeline(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, pbdPipeline->handle); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, pbdPipeline->layout, 0, 1, &descriptorPool->sets[DescriptorSet::MESH], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, pbdPipeline->layout, 1, 1, &descriptorPool->sets[DescriptorSet::SIMULATION], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, pbdPipeline->layout, 0, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::MESH], 0, nullptr |
|
|
|
|
); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, pbdPipeline->layout, 1, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::SIMULATION], 0, nullptr |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
uint32_t state; |
|
|
|
|
|
|
|
|
@ -647,12 +728,16 @@ void Application::recordPBDCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
|
|
|
|
|
for (size_t i = 0; i < k; i++) { |
|
|
|
|
state = 0; |
|
|
|
|
vkCmdPushConstants(commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state |
|
|
|
|
); |
|
|
|
|
vkCmdDispatch(commandBuffer, vertexGroupCount, 1, 1); |
|
|
|
|
computePipelineBarrier(commandBuffer); |
|
|
|
|
|
|
|
|
|
state = 1; |
|
|
|
|
vkCmdPushConstants(commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
for (uint32_t partition = 0; partition < constraintData.partitionCount; partition++) { |
|
|
|
|
auto edgePartition = constraintData.edgePartitions[partition]; |
|
|
|
@ -663,9 +748,11 @@ void Application::recordPBDCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
|
|
|
|
|
ConstraintData::Partition partitions[2] = {edgePartition, tetrahedronPartition}; |
|
|
|
|
|
|
|
|
|
vkCmdPushConstants(commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, |
|
|
|
|
offsetof(PBDPushData, edgePartition), |
|
|
|
|
sizeof(partitions), partitions); |
|
|
|
|
sizeof(partitions), partitions |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
uint32_t invocations = GetGroupCount(edgePartition.size + tetrahedronPartition.size, BLOCK_SIZE_PBD); |
|
|
|
|
vkCmdDispatch(commandBuffer, invocations, 1, 1); |
|
|
|
@ -673,7 +760,9 @@ void Application::recordPBDCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
state = 2; |
|
|
|
|
vkCmdPushConstants(commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state); |
|
|
|
|
vkCmdPushConstants( |
|
|
|
|
commandBuffer, pbdPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state |
|
|
|
|
); |
|
|
|
|
vkCmdDispatch(commandBuffer, vertexGroupCount, 1, 1); |
|
|
|
|
computePipelineBarrier(commandBuffer); |
|
|
|
|
} |
|
|
|
@ -684,7 +773,10 @@ void Application::recordNormalCommands(VkCommandBuffer commandBuffer) { |
|
|
|
|
uint32_t faceGroupCount = GetGroupCount(sizeInformation->faceCount, BLOCK_SIZE_NORMAL); |
|
|
|
|
|
|
|
|
|
vkCmdBindPipeline(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, normalPipeline->handle); |
|
|
|
|
vkCmdBindDescriptorSets(commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, normalPipeline->layout, 0, 1, &descriptorPool->sets[DescriptorSet::MESH], 0, nullptr); |
|
|
|
|
vkCmdBindDescriptorSets( |
|
|
|
|
commandBuffer, VK_PIPELINE_BIND_POINT_COMPUTE, normalPipeline->layout, 0, 1, |
|
|
|
|
&descriptorPool->sets[DescriptorSet::MESH], 0, nullptr |
|
|
|
|
); |
|
|
|
|
|
|
|
|
|
uint32_t state = 0; |
|
|
|
|
vkCmdPushConstants(commandBuffer, normalPipeline->layout, VK_SHADER_STAGE_COMPUTE_BIT, 0, sizeof(uint32_t), &state); |
|
|
|
@ -718,12 +810,14 @@ void Application::computePipelineBarrier(VkCommandBuffer commandBuffer) { |
|
|
|
|
bufferMemoryBarrier.srcQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED; |
|
|
|
|
bufferMemoryBarrier.dstQueueFamilyIndex = VK_QUEUE_FAMILY_IGNORED; |
|
|
|
|
|
|
|
|
|
vkCmdPipelineBarrier(commandBuffer, |
|
|
|
|
vkCmdPipelineBarrier( |
|
|
|
|
commandBuffer, |
|
|
|
|
VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT, VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT, |
|
|
|
|
0, |
|
|
|
|
1, &memoryBarrier, |
|
|
|
|
1, &bufferMemoryBarrier, |
|
|
|
|
0, nullptr); |
|
|
|
|
0, nullptr |
|
|
|
|
); |
|
|
|
|
} |
|
|
|
|
|
|
|
|
|
void Application::imGuiWindows() { |
|
|
|
@ -731,7 +825,8 @@ void Application::imGuiWindows() { |
|
|
|
|
ImGui_ImplGlfw_NewFrame(); |
|
|
|
|
ImGui::NewFrame(); |
|
|
|
|
|
|
|
|
|
ImGui::Begin("Performance"); { |
|
|
|
|
ImGui::Begin("Performance"); |
|
|
|
|
{ |
|
|
|
|
float updateMS = performanceInformation.updateDuration * 1000.f; |
|
|
|
|
float updateHZ = 1 / performanceInformation.recentTotalUpdateDurations.average(); |
|
|
|
|
ImGui::Text("Updates: %2.0fms | %.1fHz", updateMS, updateHZ); |
|
|
|
@ -739,14 +834,17 @@ void Application::imGuiWindows() { |
|
|
|
|
float frameMS = performanceInformation.recentFrameDurations.average() * 1000.f; |
|
|
|
|
float frameHZ = 1 / performanceInformation.recentFrameDurations.average(); |
|
|
|
|
ImGui::Text("Frames: %.2fms | %.1fHz", frameMS, frameHZ); |
|
|
|
|
} ImGui::End(); |
|
|
|
|
} |
|
|
|
|
ImGui::End(); |
|
|
|
|
|
|
|
|
|
ImGui::Begin("Scene"); { |
|
|
|
|
ImGui::Begin("Scene"); |
|
|
|
|
{ |
|
|
|
|
if (ImGui::Button("Add")) { |
|
|
|
|
addSoftBody("models/bunny_medium.ply"); |
|
|
|
|
} |
|
|
|
|
for (const auto &softBody: softBodies) { |
|
|
|
|
ImGui::Text("Some softbody"); |
|
|
|
|
} |
|
|
|
|
} ImGui::End(); |
|
|
|
|
} |
|
|
|
|
ImGui::End(); |
|
|
|
|
} |
|
|
|
|