From f07113aa7c6fea6efabba86c0e8014d1a3dfb40c Mon Sep 17 00:00:00 2001 From: chee Date: Wed, 16 Aug 2023 18:36:53 +0800 Subject: [PATCH 01/13] Fix memory usage for buffer. --- .../api/dynamic_uniform_buffers/dynamic_uniform_buffers.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/samples/api/dynamic_uniform_buffers/dynamic_uniform_buffers.cpp b/samples/api/dynamic_uniform_buffers/dynamic_uniform_buffers.cpp index eb6756144..5baf05e41 100644 --- a/samples/api/dynamic_uniform_buffers/dynamic_uniform_buffers.cpp +++ b/samples/api/dynamic_uniform_buffers/dynamic_uniform_buffers.cpp @@ -213,13 +213,13 @@ void DynamicUniformBuffers::generate_cube() vertex_buffer = std::make_unique(get_device(), vertex_buffer_size, VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, - VMA_MEMORY_USAGE_GPU_TO_CPU); + VMA_MEMORY_USAGE_CPU_TO_GPU); vertex_buffer->update(vertices.data(), vertex_buffer_size); index_buffer = std::make_unique(get_device(), index_buffer_size, VK_BUFFER_USAGE_INDEX_BUFFER_BIT, - VMA_MEMORY_USAGE_GPU_TO_CPU); + VMA_MEMORY_USAGE_CPU_TO_GPU); index_buffer->update(indices.data(), index_buffer_size); } From 59722bfb297abb00ecc79587525cd6dd62ff47c3 Mon Sep 17 00:00:00 2001 From: chee Date: Thu, 17 Aug 2023 22:05:00 +0800 Subject: [PATCH 02/13] Fix memory usage for buffer. --- .../hpp_dynamic_uniform_buffers.cpp | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp index c178cd6f7..bc80d6865 100644 --- a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp +++ b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp @@ -260,10 +260,10 @@ void HPPDynamicUniformBuffers::generate_cube() // For the sake of simplicity we won't stage the vertex data to the gpu memory // Vertex buffer vertex_buffer = - std::make_unique(*get_device(), vertex_buffer_size, vk::BufferUsageFlagBits::eVertexBuffer, VMA_MEMORY_USAGE_GPU_TO_CPU); + std::make_unique(*get_device(), vertex_buffer_size, vk::BufferUsageFlagBits::eVertexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); vertex_buffer->update(vertices.data(), vertex_buffer_size); - index_buffer = std::make_unique(*get_device(), index_buffer_size, vk::BufferUsageFlagBits::eIndexBuffer, VMA_MEMORY_USAGE_GPU_TO_CPU); + index_buffer = std::make_unique(*get_device(), index_buffer_size, vk::BufferUsageFlagBits::eIndexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); index_buffer->update(indices.data(), index_buffer_size); } From 91d44edf46d6a6748697cbf4d63c29b046829ee1 Mon Sep 17 00:00:00 2001 From: Sascha Willems Date: Thu, 17 Aug 2023 19:52:43 +0200 Subject: [PATCH 03/13] Add asciidoc readmes for ray tracint samples It looks like a recent PR reverted the initial asciidoc conversions for those sample, --- .../extensions/ray_tracing_basic/README.adoc | 26 ++++ .../extensions/ray_tracing_basic/README.md | 23 --- .../ray_tracing_extended/README.adoc | 133 ++++++++++++++++ .../extensions/ray_tracing_extended/README.md | 147 ------------------ 4 files changed, 159 insertions(+), 170 deletions(-) create mode 100644 samples/extensions/ray_tracing_basic/README.adoc delete mode 100644 samples/extensions/ray_tracing_basic/README.md create mode 100644 samples/extensions/ray_tracing_extended/README.adoc delete mode 100644 samples/extensions/ray_tracing_extended/README.md diff --git a/samples/extensions/ray_tracing_basic/README.adoc b/samples/extensions/ray_tracing_basic/README.adoc new file mode 100644 index 000000000..f4de092e8 --- /dev/null +++ b/samples/extensions/ray_tracing_basic/README.adoc @@ -0,0 +1,26 @@ +//// +- Copyright (c) 2020-2023, The Khronos Group +- +- SPDX-License-Identifier: Apache-2.0 +- +- Licensed under the Apache License, Version 2.0 the "License"; +- you may not use this file except in compliance with the License. +- You may obtain a copy of the License at +- +- http://www.apache.org/licenses/LICENSE-2.0 +- +- Unless required by applicable law or agreed to in writing, software +- distributed under the License is distributed on an "AS IS" BASIS, +- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +- See the License for the specific language governing permissions and +- limitations under the License. +- +//// + +=== Basic hardware accelerated ray tracing + +*Extensions*: https://www.khronos.org/registry/vulkan/specs/1.2-extensions/html/vkspec.html#VK_KHR_ray_tracing_pipeline[`VK_KHR_ray_tracing_pipeline`], https://www.khronos.org/registry/vulkan/specs/1.2-extensions/html/vkspec.html#VK_KHR_acceleration_structure[`VK_KHR_acceleration_structure`] + +Render a basic scene using the official cross-vendor ray tracing extension. +Shows how to setup all data structures required for ray tracing, including the bottom and top level acceleration structures for the geometry, the shader binding table and the ray tracing pipelines with shader groups for ray generation, ray hits, and ray misses. +After dispatching the rays, the final result is copied to the swapchain image. diff --git a/samples/extensions/ray_tracing_basic/README.md b/samples/extensions/ray_tracing_basic/README.md deleted file mode 100644 index b4e49c06c..000000000 --- a/samples/extensions/ray_tracing_basic/README.md +++ /dev/null @@ -1,23 +0,0 @@ - -### Basic hardware accelerated ray tracing - -**Extensions**: [```VK_KHR_ray_tracing_pipeline```](https://www.khronos.org/registry/vulkan/specs/1.2-extensions/html/vkspec.html#VK_KHR_ray_tracing_pipeline), [```VK_KHR_acceleration_structure```](https://www.khronos.org/registry/vulkan/specs/1.2-extensions/html/vkspec.html#VK_KHR_acceleration_structure) - -Render a basic scene using the official cross-vendor ray tracing extension. Shows how to setup all data structures required for ray tracing, including the bottom and top level acceleration structures for the geometry, the shader binding table and the ray tracing pipelines with shader groups for ray generation, ray hits, and ray misses. After dispatching the rays, the final result is copied to the swapchain image. diff --git a/samples/extensions/ray_tracing_extended/README.adoc b/samples/extensions/ray_tracing_extended/README.adoc new file mode 100644 index 000000000..1c6de4852 --- /dev/null +++ b/samples/extensions/ray_tracing_extended/README.adoc @@ -0,0 +1,133 @@ +//// +- Copyright (c) 2019-2023, Holochip Corporation +- +- SPDX-License-Identifier: Apache-2.0 +- +- Licensed under the Apache License, Version 2.0 the "License"; +- you may not use this file except in compliance with the License. +- You may obtain a copy of the License at +- +- http://www.apache.org/licenses/LICENSE-2.0 +- +- Unless required by applicable law or agreed to in writing, software +- distributed under the License is distributed on an "AS IS" BASIS, +- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +- See the License for the specific language governing permissions and +- limitations under the License. +- +//// += Ray-tracing: Extended features and dynamic objects + +This code sample demonstrates how to incorporate animations into a ray-traced scene, and shows how to incorporate different types of changing objects within the acceleration structures. + +== Acceleration structures + +The ray tracing acceleration structures are separated into two types: bottom-level acceleration structures (BLAS) and top-level acceleration structures (TLAS). +The BLAS contains information about each object's geometry within its own coordinate system and is built using the vertex and index data stored in a GPU buffer. +In contrast, the TLAS contains information about each instance of the geometry and its transformation (i.e. +scaling, rotation, translation, etc.). + +Each object must be represented in the BLAS, but can have any number of instances, each with its own transformation. +This allows objects to be replicated without creating an acceleration structure for each instance. + +== Objects: Static, moving, and changing + +There are three categories of objects to consider when building acceleration structures: static, moving, and changing geometry. +Static geometry includes scene data. +In this code sample, the Sponza scene has a single, non-moving instance. +In contrast, dynamic objects can have a changing transformation, changing geometry, or both. +An example of transformation-only dynamic objects in this code sample are given by the flame particle effect, which is achieved by adjusting only the location and rotation of a square billboard -- the internal geometry (and thus the billboard's BLAS) does not change. +In contrast, the refraction effect is achieved by changing both the internal geometry each frame, and the rotation (so that it faces the viewer). + +Vulkan offers methods of optimizing the acceleration structures for each type of geometry. +The `VkAccelerationStructureBuildGeometryInfoKHR` struct has flags that can either toggle "fast trace", which optimizes run-time performance at the expense of build time, or "fast build", which optimizes build time. +When constructing large, static objects such as the Sponza scene, for instance, the "fast trace" bit (`VK_BUILD_ACCELERATION_STRUCTURE_PREFER_FAST_TRACE_BIT_KHR`) is selected because the build will occur once and the model contains many points. +When constructing dynamic objects such as the refraction model, which will need a BLAS update every frame, the "fast build" bit (`VK_BUILD_ACCELERATION_STRUCTURE_PREFER_FAST_BUILD_BIT_KHR`) is selected. + +Further optimization methods can be used. +For instance, the refraction model is updated every frame by the CPU and thus uses host-visible memory. +However, because host-visible memory can incur a performance penalty, the Sponza and billboard models use a staging buffer to copy to device-exclusive memory. +An alternative method would be to use a "compute shader" to generate the refraction model each frame, but that is outside the scope of this tutorial. + +== Reference Object Data from a Closest-Hit Shader + +Though the ray-tracing pipeline uses an acceleration structure to traverse the scene's geometry, the acceleration structures themselves do not store user-defined information about the geometry and instead give the developer the flexibility to define their own custom geometry information. +This information can be encoded at the per-instance level, per-object level, or per-primitive level. + +_Per-instance level:_ The top-level acceleration structure allows instance information to encode a custom ID ( per-instance level). + +_Per-object level:_ In this code sample, this custom ID then references a struct at the per-object level containing the object ID , the index of the vertices in the vertex buffer, and the index of the (triangle) indices in the index buffer: + +---- +struct SceneInstanceData +{ + uint32_t vertex_index; + uint32_t indices_index; + uint32_t image_index; + uint32_t object_type; +}; +---- + +_Per-primitive level_ In this sample, each vertex is encoded with a per-vertex normal and texture coordinate, though other applications may wish to provide other information at the per-vertex level. +To allow the bottom-level acceleration structure to reference geometry data with a custom-defined layout, the `VkAccelerationStructureGeometryKHR` provides the ability to set geometry offsets and strides (i.e. +`vertexStride`). +In the code below, the struct `acceleration_structure_geometry` of type `VkAccelerationStructureGeometryKHR` references the data layout provided by NewVertex, which encodes the normal and texture coordinate: + +---- +acceleration_structure_geometry.geometry.triangles.vertexData = vertex_data_device_address; +acceleration_structure_geometry.geometry.triangles.maxVertex = model_buffer.num_vertices; +acceleration_structure_geometry.geometry.triangles.vertexStride = sizeof(NewVertex); +acceleration_structure_geometry.geometry.triangles.indexType = VK_INDEX_TYPE_UINT32; +acceleration_structure_geometry.geometry.triangles.indexData = index_data_device_address; +acceleration_structure_geometry.geometry.triangles.transformData = transform_matrix_device_address; +---- + +This technique allows the closest-hit shader to access pre-calculated vertex information. + +== Texture Binding and Shaders + +In a traditional raster pipeline, it is possible to render each object separately and bind its appropriate texture images during that pass. +However, in a ray-tracing pipeline, each ray during a render pass could intersect with many objects within the scene, and thus all textures must be available to the shader. +In this code sample, an array of textures (`Sampler2D[]`) is bound, and each object is associated with a given texture index. +The texture ID information is stored in the object data. + +== Ambient Occlusion and Ray-Traced Shadows + +This code sample explores two different ways to calculate lighting: ray-traced shadows and ambient occlusion, both of which are updated each frame and are triggered when a primary ray intersects a scene object (i.e. +an element of the Sponza scene). + +Ray-traced shadows are calculated by performing a test: a ray is shot from the object point in the direction of the light. +If the returned distance is less than the distance to the light source, then the object point is in a shadow. +In pseudocode: + +---- +direction = object_pt - light_pt +dist = trace_ray(object_pt, direction) +if (dist < distance(object_pt, light_pt)): + color.rgb *= 0.2 +---- + +The ambient occlusion effect is used to simulate the light diminishing effect of clustered geometry. +It's simulated by tracing rays distributed about a hemisphere centered at the intersection point with the object's normal. +The light-diminishing effect is estimated using the distance to the nearest ray intersection. +In some implementations, a hard threshold is used. +In pseudocode: + +---- +for theta,phi in angles: + hard_threshold = 10.f + direction = hemisphere_pt(object_pt, normal, theta, phi) + dist = trace_ray(object_pt, direction) + if (dist < hard_threshold): + color.rgb *= 0.2 +---- + +The code sample in this tutorial instead linearly interpolates up to the hard_threshold: + +---- +color.rgb *= min(dist, hard_threshold) / min_threshold +---- + +There are further optimizations that can be used. +One common technique is to reduce the number of generated ambient occlusion rays at each point, often shooting just a single ray. +The resulting image can then be de-noised using a separate de-noising pass, though this technique is outside the scope of this tutorial. diff --git a/samples/extensions/ray_tracing_extended/README.md b/samples/extensions/ray_tracing_extended/README.md deleted file mode 100644 index db8545e1f..000000000 --- a/samples/extensions/ray_tracing_extended/README.md +++ /dev/null @@ -1,147 +0,0 @@ - - -# Ray-tracing: Extended features and dynamic objects - -This code sample demonstrates how to incorporate animations into a ray-traced scene, and shows how to incorporate -different types of changing objects within the acceleration structures. - -## Acceleration structures - -The ray tracing acceleration structures are separated into two types: bottom-level acceleration structures (BLAS) and -top-level acceleration structures (TLAS). The BLAS contains information about each object's geometry within its own -coordinate system and is built using the vertex and index data stored in a GPU buffer. In contrast, the TLAS contains -information about each instance of the geometry and its transformation (i.e. scaling, rotation, translation, etc.). - -Each object must be represented in the BLAS, but can have any number of instances, each with its own transformation. -This allows objects to be replicated without creating an acceleration structure for each instance. - -## Objects: Static, moving, and changing - -There are three categories of objects to consider when building acceleration structures: static, moving, and changing -geometry. Static geometry includes scene data. In this code sample, the Sponza scene has a single, non-moving instance. -In contrast, dynamic objects can have a changing transformation, changing geometry, or both. An example of -transformation-only dynamic objects in this code sample are given by the flame particle effect, which is achieved by -adjusting only the location and rotation of a square billboard -- the internal geometry (and thus the billboard's BLAS) -does not change. In contrast, the refraction effect is achieved by changing both the internal geometry each frame, and -the rotation (so that it faces the viewer). - -Vulkan offers methods of optimizing the acceleration structures for each type of geometry. -The `VkAccelerationStructureBuildGeometryInfoKHR` struct has flags that can either toggle "fast trace", which optimizes -run-time performance at the expense of build time, or "fast build", which optimizes build time. When constructing large, -static objects such as the Sponza scene, for instance, the "fast trace" -bit (`VK_BUILD_ACCELERATION_STRUCTURE_PREFER_FAST_TRACE_BIT_KHR`) is selected because the build will occur once and the -model contains many points. When constructing dynamic objects such as the refraction model, which will need a BLAS -update every frame, the "fast build" bit (`VK_BUILD_ACCELERATION_STRUCTURE_PREFER_FAST_BUILD_BIT_KHR`) is selected. - -Further optimization methods can be used. For instance, the refraction model is updated every frame by the CPU and thus -uses host-visible memory. However, because host-visible memory can incur a performance penalty, the Sponza and billboard -models use a staging buffer to copy to device-exclusive memory. An alternative method would be to use a "compute shader" -to generate the refraction model each frame, but that is outside the scope of this tutorial. - -## Reference Object Data from a Closest-Hit Shader - -Though the ray-tracing pipeline uses an acceleration structure to traverse the scene's geometry, the acceleration -structures themselves do not store user-defined information about the geometry and instead give the developer the -flexibility to define their own custom geometry information. This information can be encoded at the per-instance level, -per-object level, or per-primitive level. - -*Per-instance level:* The top-level acceleration structure allows instance information to encode a custom ID ( -per-instance level). - -*Per-object level:* In this code sample, this custom ID then references a struct at the per-object level containing the -object ID , the index of the vertices in the vertex buffer, and the index of the (triangle) indices in the index buffer: - -``` -struct SceneInstanceData -{ - uint32_t vertex_index; - uint32_t indices_index; - uint32_t image_index; - uint32_t object_type; -}; -``` - -*Per-primitive level* In this sample, each vertex is encoded with a per-vertex normal and texture coordinate, though -other applications may wish to provide other information at the per-vertex level. To allow the bottom-level acceleration -structure to reference geometry data with a custom-defined layout, the `VkAccelerationStructureGeometryKHR` provides the -ability to set geometry offsets and strides (i.e. `vertexStride`). In the code below, the -struct `acceleration_structure_geometry` of type `VkAccelerationStructureGeometryKHR` references the data layout -provided by NewVertex, which encodes the normal and texture coordinate: - -``` -acceleration_structure_geometry.geometry.triangles.vertexData = vertex_data_device_address; -acceleration_structure_geometry.geometry.triangles.maxVertex = model_buffer.num_vertices; -acceleration_structure_geometry.geometry.triangles.vertexStride = sizeof(NewVertex); -acceleration_structure_geometry.geometry.triangles.indexType = VK_INDEX_TYPE_UINT32; -acceleration_structure_geometry.geometry.triangles.indexData = index_data_device_address; -acceleration_structure_geometry.geometry.triangles.transformData = transform_matrix_device_address; -``` - -This technique allows the closest-hit shader to access pre-calculated vertex information. - -## Texture Binding and Shaders - -In a traditional raster pipeline, it is possible to render each object separately and bind its appropriate texture -images during that pass. However, in a ray-tracing pipeline, each ray during a render pass could intersect with many -objects within the scene, and thus all textures must be available to the shader. In this code sample, an array of -textures (`Sampler2D[]`) is bound, and each object is associated with a given texture index. The texture ID information -is stored in the object data. - -## Ambient Occlusion and Ray-Traced Shadows - -This code sample explores two different ways to calculate lighting: ray-traced shadows and ambient occlusion, both of -which are updated each frame and are triggered when a primary ray intersects a scene object (i.e. an element of the -Sponza scene). - -Ray-traced shadows are calculated by performing a test: a ray is shot from the object point in the direction of the -light. If the returned distance is less than the distance to the light source, then the object point is in a shadow. In -pseudocode: - -``` -direction = object_pt - light_pt -dist = trace_ray(object_pt, direction) -if (dist < distance(object_pt, light_pt)): - color.rgb *= 0.2 -``` - -The ambient occlusion effect is used to simulate the light diminishing effect of clustered geometry. It's simulated by -tracing rays distributed about a hemisphere centered at the intersection point with the object's normal. The -light-diminishing effect is estimated using the distance to the nearest ray intersection. In some implementations, a -hard threshold is used. In pseudocode: - -``` -for theta,phi in angles: - hard_threshold = 10.f - direction = hemisphere_pt(object_pt, normal, theta, phi) - dist = trace_ray(object_pt, direction) - if (dist < hard_threshold): - color.rgb *= 0.2 -``` - -The code sample in this tutorial instead linearly interpolates up to the hard_threshold: - -``` -color.rgb *= min(dist, hard_threshold) / min_threshold -``` - -There are further optimizations that can be used. One common technique is to reduce the number of generated ambient -occlusion rays at each point, often shooting just a single ray. The resulting image can then be de-noised using a -separate de-noising pass, though this technique is outside the scope of this tutorial. - From 250c96ce266cffa9ec4dbc3b1c731dc87dff3860 Mon Sep 17 00:00:00 2001 From: Sascha Willems Date: Thu, 17 Aug 2023 19:56:46 +0200 Subject: [PATCH 04/13] Convert shader object sample readme to asciidoc --- .../shader_object/{README.md => README.adoc} | 257 +++++++++++------- 1 file changed, 159 insertions(+), 98 deletions(-) rename samples/extensions/shader_object/{README.md => README.adoc} (64%) diff --git a/samples/extensions/shader_object/README.md b/samples/extensions/shader_object/README.adoc similarity index 64% rename from samples/extensions/shader_object/README.md rename to samples/extensions/shader_object/README.adoc index f734566b2..68586a190 100644 --- a/samples/extensions/shader_object/README.md +++ b/samples/extensions/shader_object/README.adoc @@ -1,24 +1,23 @@ - +//// += Shader Object -# Shader Object +image::./images/shader_object_screenshot.png[Sample] -![Sample](./images/shader_object_screenshot.png) - -## Overview +== Overview This sample demonstrates how to use the `VK_EXT_shader_object` extension, which provides a way to specify shaders and state without using `VkPipeline` objects. @@ -33,50 +32,62 @@ In its default configuration, this sample automatically iterates over permutatio * The output color format * The output depth format -The models in the middle demonstrate the use of linked vertex and fragment shaders, which can only ever change in tandem. The surrounding models use unlinked shaders, where each model uses its own separate vertex, fragment, and potentially geometry shaders. +The models in the middle demonstrate the use of linked vertex and fragment shaders, which can only ever change in tandem. +The surrounding models use unlinked shaders, where each model uses its own separate vertex, fragment, and potentially geometry shaders. Post processing, which is enabled by default, applies a specified visual effect to the results of model rendering. An optional wireframe mode may be enabled if supported by the device. -This sample can be configured at runtime through a debug GUI. This UI allows you to change various aspects of the scene that would be impractical to control dynamically using pipelines. UI drawing also demonstrates interoperability between pipelines within conventional render passes (which are used to render the UI) and shader objects with dynamic rendering (which are used to render the scene). +This sample can be configured at runtime through a debug GUI. +This UI allows you to change various aspects of the scene that would be impractical to control dynamically using pipelines. +UI drawing also demonstrates interoperability between pipelines within conventional render passes (which are used to render the UI) and shader objects with dynamic rendering (which are used to render the scene). -A plot of the last 2000 CPU frame times is displayed at the bottom of the screen. This plot shows the CPU impact of swapping shaders and state at runtime. +A plot of the last 2000 CPU frame times is displayed at the bottom of the screen. +This plot shows the CPU impact of swapping shaders and state at runtime. -Shader objects may only be used within `VK_KHR_dynamic_rendering` render passes. The [dynamic rendering sample](../dynamic_rendering) explains how to use that extension in more detail. The main parts relevant to drawing with shader objects are `vkCmdBeginRenderingKHR` and `vkCmdEndRenderingKHR`. +Shader objects may only be used within `VK_KHR_dynamic_rendering` render passes. +The link:../dynamic_rendering[dynamic rendering sample] explains how to use that extension in more detail. +The main parts relevant to drawing with shader objects are `vkCmdBeginRenderingKHR` and `vkCmdEndRenderingKHR`. -## Enabling the Extension +== Enabling the Extension -In order to use shader objects, both the `VK_EXT_shader_object` extension and the `shaderObject` feature need to be enabled on the `VkDevice`. This is accomplished by including `VK_EXT_SHADER_OBJECT_EXTENSION_NAME` in `VkDeviceCreateInfo`'s `ppEnabledExtensionNames` array, and also including a `VkPhysicalDeviceShaderObjectFeaturesEXT` structure with the `shaderObject` feature set to `VK_TRUE` in its `pNext` chain. +In order to use shader objects, both the `VK_EXT_shader_object` extension and the `shaderObject` feature need to be enabled on the `VkDevice`. +This is accomplished by including `VK_EXT_SHADER_OBJECT_EXTENSION_NAME` in ``VkDeviceCreateInfo``'s `ppEnabledExtensionNames` array, and also including a `VkPhysicalDeviceShaderObjectFeaturesEXT` structure with the `shaderObject` feature set to `VK_TRUE` in its `pNext` chain. In our case, these are handled through a common abstraction shared with the other samples. -```CPP +[,CPP] +---- add_device_extension(VK_EXT_SHADER_OBJECT_EXTENSION_NAME); -``` +---- -```CPP +[,CPP] +---- // Enable Shader Object auto &requestedShaderObject = gpu.request_extension_features(VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SHADER_OBJECT_FEATURES_EXT); requestedShaderObject.shaderObject = VK_TRUE; -``` +---- -## Shader Object Creation +== Shader Object Creation `VkShaderEXT` objects are created using the `vkCreateShadersEXT` function. -```CPP +[,CPP] +---- VkResult vkCreateShadersEXT( VkDevice device, uint32_t createInfoCount, const VkShaderCreateInfoEXT* pCreateInfos, const VkAllocationCallbacks* pAllocator, VkShaderEXT* pShaders); -``` +---- -In this sample, shaders are abstracted into a `Shader` class. This class holds the created `VkShaderEXT` object, information needed for building and binding the shader. +In this sample, shaders are abstracted into a `Shader` class. +This class holds the created `VkShaderEXT` object, information needed for building and binding the shader. -```CPP +[,CPP] +---- class Shader { VkShaderStageFlagBits stage; @@ -88,21 +99,26 @@ class Shader // ... } -``` +---- The scene contains multiple models that each use multiple `Shader` objects. -## Linked Shaders +== Linked Shaders -If we know a combination of shaders will always be used together we can link the shaders together. Linked shaders allow the driver to perform cross stage optimizations that can potentially improve GPU performance when the shaders are executed. +If we know a combination of shaders will always be used together we can link the shaders together. +Linked shaders allow the driver to perform cross stage optimizations that can potentially improve GPU performance when the shaders are executed. One of the ways this sample uses linked shaders is for the skybox. -The skybox uses only a vertex and fragment shader. The vertex shader's stage is set to `VK_SHADER_STAGE_VERTEX_BIT` with its next stage set to only `VK_SHADER_STAGE_FRAGMENT_BIT`. The fragment shader's stage is set to `VK_SHADER_STAGE_FRAGMENT_BIT`, with no next stage. +The skybox uses only a vertex and fragment shader. +The vertex shader's stage is set to `VK_SHADER_STAGE_VERTEX_BIT` with its next stage set to only `VK_SHADER_STAGE_FRAGMENT_BIT`. +The fragment shader's stage is set to `VK_SHADER_STAGE_FRAGMENT_BIT`, with no next stage. -To draw with shader objects we need to create one `VkShaderEXT` object per shader stage. This means we need an instance of the `Shader` class for each shader stage. +To draw with shader objects we need to create one `VkShaderEXT` object per shader stage. +This means we need an instance of the `Shader` class for each shader stage. -```CPP +[,CPP] +---- // Create shaders with current and next stage bits and set the shaders GLSL shader data, descriptor sets, and push constants skybox_vert_shader = new Shader(VK_SHADER_STAGE_VERTEX_BIT, @@ -117,11 +133,13 @@ skybox_frag_shader = new Shader(VK_SHADER_STAGE_FRAGMENT_BIT, frag_shader_data, &descriptor_set_layouts[ShaderTypeBasic], &push_constant_ranges[ShaderTypeBasic]); -``` +---- -The `Shader` class require us to provide the entire contents of the `VkShaderCreateInfoEXT` structure including information to compile the GLSL. The constructor will compile the GLSL and fill out the `VkShaderCreateInfoEXT` structure for this `Shader`. +The `Shader` class require us to provide the entire contents of the `VkShaderCreateInfoEXT` structure including information to compile the GLSL. +The constructor will compile the GLSL and fill out the `VkShaderCreateInfoEXT` structure for this `Shader`. -```CPP +[,CPP] +---- ShaderObject::Shader::Shader(VkShaderStageFlagBits stage_, VkShaderStageFlags next_stage_, std::string shader_name_, @@ -155,18 +173,24 @@ ShaderObject::Shader::Shader(VkShaderStageFlagBits stage_, vk_shader_create_info.pPushConstantRanges = pPushConstantRange; vk_shader_create_info.pSpecializationInfo = nullptr; } -``` +---- -We want the skybox shaders to be linked, so we need to add the `VK_SHADER_CREATE_LINK_STAGE_BIT_EXT` flag to each shader's `VkShaderCreateInfoEXT`. The sample determines whether to add the flag at shader creation time depending on if `build_shader` or `build_linked_shaders` is called. `build_linked_shaders` will be called with pointers to the vertex and fragment shaders. The sample only supports linked vertex and fragment shaders and does not handle linking other shader stages such as geometry. +We want the skybox shaders to be linked, so we need to add the `VK_SHADER_CREATE_LINK_STAGE_BIT_EXT` flag to each shader's `VkShaderCreateInfoEXT`. +The sample determines whether to add the flag at shader creation time depending on if `build_shader` or `build_linked_shaders` is called. +`build_linked_shaders` will be called with pointers to the vertex and fragment shaders. +The sample only supports linked vertex and fragment shaders and does not handle linking other shader stages such as geometry. -```CPP +[,CPP] +---- // Set the fragment shader as linked to build them linked and build the shader build_linked_shaders(device, skybox_vert_shader, skybox_frag_shader); -``` +---- -To build the shaders we call `vkCreateShadersEXT` with all of the `VkShaderCreateInfoEXT`s for each of the linked shaders. The `build_linked_shaders` function adds the `VK_SHADER_CREATE_LINK_STAGE_BIT_EXT` flag to each `VkShaderCreateInfoEXT` struct, then calls `vkCreateShadersEXT` on both shaders. +To build the shaders we call `vkCreateShadersEXT` with all of the ``VkShaderCreateInfoEXT``s for each of the linked shaders. +The `build_linked_shaders` function adds the `VK_SHADER_CREATE_LINK_STAGE_BIT_EXT` flag to each `VkShaderCreateInfoEXT` struct, then calls `vkCreateShadersEXT` on both shaders. -```CPP +[,CPP] +---- void ShaderObject::build_linked_shaders(VkDevice device, ShaderObject::Shader *vert, ShaderObject::Shader *frag) { VkShaderCreateInfoEXT shader_create_infos[2]; @@ -201,23 +225,29 @@ void ShaderObject::build_linked_shaders(VkDevice device, ShaderObject::Shader *v vert->set_shader(shaderEXTs[0]); frag->set_shader(shaderEXTs[1]); } -``` +---- The skybox shaders can now be bound and used to draw. -## Unlinked Shaders +== Unlinked Shaders -Linking shaders is optional. For some use cases it may be advantageous to create shaders that are not linked. This allows arbitrary combinations of shaders to be used together at command buffer recording time, though you should keep in mind that the driver may be less able to optimize the shaders' GPU performance. +Linking shaders is optional. +For some use cases it may be advantageous to create shaders that are not linked. +This allows arbitrary combinations of shaders to be used together at command buffer recording time, though you should keep in mind that the driver may be less able to optimize the shaders' GPU performance. In order to use a given combination of unlinked shaders together, their input and output interfaces need to be compatible and their arrays of descriptor set layouts and push constants must be identical. -Some models in the scene use unlinked shaders. For the sake of convenience, this sample refers to these models as "material models". +Some models in the scene use unlinked shaders. +For the sake of convenience, this sample refers to these models as "material models". -Each of the material models binds different vertex, geometry, and fragment shaders. Each of the vertex shaders can be used with each of the geometry and fragment shaders. +Each of the material models binds different vertex, geometry, and fragment shaders. +Each of the vertex shaders can be used with each of the geometry and fragment shaders. -This sample lets you disable the geometry stage through the debug UI, so all of the vertex shaders support both geometry or fragment as a next stage. In order to allow either next stage, the vertex shaders are created with next stage flags of `VK_SHADER_STAGE_GEOMETRY_BIT | VK_SHADER_STAGE_FRAGMENT_BIT` and set the GLSL, descriptors, and push constants. +This sample lets you disable the geometry stage through the debug UI, so all of the vertex shaders support both geometry or fragment as a next stage. +In order to allow either next stage, the vertex shaders are created with next stage flags of `VK_SHADER_STAGE_GEOMETRY_BIT | VK_SHADER_STAGE_FRAGMENT_BIT` and set the GLSL, descriptors, and push constants. -```CPP +[,CPP] +---- // Create shader with current and next stage bits set the GLSL shader data, descriptor sets, and push constants material_vert_shaders.emplace_back( new Shader(VK_SHADER_STAGE_VERTEX_BIT, @@ -226,18 +256,21 @@ material_vert_shaders.emplace_back( shader_data, &descriptor_set_layouts[ShaderTypeMaterial], &push_constant_ranges[ShaderTypeMaterial])); -``` +---- -Then build the shader alone. This works similarly to the earlier skybox example, except that the `VkShaderEXT`s are created separately. +Then build the shader alone. +This works similarly to the earlier skybox example, except that the ``VkShaderEXT``s are created separately. -```CPP +[,CPP] +---- // Build shader build_shader(device, material_vert_shaders.back()); -``` +---- -The `build_shader` function is very similar to `build_linked_shaders` but only creates one `VkShaderEXT` object. +The `build_shader` function is very similar to `build_linked_shaders` but only creates one `VkShaderEXT` object. -```CPP +[,CPP] +---- void ShaderObject::build_shader(VkDevice device, ShaderObject::Shader *shader) { VkShaderEXT shaderEXT; @@ -252,29 +285,35 @@ void ShaderObject::build_shader(VkDevice device, ShaderObject::Shader *shader) shader->set_shader(shaderEXT); } -``` -## Binding Shaders +---- + +== Binding Shaders -`VkShaderEXT` objects need to be bound to the command buffer for use in subsequent `vkCmdDraw*` calls. One or more shader objects can be bound using `vkCmdBindShadersEXT`. +`VkShaderEXT` objects need to be bound to the command buffer for use in subsequent `vkCmdDraw*` calls. +One or more shader objects can be bound using `vkCmdBindShadersEXT`. -Before drawing the skybox we need to bind the relevant `VkShaderEXT` objects. This sample calls the `bind_shader` function, which simply binds the `Shader`'s `VkShaderEXT`. +Before drawing the skybox we need to bind the relevant `VkShaderEXT` objects. +This sample calls the `bind_shader` function, which simply binds the ``Shader``'s `VkShaderEXT`. -```CPP +[,CPP] +---- // Bind shaders for the skybox bind_shader(draw_cmd_buffer, skybox_vert_shader); bind_shader(draw_cmd_buffer, skybox_frag_shader); -``` +---- -```CPP +[,CPP] +---- void ShaderObject::bind_shader(VkCommandBuffer cmd_buffer, ShaderObject::Shader *shader) { vkCmdBindShadersEXT(cmd_buffer, 1, shader->get_stage(), shader->get_shader()); } -``` +---- The unlinked shaders are bound with multiple calls to `bind_shader`, one per each shader to bind. -```CPP +[,CPP] +---- void ShaderObject::bind_material_shader(VkCommandBuffer cmd_buffer, int shader_index) { CurrentShader &shader = current_material_shaders[shader_index]; @@ -284,71 +323,82 @@ void ShaderObject::bind_material_shader(VkCommandBuffer cmd_buffer, int shader_i bind_shader(cmd_buffer, material_geo_shaders[shader.geo]); bind_shader(cmd_buffer, material_frag_shaders[shader.frag]); } -``` +---- It would be equally valid for the `Shader` class to be designed to bind all of the `VkShaderEXT` objects in a single `vkCmdBindShadersEXT` call. -## Unbinding Shaders +== Unbinding Shaders + Bound shaders can be unbound by calling `vkCmdBindShadersEXT` with `pShaders` set to `nullptr` and `pStages` set to an array of stages to unbind. This sample uses a geometry shader for only some draws, so we need to unbind the geometry shader before draws that don't need them. -```CPP +[,CPP] +---- // Unbind geometry shader by binding nullptr to the geometry stage VkShaderStageFlagBits geo_stage = VK_SHADER_STAGE_GEOMETRY_BIT; vkCmdBindShadersEXT(draw_cmd_buffer, 1, &geo_stage, nullptr); -``` +---- -Before drawing with shader objects, applications are required to bind either `nullptr` or a valid `VkShaderEXT` object for every shader stage enabled on the device. A valid geometry stage shader will not be bound before the first draw so we need to bind `nullptr` to it. The same code as above can be used. +Before drawing with shader objects, applications are required to bind either `nullptr` or a valid `VkShaderEXT` object for every shader stage enabled on the device. +A valid geometry stage shader will not be bound before the first draw so we need to bind `nullptr` to it. +The same code as above can be used. This sample does not enable the `tessellationShader` device feature, so it's not necessary to bind anything to either tessellation stage. -## State Setting and Drawing +== State Setting and Drawing -The [Setting State](https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects-state) subsection of the [Shader Objects](https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects) section of the Vulkan specification lists the graphics state that needs to be set on a command buffer before `vkCmdDraw*` can be called with graphics shader objects bound. +The https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects-state[Setting State] subsection of the https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects[Shader Objects] section of the Vulkan specification lists the graphics state that needs to be set on a command buffer before `vkCmdDraw*` can be called with graphics shader objects bound. This sample demonstrates one possible way to take advantage of these rules to minimize calls into the Vulkan driver. -All of the required and common state for this sample, such as vertex input binding descriptions, is set in `set_initial_state` before any draw calls. State specific to particular draws is set on the command buffer as needed. +All of the required and common state for this sample, such as vertex input binding descriptions, is set in `set_initial_state` before any draw calls. +State specific to particular draws is set on the command buffer as needed. For the skybox, we disable culling and writing to depth. -```CPP +[,CPP] +---- // Disable depth write and use cull mode none to draw skybox vkCmdSetCullModeEXT(draw_cmd_buffer, VK_CULL_MODE_NONE); vkCmdSetDepthWriteEnableEXT(draw_cmd_buffer, VK_FALSE); -``` +---- Then we bind the descriptor sets and push constants for the draw. -```CPP +[,CPP] +---- // Bind descriptors and push constants for the skybox draw glm::mat4 model_matrix = glm::mat4(1.0f); vkCmdBindDescriptorSets(draw_cmd_buffer, VK_PIPELINE_BIND_POINT_GRAPHICS, pipeline_layout[ShaderTypeBasic], 0, 1, &descriptor_sets[ShaderTypeBasic], 0, nullptr); vkCmdPushConstants(draw_cmd_buffer, pipeline_layout[ShaderTypeBasic], VK_SHADER_STAGE_VERTEX_BIT, 0, sizeof(BasicPushConstant), &model_matrix); -``` +---- Finally, we bind the shaders and draw the model. -```CPP +[,CPP] +---- // Bind shaders for the skybox bind_shader(draw_cmd_buffer, skybox_vert_shader); bind_shader(draw_cmd_buffer, skybox_frag_shader); // Draw the skybox model draw_model(skybox, draw_cmd_buffer); -``` +---- The models that will be drawn next require different state than the skybox. -```CPP +[,CPP] +---- vkCmdSetCullModeEXT(draw_cmd_buffer, VK_CULL_MODE_BACK_BIT); vkCmdSetDepthWriteEnableEXT(draw_cmd_buffer, VK_TRUE); -``` +---- -Unlinked shaders are bound the same way as linked shaders. First we set state, bind descriptor sets, push constants, and finally bind shaders and draw the model. +Unlinked shaders are bound the same way as linked shaders. +First we set state, bind descriptor sets, push constants, and finally bind shaders and draw the model. -```CPP +[,CPP] +---- // Update and push constants for cube 1 material_push_constant.model = glm::translate(glm::vec3(1.2f, -1.f, 0)) * glm::rotate((float) elapsed_time, glm::vec3(0, 1, 0)) * glm::scale(glm::vec3(0.05f)); vkCmdPushConstants(draw_cmd_buffer, pipeline_layout[ShaderTypeMaterial], @@ -360,43 +410,54 @@ bind_material_shader(draw_cmd_buffer, 2); // Draw cube 1 draw_model(cube, draw_cmd_buffer); -``` +---- -After rendering the skybox, terrain, and all other models, an optional post processing effect is applied to the intermediate image. The final result is drawn to the screen. +After rendering the skybox, terrain, and all other models, an optional post processing effect is applied to the intermediate image. +The final result is drawn to the screen. -## Options +== Options -The debug UI allows various aspects of rendering to be controlled dynamically. It can change shaders per model, both color and depth output formats, and the post processing effect. This level of dynamism would be impractical to achieve using pipelines because the tens of thousands of permutations of state would potentially each require their own pipeline. Shader objects, on the other hand, support this kind of application architecture naturally. +The debug UI allows various aspects of rendering to be controlled dynamically. +It can change shaders per model, both color and depth output formats, and the post processing effect. +This level of dynamism would be impractical to achieve using pipelines because the tens of thousands of permutations of state would potentially each require their own pipeline. +Shader objects, on the other hand, support this kind of application architecture naturally. -## Emulation Layer +== Emulation Layer -The Vulkan SDK ships with an [emulation layer](https://github.com/KhronosGroup/Vulkan-ExtensionLayer/blob/main/docs/shader_object_layer.md) that allows `VK_EXT_shader_object` to be used on drivers that don't yet have native support for the extension. The layer is useful for applications that want to use shader objects without implementing their own pipeline based fallback path for older drivers without native support. The layer can be shipped with your application, and it will disable itself if a native implementation of `VK_EXT_shader_object` exists in the driver. +The Vulkan SDK ships with an https://github.com/KhronosGroup/Vulkan-ExtensionLayer/blob/main/docs/shader_object_layer.md[emulation layer] that allows `VK_EXT_shader_object` to be used on drivers that don't yet have native support for the extension. +The layer is useful for applications that want to use shader objects without implementing their own pipeline based fallback path for older drivers without native support. +The layer can be shipped with your application, and it will disable itself if a native implementation of `VK_EXT_shader_object` exists in the driver. The emulation layer can be enabled by adding `VK_LAYER_KHRONOS_shader_object` to `ppEnabledLayerNames` in `VkDeviceCreateInfo`. -The sample framework already has an existing abstraction normally used for enabling the validation layer. This sample repurposes this mechanism to instead load the emulation layer. +The sample framework already has an existing abstraction normally used for enabling the validation layer. +This sample repurposes this mechanism to instead load the emulation layer. -```CPP +[,CPP] +---- const std::vector ShaderObject::get_validation_layers() { return {"VK_LAYER_KHRONOS_shader_object"}; } -``` +---- -Because you can't rely on the Vulkan SDK to be installed on a user's system, this sample's method of loading the layer is not suitable for a real application. Instead, the layer needs to be shipped with the application. +Because you can't rely on the Vulkan SDK to be installed on a user's system, this sample's method of loading the layer is not suitable for a real application. +Instead, the layer needs to be shipped with the application. -There are various ways to ship the layer with an application. One method is to copy the layer's `VkLayer_khronos_shader_object.dll` and `VkLayer_khronos_shader_object.json` from the Vulkan SDK to the same directory as your application's executable and then append the path to these files to the `VK_LAYER_PATH` environment variable before your first call into the Vulkan API. +There are various ways to ship the layer with an application. +One method is to copy the layer's `VkLayer_khronos_shader_object.dll` and `VkLayer_khronos_shader_object.json` from the Vulkan SDK to the same directory as your application's executable and then append the path to these files to the `VK_LAYER_PATH` environment variable before your first call into the Vulkan API. -However the layer is packaged, the Vulkan Loader is always responsible for loading the layer. Because of this the layer's files will always need to be somewhere accessible to the loader. +However the layer is packaged, the Vulkan Loader is always responsible for loading the layer. +Because of this the layer's files will always need to be somewhere accessible to the loader. -## Additional Resources +== Additional Resources -* [You Can Use Vulkan Without Pipelines Today](https://www.khronos.org/blog/you-can-use-vulkan-without-pipelines-today) -* [Extension Proposal](https://github.com/KhronosGroup/Vulkan-Docs/blob/main/proposals/VK_EXT_shader_object.adoc) -* [Specification](https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects) -* [Emulation Layer](https://github.com/KhronosGroup/Vulkan-ExtensionLayer/blob/main/docs/shader_object_layer.md) +* https://www.khronos.org/blog/you-can-use-vulkan-without-pipelines-today[You Can Use Vulkan Without Pipelines Today] +* https://github.com/KhronosGroup/Vulkan-Docs/blob/main/proposals/VK_EXT_shader_object.adoc[Extension Proposal] +* https://registry.khronos.org/vulkan/specs/1.3-extensions/html/vkspec.html#shaders-objects[Specification] +* https://github.com/KhronosGroup/Vulkan-ExtensionLayer/blob/main/docs/shader_object_layer.md[Emulation Layer] -## Conclusion +== Conclusion Shader objects can be an invaluable tool for simplifying shader and state management in highly dynamic application architectures which don't lend themselves to practical implementation using pipelines. From e80d3031a65d0eb6ee3a92ca066b671d65124b62 Mon Sep 17 00:00:00 2001 From: Tim Lobner Date: Mon, 21 Aug 2023 14:02:16 +0200 Subject: [PATCH 05/13] added shaders to ray tracing extended example IDE folder --- samples/extensions/ray_tracing_extended/CMakeLists.txt | 6 +++++- 1 file changed, 5 insertions(+), 1 deletion(-) diff --git a/samples/extensions/ray_tracing_extended/CMakeLists.txt b/samples/extensions/ray_tracing_extended/CMakeLists.txt index 03449c945..9fe065384 100644 --- a/samples/extensions/ray_tracing_extended/CMakeLists.txt +++ b/samples/extensions/ray_tracing_extended/CMakeLists.txt @@ -24,4 +24,8 @@ add_sample_with_tags( CATEGORY ${CATEGORY_NAME} AUTHOR "Holochip Corporation" NAME "Ray tracing extended" - DESCRIPTION "Extended example of Ray Tracing highlighting the Bottom and Top Level Acceleration Structure rebuild and AO.") + DESCRIPTION "Extended example of Ray Tracing highlighting the Bottom and Top Level Acceleration Structure rebuild and AO." + SHADER_FILES_GLSL + "khr_ray_tracing_extended/raygen.rgen" + "khr_ray_tracing_extended/miss.rmiss" + "khr_ray_tracing_extended/closesthit.rchit") From 8c676d06c893de2df2eb1d3127ae890a95c61403 Mon Sep 17 00:00:00 2001 From: Dan Field Date: Wed, 23 Aug 2023 09:10:15 -0700 Subject: [PATCH 06/13] Add PendingIntent.FLAG_IMMUTABLE for Android S+ --- .../java/com/khronos/vulkan_samples/NativeSampleActivity.java | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/app/android/java/com/khronos/vulkan_samples/NativeSampleActivity.java b/app/android/java/com/khronos/vulkan_samples/NativeSampleActivity.java index 6b3280538..ad1dcb559 100644 --- a/app/android/java/com/khronos/vulkan_samples/NativeSampleActivity.java +++ b/app/android/java/com/khronos/vulkan_samples/NativeSampleActivity.java @@ -78,7 +78,7 @@ public void fatalError(final String log_file) { intent.setDataAndType(path, context.getContentResolver().getType(path)); intent.addFlags(Intent.FLAG_GRANT_READ_URI_PERMISSION); - PendingIntent pi = PendingIntent.getActivity(context, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT); + PendingIntent pi = PendingIntent.getActivity(context, 0, intent, PendingIntent.FLAG_CANCEL_CURRENT | PendingIntent.FLAG_IMMUTABLE); NotificationCompat.Builder builder = new NotificationCompat.Builder(this, Notifications.CHANNEL_ID) .setSmallIcon(R.drawable.icon) From ee05d72c89655662e0c616aab7a2cc73b7e87b0b Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andreas=20S=C3=BC=C3=9Fenbach?= Date: Mon, 28 Aug 2023 17:49:34 +0200 Subject: [PATCH 07/13] Refactored API sample hpp_terrain_tessellation. (#779) --- framework/common/hpp_vk_common.h | 10 +- framework/hpp_api_vulkan_sample.cpp | 16 +- framework/hpp_api_vulkan_sample.h | 9 +- .../hpp_compute_nbody/hpp_compute_nbody.cpp | 6 +- .../hpp_dynamic_uniform_buffers.cpp | 12 +- samples/api/hpp_hdr/hpp_hdr.cpp | 20 +- .../hpp_hello_triangle/hpp_hello_triangle.cpp | 4 +- .../api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp | 8 +- samples/api/hpp_instancing/hpp_instancing.cpp | 18 +- .../hpp_separate_image_sampler.cpp | 7 +- .../hpp_terrain_tessellation.cpp | 640 ++++++++---------- .../hpp_terrain_tessellation.h | 171 +++-- 12 files changed, 461 insertions(+), 460 deletions(-) diff --git a/framework/common/hpp_vk_common.h b/framework/common/hpp_vk_common.h index 53c79a875..070a95da5 100644 --- a/framework/common/hpp_vk_common.h +++ b/framework/common/hpp_vk_common.h @@ -176,9 +176,11 @@ inline vk::Framebuffer create_framebuffer(vk::Device device, vk::RenderPass rend inline vk::Pipeline create_graphics_pipeline(vk::Device device, vk::PipelineCache pipeline_cache, - std::array const &shader_stages, + std::vector const &shader_stages, vk::PipelineVertexInputStateCreateInfo const &vertex_input_state, vk::PrimitiveTopology primitive_topology, + uint32_t patch_control_points, + vk::PolygonMode polygon_mode, vk::CullModeFlags cull_mode, vk::FrontFace front_face, std::vector const &blend_attachment_states, @@ -188,10 +190,12 @@ inline vk::Pipeline create_graphics_pipeline(vk::Device { vk::PipelineInputAssemblyStateCreateInfo input_assembly_state({}, primitive_topology, false); + vk::PipelineTessellationStateCreateInfo tessellation_state({}, patch_control_points); + vk::PipelineViewportStateCreateInfo viewport_state({}, 1, nullptr, 1, nullptr); vk::PipelineRasterizationStateCreateInfo rasterization_state; - rasterization_state.polygonMode = vk::PolygonMode::eFill; + rasterization_state.polygonMode = polygon_mode; rasterization_state.cullMode = cull_mode; rasterization_state.frontFace = front_face; rasterization_state.lineWidth = 1.0f; @@ -208,7 +212,7 @@ inline vk::Pipeline create_graphics_pipeline(vk::Device shader_stages, &vertex_input_state, &input_assembly_state, - {}, + &tessellation_state, &viewport_state, &rasterization_state, &multisample_state, diff --git a/framework/hpp_api_vulkan_sample.cpp b/framework/hpp_api_vulkan_sample.cpp index fcf688586..0fe41ccbf 100644 --- a/framework/hpp_api_vulkan_sample.cpp +++ b/framework/hpp_api_vulkan_sample.cpp @@ -846,7 +846,7 @@ vk::ImageLayout HPPApiVulkanSample::descriptor_type_to_image_layout(vk::Descript } } -HPPTexture HPPApiVulkanSample::load_texture(const std::string &file, vkb::sg::Image::ContentType content_type) +HPPTexture HPPApiVulkanSample::load_texture(const std::string &file, vkb::sg::Image::ContentType content_type, vk::SamplerAddressMode address_mode) { HPPTexture texture; @@ -906,9 +906,9 @@ HPPTexture HPPApiVulkanSample::load_texture(const std::string &file, vkb::sg::Im sampler_create_info.magFilter = vk::Filter::eLinear; sampler_create_info.minFilter = vk::Filter::eLinear; sampler_create_info.mipmapMode = vk::SamplerMipmapMode::eLinear; - sampler_create_info.addressModeU = vk::SamplerAddressMode::eRepeat; - sampler_create_info.addressModeV = vk::SamplerAddressMode::eRepeat; - sampler_create_info.addressModeW = vk::SamplerAddressMode::eRepeat; + sampler_create_info.addressModeU = address_mode; + sampler_create_info.addressModeV = address_mode; + sampler_create_info.addressModeW = address_mode; sampler_create_info.mipLodBias = 0.0f; sampler_create_info.compareOp = vk::CompareOp::eNever; sampler_create_info.minLod = 0.0f; @@ -927,7 +927,7 @@ HPPTexture HPPApiVulkanSample::load_texture(const std::string &file, vkb::sg::Im return texture; } -HPPTexture HPPApiVulkanSample::load_texture_array(const std::string &file, vkb::sg::Image::ContentType content_type) +HPPTexture HPPApiVulkanSample::load_texture_array(const std::string &file, vkb::sg::Image::ContentType content_type, vk::SamplerAddressMode address_mode) { HPPTexture texture{}; @@ -993,9 +993,9 @@ HPPTexture HPPApiVulkanSample::load_texture_array(const std::string &file, vkb:: sampler_create_info.magFilter = vk::Filter::eLinear; sampler_create_info.minFilter = vk::Filter::eLinear; sampler_create_info.mipmapMode = vk::SamplerMipmapMode::eLinear; - sampler_create_info.addressModeU = vk::SamplerAddressMode::eClampToEdge; - sampler_create_info.addressModeV = vk::SamplerAddressMode::eClampToEdge; - sampler_create_info.addressModeW = vk::SamplerAddressMode::eClampToEdge; + sampler_create_info.addressModeU = address_mode; + sampler_create_info.addressModeV = address_mode; + sampler_create_info.addressModeW = address_mode; sampler_create_info.mipLodBias = 0.0f; sampler_create_info.compareOp = vk::CompareOp::eNever; sampler_create_info.minLod = 0.0f; diff --git a/framework/hpp_api_vulkan_sample.h b/framework/hpp_api_vulkan_sample.h index 728fe6cda..a4af8bc7f 100644 --- a/framework/hpp_api_vulkan_sample.h +++ b/framework/hpp_api_vulkan_sample.h @@ -165,15 +165,20 @@ class HPPApiVulkanSample : public vkb::HPPVulkanSample * @brief Loads in a ktx 2D texture * @param file The filename of the texture to load * @param content_type The type of content in the image file + * @param address_mode The address mode to use in u-, v-, and w-direction. Defaults to /c vk::SamplerAddressMode::eRepeat. */ - HPPTexture load_texture(const std::string &file, vkb::sg::Image::ContentType content_type); + HPPTexture + load_texture(const std::string &file, vkb::sg::Image::ContentType content_type, vk::SamplerAddressMode address_mode = vk::SamplerAddressMode::eRepeat); /** * @brief Loads in a ktx 2D texture array * @param file The filename of the texture to load * @param content_type The type of content in the image file + * @param address_mode The address mode to use in u-, v-, and w-direction. Defaults to /c vk::SamplerAddressMode::eClampToEdge. */ - HPPTexture load_texture_array(const std::string &file, vkb::sg::Image::ContentType content_type); + HPPTexture load_texture_array(const std::string &file, + vkb::sg::Image::ContentType content_type, + vk::SamplerAddressMode address_mode = vk::SamplerAddressMode::eClampToEdge); /** * @brief Loads in a ktx 2D texture cubemap diff --git a/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp b/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp index 40c55c030..f758c51d9 100644 --- a/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp +++ b/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp @@ -305,8 +305,8 @@ vk::DescriptorSetLayout HPPComputeNBody::create_graphics_descriptor_set_layout() vk::Pipeline HPPComputeNBody::create_graphics_pipeline() { // Load shaders - std::array shader_stages = {{load_shader("compute_nbody/particle.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("compute_nbody/particle.frag", vk::ShaderStageFlagBits::eFragment)}}; + std::vector shader_stages = {load_shader("compute_nbody/particle.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("compute_nbody/particle.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex bindings and attributes vk::VertexInputBindingDescription vertex_input_bindings(0, sizeof(Particle), vk::VertexInputRate::eVertex); @@ -338,6 +338,8 @@ vk::Pipeline HPPComputeNBody::create_graphics_pipeline() shader_stages, vertex_input_state, vk::PrimitiveTopology::ePointList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eNone, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, diff --git a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp index bc80d6865..4c0d8b9eb 100644 --- a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp +++ b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp @@ -95,8 +95,9 @@ bool HPPDynamicUniformBuffers::prepare(const vkb::ApplicationOptions &options) descriptor_set_layout = create_descriptor_set_layout(); pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, descriptor_set_layout}); pipeline = create_pipeline(); - descriptor_pool = create_descriptor_pool(); - descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, descriptor_set_layout); + + descriptor_pool = create_descriptor_pool(); + descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, descriptor_set_layout); update_descriptor_set(); build_command_buffers(); @@ -186,9 +187,8 @@ vk::DescriptorSetLayout HPPDynamicUniformBuffers::create_descriptor_set_layout() vk::Pipeline HPPDynamicUniformBuffers::create_pipeline() { // Load shaders - std::array shader_stages = {{load_shader("dynamic_uniform_buffers/base.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("dynamic_uniform_buffers/base.frag", - vk::ShaderStageFlagBits::eFragment)}}; + std::vector shader_stages = {load_shader("dynamic_uniform_buffers/base.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("dynamic_uniform_buffers/base.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex bindings and attributes vk::VertexInputBindingDescription vertex_input_binding(0, sizeof(Vertex), vk::VertexInputRate::eVertex); @@ -213,6 +213,8 @@ vk::Pipeline HPPDynamicUniformBuffers::create_pipeline() shader_stages, vertex_input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eNone, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, diff --git a/samples/api/hpp_hdr/hpp_hdr.cpp b/samples/api/hpp_hdr/hpp_hdr.cpp index a6570127d..cbaed173e 100644 --- a/samples/api/hpp_hdr/hpp_hdr.cpp +++ b/samples/api/hpp_hdr/hpp_hdr.cpp @@ -249,9 +249,8 @@ vk::DescriptorPool HPPHDR::create_descriptor_pool() vk::Pipeline HPPHDR::create_bloom_pipeline(uint32_t direction) { - std::array shader_stages{ - load_shader("hdr/bloom.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("hdr/bloom.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages{load_shader("hdr/bloom.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("hdr/bloom.frag", vk::ShaderStageFlagBits::eFragment)}; // Set constant parameters via specialization constants vk::SpecializationMapEntry specialization_map_entry(0, 0, sizeof(uint32_t)); @@ -282,6 +281,8 @@ vk::Pipeline HPPHDR::create_bloom_pipeline(uint32_t direction) shader_stages, {}, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eFront, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, @@ -292,8 +293,8 @@ vk::Pipeline HPPHDR::create_bloom_pipeline(uint32_t direction) vk::Pipeline HPPHDR::create_composition_pipeline() { - std::array shader_stages{load_shader("hdr/composition.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("hdr/composition.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages{load_shader("hdr/composition.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("hdr/composition.frag", vk::ShaderStageFlagBits::eFragment)}; vk::PipelineColorBlendAttachmentState blend_attachment_state; blend_attachment_state.colorWriteMask = @@ -311,6 +312,8 @@ vk::Pipeline HPPHDR::create_composition_pipeline() shader_stages, {}, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eFront, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, @@ -364,9 +367,8 @@ vk::ImageView HPPHDR::create_image_view(vk::Format format, vk::ImageUsageFlagBit vk::Pipeline HPPHDR::create_models_pipeline(uint32_t shaderType, vk::CullModeFlagBits cullMode, bool depthTestAndWrite) { - std::array shader_stages{ - load_shader("hdr/gbuffer.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("hdr/gbuffer.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages{load_shader("hdr/gbuffer.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("hdr/gbuffer.frag", vk::ShaderStageFlagBits::eFragment)}; // Set constant parameters via specialization constants vk::SpecializationMapEntry specialization_map_entry(0, 0, sizeof(uint32_t)); @@ -404,6 +406,8 @@ vk::Pipeline HPPHDR::create_models_pipeline(uint32_t shaderType, vk::CullModeFla shader_stages, vertex_input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, cullMode, vk::FrontFace::eCounterClockwise, blend_attachment_states, diff --git a/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp b/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp index 44df00ccf..541e53171 100644 --- a/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp +++ b/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp @@ -376,7 +376,7 @@ vk::Device HPPHelloTriangle::create_device(const std::vector &requ vk::Pipeline HPPHelloTriangle::create_graphics_pipeline() { // Load our SPIR-V shaders. - std::array shader_stages{ + std::vector shader_stages{ vk::PipelineShaderStageCreateInfo({}, vk::ShaderStageFlagBits::eVertex, create_shader_module("triangle.vert"), "main"), vk::PipelineShaderStageCreateInfo({}, vk::ShaderStageFlagBits::eFragment, create_shader_module("triangle.frag"), "main")}; @@ -394,6 +394,8 @@ vk::Pipeline HPPHelloTriangle::create_graphics_pipeline() shader_stages, vertex_input, vk::PrimitiveTopology::eTriangleList, // We will use triangle lists to draw geometry. + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eBack, vk::FrontFace::eClockwise, {blend_attachment}, diff --git a/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp b/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp index 14b638116..f193dfc5d 100644 --- a/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp +++ b/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp @@ -169,9 +169,9 @@ vk::DescriptorPool HPPHlslShaders::create_descriptor_pool() vk::Pipeline HPPHlslShaders::create_pipeline() { - size_t vertex_shader_index = shader_modules.size() - 2; - std::array shader_stages{{{{}, vk::ShaderStageFlagBits::eVertex, shader_modules[vertex_shader_index], "main"}, - {{}, vk::ShaderStageFlagBits::eFragment, shader_modules[vertex_shader_index + 1], "main"}}}; + size_t vertex_shader_index = shader_modules.size() - 2; + std::vector shader_stages{{{}, vk::ShaderStageFlagBits::eVertex, shader_modules[vertex_shader_index], "main"}, + {{}, vk::ShaderStageFlagBits::eFragment, shader_modules[vertex_shader_index + 1], "main"}}; // Vertex bindings and attributes vk::VertexInputBindingDescription vertex_input_binding(0, sizeof(VertexStructure), vk::VertexInputRate::eVertex); @@ -194,6 +194,8 @@ vk::Pipeline HPPHlslShaders::create_pipeline() shader_stages, vertex_input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eNone, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, diff --git a/samples/api/hpp_instancing/hpp_instancing.cpp b/samples/api/hpp_instancing/hpp_instancing.cpp index 8ea7514ca..7153bc17e 100644 --- a/samples/api/hpp_instancing/hpp_instancing.cpp +++ b/samples/api/hpp_instancing/hpp_instancing.cpp @@ -213,8 +213,8 @@ vk::DescriptorSetLayout HPPInstancing::create_descriptor_set_layout() vk::Pipeline HPPInstancing::create_planet_pipeline() { // Planet rendering pipeline - std::array shader_stages = {load_shader("instancing/planet.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("instancing/planet.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages = {load_shader("instancing/planet.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("instancing/planet.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex input bindings vk::VertexInputBindingDescription binding_description(0, sizeof(HPPVertex), vk::VertexInputRate::eVertex); @@ -247,6 +247,8 @@ vk::Pipeline HPPInstancing::create_planet_pipeline() shader_stages, input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eBack, vk::FrontFace::eClockwise, {blend_attachment_state}, @@ -257,8 +259,8 @@ vk::Pipeline HPPInstancing::create_planet_pipeline() vk::Pipeline HPPInstancing::create_rocks_pipeline() { - std::array shader_stages{load_shader("instancing/instancing.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("instancing/instancing.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages{load_shader("instancing/instancing.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("instancing/instancing.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex input bindings // The instancing pipeline uses a vertex input state with two bindings @@ -305,6 +307,8 @@ vk::Pipeline HPPInstancing::create_rocks_pipeline() shader_stages, input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eBack, vk::FrontFace::eClockwise, {blend_attachment_state}, @@ -316,8 +320,8 @@ vk::Pipeline HPPInstancing::create_rocks_pipeline() vk::Pipeline HPPInstancing::create_starfield_pipeline() { // Starfield rendering pipeline - std::array shader_stages = {load_shader("instancing/starfield.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("instancing/starfield.frag", vk::ShaderStageFlagBits::eFragment)}; + std::vector shader_stages = {load_shader("instancing/starfield.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("instancing/starfield.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex input bindings vk::VertexInputBindingDescription binding_description(0, sizeof(HPPVertex), vk::VertexInputRate::eVertex); @@ -350,6 +354,8 @@ vk::Pipeline HPPInstancing::create_starfield_pipeline() shader_stages, {}, // Vertices are generated in the vertex shader vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eNone, vk::FrontFace::eClockwise, {blend_attachment_state}, diff --git a/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp b/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp index 62b7a6420..cee233cf8 100644 --- a/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp +++ b/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp @@ -201,8 +201,9 @@ vk::DescriptorPool HPPSeparateImageSampler::create_descriptor_pool() vk::Pipeline HPPSeparateImageSampler::create_graphics_pipeline() { // Load shaders - std::array shader_stages = {{load_shader("separate_image_sampler/separate_image_sampler.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("separate_image_sampler/separate_image_sampler.frag", vk::ShaderStageFlagBits::eFragment)}}; + std::vector shader_stages = { + load_shader("separate_image_sampler/separate_image_sampler.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("separate_image_sampler/separate_image_sampler.frag", vk::ShaderStageFlagBits::eFragment)}; // Vertex bindings and attributes vk::VertexInputBindingDescription input_binding(0, sizeof(VertexStructure), vk::VertexInputRate::eVertex); @@ -228,6 +229,8 @@ vk::Pipeline HPPSeparateImageSampler::create_graphics_pipeline() shader_stages, input_state, vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, vk::CullModeFlagBits::eNone, vk::FrontFace::eCounterClockwise, {blend_attachment_state}, diff --git a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp index 5483cb204..69c9a76a6 100644 --- a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp +++ b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp @@ -21,6 +21,7 @@ #include "hpp_terrain_tessellation.h" +#include #include HPPTerrainTessellation::HPPTerrainTessellation() @@ -36,29 +37,10 @@ HPPTerrainTessellation::~HPPTerrainTessellation() // Clean up used Vulkan resources // Note : Inherited destructor cleans up resources stored in base class - device.destroyPipeline(pipelines.skysphere); - device.destroyPipeline(pipelines.terrain); - if (pipelines.wireframe) - { - device.destroyPipeline(pipelines.wireframe); - } - - device.destroyPipelineLayout(pipeline_layouts.skysphere); - device.destroyPipelineLayout(pipeline_layouts.terrain); - - device.destroyDescriptorSetLayout(descriptor_set_layouts.skysphere); - device.destroyDescriptorSetLayout(descriptor_set_layouts.terrain); - - device.destroySampler(textures.heightmap.sampler); - device.destroySampler(textures.skysphere.sampler); - device.destroySampler(textures.terrain_array.sampler); - - if (query_pool) - { - device.destroyQueryPool(query_pool); - device.destroyBuffer(query_result.buffer); - device.freeMemory(query_result.memory); - } + sky_sphere.destroy(device); + terrain.destroy(device); + wireframe.destroy(device); + statistics.destroy(device); } } @@ -69,24 +51,19 @@ bool HPPTerrainTessellation::prepare(const vkb::ApplicationOptions &options) return false; } - // Note: Using reversed depth-buffer for increased precision, so Znear and Zfar are flipped - camera.type = vkb::CameraType::FirstPerson; - camera.set_perspective(60.0f, static_cast(extent.width) / static_cast(extent.height), 512.0f, 0.1f); - camera.set_rotation(glm::vec3(-12.0f, 159.0f, 0.0f)); - camera.set_translation(glm::vec3(18.0f, 22.5f, 57.5f)); - camera.translation_speed = 7.5f; - + prepare_camera(); load_assets(); generate_terrain(); - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) - { - setup_query_result_buffer(); - } prepare_uniform_buffers(); - setup_descriptor_set_layouts(); - prepare_pipelines(); - setup_descriptor_pool(); - setup_descriptor_sets(); + + descriptor_pool = create_descriptor_pool(); + + prepare_sky_sphere(); + prepare_terrain(); + prepare_wireframe(); + + prepare_statistics(); + build_command_buffers(); prepared = true; return true; @@ -95,7 +72,8 @@ bool HPPTerrainTessellation::prepare(const vkb::ApplicationOptions &options) void HPPTerrainTessellation::request_gpu_features(vkb::core::HPPPhysicalDevice &gpu) { // Tessellation shader support is required for this example - if (!gpu.get_features().tessellationShader) + auto &available_features = gpu.get_features(); + if (!available_features.tessellationShader) { throw vkb::VulkanException(VK_ERROR_FEATURE_NOT_PRESENT, "Selected GPU does not support tessellation shaders!"); } @@ -105,13 +83,16 @@ void HPPTerrainTessellation::request_gpu_features(vkb::core::HPPPhysicalDevice & requested_features.tessellationShader = true; // Fill mode non solid is required for wireframe display - requested_features.fillModeNonSolid = gpu.get_features().fillModeNonSolid; + requested_features.fillModeNonSolid = available_features.fillModeNonSolid; + wireframe.supported = available_features.fillModeNonSolid; // Pipeline statistics - requested_features.pipelineStatisticsQuery = gpu.get_features().pipelineStatisticsQuery; + requested_features.pipelineStatisticsQuery = available_features.pipelineStatisticsQuery; + statistics.query_supported = available_features.pipelineStatisticsQuery; // Enable anisotropic filtering if supported - requested_features.samplerAnisotropy = gpu.get_features().samplerAnisotropy; + requested_features.samplerAnisotropy = available_features.samplerAnisotropy; + terrain.sampler_anisotropy_supported = available_features.samplerAnisotropy; } void HPPTerrainTessellation::build_command_buffers() @@ -123,55 +104,55 @@ void HPPTerrainTessellation::build_command_buffers() for (int32_t i = 0; i < draw_cmd_buffers.size(); ++i) { - draw_cmd_buffers[i].begin(command_buffer_begin_info); + auto command_buffer = draw_cmd_buffers[i]; + command_buffer.begin(command_buffer_begin_info); - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) + if (statistics.query_supported) { - draw_cmd_buffers[i].resetQueryPool(query_pool, 0, 2); + command_buffer.resetQueryPool(statistics.query_pool, 0, 2); } render_pass_begin_info.framebuffer = framebuffers[i]; - draw_cmd_buffers[i].beginRenderPass(render_pass_begin_info, vk::SubpassContents::eInline); + command_buffer.beginRenderPass(render_pass_begin_info, vk::SubpassContents::eInline); vk::Viewport viewport(0.0f, 0.0f, static_cast(extent.width), static_cast(extent.height), 0.0f, 1.0f); - draw_cmd_buffers[i].setViewport(0, viewport); + command_buffer.setViewport(0, viewport); vk::Rect2D scissor({0, 0}, extent); - draw_cmd_buffers[i].setScissor(0, scissor); - - draw_cmd_buffers[i].setLineWidth(1.0f); + command_buffer.setScissor(0, scissor); vk::DeviceSize offset = 0; // Skysphere - draw_cmd_buffers[i].bindPipeline(vk::PipelineBindPoint::eGraphics, pipelines.skysphere); - draw_cmd_buffers[i].bindDescriptorSets(vk::PipelineBindPoint::eGraphics, pipeline_layouts.skysphere, 0, descriptor_sets.skysphere, {}); - draw_model(skysphere, draw_cmd_buffers[i]); + command_buffer.bindPipeline(vk::PipelineBindPoint::eGraphics, sky_sphere.pipeline); + command_buffer.bindDescriptorSets(vk::PipelineBindPoint::eGraphics, sky_sphere.pipeline_layout, 0, sky_sphere.descriptor_set, {}); + draw_model(sky_sphere.geometry, command_buffer); // Terrain - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) + if (statistics.query_supported) { // Begin pipeline statistics query - draw_cmd_buffers[i].beginQuery(query_pool, 0, {}); + command_buffer.beginQuery(statistics.query_pool, 0, {}); } // Render - draw_cmd_buffers[i].bindPipeline(vk::PipelineBindPoint::eGraphics, wireframe ? pipelines.wireframe : pipelines.terrain); - draw_cmd_buffers[i].bindDescriptorSets(vk::PipelineBindPoint::eGraphics, pipeline_layouts.terrain, 0, descriptor_sets.terrain, {}); - draw_cmd_buffers[i].bindVertexBuffers(0, terrain.vertices->get_handle(), offset); - draw_cmd_buffers[i].bindIndexBuffer(terrain.indices->get_handle(), 0, vk::IndexType::eUint32); - draw_cmd_buffers[i].drawIndexed(terrain.index_count, 1, 0, 0, 0); - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) + command_buffer.bindPipeline(vk::PipelineBindPoint::eGraphics, wireframe.enabled ? wireframe.pipeline : terrain.pipeline); + command_buffer.bindDescriptorSets(vk::PipelineBindPoint::eGraphics, terrain.pipeline_layout, 0, terrain.descriptor_set, {}); + command_buffer.bindVertexBuffers(0, terrain.vertices->get_handle(), offset); + command_buffer.bindIndexBuffer(terrain.indices->get_handle(), 0, vk::IndexType::eUint32); + command_buffer.drawIndexed(terrain.index_count, 1, 0, 0, 0); + + if (statistics.query_supported) { // End pipeline statistics query - draw_cmd_buffers[i].endQuery(query_pool, 0); + command_buffer.endQuery(statistics.query_pool, 0); } - draw_ui(draw_cmd_buffers[i]); + draw_ui(command_buffer); - draw_cmd_buffers[i].endRenderPass(); + command_buffer.endRenderPass(); - draw_cmd_buffers[i].end(); + command_buffer.end(); } } @@ -179,28 +160,28 @@ void HPPTerrainTessellation::on_update_ui_overlay(vkb::HPPDrawer &drawer) { if (drawer.header("Settings")) { - if (drawer.checkbox("Tessellation", &tessellation)) + if (drawer.checkbox("Tessellation", &terrain.tessellation_enabled)) { update_uniform_buffers(); } - if (drawer.input_float("Factor", &ubo_tess.tessellation_factor, 0.05f, 2)) + if (drawer.input_float("Factor", &terrain.tessellation.tessellation_factor, 0.05f, 2)) { update_uniform_buffers(); } - if (get_device()->get_gpu().get_features().fillModeNonSolid) + if (wireframe.supported) { - if (drawer.checkbox("Wireframe", &wireframe)) + if (drawer.checkbox("Wireframe", &wireframe.enabled)) { build_command_buffers(); } } } - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) + if (statistics.query_supported) { if (drawer.header("Pipeline statistics")) { - drawer.text("VS invocations: %d", pipeline_stats[0]); - drawer.text("TE invocations: %d", pipeline_stats[1]); + drawer.text("VS invocations: %d", statistics.results[0]); + drawer.text("TE invocations: %d", statistics.results[1]); } } } @@ -219,6 +200,123 @@ void HPPTerrainTessellation::view_changed() update_uniform_buffers(); } +vk::DescriptorPool HPPTerrainTessellation::create_descriptor_pool() +{ + std::array pool_sizes = {{{vk::DescriptorType::eUniformBuffer, 3}, {vk::DescriptorType::eCombinedImageSampler, 3}}}; + return get_device()->get_handle().createDescriptorPool({{}, 2, pool_sizes}); +} + +vk::DescriptorSetLayout HPPTerrainTessellation::create_sky_sphere_descriptor_set_layout() +{ + std::array layout_bindings = { + {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eVertex}, + {1, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; + + vk::DescriptorSetLayoutCreateInfo skysphere_descriptor_layout({}, layout_bindings); + return get_device()->get_handle().createDescriptorSetLayout(skysphere_descriptor_layout); +} + +vk::Pipeline HPPTerrainTessellation::create_sky_sphere_pipeline() +{ + std::vector shader_stages = { + load_shader("terrain_tessellation/skysphere.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("terrain_tessellation/skysphere.frag", vk::ShaderStageFlagBits::eFragment)}; + + // Vertex bindings an attributes + // Binding description + std::array vertex_input_bindings = {{{0, sizeof(HPPVertex), vk::VertexInputRate::eVertex}}}; + + // Attribute descriptions + std::array vertex_input_attributes = {{{0, 0, vk::Format::eR32G32B32Sfloat, 0}, // Position + {1, 0, vk::Format::eR32G32B32Sfloat, sizeof(float) * 3}, // Normal + {2, 0, vk::Format::eR32G32Sfloat, sizeof(float) * 6}}}; // UV + + vk::PipelineVertexInputStateCreateInfo vertex_input_state({}, vertex_input_bindings, vertex_input_attributes); + + vk::PipelineColorBlendAttachmentState blend_attachment_state; + blend_attachment_state.colorWriteMask = + vk::ColorComponentFlagBits::eR | vk::ColorComponentFlagBits::eG | vk::ColorComponentFlagBits::eB | vk::ColorComponentFlagBits::eA; + + // Note: Using reversed depth-buffer for increased precision, so Greater depth values are kept + vk::PipelineDepthStencilStateCreateInfo depth_stencil_state; + depth_stencil_state.depthCompareOp = vk::CompareOp::eGreater; + depth_stencil_state.depthTestEnable = true; + depth_stencil_state.depthWriteEnable = false; + depth_stencil_state.back.compareOp = vk::CompareOp::eGreater; + depth_stencil_state.front = depth_stencil_state.back; + + // For the sky_sphere use triangle list topology + return vkb::common::create_graphics_pipeline(get_device()->get_handle(), + pipeline_cache, + shader_stages, + vertex_input_state, + vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, + vk::CullModeFlagBits::eBack, + vk::FrontFace::eCounterClockwise, + {blend_attachment_state}, + depth_stencil_state, + sky_sphere.pipeline_layout, + render_pass); +} + +vk::DescriptorSetLayout HPPTerrainTessellation::create_terrain_descriptor_set_layout() +{ + // Terrain + std::array layout_bindings = { + {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eTessellationControl | vk::ShaderStageFlagBits::eTessellationEvaluation}, + {1, + vk::DescriptorType::eCombinedImageSampler, + 1, + vk::ShaderStageFlagBits::eTessellationControl | vk::ShaderStageFlagBits::eTessellationEvaluation | vk::ShaderStageFlagBits::eFragment}, + {2, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; + + vk::DescriptorSetLayoutCreateInfo terrain_descriptor_layout({}, layout_bindings); + return get_device()->get_handle().createDescriptorSetLayout(terrain_descriptor_layout); +} + +vk::Pipeline HPPTerrainTessellation::create_terrain_pipeline(vk::PolygonMode polygon_mode) +{ + // Vertex bindings an attributes + // Binding description + std::array vertex_input_bindings = { + {{0, sizeof(HPPTerrainTessellation::Vertex), vk::VertexInputRate::eVertex}}}; + + // Attribute descriptions + std::array vertex_input_attributes = {{{0, 0, vk::Format::eR32G32B32Sfloat, 0}, // Position + {1, 0, vk::Format::eR32G32B32Sfloat, sizeof(float) * 3}, // Normal + {2, 0, vk::Format::eR32G32Sfloat, sizeof(float) * 6}}}; // UV + + vk::PipelineVertexInputStateCreateInfo vertex_input_state({}, vertex_input_bindings, vertex_input_attributes); + + vk::PipelineColorBlendAttachmentState blend_attachment_state; + blend_attachment_state.colorWriteMask = + vk::ColorComponentFlagBits::eR | vk::ColorComponentFlagBits::eG | vk::ColorComponentFlagBits::eB | vk::ColorComponentFlagBits::eA; + + // Note: Using reversed depth-buffer for increased precision, so Greater depth values are kept + vk::PipelineDepthStencilStateCreateInfo depth_stencil_state; + depth_stencil_state.depthCompareOp = vk::CompareOp::eGreater; + depth_stencil_state.depthTestEnable = true; + depth_stencil_state.depthWriteEnable = true; + depth_stencil_state.back.compareOp = vk::CompareOp::eGreater; + depth_stencil_state.front = depth_stencil_state.back; + + return vkb::common::create_graphics_pipeline(get_device()->get_handle(), + pipeline_cache, + terrain.shader_stages, + vertex_input_state, + vk::PrimitiveTopology::ePatchList, + 4, // we render the terrain as a grid of quad patches + polygon_mode, + vk::CullModeFlagBits::eBack, + vk::FrontFace::eCounterClockwise, + {blend_attachment_state}, + depth_stencil_state, + terrain.pipeline_layout, + render_pass); +} + void HPPTerrainTessellation::draw() { HPPApiVulkanSample::prepare_frame(); @@ -229,11 +327,11 @@ void HPPTerrainTessellation::draw() // Submit to queue queue.submit(submit_info); - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) + if (statistics.query_supported) { // Read query results for displaying in next frame - pipeline_stats = - get_device()->get_handle().getQueryPoolResult>(query_pool, 0, 1, sizeof(pipeline_stats), vk::QueryResultFlagBits::e64).value; + statistics.results = + get_device()->get_handle().getQueryPoolResult>(statistics.query_pool, 0, 1, sizeof(statistics.results), vk::QueryResultFlagBits::e64).value; } HPPApiVulkanSample::submit_frame(); @@ -248,23 +346,20 @@ void HPPTerrainTessellation::generate_terrain() std::vector vertices; vertices.resize(vertex_count); - const float wx = 2.0f; - const float wy = 2.0f; - for (auto x = 0; x < patch_size; x++) { for (auto y = 0; y < patch_size; y++) { uint32_t index = (x + y * patch_size); - vertices[index].pos[0] = x * wx + wx / 2.0f - static_cast(patch_size) * wx / 2.0f; + vertices[index].pos[0] = 2.0f * x + 1.0f - patch_size; vertices[index].pos[1] = 0.0f; - vertices[index].pos[2] = y * wy + wy / 2.0f - static_cast(patch_size) * wy / 2.0f; + vertices[index].pos[2] = 2.0f * y * 1.0f - patch_size; vertices[index].uv = glm::vec2(static_cast(x) / patch_size, static_cast(y) / patch_size) * uv_scale; } } // Calculate normals from height map using a sobel filter - vkb::HeightMap heightmap("textures/terrain_heightmap_r16.ktx", patch_size); + vkb::HeightMap height_map("textures/terrain_heightmap_r16.ktx", patch_size); for (auto x = 0; x < patch_size; x++) { for (auto y = 0; y < patch_size; y++) @@ -275,7 +370,7 @@ void HPPTerrainTessellation::generate_terrain() { for (auto hy = -1; hy <= 1; hy++) { - heights[hx + 1][hy + 1] = heightmap.get_height(x + hx, y + hy); + heights[hx + 1][hy + 1] = height_map.get_height(x + hx, y + hy); } } @@ -334,318 +429,169 @@ void HPPTerrainTessellation::generate_terrain() terrain.indices = std::make_unique( *get_device(), index_buffer_size, vk::BufferUsageFlagBits::eIndexBuffer | vk::BufferUsageFlagBits::eTransferDst, VMA_MEMORY_USAGE_GPU_ONLY); - // Copy from staging buffers - vk::CommandBuffer copy_command = device->create_command_buffer(vk::CommandBufferLevel::ePrimary, true); + vk::Device vkDevice = get_device()->get_handle(); + // Copy from staging buffers + vk::CommandBuffer copy_command = vkb::common::allocate_command_buffer(vkDevice, device->get_command_pool().get_handle()); + copy_command.begin(vk::CommandBufferBeginInfo()); copy_command.copyBuffer(vertex_staging.buffer, terrain.vertices->get_handle(), {{0, 0, vertex_buffer_size}}); - copy_command.copyBuffer(index_staging.buffer, terrain.indices->get_handle(), {{0, 0, index_buffer_size}}); + get_device()->flush_command_buffer(copy_command, queue, true); - device->flush_command_buffer(copy_command, queue, true); - - get_device()->get_handle().destroyBuffer(vertex_staging.buffer); - get_device()->get_handle().freeMemory(vertex_staging.memory); - get_device()->get_handle().destroyBuffer(index_staging.buffer); - get_device()->get_handle().freeMemory(index_staging.memory); + vkDevice.destroyBuffer(vertex_staging.buffer); + vkDevice.freeMemory(vertex_staging.memory); + vkDevice.destroyBuffer(index_staging.buffer); + vkDevice.freeMemory(index_staging.memory); } void HPPTerrainTessellation::load_assets() { - skysphere = load_model("scenes/geosphere.gltf"); - - textures.skysphere = load_texture("textures/skysphere_rgba.ktx", vkb::sg::Image::Color); - // Terrain textures are stored in a texture array with layers corresponding to terrain height - textures.terrain_array = load_texture_array("textures/terrain_texturearray_rgba.ktx", vkb::sg::Image::Color); - - // Height data is stored in a one-channel texture - textures.heightmap = load_texture("textures/terrain_heightmap_r16.ktx", vkb::sg::Image::Other); - - // Setup a mirroring sampler for the height map - vk::SamplerCreateInfo sampler_create_info; - sampler_create_info.magFilter = vk::Filter::eLinear; - sampler_create_info.minFilter = vk::Filter::eLinear; - sampler_create_info.mipmapMode = vk::SamplerMipmapMode::eLinear; - sampler_create_info.addressModeU = vk::SamplerAddressMode::eMirroredRepeat; - sampler_create_info.addressModeV = sampler_create_info.addressModeU; - sampler_create_info.addressModeW = sampler_create_info.addressModeU; - sampler_create_info.maxAnisotropy = 1.0f; - sampler_create_info.compareOp = vk::CompareOp::eNever; - sampler_create_info.minLod = 0.0f; - sampler_create_info.maxLod = static_cast(textures.heightmap.image->get_mipmaps().size()); - sampler_create_info.borderColor = vk::BorderColor::eFloatOpaqueWhite; - get_device()->get_handle().destroySampler(textures.heightmap.sampler); - textures.heightmap.sampler = get_device()->get_handle().createSampler(sampler_create_info); - - // Setup a repeating sampler for the terrain texture layers - sampler_create_info.addressModeU = vk::SamplerAddressMode::eRepeat; - sampler_create_info.addressModeV = sampler_create_info.addressModeU; - sampler_create_info.addressModeW = sampler_create_info.addressModeU; - sampler_create_info.maxLod = static_cast(textures.terrain_array.image->get_mipmaps().size()); - if (get_device()->get_gpu().get_features().samplerAnisotropy) - { - sampler_create_info.maxAnisotropy = 4.0f; - sampler_create_info.anisotropyEnable = true; - } - get_device()->get_handle().destroySampler(textures.terrain_array.sampler); - textures.terrain_array.sampler = get_device()->get_handle().createSampler(sampler_create_info); -} - -void HPPTerrainTessellation::prepare_pipelines() -{ - std::array terrain_shader_stages = {{load_shader("terrain_tessellation/terrain.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("terrain_tessellation/terrain.frag", vk::ShaderStageFlagBits::eFragment), - load_shader("terrain_tessellation/terrain.tesc", vk::ShaderStageFlagBits::eTessellationControl), - load_shader("terrain_tessellation/terrain.tese", vk::ShaderStageFlagBits::eTessellationEvaluation)}}; - - // Vertex bindings an attributes - // Binding description - std::array vertex_input_bindings = {{{0, sizeof(HPPTerrainTessellation::Vertex), vk::VertexInputRate::eVertex}}}; + sky_sphere.geometry = load_model("scenes/geosphere.gltf"); + sky_sphere.texture = load_texture("textures/skysphere_rgba.ktx", vkb::sg::Image::Color); - // Attribute descriptions - std::array vertex_input_attributes = {{{0, 0, vk::Format::eR32G32B32Sfloat, 0}, // Position - {1, 0, vk::Format::eR32G32B32Sfloat, sizeof(float) * 3}, // Normal - {2, 0, vk::Format::eR32G32Sfloat, sizeof(float) * 6}}}; // UV - - vk::PipelineVertexInputStateCreateInfo vertex_input_state({}, vertex_input_bindings, vertex_input_attributes); + // Terrain textures are stored in a texture array with layers corresponding to terrain height; create a repeating sampler + terrain.terrain_array = load_texture_array("textures/terrain_texturearray_rgba.ktx", vkb::sg::Image::Color, vk::SamplerAddressMode::eRepeat); - vk::PipelineInputAssemblyStateCreateInfo input_assembly_state({}, vk::PrimitiveTopology::ePatchList, false); - - // We render the terrain as a grid of quad patches - vk::PipelineTessellationStateCreateInfo tessellation_state({}, 4); - - vk::PipelineViewportStateCreateInfo viewport_state({}, 1, nullptr, 1, nullptr); - - vk::PipelineRasterizationStateCreateInfo rasterization_state; - rasterization_state.polygonMode = vk::PolygonMode::eFill; - rasterization_state.cullMode = vk::CullModeFlagBits::eBack; - rasterization_state.frontFace = vk::FrontFace::eCounterClockwise; - rasterization_state.lineWidth = 1.0f; - - vk::PipelineMultisampleStateCreateInfo multisample_state({}, vk::SampleCountFlagBits::e1); + // Height data is stored in a one-channel texture; create a mirroring sampler + terrain.height_map = load_texture("textures/terrain_heightmap_r16.ktx", vkb::sg::Image::Other, vk::SamplerAddressMode::eMirroredRepeat); +} - // Note: Using reversed depth-buffer for increased precision, so Greater depth values are kept - vk::PipelineDepthStencilStateCreateInfo depth_stencil_state; - depth_stencil_state.depthCompareOp = vk::CompareOp::eGreater; - depth_stencil_state.depthTestEnable = true; - depth_stencil_state.depthWriteEnable = true; - depth_stencil_state.back.compareOp = vk::CompareOp::eGreater; - depth_stencil_state.front = depth_stencil_state.back; +void HPPTerrainTessellation::prepare_camera() +{ + // Note: Using reversed depth-buffer for increased precision, so Znear and Zfar are flipped + camera.type = vkb::CameraType::FirstPerson; + camera.set_perspective(60.0f, static_cast(extent.width) / static_cast(extent.height), 512.0f, 0.1f); + camera.set_rotation(glm::vec3(-12.0f, 159.0f, 0.0f)); + camera.set_translation(glm::vec3(18.0f, 22.5f, 57.5f)); + camera.translation_speed = 7.5f; +} - vk::PipelineColorBlendAttachmentState blend_attachment_state; - blend_attachment_state.colorWriteMask = - vk::ColorComponentFlagBits::eR | vk::ColorComponentFlagBits::eG | vk::ColorComponentFlagBits::eB | vk::ColorComponentFlagBits::eA; +void HPPTerrainTessellation::prepare_sky_sphere() +{ + sky_sphere.descriptor_set_layout = create_sky_sphere_descriptor_set_layout(); + sky_sphere.pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, sky_sphere.descriptor_set_layout}); + sky_sphere.pipeline = create_sky_sphere_pipeline(); + sky_sphere.descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, {sky_sphere.descriptor_set_layout}); + update_sky_sphere_descriptor_set(); +} - vk::PipelineColorBlendStateCreateInfo color_blend_state({}, false, {}, blend_attachment_state); - - std::array dynamic_state_enables = {vk::DynamicState::eViewport, vk::DynamicState::eScissor, vk::DynamicState::eLineWidth}; - - vk::PipelineDynamicStateCreateInfo dynamic_state({}, dynamic_state_enables); - - vk::GraphicsPipelineCreateInfo pipeline_create_info({}, - terrain_shader_stages, - &vertex_input_state, - &input_assembly_state, - &tessellation_state, - &viewport_state, - &rasterization_state, - &multisample_state, - &depth_stencil_state, - &color_blend_state, - &dynamic_state, - pipeline_layouts.terrain, - render_pass, - {}, - {}, - -1); - - vk::Result result; - std::tie(result, pipelines.terrain) = get_device()->get_handle().createGraphicsPipeline(pipeline_cache, pipeline_create_info); - assert(result == vk::Result::eSuccess); - - // Terrain wireframe pipeline - if (get_device()->get_gpu().get_features().fillModeNonSolid) +void HPPTerrainTessellation::prepare_statistics() +{ + if (statistics.query_supported) { - rasterization_state.polygonMode = vk::PolygonMode::eLine; - std::tie(result, pipelines.wireframe) = get_device()->get_handle().createGraphicsPipeline(pipeline_cache, pipeline_create_info); - assert(result == vk::Result::eSuccess); - }; + // Create query pool + vk::QueryPoolCreateInfo query_pool_info({}, + vk::QueryType::ePipelineStatistics, + 2, + vk::QueryPipelineStatisticFlagBits::eVertexShaderInvocations | + vk::QueryPipelineStatisticFlagBits::eTessellationEvaluationShaderInvocations); + statistics.query_pool = get_device()->get_handle().createQueryPool(query_pool_info); + } +} - // Skysphere pipeline - - // Stride from glTF model vertex layout - vertex_input_bindings[0].stride = sizeof(HPPVertex); - - rasterization_state.polygonMode = vk::PolygonMode::eFill; - // Revert to triangle list topology - input_assembly_state.topology = vk::PrimitiveTopology::eTriangleList; - // Reset tessellation state - pipeline_create_info.pTessellationState = nullptr; - // Don't write to depth buffer - depth_stencil_state.depthWriteEnable = false; - pipeline_create_info.layout = pipeline_layouts.skysphere; - std::array skysphere_shader_stages = {{load_shader("terrain_tessellation/skysphere.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("terrain_tessellation/skysphere.frag", vk::ShaderStageFlagBits::eFragment)}}; - pipeline_create_info.setStages(skysphere_shader_stages); - - std::tie(result, pipelines.skysphere) = get_device()->get_handle().createGraphicsPipeline(pipeline_cache, pipeline_create_info); - assert(result == vk::Result::eSuccess); +void HPPTerrainTessellation::prepare_terrain() +{ + terrain.shader_stages = {load_shader("terrain_tessellation/terrain.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("terrain_tessellation/terrain.frag", vk::ShaderStageFlagBits::eFragment), + load_shader("terrain_tessellation/terrain.tesc", vk::ShaderStageFlagBits::eTessellationControl), + load_shader("terrain_tessellation/terrain.tese", vk::ShaderStageFlagBits::eTessellationEvaluation)}; + + terrain.descriptor_set_layout = create_terrain_descriptor_set_layout(); + terrain.pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, terrain.descriptor_set_layout}); + terrain.pipeline = create_terrain_pipeline(vk::PolygonMode::eFill); + terrain.descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, {terrain.descriptor_set_layout}); + update_terrain_descriptor_set(); } // Prepare and initialize uniform buffer containing shader uniforms void HPPTerrainTessellation::prepare_uniform_buffers() { // Shared tessellation shader stages uniform buffer - uniform_buffers.terrain_tessellation = - std::make_unique(*get_device(), sizeof(ubo_tess), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); + terrain.tessellation_buffer = + std::make_unique(*get_device(), sizeof(terrain.tessellation), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); // Skysphere vertex shader uniform buffer - uniform_buffers.skysphere_vertex = - std::make_unique(*get_device(), sizeof(ubo_vs), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); + sky_sphere.transform_buffer = + std::make_unique(*get_device(), sizeof(sky_sphere.transform), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); update_uniform_buffers(); } -void HPPTerrainTessellation::setup_descriptor_pool() +void HPPTerrainTessellation::prepare_wireframe() { - std::array pool_sizes = {{{vk::DescriptorType::eUniformBuffer, 3}, {vk::DescriptorType::eCombinedImageSampler, 3}}}; - - vk::DescriptorPoolCreateInfo descriptor_pool_create_info({}, 2, pool_sizes); - - descriptor_pool = get_device()->get_handle().createDescriptorPool(descriptor_pool_create_info); + if (wireframe.supported) + { + // wireframe mode uses nearly the same settings as the terrain mode... just vk::PolygonMode::eLine, instead of vk::PolygonMode::eFill + wireframe.pipeline = create_terrain_pipeline(vk::PolygonMode::eLine); + }; } -void HPPTerrainTessellation::setup_descriptor_set_layouts() +void HPPTerrainTessellation::update_uniform_buffers() { - // Terrain - std::array terrain_layout_bindings = { - {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eTessellationControl | vk::ShaderStageFlagBits::eTessellationEvaluation}, - {1, - vk::DescriptorType::eCombinedImageSampler, - 1, - vk::ShaderStageFlagBits::eTessellationControl | vk::ShaderStageFlagBits::eTessellationEvaluation | vk::ShaderStageFlagBits::eFragment}, - {2, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; + // Tessellation + terrain.tessellation.projection = camera.matrices.perspective; + terrain.tessellation.modelview = camera.matrices.view * glm::mat4(1.0f); + terrain.tessellation.light_pos.y = -0.5f - terrain.tessellation.displacement_factor; // todo: Not used yet + terrain.tessellation.viewport_dim = glm::vec2(static_cast(extent.width), static_cast(extent.height)); - vk::DescriptorSetLayoutCreateInfo terrain_descriptor_layout({}, terrain_layout_bindings); - descriptor_set_layouts.terrain = get_device()->get_handle().createDescriptorSetLayout(terrain_descriptor_layout); -#if defined(ANDROID) - vk::PipelineLayoutCreateInfo terrain_pipeline_layout_create_info({}, 1, &descriptor_set_layouts.terrain); -#else - vk::PipelineLayoutCreateInfo terrain_pipeline_layout_create_info({}, descriptor_set_layouts.terrain); -#endif - pipeline_layouts.terrain = get_device()->get_handle().createPipelineLayout(terrain_pipeline_layout_create_info); - - // Skysphere - std::array skysphere_layout_bindings = { - {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eVertex}, - {1, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; + frustum.update(terrain.tessellation.projection * terrain.tessellation.modelview); + memcpy(terrain.tessellation.frustum_planes, frustum.get_planes().data(), sizeof(glm::vec4) * 6); + + float saved_factor = terrain.tessellation.tessellation_factor; + if (!terrain.tessellation_enabled) + { + // Setting this to zero sets all tessellation factors to 1.0 in the shader + terrain.tessellation.tessellation_factor = 0.0f; + } + + terrain.tessellation_buffer->convert_and_update(terrain.tessellation); - vk::DescriptorSetLayoutCreateInfo skysphere_descriptor_layout({}, skysphere_layout_bindings); - descriptor_set_layouts.skysphere = get_device()->get_handle().createDescriptorSetLayout(skysphere_descriptor_layout); -#if defined(ANDROID) - vk::PipelineLayoutCreateInfo skysphere_pipeline_layout_create_info({}, 1, &descriptor_set_layouts.skysphere); -#else - vk::PipelineLayoutCreateInfo skysphere_pipeline_layout_create_info({}, descriptor_set_layouts.skysphere); -#endif - pipeline_layouts.skysphere = get_device()->get_handle().createPipelineLayout(skysphere_pipeline_layout_create_info); + if (!terrain.tessellation_enabled) + { + terrain.tessellation.tessellation_factor = saved_factor; + } + + // Skysphere vertex shader + sky_sphere.transform = camera.matrices.perspective * glm::mat4(glm::mat3(camera.matrices.view)); + sky_sphere.transform_buffer->convert_and_update(sky_sphere.transform); } -void HPPTerrainTessellation::setup_descriptor_sets() +void HPPTerrainTessellation::update_sky_sphere_descriptor_set() { - // Terrain -#if defined(ANDROID) - vk::DescriptorSetAllocateInfo alloc_info(descriptor_pool, 1, &descriptor_set_layouts.terrain); -#else - vk::DescriptorSetAllocateInfo alloc_info(descriptor_pool, descriptor_set_layouts.terrain); -#endif - descriptor_sets.terrain = get_device()->get_handle().allocateDescriptorSets(alloc_info).front(); - - vk::DescriptorBufferInfo terrain_buffer_descriptor(uniform_buffers.terrain_tessellation->get_handle(), 0, VK_WHOLE_SIZE); - vk::DescriptorImageInfo heightmap_image_descriptor( - textures.heightmap.sampler, - textures.heightmap.image->get_vk_image_view().get_handle(), - descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, textures.heightmap.image->get_vk_image_view().get_format())); - vk::DescriptorImageInfo terrainmap_image_descriptor( - textures.terrain_array.sampler, - textures.terrain_array.image->get_vk_image_view().get_handle(), - descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, textures.terrain_array.image->get_vk_image_view().get_format())); - std::array terrain_write_descriptor_sets = { - {{descriptor_sets.terrain, 0, {}, vk::DescriptorType::eUniformBuffer, {}, terrain_buffer_descriptor}, - {descriptor_sets.terrain, 1, {}, vk::DescriptorType::eCombinedImageSampler, heightmap_image_descriptor}, - {descriptor_sets.terrain, 2, {}, vk::DescriptorType::eCombinedImageSampler, terrainmap_image_descriptor}}}; - get_device()->get_handle().updateDescriptorSets(terrain_write_descriptor_sets, {}); + vk::DescriptorBufferInfo skysphere_buffer_descriptor(sky_sphere.transform_buffer->get_handle(), 0, VK_WHOLE_SIZE); - // Skysphere - alloc_info.setSetLayouts(descriptor_set_layouts.skysphere); - descriptor_sets.skysphere = get_device()->get_handle().allocateDescriptorSets(alloc_info).front(); + vk::DescriptorImageInfo skysphere_image_descriptor( + sky_sphere.texture.sampler, + sky_sphere.texture.image->get_vk_image_view().get_handle(), + descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, sky_sphere.texture.image->get_vk_image_view().get_format())); - vk::DescriptorBufferInfo skysphere_buffer_descriptor(uniform_buffers.skysphere_vertex->get_handle(), 0, VK_WHOLE_SIZE); - vk::DescriptorImageInfo skysphere_image_descriptor( - textures.skysphere.sampler, - textures.skysphere.image->get_vk_image_view().get_handle(), - descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, textures.skysphere.image->get_vk_image_view().get_format())); std::array skysphere_write_descriptor_sets = { - {{descriptor_sets.skysphere, 0, {}, vk::DescriptorType::eUniformBuffer, {}, skysphere_buffer_descriptor}, - {descriptor_sets.skysphere, 1, {}, vk::DescriptorType::eCombinedImageSampler, skysphere_image_descriptor}}}; + {{sky_sphere.descriptor_set, 0, {}, vk::DescriptorType::eUniformBuffer, {}, skysphere_buffer_descriptor}, + {sky_sphere.descriptor_set, 1, {}, vk::DescriptorType::eCombinedImageSampler, skysphere_image_descriptor}}}; + get_device()->get_handle().updateDescriptorSets(skysphere_write_descriptor_sets, {}); } -// Setup pool and buffer for storing pipeline statistics results -void HPPTerrainTessellation::setup_query_result_buffer() +void HPPTerrainTessellation::update_terrain_descriptor_set() { - const uint32_t buffer_size = 2 * sizeof(uint64_t); - - vk::BufferCreateInfo buffer_create_info({}, buffer_size, vk::BufferUsageFlagBits::eUniformBuffer | vk::BufferUsageFlagBits::eTransferDst); - - // Results are saved in a host visible buffer for easy access by the application - query_result.buffer = get_device()->get_handle().createBuffer(buffer_create_info); - vk::MemoryRequirements memory_requirements = get_device()->get_handle().getBufferMemoryRequirements(query_result.buffer); - vk::MemoryAllocateInfo memory_allocation( - memory_requirements.size, - get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, - vk::MemoryPropertyFlagBits::eHostVisible | vk::MemoryPropertyFlagBits::eHostCoherent)); - query_result.memory = get_device()->get_handle().allocateMemory(memory_allocation); - get_device()->get_handle().bindBufferMemory(query_result.buffer, query_result.memory, 0); - - // Create query pool - if (get_device()->get_gpu().get_features().pipelineStatisticsQuery) - { - vk::QueryPoolCreateInfo query_pool_info({}, - vk::QueryType::ePipelineStatistics, - 2, - vk::QueryPipelineStatisticFlagBits::eVertexShaderInvocations | - vk::QueryPipelineStatisticFlagBits::eTessellationEvaluationShaderInvocations); - query_pool = get_device()->get_handle().createQueryPool(query_pool_info); - } -} + vk::DescriptorBufferInfo terrain_buffer_descriptor(terrain.tessellation_buffer->get_handle(), 0, VK_WHOLE_SIZE); -void HPPTerrainTessellation::update_uniform_buffers() -{ - // Tessellation - ubo_tess.projection = camera.matrices.perspective; - ubo_tess.modelview = camera.matrices.view * glm::mat4(1.0f); - ubo_tess.light_pos.y = -0.5f - ubo_tess.displacement_factor; // todo: Not used yet - ubo_tess.viewport_dim = glm::vec2(static_cast(extent.width), static_cast(extent.height)); - - frustum.update(ubo_tess.projection * ubo_tess.modelview); - memcpy(ubo_tess.frustum_planes, frustum.get_planes().data(), sizeof(glm::vec4) * 6); - - float saved_factor = ubo_tess.tessellation_factor; - if (!tessellation) - { - // Setting this to zero sets all tessellation factors to 1.0 in the shader - ubo_tess.tessellation_factor = 0.0f; - } + vk::DescriptorImageInfo heightmap_image_descriptor( + terrain.height_map.sampler, + terrain.height_map.image->get_vk_image_view().get_handle(), + descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, terrain.height_map.image->get_vk_image_view().get_format())); - uniform_buffers.terrain_tessellation->convert_and_update(ubo_tess); + vk::DescriptorImageInfo terrainmap_image_descriptor( + terrain.terrain_array.sampler, + terrain.terrain_array.image->get_vk_image_view().get_handle(), + descriptor_type_to_image_layout(vk::DescriptorType::eCombinedImageSampler, terrain.terrain_array.image->get_vk_image_view().get_format())); - if (!tessellation) - { - ubo_tess.tessellation_factor = saved_factor; - } + std::array terrain_write_descriptor_sets = { + {{terrain.descriptor_set, 0, {}, vk::DescriptorType::eUniformBuffer, {}, terrain_buffer_descriptor}, + {terrain.descriptor_set, 1, {}, vk::DescriptorType::eCombinedImageSampler, heightmap_image_descriptor}, + {terrain.descriptor_set, 2, {}, vk::DescriptorType::eCombinedImageSampler, terrainmap_image_descriptor}}}; - // Skysphere vertex shader - ubo_vs.mvp = camera.matrices.perspective * glm::mat4(glm::mat3(camera.matrices.view)); - uniform_buffers.skysphere_vertex->convert_and_update(ubo_vs.mvp); + get_device()->get_handle().updateDescriptorSets(terrain_write_descriptor_sets, {}); } std::unique_ptr create_hpp_terrain_tessellation() diff --git a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.h b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.h index 3eede40d5..88dd1d2ed 100644 --- a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.h +++ b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.h @@ -32,60 +32,49 @@ class HPPTerrainTessellation : public HPPApiVulkanSample ~HPPTerrainTessellation(); private: - struct DescriptorSetLayouts + struct SkySphere { - vk::DescriptorSetLayout terrain; - vk::DescriptorSetLayout skysphere; + vk::DescriptorSet descriptor_set; + vk::DescriptorSetLayout descriptor_set_layout; + vk::PipelineLayout pipeline_layout; + vk::Pipeline pipeline; + + std::unique_ptr geometry; + + HPPTexture texture; + + glm::mat4 transform; + std::unique_ptr transform_buffer; + + void destroy(vk::Device device) + { + device.destroyPipeline(pipeline); + device.destroyPipelineLayout(pipeline_layout); + device.destroyDescriptorSetLayout(descriptor_set_layout); + // the descriptor_set is implicitly freed by destroying its DescriptorPool + device.destroySampler(texture.sampler); + } }; - struct DescriptorSets + struct Statistics { - vk::DescriptorSet terrain; - vk::DescriptorSet skysphere; - }; - - struct PipelineLayouts - { - vk::PipelineLayout terrain; - vk::PipelineLayout skysphere; - }; - - struct Pipelines - { - vk::Pipeline terrain; - vk::Pipeline wireframe = nullptr; - vk::Pipeline skysphere; - }; - - // Pipeline statistics - struct QueryResults - { - vk::Buffer buffer; - vk::DeviceMemory memory; - }; + bool query_supported = false; - // Skysphere vertex shader stage - struct SkySphereUBO - { - glm::mat4 mvp; - }; + vk::QueryPool query_pool; - struct Textures - { - HPPTexture heightmap; - HPPTexture skysphere; - HPPTexture terrain_array; - }; + std::array results = {{0}}; - struct Terrain - { - std::unique_ptr vertices; - std::unique_ptr indices; - uint32_t index_count; + void destroy(vk::Device device) + { + if (query_supported) + { + device.destroyQueryPool(query_pool); + } + } }; // Shared values for tessellation control and evaluation stages - struct TessellationUBO + struct Tessellation { glm::mat4 projection; glm::mat4 modelview; @@ -98,10 +87,37 @@ class HPPTerrainTessellation : public HPPApiVulkanSample float tessellated_edge_size = 20.0f; }; - struct UniformBuffers + struct Terrain { - std::unique_ptr terrain_tessellation; - std::unique_ptr skysphere_vertex; + vk::DescriptorSet descriptor_set; + vk::DescriptorSetLayout descriptor_set_layout; + vk::PipelineLayout pipeline_layout; + vk::Pipeline pipeline; + + std::unique_ptr vertices; + std::unique_ptr indices; + uint32_t index_count; + + HPPTexture height_map; + HPPTexture terrain_array; + + bool sampler_anisotropy_supported = false; + + std::vector shader_stages; + + Tessellation tessellation; + std::unique_ptr tessellation_buffer; + bool tessellation_enabled = true; + + void destroy(vk::Device device) + { + device.destroyPipeline(pipeline); + device.destroyPipelineLayout(pipeline_layout); + device.destroyDescriptorSetLayout(descriptor_set_layout); + // the descriptor_set is implicitly freed by destroying its DescriptorPool + device.destroySampler(height_map.sampler); + device.destroySampler(terrain_array.sampler); + } }; struct Vertex @@ -111,6 +127,19 @@ class HPPTerrainTessellation : public HPPApiVulkanSample glm::vec2 uv; }; + struct Wireframe + { + bool supported = false; + bool enabled = false; + + vk::Pipeline pipeline; + + void destroy(vk::Device device) + { + device.destroyPipeline(pipeline); + } + }; + private: // from vkb::Application bool prepare(const vkb::ApplicationOptions &options) override; @@ -124,34 +153,30 @@ class HPPTerrainTessellation : public HPPApiVulkanSample void render(float delta_time) override; void view_changed() override; - void draw(); - void generate_terrain(); - void load_assets(); - void prepare_pipelines(); - void prepare_uniform_buffers(); - void setup_descriptor_pool(); - void setup_descriptor_set_layouts(); - void setup_descriptor_sets(); - void setup_query_result_buffer(); - void update_uniform_buffers(); + vk::DescriptorPool create_descriptor_pool(); + vk::DescriptorSetLayout create_sky_sphere_descriptor_set_layout(); + vk::Pipeline create_sky_sphere_pipeline(); + vk::DescriptorSetLayout create_terrain_descriptor_set_layout(); + vk::Pipeline create_terrain_pipeline(vk::PolygonMode polygon_mode); + void draw(); + void generate_terrain(); + void load_assets(); + void prepare_camera(); + void prepare_sky_sphere(); + void prepare_statistics(); + void prepare_terrain(); + void prepare_uniform_buffers(); + void prepare_wireframe(); + void update_uniform_buffers(); + void update_sky_sphere_descriptor_set(); + void update_terrain_descriptor_set(); private: - DescriptorSetLayouts descriptor_set_layouts; - DescriptorSets descriptor_sets; - vkb::Frustum frustum; // View frustum passed to tessellation control shader for culling - PipelineLayouts pipeline_layouts; - std::array pipeline_stats = {{0}}; - Pipelines pipelines; - vk::QueryPool query_pool = nullptr; - QueryResults query_result; - std::unique_ptr skysphere; - bool tessellation = true; - Textures textures; - Terrain terrain; - TessellationUBO ubo_tess; - SkySphereUBO ubo_vs; - UniformBuffers uniform_buffers; - bool wireframe = false; + vkb::Frustum frustum; // View frustum passed to tessellation control shader for culling + SkySphere sky_sphere; + Statistics statistics; + Terrain terrain; + Wireframe wireframe; }; std::unique_ptr create_hpp_terrain_tessellation(); From 68a1f53602f170b83a4728e94893ea0a9cbaf75c Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andreas=20S=C3=BC=C3=9Fenbach?= Date: Mon, 28 Aug 2023 17:56:26 +0200 Subject: [PATCH 08/13] Simplify RenderFrame::allocate_buffer (#771) * Simplify RenderFrame::allocate_buffer * Fix glitch in BufferPool::request_buffer_block --- framework/buffer_pool.cpp | 63 +++++++++++++++------------- framework/buffer_pool.h | 21 +++++++--- framework/rendering/render_frame.cpp | 20 +++------ 3 files changed, 54 insertions(+), 50 deletions(-) diff --git a/framework/buffer_pool.cpp b/framework/buffer_pool.cpp index 658ddd9ea..df99d7083 100644 --- a/framework/buffer_pool.cpp +++ b/framework/buffer_pool.cpp @@ -1,4 +1,4 @@ -/* Copyright (c) 2019-2022, Arm Limited and Contributors +/* Copyright (c) 2019-2023, Arm Limited and Contributors * * SPDX-License-Identifier: Apache-2.0 * @@ -50,21 +50,29 @@ BufferBlock::BufferBlock(Device &device, VkDeviceSize size, VkBufferUsageFlags u } } -BufferAllocation BufferBlock::allocate(const uint32_t allocate_size) +VkDeviceSize BufferBlock::aligned_offset() const { - assert(allocate_size > 0 && "Allocation size must be greater than zero"); + return (offset + alignment - 1) & ~(alignment - 1); +} - auto aligned_offset = (offset + alignment - 1) & ~(alignment - 1); +bool BufferBlock::can_allocate(VkDeviceSize size) const +{ + assert(size > 0 && "Allocation size must be greater than zero"); + return (aligned_offset() + size <= buffer.get_size()); +} - if (aligned_offset + allocate_size > buffer.get_size()) +BufferAllocation BufferBlock::allocate(VkDeviceSize size) +{ + if (can_allocate(size)) { - // No more space available from the underlying buffer, return empty allocation - return BufferAllocation{}; + // Move the current offset and return an allocation + auto aligned = aligned_offset(); + offset = aligned + size; + return BufferAllocation{buffer, size, aligned}; } - // Move the current offset and return an allocation - offset = aligned_offset + allocate_size; - return BufferAllocation{buffer, allocate_size, aligned_offset}; + // No more space available from the underlying buffer, return empty allocation + return BufferAllocation{}; } VkDeviceSize BufferBlock::get_size() const @@ -87,28 +95,25 @@ BufferPool::BufferPool(Device &device, VkDeviceSize block_size, VkBufferUsageFla BufferBlock &BufferPool::request_buffer_block(const VkDeviceSize minimum_size, bool minimal) { - // Find the first block in the range of the inactive blocks - // which can fit the minimum size - auto it = std::upper_bound(buffer_blocks.begin() + active_buffer_block_count, buffer_blocks.end(), minimum_size, - [](const VkDeviceSize &a, const std::unique_ptr &b) -> bool { return a <= b->get_size(); }); - - if (it != buffer_blocks.end()) + // Find a block in the range of the blocks which can fit the minimum size + auto it = minimal ? std::find_if(buffer_blocks.begin(), + buffer_blocks.end(), + [&minimum_size](const std::unique_ptr &buffer_block) { return (buffer_block->get_size() == minimum_size) && buffer_block->can_allocate(minimum_size); }) : + std::find_if(buffer_blocks.begin(), + buffer_blocks.end(), + [&minimum_size](const std::unique_ptr &buffer_block) { return buffer_block->can_allocate(minimum_size); }); + + if (it == buffer_blocks.end()) { - // Recycle inactive block - active_buffer_block_count++; - return *it->get(); - } - - LOGD("Building #{} buffer block ({})", buffer_blocks.size(), usage); - - VkDeviceSize new_block_size = minimal ? minimum_size : std::max(block_size, minimum_size); + LOGD("Building #{} buffer block ({})", buffer_blocks.size(), usage); - // Create a new block, store and return it - buffer_blocks.emplace_back(std::make_unique(device, new_block_size, usage, memory_usage)); + VkDeviceSize new_block_size = minimal ? minimum_size : std::max(block_size, minimum_size); - auto &block = buffer_blocks[active_buffer_block_count++]; + // Create a new block and get the iterator on it + it = buffer_blocks.emplace(buffer_blocks.end(), std::make_unique(device, new_block_size, usage, memory_usage)); + } - return *block.get(); + return *it->get(); } void BufferPool::reset() @@ -117,8 +122,6 @@ void BufferPool::reset() { buffer_block->reset(); } - - active_buffer_block_count = 0; } BufferAllocation::BufferAllocation(core::Buffer &buffer, VkDeviceSize size, VkDeviceSize offset) : diff --git a/framework/buffer_pool.h b/framework/buffer_pool.h index 1ece24db6..ab0cd2ee2 100644 --- a/framework/buffer_pool.h +++ b/framework/buffer_pool.h @@ -1,4 +1,4 @@ -/* Copyright (c) 2019-2022, Arm Limited and Contributors +/* Copyright (c) 2019-2023, Arm Limited and Contributors * * SPDX-License-Identifier: Apache-2.0 * @@ -75,15 +75,29 @@ class BufferBlock public: BufferBlock(Device &device, VkDeviceSize size, VkBufferUsageFlags usage, VmaMemoryUsage memory_usage); + /** + * @brief check if this BufferBlock can allocate a given amount of memory + * @param size the number of bytes to check + * @return \c true if \a size bytes can be allocated from this \c BufferBlock, otherwise \c false. + */ + bool can_allocate(VkDeviceSize size) const; + /** * @return An usable view on a portion of the underlying buffer */ - BufferAllocation allocate(uint32_t size); + BufferAllocation allocate(VkDeviceSize size); VkDeviceSize get_size() const; void reset(); + private: + /** + * @ brief Determine the current aligned offset. + * @return The current aligned offset. + */ + VkDeviceSize aligned_offset() const; + private: core::Buffer buffer; @@ -131,8 +145,5 @@ class BufferPool VkBufferUsageFlags usage{}; VmaMemoryUsage memory_usage{}; - - /// Numbers of active blocks from the start of buffer_blocks - uint32_t active_buffer_block_count{0}; }; } // namespace vkb diff --git a/framework/rendering/render_frame.cpp b/framework/rendering/render_frame.cpp index 4ce862e9b..e0e6436e1 100644 --- a/framework/rendering/render_frame.cpp +++ b/framework/rendering/render_frame.cpp @@ -289,23 +289,13 @@ BufferAllocation RenderFrame::allocate_buffer(const VkBufferUsageFlags usage, co bool want_minimal_block = buffer_allocation_strategy == BufferAllocationStrategy::OneAllocationPerBuffer; - if (want_minimal_block || !buffer_block) + if (want_minimal_block || !buffer_block || !buffer_block->can_allocate(size)) { - // If there is no block associated with the pool or we are creating a buffer for each allocation, - // request a new buffer block - buffer_block = &buffer_pool.request_buffer_block(to_u32(size), want_minimal_block); + // If we are creating a buffer for each allocation of there is no block associated with the pool or the current block is too small + // for this allocation, request a new buffer block + buffer_block = &buffer_pool.request_buffer_block(size, want_minimal_block); } - auto data = buffer_block->allocate(to_u32(size)); - - // Check if the buffer block can allocate the requested size - if (data.empty()) - { - buffer_block = &buffer_pool.request_buffer_block(to_u32(size), want_minimal_block); - - data = buffer_block->allocate(to_u32(size)); - } - - return data; + return buffer_block->allocate(to_u32(size)); } } // namespace vkb From 9a134552d64b2ebb16b9628ccae27f07e8c05290 Mon Sep 17 00:00:00 2001 From: Tom Atkinson Date: Mon, 28 Aug 2023 20:33:43 +0100 Subject: [PATCH 09/13] Validation Layer Fixes (#774) * Validation Layer Fixes * review fixes * remove USE_VALIDATION_LAYERS --- framework/core/hpp_instance.cpp | 24 ++++++++++++++++-------- framework/core/instance.cpp | 25 +++++++++++++++++-------- 2 files changed, 33 insertions(+), 16 deletions(-) diff --git a/framework/core/hpp_instance.cpp b/framework/core/hpp_instance.cpp index 7bc78db63..d798f9325 100644 --- a/framework/core/hpp_instance.cpp +++ b/framework/core/hpp_instance.cpp @@ -21,11 +21,19 @@ #include #include +#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +# define USE_VALIDATION_LAYERS +#endif + +#if defined(USE_VALIDATION_LAYERS) && (defined(VKB_VALIDATION_LAYERS_GPU_ASSISTED) || defined(VKB_VALIDATION_LAYERS_BEST_PRACTICES) || defined(VKB_VALIDATION_LAYERS_SYNCHRONIZATION)) +# define USE_VALIDATION_LAYER_FEATURES +#endif + namespace vkb { namespace { -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS VKAPI_ATTR VkBool32 VKAPI_CALL debug_utils_messenger_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity, VkDebugUtilsMessageTypeFlagsEXT message_type, const VkDebugUtilsMessengerCallbackDataEXT *callback_data, void *user_data) @@ -181,7 +189,7 @@ HPPInstance::HPPInstance(const std::string &applicati { std::vector available_instance_extensions = vk::enumerateInstanceExtensionProperties(); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS // Check if VK_EXT_debug_utils is supported, which supersedes VK_EXT_Debug_Report const bool has_debug_utils = enable_extension(VK_EXT_DEBUG_UTILS_EXTENSION_NAME, available_instance_extensions, enabled_extensions); @@ -204,7 +212,7 @@ HPPInstance::HPPInstance(const std::string &applicati enable_extension(VK_KHR_PORTABILITY_ENUMERATION_EXTENSION_NAME, available_instance_extensions, enabled_extensions); #endif -#if (defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS)) && (defined(VKB_VALIDATION_LAYERS_GPU_ASSISTED) || defined(VKB_VALIDATION_LAYERS_BEST_PRACTICES)) +#ifdef USE_VALIDATION_LAYER_FEATURES bool validation_features = false; { std::vector available_layer_instance_extensions = vk::enumerateInstanceExtensionProperties(std::string("VK_LAYER_KHRONOS_validation")); @@ -270,7 +278,7 @@ HPPInstance::HPPInstance(const std::string &applicati std::vector requested_validation_layers(required_validation_layers); -#ifdef VKB_VALIDATION_LAYERS +#ifdef USE_VALIDATION_LAYERS // Determine the optimal validation layers to enable that are necessary for useful debugging std::vector optimal_validation_layers = get_optimal_validation_layers(supported_validation_layers); requested_validation_layers.insert(requested_validation_layers.end(), optimal_validation_layers.begin(), optimal_validation_layers.end()); @@ -293,7 +301,7 @@ HPPInstance::HPPInstance(const std::string &applicati vk::InstanceCreateInfo instance_info({}, &app_info, requested_validation_layers, enabled_extensions); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS vk::DebugUtilsMessengerCreateInfoEXT debug_utils_create_info; vk::DebugReportCallbackCreateInfoEXT debug_report_create_info; if (has_debug_utils) @@ -319,7 +327,7 @@ HPPInstance::HPPInstance(const std::string &applicati instance_info.flags |= vk::InstanceCreateFlagBits::eEnumeratePortabilityKHR; #endif -#if (defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS)) && (defined(VKB_VALIDATION_LAYERS_GPU_ASSISTED) || defined(VKB_VALIDATION_LAYERS_BEST_PRACTICES)) +#ifdef USE_VALIDATION_LAYER_FEATURES vk::ValidationFeaturesEXT validation_features_info; std::vector enable_features{}; if (validation_features) @@ -346,7 +354,7 @@ HPPInstance::HPPInstance(const std::string &applicati // Need to load volk for all the not-yet Vulkan-Hpp calls volkLoadInstance(handle); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS if (has_debug_utils) { debug_utils_messenger = handle.createDebugUtilsMessengerEXT(debug_utils_create_info); @@ -375,7 +383,7 @@ HPPInstance::HPPInstance(vk::Instance instance) : HPPInstance::~HPPInstance() { -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS if (debug_utils_messenger) { handle.destroyDebugUtilsMessengerEXT(debug_utils_messenger); diff --git a/framework/core/instance.cpp b/framework/core/instance.cpp index 70e9465d9..884e17df1 100644 --- a/framework/core/instance.cpp +++ b/framework/core/instance.cpp @@ -22,11 +22,19 @@ #include #include +#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +# define USE_VALIDATION_LAYERS +#endif + +#if defined(USE_VALIDATION_LAYERS) && (defined(VKB_VALIDATION_LAYERS_GPU_ASSISTED) || defined(VKB_VALIDATION_LAYERS_BEST_PRACTICES) || defined(VKB_VALIDATION_LAYERS_SYNCHRONIZATION)) +# define USE_VALIDATION_LAYER_FEATURES +#endif + namespace vkb { namespace { -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS VKAPI_ATTR VkBool32 VKAPI_CALL debug_utils_messenger_callback(VkDebugUtilsMessageSeverityFlagBitsEXT message_severity, VkDebugUtilsMessageTypeFlagsEXT message_type, const VkDebugUtilsMessengerCallbackDataEXT *callback_data, @@ -185,7 +193,7 @@ Instance::Instance(const std::string &application_nam std::vector available_instance_extensions(instance_extension_count); VK_CHECK(vkEnumerateInstanceExtensionProperties(nullptr, &instance_extension_count, available_instance_extensions.data())); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS // Check if VK_EXT_debug_utils is supported, which supersedes VK_EXT_Debug_Report const bool has_debug_utils = enable_extension(VK_EXT_DEBUG_UTILS_EXTENSION_NAME, available_instance_extensions, enabled_extensions); @@ -208,7 +216,7 @@ Instance::Instance(const std::string &application_nam enable_extension(VK_KHR_PORTABILITY_ENUMERATION_EXTENSION_NAME, available_instance_extensions, enabled_extensions); #endif -#if (defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS)) +#ifdef USE_VALIDATION_LAYER_FEATURES bool validation_features = false; { uint32_t layer_instance_extension_count; @@ -282,7 +290,7 @@ Instance::Instance(const std::string &application_nam std::vector requested_validation_layers(required_validation_layers); -#ifdef VKB_VALIDATION_LAYERS +#ifdef USE_VALIDATION_LAYERS // Determine the optimal validation layers to enable that are necessary for useful debugging std::vector optimal_validation_layers = get_optimal_validation_layers(supported_validation_layers); requested_validation_layers.insert(requested_validation_layers.end(), optimal_validation_layers.begin(), optimal_validation_layers.end()); @@ -319,7 +327,7 @@ Instance::Instance(const std::string &application_nam instance_info.enabledLayerCount = to_u32(requested_validation_layers.size()); instance_info.ppEnabledLayerNames = requested_validation_layers.data(); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS VkDebugUtilsMessengerCreateInfoEXT debug_utils_create_info = {VK_STRUCTURE_TYPE_DEBUG_UTILS_MESSENGER_CREATE_INFO_EXT}; VkDebugReportCallbackCreateInfoEXT debug_report_create_info = {VK_STRUCTURE_TYPE_DEBUG_REPORT_CREATE_INFO_EXT}; if (has_debug_utils) @@ -338,12 +346,13 @@ Instance::Instance(const std::string &application_nam instance_info.pNext = &debug_report_create_info; } #endif + #if (defined(VKB_ENABLE_PORTABILITY)) instance_info.flags |= VK_INSTANCE_CREATE_ENUMERATE_PORTABILITY_BIT_KHR; #endif // Some of the specialized layers need to be enabled explicitly -#if (defined(VKB_VALIDATION_LAYERS_GPU_ASSISTED) || defined(VKB_VALIDATION_LAYERS_BEST_PRACTICES) || defined(VKB_VALIDATION_LAYERS_SYNCHRONIZATION)) +#ifdef USE_VALIDATION_LAYER_FEATURES VkValidationFeaturesEXT validation_features_info = {VK_STRUCTURE_TYPE_VALIDATION_FEATURES_EXT}; std::vector enable_features{}; if (validation_features) @@ -375,7 +384,7 @@ Instance::Instance(const std::string &application_nam volkLoadInstance(handle); -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS if (has_debug_utils) { result = vkCreateDebugUtilsMessengerEXT(handle, &debug_utils_create_info, nullptr, &debug_utils_messenger); @@ -412,7 +421,7 @@ Instance::Instance(VkInstance instance) : Instance::~Instance() { -#if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) +#ifdef USE_VALIDATION_LAYERS if (debug_utils_messenger != VK_NULL_HANDLE) { vkDestroyDebugUtilsMessengerEXT(handle, debug_utils_messenger, nullptr); From 5eb9b181bd1e07b7513a5277db4a399838d43824 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andreas=20S=C3=BC=C3=9Fenbach?= Date: Mon, 11 Sep 2023 17:06:16 +0200 Subject: [PATCH 10/13] Refactored API sample hpp_texture_loading. (#793) --- .../hpp_compute_nbody/hpp_compute_nbody.cpp | 35 +- .../hpp_dynamic_uniform_buffers.cpp | 54 +- samples/api/hpp_hdr/hpp_hdr.cpp | 41 +- .../hpp_hello_triangle/hpp_hello_triangle.cpp | 42 +- .../api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp | 60 +- samples/api/hpp_instancing/hpp_instancing.cpp | 58 +- .../hpp_separate_image_sampler.cpp | 66 +- .../hpp_terrain_tessellation.cpp | 39 +- .../hpp_texture_loading.cpp | 628 ++++++++---------- .../hpp_texture_loading/hpp_texture_loading.h | 97 +-- third_party/vulkan | 2 +- 11 files changed, 543 insertions(+), 579 deletions(-) diff --git a/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp b/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp index f758c51d9..e2a16d102 100644 --- a/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp +++ b/samples/api/hpp_compute_nbody/hpp_compute_nbody.cpp @@ -44,20 +44,20 @@ HPPComputeNBody::~HPPComputeNBody() bool HPPComputeNBody::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) - { - return false; - } + assert(!prepared); - load_assets(); + if (HPPApiVulkanSample::prepare(options)) + { + load_assets(); + descriptor_pool = create_descriptor_pool(); + prepare_graphics(); + prepare_compute(); + build_command_buffers(); - descriptor_pool = create_descriptor_pool(); + prepared = true; + } - prepare_graphics(); - prepare_compute(); - build_command_buffers(); - prepared = true; - return true; + return prepared; } bool HPPComputeNBody::resize(const uint32_t width, const uint32_t height) @@ -145,13 +145,14 @@ void HPPComputeNBody::build_command_buffers() void HPPComputeNBody::render(float delta_time) { - if (!prepared) - return; - draw(); - update_compute_uniform_buffers(delta_time); - if (camera.updated) + if (prepared) { - update_graphics_uniform_buffers(); + draw(); + update_compute_uniform_buffers(delta_time); + if (camera.updated) + { + update_graphics_uniform_buffers(); + } } } diff --git a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp index 4c0d8b9eb..fd2d6f4e1 100644 --- a/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp +++ b/samples/api/hpp_dynamic_uniform_buffers/hpp_dynamic_uniform_buffers.cpp @@ -83,26 +83,25 @@ void HPPDynamicUniformBuffers::aligned_free(void *data) bool HPPDynamicUniformBuffers::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) { - return false; + prepare_camera(); + generate_cube(); + prepare_uniform_buffers(); + descriptor_set_layout = create_descriptor_set_layout(); + pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, descriptor_set_layout}); + pipeline = create_pipeline(); + descriptor_pool = create_descriptor_pool(); + descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, descriptor_set_layout); + update_descriptor_set(); + build_command_buffers(); + + prepared = true; } - prepare_camera(); - generate_cube(); - prepare_uniform_buffers(); - - descriptor_set_layout = create_descriptor_set_layout(); - pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, descriptor_set_layout}); - pipeline = create_pipeline(); - - descriptor_pool = create_descriptor_pool(); - descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, descriptor_set_layout); - - update_descriptor_set(); - build_command_buffers(); - prepared = true; - return true; + return prepared; } bool HPPDynamicUniformBuffers::resize(const uint32_t width, const uint32_t height) @@ -152,18 +151,17 @@ void HPPDynamicUniformBuffers::build_command_buffers() void HPPDynamicUniformBuffers::render(float delta_time) { - if (!prepared) - { - return; - } - draw(); - if (!paused) - { - update_dynamic_uniform_buffer(delta_time); - } - if (camera.updated) + if (prepared) { - update_uniform_buffers(); + draw(); + if (!paused) + { + update_dynamic_uniform_buffer(delta_time); + } + if (camera.updated) + { + update_uniform_buffers(); + } } } diff --git a/samples/api/hpp_hdr/hpp_hdr.cpp b/samples/api/hpp_hdr/hpp_hdr.cpp index cbaed173e..d741aad85 100644 --- a/samples/api/hpp_hdr/hpp_hdr.cpp +++ b/samples/api/hpp_hdr/hpp_hdr.cpp @@ -43,23 +43,23 @@ HPPHDR::~HPPHDR() bool HPPHDR::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + if (HPPApiVulkanSample::prepare(options)) { - return false; + prepare_camera(); + load_assets(); + prepare_uniform_buffers(); + prepare_offscreen_buffer(); + descriptor_pool = create_descriptor_pool(); + setup_bloom(); + setup_composition(); + setup_models(); + build_command_buffers(); + + prepared = true; } - prepare_camera(); - load_assets(); - prepare_uniform_buffers(); - prepare_offscreen_buffer(); - - descriptor_pool = create_descriptor_pool(); - setup_bloom(); - setup_composition(); - setup_models(); - build_command_buffers(); - prepared = true; - return true; + return prepared; } bool HPPHDR::resize(const uint32_t width, const uint32_t height) @@ -210,14 +210,13 @@ void HPPHDR::on_update_ui_overlay(vkb::HPPDrawer &drawer) void HPPHDR::render(float delta_time) { - if (!prepared) - { - return; - } - draw(); - if (camera.updated) + if (prepared) { - update_uniform_buffers(); + draw(); + if (camera.updated) + { + update_uniform_buffers(); + } } } diff --git a/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp b/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp index 541e53171..0d8c73042 100644 --- a/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp +++ b/samples/api/hpp_hello_triangle/hpp_hello_triangle.cpp @@ -195,40 +195,38 @@ HPPHelloTriangle::~HPPHelloTriangle() bool HPPHelloTriangle::prepare(const vkb::ApplicationOptions &options) { - if (!Application::prepare(options)) + if (Application::prepare(options)) { - return false; - } - - instance = create_instance({VK_KHR_SURFACE_EXTENSION_NAME}, {}); + instance = create_instance({VK_KHR_SURFACE_EXTENSION_NAME}, {}); #if defined(VKB_DEBUG) || defined(VKB_VALIDATION_LAYERS) - debug_utils_messenger = instance.createDebugUtilsMessengerEXT(debug_utils_create_info); + debug_utils_messenger = instance.createDebugUtilsMessengerEXT(debug_utils_create_info); #endif - select_physical_device_and_surface(); + select_physical_device_and_surface(); - const vkb::Window::Extent &extent = options.window->get_extent(); - swapchain_data.extent.width = extent.width; - swapchain_data.extent.height = extent.height; + const vkb::Window::Extent &extent = options.window->get_extent(); + swapchain_data.extent.width = extent.width; + swapchain_data.extent.height = extent.height; - // create a device - device = create_device({VK_KHR_SWAPCHAIN_EXTENSION_NAME}); + // create a device + device = create_device({VK_KHR_SWAPCHAIN_EXTENSION_NAME}); - // get the (graphics) queue - queue = device.getQueue(graphics_queue_index, 0); + // get the (graphics) queue + queue = device.getQueue(graphics_queue_index, 0); - init_swapchain(); + init_swapchain(); - // Create the necessary objects for rendering. - render_pass = create_render_pass(); + // Create the necessary objects for rendering. + render_pass = create_render_pass(); - // Create a blank pipeline layout. - // We are not binding any resources to the pipeline in this first sample. - pipeline_layout = device.createPipelineLayout({}); + // Create a blank pipeline layout. + // We are not binding any resources to the pipeline in this first sample. + pipeline_layout = device.createPipelineLayout({}); - pipeline = create_graphics_pipeline(); + pipeline = create_graphics_pipeline(); - init_framebuffers(); + init_framebuffers(); + } return true; } diff --git a/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp b/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp index f193dfc5d..bbab62712 100644 --- a/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp +++ b/samples/api/hpp_hlsl_shaders/hpp_hlsl_shaders.cpp @@ -53,36 +53,37 @@ HPPHlslShaders::~HPPHlslShaders() bool HPPHlslShaders::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) { - return false; + load_assets(); + generate_quad(); + + // Vertex shader uniform buffer block + uniform_buffer_vs = std::make_unique(*get_device(), + sizeof(ubo_vs), + vk::BufferUsageFlagBits::eUniformBuffer, + VMA_MEMORY_USAGE_CPU_TO_GPU); + update_uniform_buffers(); + + // We separate the descriptor sets for the uniform buffer + image and samplers, so we don't need to duplicate the descriptors for the former + base_descriptor_set_layout = create_base_descriptor_set_layout(); + sampler_descriptor_set_layout = create_sampler_descriptor_set_layout(); + + pipeline_layout = create_pipeline_layout(); + shader_modules = {create_shader_module("hlsl_shaders/hlsl_shader.vert", vk::ShaderStageFlagBits::eVertex), + create_shader_module("hlsl_shaders/hlsl_shader.frag", vk::ShaderStageFlagBits::eFragment)}; + pipeline = create_pipeline(); + descriptor_pool = create_descriptor_pool(); + base_descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, base_descriptor_set_layout); + update_descriptor_sets(); + build_command_buffers(); + + prepared = true; } - load_assets(); - generate_quad(); - - // Vertex shader uniform buffer block - uniform_buffer_vs = std::make_unique(*get_device(), - sizeof(ubo_vs), - vk::BufferUsageFlagBits::eUniformBuffer, - VMA_MEMORY_USAGE_CPU_TO_GPU); - update_uniform_buffers(); - - // We separate the descriptor sets for the uniform buffer + image and samplers, so we don't need to duplicate the descriptors for the former - base_descriptor_set_layout = create_base_descriptor_set_layout(); - sampler_descriptor_set_layout = create_sampler_descriptor_set_layout(); - - pipeline_layout = create_pipeline_layout(); - shader_modules.push_back(create_shader_module("hlsl_shaders/hlsl_shader.vert", vk::ShaderStageFlagBits::eVertex)); - shader_modules.push_back(create_shader_module("hlsl_shaders/hlsl_shader.frag", vk::ShaderStageFlagBits::eFragment)); - pipeline = create_pipeline(); - - descriptor_pool = create_descriptor_pool(); - base_descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, base_descriptor_set_layout); - update_descriptor_sets(); - build_command_buffers(); - prepared = true; - return true; + return prepared; } // Enable physical device features required for this example @@ -139,11 +140,10 @@ void HPPHlslShaders::build_command_buffers() void HPPHlslShaders::render(float delta_time) { - if (!prepared) + if (prepared) { - return; + draw(); } - draw(); } void HPPHlslShaders::view_changed() diff --git a/samples/api/hpp_instancing/hpp_instancing.cpp b/samples/api/hpp_instancing/hpp_instancing.cpp index 7153bc17e..8dffef914 100644 --- a/samples/api/hpp_instancing/hpp_instancing.cpp +++ b/samples/api/hpp_instancing/hpp_instancing.cpp @@ -46,38 +46,38 @@ HPPInstancing::~HPPInstancing() bool HPPInstancing::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) { - return false; + initialize_camera(); + load_assets(); + prepare_instance_data(); + prepare_uniform_buffers(); + vk::Device device = get_device()->get_handle(); + descriptor_set_layout = create_descriptor_set_layout(); + pipeline_layout = device.createPipelineLayout({{}, descriptor_set_layout}); + descriptor_pool = create_descriptor_pool(); + + // setup planet + planet.pipeline = create_planet_pipeline(); + planet.descriptor_set = vkb::common::allocate_descriptor_set(device, descriptor_pool, descriptor_set_layout); + update_planet_descriptor_set(); + + // setup rocks + rocks.pipeline = create_rocks_pipeline(); + rocks.descriptor_set = vkb::common::allocate_descriptor_set(device, descriptor_pool, descriptor_set_layout); + update_rocks_descriptor_set(); + + // setup starfield + starfield_pipeline = create_starfield_pipeline(); + + build_command_buffers(); + + prepared = true; } - initialize_camera(); - load_assets(); - prepare_instance_data(); - prepare_uniform_buffers(); - - vk::Device device = get_device()->get_handle(); - - descriptor_set_layout = create_descriptor_set_layout(); - pipeline_layout = device.createPipelineLayout({{}, descriptor_set_layout}); - descriptor_pool = create_descriptor_pool(); - - // setup planet - planet.pipeline = create_planet_pipeline(); - planet.descriptor_set = vkb::common::allocate_descriptor_set(device, descriptor_pool, descriptor_set_layout); - update_planet_descriptor_set(); - - // setup rocks - rocks.pipeline = create_rocks_pipeline(); - rocks.descriptor_set = vkb::common::allocate_descriptor_set(device, descriptor_pool, descriptor_set_layout); - update_rocks_descriptor_set(); - - // setup starfield - starfield_pipeline = create_starfield_pipeline(); - - build_command_buffers(); - prepared = true; - return true; + return prepared; } bool HPPInstancing::resize(const uint32_t width, const uint32_t height) diff --git a/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp b/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp index cee233cf8..0ff5654e5 100644 --- a/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp +++ b/samples/api/hpp_separate_image_sampler/hpp_separate_image_sampler.cpp @@ -53,47 +53,44 @@ HPPSeparateImageSampler::~HPPSeparateImageSampler() bool HPPSeparateImageSampler::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) { - return false; - } + load_assets(); + generate_quad(); + prepare_uniform_buffers(); - load_assets(); - generate_quad(); - prepare_uniform_buffers(); + // Create two samplers with different options, first one with linear filtering, the second one with nearest filtering + samplers = {{create_sampler(vk::Filter::eLinear), create_sampler(vk::Filter::eNearest)}}; - // Create two samplers with different options - // First sampler with linear filtering - samplers[0] = create_sampler(vk::Filter::eLinear); - // Second sampler with nearest filtering - samplers[1] = create_sampler(vk::Filter::eNearest); + descriptor_pool = create_descriptor_pool(); - descriptor_pool = create_descriptor_pool(); + // We separate the descriptor sets for the uniform buffer + image and samplers, so we don't need to duplicate the descriptors for the former + // Descriptors set for the uniform buffer and the image + base_descriptor_set_layout = create_base_descriptor_set_layout(); + base_descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, base_descriptor_set_layout); + update_base_descriptor_set(); - vk::Device device = get_device()->get_handle(); + // Sets for each of the sampler + sampler_descriptor_set_layout = create_sampler_descriptor_set_layout(); + for (size_t i = 0; i < sampler_descriptor_sets.size(); i++) + { + sampler_descriptor_sets[i] = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, sampler_descriptor_set_layout); + update_sampler_descriptor_set(i); + } - // We separate the descriptor sets for the uniform buffer + image and samplers, so we don't need to duplicate the descriptors for the former - // Descriptors set for the uniform buffer and the image - base_descriptor_set_layout = create_base_descriptor_set_layout(); - base_descriptor_set = vkb::common::allocate_descriptor_set(device, descriptor_pool, base_descriptor_set_layout); - update_base_descriptor_set(); + // Pipeline layout + // Set layout for the base descriptors in set 0 and set layout for the sampler descriptors in set 1 + pipeline_layout = create_pipeline_layout({base_descriptor_set_layout, sampler_descriptor_set_layout}); + pipeline = create_graphics_pipeline(); - // Sets for each of the sampler - sampler_descriptor_set_layout = create_sampler_descriptor_set_layout(); - for (size_t i = 0; i < sampler_descriptor_sets.size(); i++) - { - sampler_descriptor_sets[i] = vkb::common::allocate_descriptor_set(device, descriptor_pool, sampler_descriptor_set_layout); - update_sampler_descriptor_set(i); - } + build_command_buffers(); - // Pipeline layout - // Set layout for the base descriptors in set 0 and set layout for the sampler descriptors in set 1 - pipeline_layout = create_pipeline_layout({base_descriptor_set_layout, sampler_descriptor_set_layout}); - pipeline = create_graphics_pipeline(); + prepared = true; + } - build_command_buffers(); - prepared = true; - return true; + return prepared; } // Enable physical device features required for this example @@ -163,11 +160,10 @@ void HPPSeparateImageSampler::on_update_ui_overlay(vkb::HPPDrawer &drawer) void HPPSeparateImageSampler::render(float delta_time) { - if (!prepared) + if (prepared) { - return; + draw(); } - draw(); } void HPPSeparateImageSampler::view_changed() diff --git a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp index 69c9a76a6..991d030b1 100644 --- a/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp +++ b/samples/api/hpp_terrain_tessellation/hpp_terrain_tessellation.cpp @@ -46,27 +46,25 @@ HPPTerrainTessellation::~HPPTerrainTessellation() bool HPPTerrainTessellation::prepare(const vkb::ApplicationOptions &options) { - if (!HPPApiVulkanSample::prepare(options)) + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) { - return false; + prepare_camera(); + load_assets(); + generate_terrain(); + prepare_uniform_buffers(); + descriptor_pool = create_descriptor_pool(); + prepare_sky_sphere(); + prepare_terrain(); + prepare_wireframe(); + prepare_statistics(); + build_command_buffers(); + + prepared = true; } - prepare_camera(); - load_assets(); - generate_terrain(); - prepare_uniform_buffers(); - - descriptor_pool = create_descriptor_pool(); - - prepare_sky_sphere(); - prepare_terrain(); - prepare_wireframe(); - - prepare_statistics(); - - build_command_buffers(); - prepared = true; - return true; + return prepared; } void HPPTerrainTessellation::request_gpu_features(vkb::core::HPPPhysicalDevice &gpu) @@ -188,11 +186,10 @@ void HPPTerrainTessellation::on_update_ui_overlay(vkb::HPPDrawer &drawer) void HPPTerrainTessellation::render(float delta_time) { - if (!prepared) + if (prepared) { - return; + draw(); } - draw(); } void HPPTerrainTessellation::view_changed() diff --git a/samples/api/hpp_texture_loading/hpp_texture_loading.cpp b/samples/api/hpp_texture_loading/hpp_texture_loading.cpp index 5123d4e3f..af07a64cb 100644 --- a/samples/api/hpp_texture_loading/hpp_texture_loading.cpp +++ b/samples/api/hpp_texture_loading/hpp_texture_loading.cpp @@ -21,11 +21,14 @@ #include +#include + HPPTextureLoading::HPPTextureLoading() { + title = "HPP Texture loading"; + zoom = -2.5f; rotation = {0.0f, 15.0f, 0.0f}; - title = "HPP Texture loading"; } HPPTextureLoading::~HPPTextureLoading() @@ -36,18 +39,38 @@ HPPTextureLoading::~HPPTextureLoading() // Clean up used Vulkan resources // Note : Inherited destructor cleans up resources stored in base class - device.destroyPipeline(pipeline); - device.destroyPipelineLayout(pipeline_layout); device.destroyDescriptorSetLayout(descriptor_set_layout); + texture.destroy(device); } - destroy_texture(texture); - vertex_buffer.reset(); index_buffer.reset(); - uniform_buffer_vs.reset(); + vertex_shader_data_buffer.reset(); +} + +bool HPPTextureLoading::prepare(const vkb::ApplicationOptions &options) +{ + assert(!prepared); + + if (HPPApiVulkanSample::prepare(options)) + { + load_texture(); + generate_quad(); + prepare_uniform_buffers(); + descriptor_set_layout = create_descriptor_set_layout(); + pipeline_layout = get_device()->get_handle().createPipelineLayout({{}, descriptor_set_layout}); + pipeline = create_pipeline(); + descriptor_pool = create_descriptor_pool(); + descriptor_set = vkb::common::allocate_descriptor_set(get_device()->get_handle(), descriptor_pool, {descriptor_set_layout}); + update_descriptor_set(); + build_command_buffers(); + + prepared = true; + } + + return prepared; } // Enable physical device features required for this example @@ -60,6 +83,178 @@ void HPPTextureLoading::request_gpu_features(vkb::core::HPPPhysicalDevice &gpu) } } +void HPPTextureLoading::build_command_buffers() +{ + vk::CommandBufferBeginInfo command_buffer_begin_info; + + vk::ClearValue clear_values[2]; + clear_values[0].color = default_clear_color; + clear_values[1].depthStencil = vk::ClearDepthStencilValue(0.0f, 0); + + vk::RenderPassBeginInfo render_pass_begin_info; + render_pass_begin_info.renderPass = render_pass; + render_pass_begin_info.renderArea.offset.x = 0; + render_pass_begin_info.renderArea.offset.y = 0; + render_pass_begin_info.renderArea.extent = extent; + render_pass_begin_info.clearValueCount = 2; + render_pass_begin_info.pClearValues = clear_values; + + for (int32_t i = 0; i < draw_cmd_buffers.size(); ++i) + { + auto command_buffer = draw_cmd_buffers[i]; + + // Set target frame buffer + render_pass_begin_info.framebuffer = framebuffers[i]; + + command_buffer.begin(command_buffer_begin_info); + + command_buffer.beginRenderPass(render_pass_begin_info, vk::SubpassContents::eInline); + + vk::Viewport viewport(0.0f, 0.0f, static_cast(extent.width), static_cast(extent.height), 0.0f, 1.0f); + command_buffer.setViewport(0, viewport); + + vk::Rect2D scissor({0, 0}, extent); + command_buffer.setScissor(0, scissor); + + command_buffer.bindDescriptorSets(vk::PipelineBindPoint::eGraphics, pipeline_layout, 0, descriptor_set, {}); + command_buffer.bindPipeline(vk::PipelineBindPoint::eGraphics, pipeline); + + vk::DeviceSize offset = 0; + command_buffer.bindVertexBuffers(0, vertex_buffer->get_handle(), offset); + command_buffer.bindIndexBuffer(index_buffer->get_handle(), 0, vk::IndexType::eUint32); + + command_buffer.drawIndexed(index_count, 1, 0, 0, 0); + + draw_ui(command_buffer); + + command_buffer.endRenderPass(); + + command_buffer.end(); + } +} + +void HPPTextureLoading::on_update_ui_overlay(vkb::HPPDrawer &drawer) +{ + if (drawer.header("Settings")) + { + if (drawer.slider_float("LOD bias", &vertex_shader_data.lod_bias, 0.0f, static_cast(texture.mip_levels))) + { + update_uniform_buffers(); + } + } +} + +void HPPTextureLoading::render(float delta_time) +{ + if (prepared) + { + draw(); + } +} + +void HPPTextureLoading::view_changed() +{ + update_uniform_buffers(); +} + +vk::DescriptorPool HPPTextureLoading::create_descriptor_pool() +{ + // Example uses one ubo and one image sampler + std::array pool_sizes = {{{vk::DescriptorType::eUniformBuffer, 1}, {vk::DescriptorType::eCombinedImageSampler, 1}}}; + + return get_device()->get_handle().createDescriptorPool({{}, 2, pool_sizes}); +} + +vk::DescriptorSetLayout HPPTextureLoading::create_descriptor_set_layout() +{ + std::array set_layout_bindings = { + {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eVertex}, + {1, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; + + return get_device()->get_handle().createDescriptorSetLayout({{}, set_layout_bindings}); +} + +vk::Pipeline HPPTextureLoading::create_pipeline() +{ + // Load shaders + std::vector shader_stages = {{load_shader("texture_loading/texture.vert", vk::ShaderStageFlagBits::eVertex), + load_shader("texture_loading/texture.frag", vk::ShaderStageFlagBits::eFragment)}}; + + // Vertex bindings and attributes + vk::VertexInputBindingDescription vertex_input_binding(0, sizeof(Vertex), vk::VertexInputRate::eVertex); + std::array vertex_input_attributes = { + {{0, 0, vk::Format::eR32G32B32Sfloat, offsetof(Vertex, pos)}, // Location 0 : Position + {1, 0, vk::Format::eR32G32Sfloat, offsetof(Vertex, uv)}, // Location 1: Texture Coordinates + {2, 0, vk::Format::eR32G32B32Sfloat, offsetof(Vertex, normal)}}}; // Location 2 : Normal + vk::PipelineVertexInputStateCreateInfo vertex_input_state({}, vertex_input_binding, vertex_input_attributes); + + vk::PipelineColorBlendAttachmentState blend_attachment_state; + blend_attachment_state.colorWriteMask = + vk::ColorComponentFlagBits::eR | vk::ColorComponentFlagBits::eG | vk::ColorComponentFlagBits::eB | vk::ColorComponentFlagBits::eA; + + // Note: Using reversed depth-buffer for increased precision, so Greater depth values are kept + vk::PipelineDepthStencilStateCreateInfo depth_stencil_state; + depth_stencil_state.depthTestEnable = true; + depth_stencil_state.depthWriteEnable = true; + depth_stencil_state.depthCompareOp = vk::CompareOp::eGreater; + depth_stencil_state.back.compareOp = vk::CompareOp::eGreater; + + return vkb::common::create_graphics_pipeline(get_device()->get_handle(), + pipeline_cache, + shader_stages, + vertex_input_state, + vk::PrimitiveTopology::eTriangleList, + 0, + vk::PolygonMode::eFill, + vk::CullModeFlagBits::eNone, + vk::FrontFace::eCounterClockwise, + {blend_attachment_state}, + depth_stencil_state, + pipeline_layout, + render_pass); +} + +void HPPTextureLoading::draw() +{ + HPPApiVulkanSample::prepare_frame(); + + // Command buffer to be submitted to the queue + submit_info.setCommandBuffers(draw_cmd_buffers[current_buffer]); + + // Submit to queue + queue.submit(submit_info); + + HPPApiVulkanSample::submit_frame(); +} + +void HPPTextureLoading::generate_quad() +{ + // Setup vertices for a single uv-mapped quad made from two triangles + std::vector vertices = {{{1.0f, 1.0f, 0.0f}, {1.0f, 1.0f}, {0.0f, 0.0f, 1.0f}}, + {{-1.0f, 1.0f, 0.0f}, {0.0f, 1.0f}, {0.0f, 0.0f, 1.0f}}, + {{-1.0f, -1.0f, 0.0f}, {0.0f, 0.0f}, {0.0f, 0.0f, 1.0f}}, + {{1.0f, -1.0f, 0.0f}, {1.0f, 0.0f}, {0.0f, 0.0f, 1.0f}}}; + + // Setup indices + std::vector indices = {0, 1, 2, 2, 3, 0}; + index_count = static_cast(indices.size()); + + auto vertex_buffer_size = vkb::to_u32(vertices.size() * sizeof(Vertex)); + auto index_buffer_size = vkb::to_u32(indices.size() * sizeof(uint32_t)); + + // Create buffers + // For the sake of simplicity we won't stage the vertex data to the gpu memory + // Vertex buffer + vertex_buffer = std::make_unique( + *get_device(), vertex_buffer_size, vk::BufferUsageFlagBits::eTransferDst | vk::BufferUsageFlagBits::eVertexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); + vertex_buffer->update(vertices.data(), vertex_buffer_size); + + // Index buffer + index_buffer = std::make_unique( + *get_device(), index_buffer_size, vk::BufferUsageFlagBits::eTransferDst | vk::BufferUsageFlagBits::eIndexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); + index_buffer->update(indices.data(), index_buffer_size); +} + /* Upload texture image data to the GPU @@ -108,6 +303,8 @@ void HPPTextureLoading::load_texture() use_staging = !(format_properties.linearTilingFeatures & vk::FormatFeatureFlagBits::eSampledImage); } + vk::Device device = get_device()->get_handle(); + if (use_staging) { // Copy data to an optimal tiled image @@ -117,22 +314,23 @@ void HPPTextureLoading::load_texture() // This buffer will be the data source for copying texture data to the optimal tiled image on the device // This buffer is used as a transfer source for the buffer copy vk::BufferCreateInfo buffer_create_info({}, ktx_texture->dataSize, vk::BufferUsageFlagBits::eTransferSrc, vk::SharingMode::eExclusive, {}); - vk::Buffer staging_buffer = get_device()->get_handle().createBuffer(buffer_create_info); + vk::Buffer staging_buffer = device.createBuffer(buffer_create_info); // Get memory requirements for the staging buffer (alignment, memory type bits) - vk::MemoryRequirements memory_requirements = get_device()->get_handle().getBufferMemoryRequirements(staging_buffer); - vk::MemoryAllocateInfo memory_allocate_info( - memory_requirements.size, - // Get memory type index for a host visible buffer - get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, vk::MemoryPropertyFlagBits::eHostVisible | vk::MemoryPropertyFlagBits::eHostCoherent)); - vk::DeviceMemory staging_memory = get_device()->get_handle().allocateMemory(memory_allocate_info); - get_device()->get_handle().bindBufferMemory(staging_buffer, staging_memory, 0); + vk::MemoryRequirements memory_requirements = device.getBufferMemoryRequirements(staging_buffer); - // Copy texture data into host local staging buffer + // Get memory type index for a host visible buffer + uint32_t memory_type = get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, + vk::MemoryPropertyFlagBits::eHostVisible | vk::MemoryPropertyFlagBits::eHostCoherent); + + vk::MemoryAllocateInfo memory_allocate_info(memory_requirements.size, memory_type); + vk::DeviceMemory staging_memory = device.allocateMemory(memory_allocate_info); + device.bindBufferMemory(staging_buffer, staging_memory, 0); - uint8_t *data = reinterpret_cast(get_device()->get_handle().mapMemory(staging_memory, 0, memory_requirements.size)); + // Copy texture data into host local staging buffer + uint8_t *data = reinterpret_cast(device.mapMemory(staging_memory, 0, memory_requirements.size)); memcpy(data, ktx_texture->pData, ktx_texture->dataSize); - get_device()->get_handle().unmapMemory(staging_memory); + device.unmapMemory(staging_memory); // Setup buffer copy regions for each mip level std::vector buffer_copy_regions(texture.mip_levels); @@ -153,39 +351,35 @@ void HPPTextureLoading::load_texture() // Create optimal tiled target image on the device vk::ImageCreateInfo image_create_info; - image_create_info.imageType = vk::ImageType::e2D; - image_create_info.format = format; - image_create_info.mipLevels = texture.mip_levels; - image_create_info.arrayLayers = 1; - image_create_info.samples = vk::SampleCountFlagBits::e1; - image_create_info.tiling = vk::ImageTiling::eOptimal; - image_create_info.sharingMode = vk::SharingMode::eExclusive; - // Set initial layout of the image to undefined - image_create_info.initialLayout = vk::ImageLayout::eUndefined; + image_create_info.imageType = vk::ImageType::e2D; + image_create_info.format = format; + image_create_info.mipLevels = texture.mip_levels; + image_create_info.arrayLayers = 1; + image_create_info.samples = vk::SampleCountFlagBits::e1; + image_create_info.tiling = vk::ImageTiling::eOptimal; + image_create_info.sharingMode = vk::SharingMode::eExclusive; + image_create_info.initialLayout = vk::ImageLayout::eUndefined; // Set initial layout of the image to undefined image_create_info.extent = vk::Extent3D(texture.extent, 1); image_create_info.usage = vk::ImageUsageFlagBits::eTransferDst | vk::ImageUsageFlagBits::eSampled; - texture.image = get_device()->get_handle().createImage(image_create_info); + texture.image = device.createImage(image_create_info); - memory_requirements = get_device()->get_handle().getImageMemoryRequirements(texture.image); - memory_allocate_info = {memory_requirements.size, - get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, vk::MemoryPropertyFlagBits::eDeviceLocal)}; - texture.device_memory = get_device()->get_handle().allocateMemory(memory_allocate_info); - get_device()->get_handle().bindImageMemory(texture.image, texture.device_memory, 0); + memory_requirements = device.getImageMemoryRequirements(texture.image); + memory_type = get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, vk::MemoryPropertyFlagBits::eDeviceLocal); + memory_allocate_info = {memory_requirements.size, memory_type}; + texture.device_memory = device.allocateMemory(memory_allocate_info); + device.bindImageMemory(texture.image, texture.device_memory, 0); - vk::CommandBuffer copy_command = get_device()->create_command_buffer(vk::CommandBufferLevel::ePrimary, true); + vk::CommandBuffer copy_command = vkb::common::allocate_command_buffer(device, get_device()->get_command_pool().get_handle()); + copy_command.begin(vk::CommandBufferBeginInfo()); // Image memory barriers for the texture image // The sub resource range describes the regions of the image that will be transitioned using the memory barriers below vk::ImageSubresourceRange subresource_range; - // Image only contains color data - subresource_range.aspectMask = vk::ImageAspectFlagBits::eColor; - // Start at first mip level - subresource_range.baseMipLevel = 0; - // We will transition on all mip levels - subresource_range.levelCount = texture.mip_levels; - // The 2D texture only has one layer - subresource_range.layerCount = 1; + subresource_range.aspectMask = vk::ImageAspectFlagBits::eColor; // Image contains only color data + subresource_range.baseMipLevel = 0; // Start at first mip level + subresource_range.levelCount = texture.mip_levels; // We will transition on all mip levels + subresource_range.layerCount = 1; // The 2D texture only has one layer // Transition the texture image layout to transfer target, so we can safely copy our buffer data to it. vk::ImageMemoryBarrier image_memory_barrier; @@ -223,8 +417,8 @@ void HPPTextureLoading::load_texture() get_device()->flush_command_buffer(copy_command, queue, true); // Clean up staging resources - get_device()->get_handle().destroyBuffer(staging_buffer); - get_device()->get_handle().freeMemory(staging_memory); + device.destroyBuffer(staging_buffer); + device.freeMemory(staging_memory); } else { @@ -242,23 +436,25 @@ void HPPTextureLoading::load_texture() image_create_info.sharingMode = vk::SharingMode::eExclusive; image_create_info.initialLayout = vk::ImageLayout::ePreinitialized; image_create_info.extent = vk::Extent3D(texture.extent, 1); - vk::Image mappable_image = get_device()->get_handle().createImage(image_create_info); + vk::Image mappable_image = device.createImage(image_create_info); // Get memory requirements for this image like size and alignment - vk::MemoryRequirements memory_requirements = get_device()->get_handle().getImageMemoryRequirements(mappable_image); + vk::MemoryRequirements memory_requirements = device.getImageMemoryRequirements(mappable_image); + + // Get memory type that can be mapped to host memory + uint32_t memory_type = get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, + vk::MemoryPropertyFlagBits::eHostVisible | vk::MemoryPropertyFlagBits::eHostCoherent); + // Set memory allocation size to required memory size - vk::MemoryAllocateInfo memory_allocate_info( - memory_requirements.size, - // Get memory type that can be mapped to host memory - get_device()->get_gpu().get_memory_type(memory_requirements.memoryTypeBits, vk::MemoryPropertyFlagBits::eHostVisible | vk::MemoryPropertyFlagBits::eHostCoherent)); - vk::DeviceMemory mappable_memory = get_device()->get_handle().allocateMemory(memory_allocate_info); - get_device()->get_handle().bindImageMemory(mappable_image, mappable_memory, 0); + vk::MemoryAllocateInfo memory_allocate_info(memory_requirements.size, memory_type); + vk::DeviceMemory mappable_memory = device.allocateMemory(memory_allocate_info); + device.bindImageMemory(mappable_image, mappable_memory, 0); // Map image memory - void *data = get_device()->get_handle().mapMemory(mappable_memory, 0, memory_requirements.size); + void *data = device.mapMemory(mappable_memory, 0, memory_requirements.size); // Copy image data of the first mip level into memory memcpy(data, ktx_texture->pData, ktxTexture_GetImageSize(ktx_texture, 0)); - get_device()->get_handle().unmapMemory(mappable_memory); + device.unmapMemory(mappable_memory); // Linear tiled images don't need to be staged and can be directly used as textures texture.image = mappable_image; @@ -266,7 +462,8 @@ void HPPTextureLoading::load_texture() texture.image_layout = vk::ImageLayout::eShaderReadOnlyOptimal; // Setup image memory barrier transfer image to shader read layout - vk::CommandBuffer copy_command = get_device()->create_command_buffer(vk::CommandBufferLevel::ePrimary, true); + vk::CommandBuffer copy_command = vkb::common::allocate_command_buffer(device, get_device()->get_command_pool().get_handle()); + copy_command.begin(vk::CommandBufferBeginInfo()); // The sub resource range describes the regions of the image we will be transition vk::ImageSubresourceRange subresource_range(vk::ImageAspectFlagBits::eColor, 0, 1, 0, 1); @@ -295,18 +492,18 @@ void HPPTextureLoading::load_texture() // This separates all the sampling information from the texture data. This means you could have multiple sampler objects for the same texture with different settings // Note: Similar to the samplers available with OpenGL 3.3 vk::SamplerCreateInfo sampler; - sampler.magFilter = vk::Filter::eLinear; - sampler.minFilter = vk::Filter::eLinear; - sampler.mipmapMode = vk::SamplerMipmapMode::eLinear; - sampler.addressModeU = vk::SamplerAddressMode::eRepeat; - sampler.addressModeV = vk::SamplerAddressMode::eRepeat; - sampler.addressModeW = vk::SamplerAddressMode::eRepeat; - sampler.mipLodBias = 0.0f; - sampler.compareOp = vk::CompareOp::eNever; - sampler.minLod = 0.0f; - // Set max level-of-detail to mip level count of the texture - sampler.maxLod = (use_staging) ? static_cast(texture.mip_levels) : 0.0f; + sampler.magFilter = vk::Filter::eLinear; + sampler.minFilter = vk::Filter::eLinear; + sampler.mipmapMode = vk::SamplerMipmapMode::eLinear; + sampler.addressModeU = vk::SamplerAddressMode::eRepeat; + sampler.addressModeV = vk::SamplerAddressMode::eRepeat; + sampler.addressModeW = vk::SamplerAddressMode::eRepeat; + sampler.mipLodBias = 0.0f; + sampler.compareOp = vk::CompareOp::eNever; + sampler.minLod = 0.0f; + sampler.maxLod = (use_staging) ? static_cast(texture.mip_levels) : 0.0f; // Set max level-of-detail to mip level count of the texture sampler.maxAnisotropy = 1.0f; + // Enable anisotropic filtering // This feature is optional, so we must check if it's supported on the device if (get_device()->get_gpu().get_features().samplerAnisotropy) @@ -322,170 +519,47 @@ void HPPTextureLoading::load_texture() sampler.anisotropyEnable = false; } sampler.borderColor = vk::BorderColor::eFloatOpaqueWhite; - texture.sampler = get_device()->get_handle().createSampler(sampler); + texture.sampler = device.createSampler(sampler); // Create image view // Textures are not directly accessed by the shaders and // are abstracted by image views containing additional // information and sub resource ranges - vk::ImageViewCreateInfo view; - view.viewType = vk::ImageViewType::e2D; - view.format = format; - view.components = {vk::ComponentSwizzle::eR, vk::ComponentSwizzle::eG, vk::ComponentSwizzle::eB, vk::ComponentSwizzle::eA}; - // The subresource range describes the set of mip levels (and array layers) that can be accessed through this image view + vk::ImageViewCreateInfo image_view_create_info; + image_view_create_info.viewType = vk::ImageViewType::e2D; + image_view_create_info.format = format; + image_view_create_info.components = {vk::ComponentSwizzle::eR, vk::ComponentSwizzle::eG, vk::ComponentSwizzle::eB, vk::ComponentSwizzle::eA}; + // The subresource range describes the set of mip levels (and array layers) that can be accessed through this image image_view_create_info // It's possible to create multiple image views for a single image referring to different (and/or overlapping) ranges of the image - view.subresourceRange.aspectMask = vk::ImageAspectFlagBits::eColor; - view.subresourceRange.baseMipLevel = 0; - view.subresourceRange.baseArrayLayer = 0; - view.subresourceRange.layerCount = 1; + image_view_create_info.subresourceRange.aspectMask = vk::ImageAspectFlagBits::eColor; + image_view_create_info.subresourceRange.baseMipLevel = 0; + image_view_create_info.subresourceRange.baseArrayLayer = 0; + image_view_create_info.subresourceRange.layerCount = 1; // Linear tiling usually won't support mip maps // Only set mip map count if optimal tiling is used - view.subresourceRange.levelCount = (use_staging) ? texture.mip_levels : 1; - // The view will be based on the texture's image - view.image = texture.image; - texture.view = get_device()->get_handle().createImageView(view); + image_view_create_info.subresourceRange.levelCount = (use_staging) ? texture.mip_levels : 1; + // The image_view_create_info will be based on the texture's image + image_view_create_info.image = texture.image; + texture.image_view = device.createImageView(image_view_create_info); } -// Free all Vulkan resources used by a texture object -void HPPTextureLoading::destroy_texture(Texture texture) -{ - get_device()->get_handle().destroyImageView(texture.view); - get_device()->get_handle().destroyImage(texture.image); - get_device()->get_handle().destroySampler(texture.sampler); - get_device()->get_handle().freeMemory(texture.device_memory); -} - -void HPPTextureLoading::build_command_buffers() -{ - vk::CommandBufferBeginInfo command_buffer_begin_info; - - vk::ClearValue clear_values[2]; - clear_values[0].color = default_clear_color; - clear_values[1].depthStencil = vk::ClearDepthStencilValue(0.0f, 0); - - vk::RenderPassBeginInfo render_pass_begin_info; - render_pass_begin_info.renderPass = render_pass; - render_pass_begin_info.renderArea.offset.x = 0; - render_pass_begin_info.renderArea.offset.y = 0; - render_pass_begin_info.renderArea.extent = extent; - render_pass_begin_info.clearValueCount = 2; - render_pass_begin_info.pClearValues = clear_values; - - for (int32_t i = 0; i < draw_cmd_buffers.size(); ++i) - { - // Set target frame buffer - render_pass_begin_info.framebuffer = framebuffers[i]; - - draw_cmd_buffers[i].begin(command_buffer_begin_info); - - draw_cmd_buffers[i].beginRenderPass(render_pass_begin_info, vk::SubpassContents::eInline); - - vk::Viewport viewport(0.0f, 0.0f, static_cast(extent.width), static_cast(extent.height), 0.0f, 1.0f); - draw_cmd_buffers[i].setViewport(0, viewport); - - vk::Rect2D scissor({0, 0}, extent); - draw_cmd_buffers[i].setScissor(0, scissor); - - draw_cmd_buffers[i].bindDescriptorSets(vk::PipelineBindPoint::eGraphics, pipeline_layout, 0, descriptor_set, {}); - draw_cmd_buffers[i].bindPipeline(vk::PipelineBindPoint::eGraphics, pipeline); - - vk::DeviceSize offset = 0; - draw_cmd_buffers[i].bindVertexBuffers(0, vertex_buffer->get_handle(), offset); - draw_cmd_buffers[i].bindIndexBuffer(index_buffer->get_handle(), 0, vk::IndexType::eUint32); - - draw_cmd_buffers[i].drawIndexed(index_count, 1, 0, 0, 0); - - draw_ui(draw_cmd_buffers[i]); - - draw_cmd_buffers[i].endRenderPass(); - - draw_cmd_buffers[i].end(); - } -} - -void HPPTextureLoading::draw() -{ - HPPApiVulkanSample::prepare_frame(); - - // Command buffer to be submitted to the queue - submit_info.setCommandBuffers(draw_cmd_buffers[current_buffer]); - - // Submit to queue - queue.submit(submit_info); - - HPPApiVulkanSample::submit_frame(); -} - -void HPPTextureLoading::generate_quad() -{ - // Setup vertices for a single uv-mapped quad made from two triangles - std::vector vertices = - { - {{1.0f, 1.0f, 0.0f}, {1.0f, 1.0f}, {0.0f, 0.0f, 1.0f}}, - {{-1.0f, 1.0f, 0.0f}, {0.0f, 1.0f}, {0.0f, 0.0f, 1.0f}}, - {{-1.0f, -1.0f, 0.0f}, {0.0f, 0.0f}, {0.0f, 0.0f, 1.0f}}, - {{1.0f, -1.0f, 0.0f}, {1.0f, 0.0f}, {0.0f, 0.0f, 1.0f}}}; - - // Setup indices - std::vector indices = {0, 1, 2, 2, 3, 0}; - index_count = static_cast(indices.size()); - - auto vertex_buffer_size = vkb::to_u32(vertices.size() * sizeof(HPPTextureLoadingVertexStructure)); - auto index_buffer_size = vkb::to_u32(indices.size() * sizeof(uint32_t)); - - // Create buffers - // For the sake of simplicity we won't stage the vertex data to the gpu memory - // Vertex buffer - vertex_buffer = std::make_unique( - *get_device(), vertex_buffer_size, vk::BufferUsageFlagBits::eTransferDst | vk::BufferUsageFlagBits::eVertexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); - vertex_buffer->update(vertices.data(), vertex_buffer_size); - - // Index buffer - index_buffer = std::make_unique( - *get_device(), index_buffer_size, vk::BufferUsageFlagBits::eTransferDst | vk::BufferUsageFlagBits::eIndexBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); - index_buffer->update(indices.data(), index_buffer_size); -} - -void HPPTextureLoading::setup_descriptor_pool() -{ - // Example uses one ubo and one image sampler - std::array pool_sizes = {{{vk::DescriptorType::eUniformBuffer, 1}, {vk::DescriptorType::eCombinedImageSampler, 1}}}; - - vk::DescriptorPoolCreateInfo descriptor_pool_create_info({}, 2, pool_sizes); - - descriptor_pool = get_device()->get_handle().createDescriptorPool(descriptor_pool_create_info); -} - -void HPPTextureLoading::setup_descriptor_set_layout() +// Prepare and initialize uniform buffer containing shader uniforms +void HPPTextureLoading::prepare_uniform_buffers() { - std::array set_layout_bindings = { - {{0, vk::DescriptorType::eUniformBuffer, 1, vk::ShaderStageFlagBits::eVertex}, - {1, vk::DescriptorType::eCombinedImageSampler, 1, vk::ShaderStageFlagBits::eFragment}}}; - - vk::DescriptorSetLayoutCreateInfo descriptor_layout({}, set_layout_bindings); - - descriptor_set_layout = get_device()->get_handle().createDescriptorSetLayout(descriptor_layout); - -#if defined(ANDROID) - vk::PipelineLayoutCreateInfo pipeline_layout_create_info({}, 1, &descriptor_set_layout); -#else - vk::PipelineLayoutCreateInfo pipeline_layout_create_info({}, descriptor_set_layout); -#endif + // Vertex shader uniform buffer block + vertex_shader_data_buffer = + std::make_unique(*get_device(), sizeof(vertex_shader_data), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); - pipeline_layout = get_device()->get_handle().createPipelineLayout(pipeline_layout_create_info); + update_uniform_buffers(); } -void HPPTextureLoading::setup_descriptor_set() +void HPPTextureLoading::update_descriptor_set() { - vk::DescriptorSetAllocateInfo alloc_info(descriptor_pool, 1, &descriptor_set_layout); - - descriptor_set = get_device()->get_handle().allocateDescriptorSets(alloc_info).front(); - - vk::DescriptorBufferInfo buffer_descriptor(uniform_buffer_vs->get_handle(), 0, VK_WHOLE_SIZE); + vk::DescriptorBufferInfo buffer_descriptor(vertex_shader_data_buffer->get_handle(), 0, VK_WHOLE_SIZE); // Setup a descriptor image info for the current texture to be used as a combined image sampler vk::DescriptorImageInfo image_descriptor; - image_descriptor.imageView = texture.view; // The image's view (images are never directly accessed by the shader, but rather through views defining subresources) + image_descriptor.imageView = texture.image_view; // The image's view (images are never directly accessed by the shader, but rather through views defining subresources) image_descriptor.sampler = texture.sampler; // The sampler (Telling the pipeline how to sample the texture, including repeat, border, etc.) image_descriptor.imageLayout = texture.image_layout; // The current layout of the image (Note: Should always fit the actual use, e.g. shader read) @@ -499,136 +573,20 @@ void HPPTextureLoading::setup_descriptor_set() get_device()->get_handle().updateDescriptorSets(write_descriptor_sets, {}); } -void HPPTextureLoading::prepare_pipeline() -{ - vk::PipelineInputAssemblyStateCreateInfo input_assembly_state({}, vk::PrimitiveTopology::eTriangleList); - - vk::PipelineRasterizationStateCreateInfo rasterization_state; - rasterization_state.polygonMode = vk::PolygonMode::eFill; - rasterization_state.cullMode = vk::CullModeFlagBits::eNone; - rasterization_state.frontFace = vk::FrontFace::eCounterClockwise; - rasterization_state.lineWidth = 1.0f; - - vk::PipelineColorBlendAttachmentState blend_attachment_state; - blend_attachment_state.colorWriteMask = - vk::ColorComponentFlagBits::eR | vk::ColorComponentFlagBits::eG | vk::ColorComponentFlagBits::eB | vk::ColorComponentFlagBits::eA; - - vk::PipelineColorBlendStateCreateInfo color_blend_state({}, {}, {}, blend_attachment_state); - - // Note: Using reversed depth-buffer for increased precision, so Greater depth values are kept - vk::PipelineDepthStencilStateCreateInfo depth_stencil_state; - depth_stencil_state.depthTestEnable = true; - depth_stencil_state.depthWriteEnable = true; - depth_stencil_state.depthCompareOp = vk::CompareOp::eGreater; - depth_stencil_state.back.compareOp = vk::CompareOp::eGreater; - - vk::PipelineViewportStateCreateInfo viewport_state({}, 1, nullptr, 1, nullptr); - - vk::PipelineMultisampleStateCreateInfo multisample_state({}, vk::SampleCountFlagBits::e1); - - std::array dynamic_state_enables = {{vk::DynamicState::eViewport, vk::DynamicState::eScissor}}; - vk::PipelineDynamicStateCreateInfo dynamic_state({}, dynamic_state_enables); - - // Load shaders - std::array shader_stages = {{load_shader("texture_loading/texture.vert", vk::ShaderStageFlagBits::eVertex), - load_shader("texture_loading/texture.frag", vk::ShaderStageFlagBits::eFragment)}}; - - // Vertex bindings and attributes - vk::VertexInputBindingDescription vertex_input_binding(0, sizeof(HPPTextureLoadingVertexStructure), vk::VertexInputRate::eVertex); - std::array vertex_input_attributes = { - {{0, 0, vk::Format::eR32G32B32Sfloat, offsetof(HPPTextureLoadingVertexStructure, pos)}, // Location 0 : Position - {1, 0, vk::Format::eR32G32Sfloat, offsetof(HPPTextureLoadingVertexStructure, uv)}, // Location 1: Texture Coordinates - {2, 0, vk::Format::eR32G32B32Sfloat, offsetof(HPPTextureLoadingVertexStructure, normal)}}}; // Location 2 : Normal - vk::PipelineVertexInputStateCreateInfo vertex_input_state({}, vertex_input_binding, vertex_input_attributes); - - vk::GraphicsPipelineCreateInfo pipeline_create_info({}, - shader_stages, - &vertex_input_state, - &input_assembly_state, - {}, - &viewport_state, - &rasterization_state, - &multisample_state, - &depth_stencil_state, - &color_blend_state, - &dynamic_state, - pipeline_layout, - render_pass, - {}, - {}, - -1); - - vk::Result result; - std::tie(result, pipeline) = get_device()->get_handle().createGraphicsPipeline(pipeline_cache, pipeline_create_info); - assert(result == vk::Result::eSuccess); -} - -// Prepare and initialize uniform buffer containing shader uniforms -void HPPTextureLoading::prepare_uniform_buffers() -{ - // Vertex shader uniform buffer block - uniform_buffer_vs = std::make_unique(*get_device(), sizeof(ubo_vs), vk::BufferUsageFlagBits::eUniformBuffer, VMA_MEMORY_USAGE_CPU_TO_GPU); - - update_uniform_buffers(); -} - void HPPTextureLoading::update_uniform_buffers() { // Vertex shader - ubo_vs.projection = glm::perspective(glm::radians(60.0f), static_cast(extent.width) / static_cast(extent.height), 0.001f, 256.0f); - glm::mat4 view_matrix = glm::translate(glm::mat4(1.0f), glm::vec3(0.0f, 0.0f, zoom)); - - ubo_vs.model = view_matrix * glm::translate(glm::mat4(1.0f), camera_pos); - ubo_vs.model = glm::rotate(ubo_vs.model, glm::radians(rotation.x), glm::vec3(1.0f, 0.0f, 0.0f)); - ubo_vs.model = glm::rotate(ubo_vs.model, glm::radians(rotation.y), glm::vec3(0.0f, 1.0f, 0.0f)); - ubo_vs.model = glm::rotate(ubo_vs.model, glm::radians(rotation.z), glm::vec3(0.0f, 0.0f, 1.0f)); - - ubo_vs.view_pos = glm::vec4(0.0f, 0.0f, -zoom, 0.0f); - - uniform_buffer_vs->convert_and_update(ubo_vs); -} - -bool HPPTextureLoading::prepare(const vkb::ApplicationOptions &options) -{ - if (!HPPApiVulkanSample::prepare(options)) - { - return false; - } - load_texture(); - generate_quad(); - prepare_uniform_buffers(); - setup_descriptor_set_layout(); - prepare_pipeline(); - setup_descriptor_pool(); - setup_descriptor_set(); - build_command_buffers(); - prepared = true; - return true; -} + vertex_shader_data.projection = glm::perspective(glm::radians(60.0f), static_cast(extent.width) / static_cast(extent.height), 0.001f, 256.0f); + glm::mat4 view_matrix = glm::translate(glm::mat4(1.0f), glm::vec3(0.0f, 0.0f, zoom)); -void HPPTextureLoading::render(float delta_time) -{ - if (!prepared) - { - return; - } - draw(); -} + vertex_shader_data.model = view_matrix * glm::translate(glm::mat4(1.0f), camera_pos); + vertex_shader_data.model = glm::rotate(vertex_shader_data.model, glm::radians(rotation.x), glm::vec3(1.0f, 0.0f, 0.0f)); + vertex_shader_data.model = glm::rotate(vertex_shader_data.model, glm::radians(rotation.y), glm::vec3(0.0f, 1.0f, 0.0f)); + vertex_shader_data.model = glm::rotate(vertex_shader_data.model, glm::radians(rotation.z), glm::vec3(0.0f, 0.0f, 1.0f)); -void HPPTextureLoading::view_changed() -{ - update_uniform_buffers(); -} + vertex_shader_data.view_pos = glm::vec4(0.0f, 0.0f, -zoom, 0.0f); -void HPPTextureLoading::on_update_ui_overlay(vkb::HPPDrawer &drawer) -{ - if (drawer.header("Settings")) - { - if (drawer.slider_float("LOD bias", &ubo_vs.lod_bias, 0.0f, static_cast(texture.mip_levels))) - { - update_uniform_buffers(); - } - } + vertex_shader_data_buffer->convert_and_update(vertex_shader_data); } std::unique_ptr create_hpp_texture_loading() diff --git a/samples/api/hpp_texture_loading/hpp_texture_loading.h b/samples/api/hpp_texture_loading/hpp_texture_loading.h index 6d3d58f8b..af2410cc6 100644 --- a/samples/api/hpp_texture_loading/hpp_texture_loading.h +++ b/samples/api/hpp_texture_loading/hpp_texture_loading.h @@ -25,67 +25,84 @@ #include -// Vertex layout for this example -struct HPPTextureLoadingVertexStructure -{ - float pos[3]; - float uv[2]; - float normal[3]; -}; - class HPPTextureLoading : public HPPApiVulkanSample { public: + HPPTextureLoading(); + ~HPPTextureLoading(); + + private: // Contains all Vulkan objects that are required to store and use a texture // Note that this repository contains a texture class (vulkan_texture.h) that encapsulates texture loading functionality in a class that is used in subsequent demos struct Texture { - vk::Sampler sampler; + vk::DeviceMemory device_memory; vk::Image image; vk::ImageLayout image_layout; - vk::DeviceMemory device_memory; - vk::ImageView view; + vk::ImageView image_view; + vk::Sampler sampler; vk::Extent2D extent; uint32_t mip_levels; - } texture; - std::unique_ptr vertex_buffer; - std::unique_ptr index_buffer; - uint32_t index_count; + void destroy(vk::Device device) + { + device.destroyImageView(image_view); + device.destroyImage(image); + device.destroySampler(sampler); + device.freeMemory(device_memory); + } + }; - std::unique_ptr uniform_buffer_vs; + // Vertex layout for this example + struct Vertex + { + float pos[3]; + float uv[2]; + float normal[3]; + }; - struct + struct VertexShaderData { glm::mat4 projection; glm::mat4 model; glm::vec4 view_pos; float lod_bias = 0.0f; - } ubo_vs; + }; - vk::Pipeline pipeline; - vk::PipelineLayout pipeline_layout; - vk::DescriptorSet descriptor_set; - vk::DescriptorSetLayout descriptor_set_layout; + private: + // from vkb::Application + bool prepare(const vkb::ApplicationOptions &options) override; - HPPTextureLoading(); - ~HPPTextureLoading(); - virtual void request_gpu_features(vkb::core::HPPPhysicalDevice &gpu) override; - void load_texture(); - void destroy_texture(Texture texture); - void build_command_buffers() override; - void draw(); - void generate_quad(); - void setup_descriptor_pool(); - void setup_descriptor_set_layout(); - void setup_descriptor_set(); - void prepare_pipeline(); - void prepare_uniform_buffers(); - void update_uniform_buffers(); - bool prepare(const vkb::ApplicationOptions &options) override; - virtual void render(float delta_time) override; - virtual void view_changed() override; - virtual void on_update_ui_overlay(vkb::HPPDrawer &drawer) override; + // from HPPVulkanSample + void request_gpu_features(vkb::core::HPPPhysicalDevice &gpu) override; + + // from HPPApiVulkanSample + void build_command_buffers() override; + void on_update_ui_overlay(vkb::HPPDrawer &drawer) override; + void render(float delta_time) override; + void view_changed() override; + + vk::DescriptorPool create_descriptor_pool(); + vk::DescriptorSetLayout create_descriptor_set_layout(); + vk::Pipeline create_pipeline(); + void draw(); + void generate_quad(); + void load_texture(); + void prepare_uniform_buffers(); + void update_descriptor_set(); + void update_uniform_buffers(); + + private: + vk::DescriptorSet descriptor_set; + vk::DescriptorSetLayout descriptor_set_layout; + std::unique_ptr index_buffer; + uint32_t index_count; + vk::Pipeline pipeline; + vk::PipelineLayout pipeline_layout; + Texture texture; + std::unique_ptr vertex_buffer; + VertexShaderData vertex_shader_data; + std::unique_ptr vertex_shader_data_buffer; }; std::unique_ptr create_hpp_texture_loading(); diff --git a/third_party/vulkan b/third_party/vulkan index 9e61870ec..a0c76b4ef 160000 --- a/third_party/vulkan +++ b/third_party/vulkan @@ -1 +1 @@ -Subproject commit 9e61870ecbd32514113b467e0a0c46f60ed222c7 +Subproject commit a0c76b4ef76e219483755ff61dce6b67ff79f24b From 23d9748f636d4a67214a28af9128ca1e6d7fba2f Mon Sep 17 00:00:00 2001 From: Krzysztof Dmitruk <127966594+Krzysztof-Dmitruk-Mobica@users.noreply.github.com> Date: Mon, 11 Sep 2023 17:07:37 +0200 Subject: [PATCH 11/13] Add a new sample - dynamic line rasterization (#790) * Add a new sample - dynamic line rasterization * Add a screenshot to the readme file --- samples/README.adoc | 6 + .../dynamic_line_rasterization/CMakeLists.txt | 33 ++ .../dynamic_line_rasterization/README.adoc | 63 +++ .../dynamic_line_rasterization.cpp | 515 ++++++++++++++++++ .../dynamic_line_rasterization.h | 91 ++++ .../dynamic_line_rasterization/screenshot.png | Bin 0 -> 107422 bytes shaders/dynamic_line_rasterization/base.frag | 27 + shaders/dynamic_line_rasterization/base.vert | 35 ++ shaders/dynamic_line_rasterization/grid.frag | 58 ++ shaders/dynamic_line_rasterization/grid.vert | 55 ++ 10 files changed, 883 insertions(+) create mode 100644 samples/extensions/dynamic_line_rasterization/CMakeLists.txt create mode 100644 samples/extensions/dynamic_line_rasterization/README.adoc create mode 100644 samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.cpp create mode 100644 samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.h create mode 100644 samples/extensions/dynamic_line_rasterization/screenshot.png create mode 100644 shaders/dynamic_line_rasterization/base.frag create mode 100644 shaders/dynamic_line_rasterization/base.vert create mode 100644 shaders/dynamic_line_rasterization/grid.frag create mode 100644 shaders/dynamic_line_rasterization/grid.vert diff --git a/samples/README.adoc b/samples/README.adoc index b5e5491ef..2819e3ed3 100644 --- a/samples/README.adoc +++ b/samples/README.adoc @@ -480,6 +480,12 @@ Demonstrate how to create multiple color blend attachments and then toggle them Demonstrate how to use shader objects. +=== link:./extensions/dynamic_line_rasterization[Dynamic line rasterization] + +*Extensions:* https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/VK_EXT_line_rasterization.html[`VK_EXT_line_rasterization`], https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/VK_EXT_extended_dynamic_state3.html[`VK_EXT_extended_dynamic_state3`] + +Demonstrate methods for dynamically customizing the appearance of the rendered lines. + == Tooling Samples The goal of these samples is to demonstrate usage of tooling functions and libraries that are not directly part of the api. diff --git a/samples/extensions/dynamic_line_rasterization/CMakeLists.txt b/samples/extensions/dynamic_line_rasterization/CMakeLists.txt new file mode 100644 index 000000000..4693789bd --- /dev/null +++ b/samples/extensions/dynamic_line_rasterization/CMakeLists.txt @@ -0,0 +1,33 @@ +# Copyright (c) 2023, Mobica Limited +# +# SPDX-License-Identifier: Apache-2.0 +# +# Licensed under the Apache License, Version 2.0 the "License"; +# you may not use this file except in compliance with the License. +# You may obtain a copy of the License at +# +# http://www.apache.org/licenses/LICENSE-2.0 +# +# Unless required by applicable law or agreed to in writing, software +# distributed under the License is distributed on an "AS IS" BASIS, +# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +# See the License for the specific language governing permissions and +# limitations under the License. +# + +get_filename_component(FOLDER_NAME ${CMAKE_CURRENT_LIST_DIR} NAME) +get_filename_component(PARENT_DIR ${CMAKE_CURRENT_LIST_DIR} PATH) +get_filename_component(CATEGORY_NAME ${PARENT_DIR} NAME) + +add_sample( + ID ${FOLDER_NAME} + CATEGORY ${CATEGORY_NAME} + AUTHOR "Khronos" + NAME "dynamic_line_rasterization" + DESCRIPTION "Sample description" + SHADER_FILES_GLSL + "dynamic_line_rasterization/base.vert" + "dynamic_line_rasterization/base.frag" + "dynamic_line_rasterization/grid.vert" + "dynamic_line_rasterization/grid.frag" + ) diff --git a/samples/extensions/dynamic_line_rasterization/README.adoc b/samples/extensions/dynamic_line_rasterization/README.adoc new file mode 100644 index 000000000..47226e685 --- /dev/null +++ b/samples/extensions/dynamic_line_rasterization/README.adoc @@ -0,0 +1,63 @@ +//// +- Copyright (c) 2023, Mobica Limited +- +- SPDX-License-Identifier: Apache-2.0 +- +- Licensed under the Apache License, Version 2.0 the "License"; +- you may not use this file except in compliance with the License. +- You may obtain a copy of the License at +- +- http://www.apache.org/licenses/LICENSE-2.0 +- +- Unless required by applicable law or agreed to in writing, software +- distributed under the License is distributed on an "AS IS" BASIS, +- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +- See the License for the specific language governing permissions and +- limitations under the License. +- +//// + += Dynamic line rasterization + +image::screenshot.png[] + +== Overview + +This sample demonstrates functions from various extensions related to dynamic line rasterization. These functions can be useful for developing CAD applications. + +* From the `VK_EXT_line_rasterization` extension: +** `vkCmdSetLineStippleEXT` - sets the stipple pattern. +* From the `VK_EXT_extended_dynamic_state3` extension: +** `vkCmdSetPolygonModeEXT` - sets how defined primitives should be rasterized. +** `vkCmdSetLineRasterizationModeEXT` - sets the algorithm for line rasterization. +** `vkCmdSetLineStippleEnableEXT` - toggles stippling for lines. +* And also from the core Vulkan: +** `vkCmdSetLineWidth` - sets the line width. +** `vkCmdSetPrimitiveTopologyEXT` - defines which type of primitives is being drawn. + +== The sample + +Dynamic line rasterization contains a wireframed cube whose appearance can be modified by the user. The cube edges and filling are rendered in a single pipeline, using a different set of indices. The `vkCmdSetPrimitiveTopologyEXT` and `vkCmdSetPolygonModeEXT` functions are used to change the way they are rendered. + +Users can modify the line width (`vkCmdSetLineWidth`) and choose how the line is drawn (`vkCmdSetLineRasterizationModeEXT`). The sample also demonstrates the ability to stipple the line. Stippling is defined by two variables: + +** `lineStipplePattern` - a `uint16_t` where each bit represents whether a point on the line is colored (1) or transparent (0). +** `lineStippleFactor` - a factor used to determine how many consecutive points are affected by a single pattern bit. + +The sample also contains a grid rendered beneath the cube using a different pipeline. This grid represents another approach to line rasterization based on the fragment shader. Consequently, the appearance of the gridlines cannot be modified by the user. + +== Credits + +The infinite grid shader is based on the code from the https://asliceofrendering.com/scene%20helper/2020/01/05/InfiniteGrid/[asliceofrendering.com] blog. + +== Documentation links + +https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/vkCmdSetLineStippleEXT.html + +https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/vkCmdSetPolygonModeEXT.html + +https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/vkCmdSetLineRasterizationModeEXT.html + +https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/vkCmdSetLineStippleEnableEXT.html + +https://registry.khronos.org/vulkan/specs/1.3-extensions/man/html/vkCmdSetLineWidth.html diff --git a/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.cpp b/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.cpp new file mode 100644 index 000000000..c0e628fc5 --- /dev/null +++ b/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.cpp @@ -0,0 +1,515 @@ +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "dynamic_line_rasterization.h" + +DynamicLineRasterization::DynamicLineRasterization() +{ + add_device_extension(VK_EXT_EXTENDED_DYNAMIC_STATE_EXTENSION_NAME); + add_device_extension(VK_EXT_EXTENDED_DYNAMIC_STATE_3_EXTENSION_NAME); + add_device_extension(VK_EXT_LINE_RASTERIZATION_EXTENSION_NAME); +} + +DynamicLineRasterization::~DynamicLineRasterization() +{ + if (device) + { + vkDestroyPipelineLayout(get_device().get_handle(), pipeline_layout, nullptr); + + vkDestroyPipeline(get_device().get_handle(), pipelines.object, nullptr); + vkDestroyPipeline(get_device().get_handle(), pipelines.grid, nullptr); + + vkDestroyDescriptorSetLayout(get_device().get_handle(), descriptor_set_layout, nullptr); + vkDestroyDescriptorPool(get_device().get_handle(), descriptor_pool, nullptr); + } +} + +bool DynamicLineRasterization::prepare(const vkb::ApplicationOptions &options) +{ + if (!ApiVulkanSample::prepare(options)) + { + return false; + } + + camera.type = vkb::CameraType::LookAt; + camera.set_position({0.0f, 1.0f, -5.0f}); + camera.set_rotation({-15.0f, 15.0f, 0.0f}); + camera.set_perspective(45.0f, static_cast(width) / static_cast(height), 128.0f, 0.1f); + + prepare_uniform_buffers(); + prepare_scene(); + setup_descriptor_pool(); + create_descriptor_set_layout(); + create_descriptor_set(); + create_pipelines(); + build_command_buffers(); + + prepared = true; + + return true; +} + +void DynamicLineRasterization::prepare_scene() +{ + std::vector vertices = { + {-1.0f, -1.0f, 1.0f}, + {1.0f, -1.0f, 1.0f}, + {1.0f, 1.0f, 1.0f}, + {-1.0f, 1.0f, 1.0f}, + + {-1.0f, -1.0f, -1.0f}, + {1.0f, -1.0f, -1.0f}, + {1.0f, 1.0f, -1.0f}, + {-1.0f, 1.0f, -1.0f}}; + + std::vector cube_indices = { + 0, 1, 2, + 2, 3, 0, + + 4, 5, 6, + 6, 7, 4, + + 0, 3, 7, + 7, 4, 0, + + 1, 5, 6, + 6, 2, 1, + + 3, 2, 6, + 6, 7, 3, + + 0, 4, 5, + 5, 1, 0}; + + // Indices of the edges of the cube + std::vector edges_indices = { + 0, 1, + 1, 2, + 2, 3, + 3, 0, + + 4, 5, + 5, 6, + 6, 7, + 7, 4, + + 0, 4, + 1, 5, + 2, 6, + 3, 7}; + + cube_index_count = static_cast(cube_indices.size()); + edges_index_count = static_cast(edges_indices.size()); + uint32_t vertex_buffer_size = vertices.size() * sizeof(glm::vec3); + uint32_t cube_index_buffer_size = cube_indices.size() * sizeof(uint32_t); + uint32_t edges_index_buffer_size = edges_indices.size() * sizeof(uint32_t); + + vertex_buffer = std::make_unique(get_device(), + vertex_buffer_size, + VK_BUFFER_USAGE_VERTEX_BUFFER_BIT, + VMA_MEMORY_USAGE_CPU_TO_GPU); + vertex_buffer->update(vertices.data(), vertex_buffer_size); + + cube_index_buffer = std::make_unique(get_device(), + cube_index_buffer_size, + VK_BUFFER_USAGE_INDEX_BUFFER_BIT, + VMA_MEMORY_USAGE_CPU_TO_GPU); + cube_index_buffer->update(cube_indices.data(), cube_index_buffer_size); + + edges_index_buffer = std::make_unique(get_device(), + edges_index_buffer_size, + VK_BUFFER_USAGE_INDEX_BUFFER_BIT, + VMA_MEMORY_USAGE_CPU_TO_GPU); + edges_index_buffer->update(edges_indices.data(), edges_index_buffer_size); + + fill_color = glm::vec4(0.957f, 0.384f, 0.024f, 0.1f); + edge_color = glm::vec4(0.957f, 0.384f, 0.024f, 1.0f); + + // Fill the first half of the stipple array with 'true' values for the initial stipple pattern. + std::fill(gui_settings.stipple_pattern_arr.begin(), gui_settings.stipple_pattern_arr.begin() + 8, true); + gui_settings.stipple_pattern = array_to_uint16(gui_settings.stipple_pattern_arr); +} + +void DynamicLineRasterization::setup_descriptor_pool() +{ + std::vector pool_sizes = { + vkb::initializers::descriptor_pool_size(VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, 2u), + }; + + VkDescriptorPoolCreateInfo descriptor_pool_create_info = + vkb::initializers::descriptor_pool_create_info( + static_cast(pool_sizes.size()), + pool_sizes.data(), + static_cast(pool_sizes.size())); + VK_CHECK(vkCreateDescriptorPool(get_device().get_handle(), &descriptor_pool_create_info, nullptr, &descriptor_pool)); +} + +void DynamicLineRasterization::create_pipelines() +{ + VkPipelineInputAssemblyStateCreateInfo input_assembly_state = + vkb::initializers::pipeline_input_assembly_state_create_info( + VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST, + 0, + VK_FALSE); + + VkPipelineRasterizationStateCreateInfo rasterization_state = + vkb::initializers::pipeline_rasterization_state_create_info( + VK_POLYGON_MODE_FILL, + VK_CULL_MODE_NONE, + VK_FRONT_FACE_COUNTER_CLOCKWISE, + 0); + + VkPipelineColorBlendAttachmentState blend_attachment_state = + vkb::initializers::pipeline_color_blend_attachment_state( + VK_COLOR_COMPONENT_R_BIT | VK_COLOR_COMPONENT_G_BIT | VK_COLOR_COMPONENT_B_BIT | VK_COLOR_COMPONENT_A_BIT, + VK_TRUE); + + blend_attachment_state.colorBlendOp = VK_BLEND_OP_ADD; + blend_attachment_state.srcColorBlendFactor = VK_BLEND_FACTOR_SRC_ALPHA; + blend_attachment_state.dstColorBlendFactor = VK_BLEND_FACTOR_ONE_MINUS_SRC_ALPHA; + blend_attachment_state.alphaBlendOp = VK_BLEND_OP_ADD; + blend_attachment_state.srcAlphaBlendFactor = VK_BLEND_FACTOR_ONE; + blend_attachment_state.dstAlphaBlendFactor = VK_BLEND_FACTOR_ZERO; + + VkPipelineColorBlendStateCreateInfo color_blend_state = + vkb::initializers::pipeline_color_blend_state_create_info( + 1, + &blend_attachment_state); + + VkPipelineDepthStencilStateCreateInfo depth_stencil_state = + vkb::initializers::pipeline_depth_stencil_state_create_info( + VK_FALSE, + VK_FALSE, + VK_COMPARE_OP_NEVER); + + VkPipelineViewportStateCreateInfo viewport_state = + vkb::initializers::pipeline_viewport_state_create_info(1, 1, 0); + + VkPipelineMultisampleStateCreateInfo multisample_state = + vkb::initializers::pipeline_multisample_state_create_info( + VK_SAMPLE_COUNT_1_BIT, + 0); + + std::vector dynamic_state_enables = { + VK_DYNAMIC_STATE_VIEWPORT, + VK_DYNAMIC_STATE_SCISSOR, + VK_DYNAMIC_STATE_PRIMITIVE_TOPOLOGY, + VK_DYNAMIC_STATE_POLYGON_MODE_EXT, + VK_DYNAMIC_STATE_LINE_RASTERIZATION_MODE_EXT, + VK_DYNAMIC_STATE_LINE_STIPPLE_ENABLE_EXT, + VK_DYNAMIC_STATE_LINE_STIPPLE_EXT, + VK_DYNAMIC_STATE_LINE_WIDTH}; + + VkPipelineDynamicStateCreateInfo dynamic_state = + vkb::initializers::pipeline_dynamic_state_create_info( + dynamic_state_enables.data(), + static_cast(dynamic_state_enables.size()), + 0); + + const std::vector vertex_input_bindings = { + vkb::initializers::vertex_input_binding_description(0, sizeof(glm::vec3), VK_VERTEX_INPUT_RATE_VERTEX), + }; + const std::vector vertex_input_attributes = { + vkb::initializers::vertex_input_attribute_description(0, 0, VK_FORMAT_R32G32B32_SFLOAT, 0), + }; + + VkPipelineVertexInputStateCreateInfo vertex_input_state = vkb::initializers::pipeline_vertex_input_state_create_info(); + vertex_input_state.vertexBindingDescriptionCount = static_cast(vertex_input_bindings.size()); + vertex_input_state.pVertexBindingDescriptions = vertex_input_bindings.data(); + vertex_input_state.vertexAttributeDescriptionCount = static_cast(vertex_input_attributes.size()); + vertex_input_state.pVertexAttributeDescriptions = vertex_input_attributes.data(); + + std::array shader_stages{}; + shader_stages[0] = load_shader("dynamic_line_rasterization/base.vert", VK_SHADER_STAGE_VERTEX_BIT); + shader_stages[1] = load_shader("dynamic_line_rasterization/base.frag", VK_SHADER_STAGE_FRAGMENT_BIT); + + VkGraphicsPipelineCreateInfo graphics_create{VK_STRUCTURE_TYPE_GRAPHICS_PIPELINE_CREATE_INFO}; + graphics_create.pNext = VK_NULL_HANDLE; + graphics_create.renderPass = render_pass; + graphics_create.pInputAssemblyState = &input_assembly_state; + graphics_create.pRasterizationState = &rasterization_state; + graphics_create.pColorBlendState = &color_blend_state; + graphics_create.pMultisampleState = &multisample_state; + graphics_create.pViewportState = &viewport_state; + graphics_create.pDepthStencilState = &depth_stencil_state; + graphics_create.pDynamicState = &dynamic_state; + graphics_create.pVertexInputState = &vertex_input_state; + graphics_create.pTessellationState = VK_NULL_HANDLE; + graphics_create.stageCount = 2; + graphics_create.pStages = shader_stages.data(); + graphics_create.layout = pipeline_layout; + + VK_CHECK(vkCreateGraphicsPipelines(get_device().get_handle(), + pipeline_cache, + 1, + &graphics_create, + VK_NULL_HANDLE, + &pipelines.object)); + + shader_stages[0] = load_shader("dynamic_line_rasterization/grid.vert", VK_SHADER_STAGE_VERTEX_BIT); + shader_stages[1] = load_shader("dynamic_line_rasterization/grid.frag", VK_SHADER_STAGE_FRAGMENT_BIT); + graphics_create.pStages = shader_stages.data(); + vertex_input_state = vkb::initializers::pipeline_vertex_input_state_create_info(); + graphics_create.pVertexInputState = &vertex_input_state; + + vkCreateGraphicsPipelines(get_device().get_handle(), pipeline_cache, 1, &graphics_create, VK_NULL_HANDLE, &pipelines.grid); +} + +void DynamicLineRasterization::prepare_uniform_buffers() +{ + camera_ubo = std::make_unique(get_device(), sizeof(CameraUbo), VK_BUFFER_USAGE_UNIFORM_BUFFER_BIT, VMA_MEMORY_USAGE_CPU_TO_GPU); +} + +void DynamicLineRasterization::update_uniform_buffers() +{ + CameraUbo cam; + cam.model = glm::mat4(1.0f); + cam.model = glm::translate(cam.model, glm::vec3(0.0f)); + cam.view = camera.matrices.view; + cam.projection = camera.matrices.perspective; + + camera_ubo->convert_and_update(cam); + + build_command_buffers(); +} + +void DynamicLineRasterization::create_descriptor_set() +{ + VkDescriptorSetAllocateInfo alloc_info = vkb::initializers::descriptor_set_allocate_info(descriptor_pool, &descriptor_set_layout, 1u); + VK_CHECK(vkAllocateDescriptorSets(get_device().get_handle(), &alloc_info, &descriptor_set)); + + VkDescriptorBufferInfo buffer_descriptor = create_descriptor(*camera_ubo); + + std::vector write_descriptor_sets = { + vkb::initializers::write_descriptor_set(descriptor_set, VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, 0u, &buffer_descriptor), + }; + + vkUpdateDescriptorSets(get_device().get_handle(), static_cast(write_descriptor_sets.size()), write_descriptor_sets.data(), 0u, nullptr); +} + +void DynamicLineRasterization::create_descriptor_set_layout() +{ + std::vector set_layout_bindings = { + vkb::initializers::descriptor_set_layout_binding(VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, VK_SHADER_STAGE_VERTEX_BIT, 0u), + vkb::initializers::descriptor_set_layout_binding(VK_DESCRIPTOR_TYPE_UNIFORM_BUFFER, VK_SHADER_STAGE_FRAGMENT_BIT, 1u)}; + + VkDescriptorSetLayoutCreateInfo descriptor_set_layout_create_info = vkb::initializers::descriptor_set_layout_create_info(set_layout_bindings); + VK_CHECK(vkCreateDescriptorSetLayout(get_device().get_handle(), &descriptor_set_layout_create_info, nullptr, &descriptor_set_layout)); + + VkPushConstantRange push_constant_range = + vkb::initializers::push_constant_range(VK_SHADER_STAGE_FRAGMENT_BIT, sizeof(glm::vec4), 0); + + VkPipelineLayoutCreateInfo pipeline_layout_create_info = vkb::initializers::pipeline_layout_create_info(&descriptor_set_layout); + + pipeline_layout_create_info.pushConstantRangeCount = 1; + pipeline_layout_create_info.pPushConstantRanges = &push_constant_range; + + VK_CHECK(vkCreatePipelineLayout(get_device().get_handle(), &pipeline_layout_create_info, nullptr, &pipeline_layout)); +} + +void DynamicLineRasterization::draw() +{ + ApiVulkanSample::prepare_frame(); + submit_info.commandBufferCount = 1; + submit_info.pCommandBuffers = &draw_cmd_buffers[current_buffer]; + + VK_CHECK(vkQueueSubmit(queue, 1u, &submit_info, VK_NULL_HANDLE)); + ApiVulkanSample::submit_frame(); + + VkPipelineStageFlags wait_stage_mask = VK_PIPELINE_STAGE_COMPUTE_SHADER_BIT; +} + +void DynamicLineRasterization::render(float delta_time) +{ + if (!prepared) + return; + + draw(); + + if (camera.updated) + update_uniform_buffers(); +} + +void DynamicLineRasterization::build_command_buffers() +{ + VkCommandBufferBeginInfo command_buffer_begin_info = vkb::initializers::command_buffer_begin_info(); + + std::array clear_values; + clear_values[0].color = {{0.05f, 0.05f, 0.05f, 1.0f}}; + clear_values[1].depthStencil = {0.0f, 0u}; + + VkRenderPassBeginInfo render_pass_begin_info = vkb::initializers::render_pass_begin_info(); + render_pass_begin_info.renderPass = render_pass; + render_pass_begin_info.renderArea.extent.width = width; + render_pass_begin_info.renderArea.extent.height = height; + render_pass_begin_info.clearValueCount = static_cast(clear_values.size()); + render_pass_begin_info.pClearValues = clear_values.data(); + + for (uint32_t i = 0u; i < draw_cmd_buffers.size(); ++i) + { + render_pass_begin_info.framebuffer = framebuffers[i]; + auto &cmd_buff = draw_cmd_buffers[i]; + + VK_CHECK(vkBeginCommandBuffer(cmd_buff, &command_buffer_begin_info)); + + vkCmdBeginRenderPass(cmd_buff, &render_pass_begin_info, VK_SUBPASS_CONTENTS_INLINE); + + VkViewport viewport = vkb::initializers::viewport(static_cast(width), static_cast(height), 0.0f, 1.0f); + vkCmdSetViewport(cmd_buff, 0u, 1u, &viewport); + + VkRect2D scissor = vkb::initializers::rect2D(width, height, 0, 0); + vkCmdSetScissor(cmd_buff, 0u, 1u, &scissor); + + vkCmdBindDescriptorSets(cmd_buff, VK_PIPELINE_BIND_POINT_GRAPHICS, pipeline_layout, 0u, 1u, &descriptor_set, 0u, nullptr); + + // While dynamic parameterization is not utilized for the grid, it should be called before the first draw command to prevent validation layer warnings. + vkCmdSetLineRasterizationModeEXT(cmd_buff, static_cast(gui_settings.selected_rasterization_mode)); + vkCmdSetLineWidth(cmd_buff, gui_settings.line_width); + vkCmdSetLineStippleEnableEXT(cmd_buff, static_cast(gui_settings.stipple_enabled)); + vkCmdSetLineStippleEXT(cmd_buff, gui_settings.stipple_factor, gui_settings.stipple_pattern); + vkCmdSetPrimitiveTopologyEXT(cmd_buff, VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST); + vkCmdSetPolygonModeEXT(cmd_buff, VK_POLYGON_MODE_FILL); + + // Draw the grid + if (gui_settings.grid_enabled) + { + vkCmdBindPipeline(cmd_buff, VK_PIPELINE_BIND_POINT_GRAPHICS, pipelines.grid); + vkCmdDraw(cmd_buff, 6, 1, 0, 0); + } + + vkCmdBindPipeline(cmd_buff, VK_PIPELINE_BIND_POINT_GRAPHICS, pipelines.object); + VkDeviceSize offsets[1] = {0}; + vkCmdBindVertexBuffers(cmd_buff, 0, 1, vertex_buffer->get(), offsets); + + // Fill the cube + if (gui_settings.fill_enabled) + { + vkCmdBindIndexBuffer(cmd_buff, cube_index_buffer->get_handle(), 0, VK_INDEX_TYPE_UINT32); + vkCmdPushConstants(cmd_buff, pipeline_layout, VK_SHADER_STAGE_FRAGMENT_BIT, 0, sizeof(glm::vec4), &fill_color); + vkCmdSetPrimitiveTopologyEXT(cmd_buff, VK_PRIMITIVE_TOPOLOGY_TRIANGLE_LIST); + vkCmdSetPolygonModeEXT(draw_cmd_buffers[i], VK_POLYGON_MODE_FILL); + + vkCmdDrawIndexed(cmd_buff, cube_index_count, 1, 0, 0, 0); + } + + // Draw the cube edges + vkCmdBindIndexBuffer(cmd_buff, edges_index_buffer->get_handle(), 0, VK_INDEX_TYPE_UINT32); + vkCmdPushConstants(cmd_buff, pipeline_layout, VK_SHADER_STAGE_FRAGMENT_BIT, 0, sizeof(glm::vec4), &edge_color); + vkCmdSetPrimitiveTopologyEXT(cmd_buff, VK_PRIMITIVE_TOPOLOGY_LINE_LIST); + vkCmdSetPolygonModeEXT(cmd_buff, VK_POLYGON_MODE_LINE); + + vkCmdDrawIndexed(cmd_buff, edges_index_count, 1, 0, 0, 0); + + draw_ui(cmd_buff); + + vkCmdEndRenderPass(cmd_buff); + + VK_CHECK(vkEndCommandBuffer(cmd_buff)); + } +} + +void DynamicLineRasterization::request_gpu_features(vkb::PhysicalDevice &gpu) +{ + { + auto &features = gpu.request_extension_features( + VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_LINE_RASTERIZATION_FEATURES_EXT); + features.smoothLines = VK_TRUE; + features.stippledSmoothLines = VK_TRUE; + features.bresenhamLines = VK_TRUE; + features.stippledBresenhamLines = VK_TRUE; + features.rectangularLines = VK_TRUE; + features.stippledRectangularLines = VK_TRUE; + } + { + auto &features = + gpu.request_extension_features(VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTENDED_DYNAMIC_STATE_FEATURES_EXT); + features.extendedDynamicState = VK_TRUE; + } + { + auto &features = + gpu.request_extension_features(VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTENDED_DYNAMIC_STATE_3_FEATURES_EXT); + features.extendedDynamicState3PolygonMode = VK_TRUE; + features.extendedDynamicState3LineRasterizationMode = VK_TRUE; + features.extendedDynamicState3LineStippleEnable = VK_TRUE; + } + { + auto &features = gpu.get_mutable_requested_features(); + features.fillModeNonSolid = VK_TRUE; + features.wideLines = VK_TRUE; + } +} + +void DynamicLineRasterization::on_update_ui_overlay(vkb::Drawer &drawer) +{ + auto build_command_buffers_when = [this](bool drawer_action) { + if (drawer_action) + build_command_buffers(); + }; + + auto uint16_to_hex_string = [](const char *caption, uint16_t value) { + std::stringstream stream; + stream << caption << std::hex << value; + return stream.str(); + }; + + if (drawer.header("Primitive options")) + { + build_command_buffers_when(drawer.checkbox("Fill", &gui_settings.fill_enabled)); + build_command_buffers_when(drawer.checkbox("Grid", &gui_settings.grid_enabled)); + build_command_buffers_when(drawer.combo_box("Rasterization mode", &gui_settings.selected_rasterization_mode, gui_settings.rasterization_mode_names)); + build_command_buffers_when(drawer.slider_float("Line width", &gui_settings.line_width, 1.0f, 64.0f)); + build_command_buffers_when(drawer.checkbox("Stipple enabled", &gui_settings.stipple_enabled)); + // The stipple factor has a maximum value of 256. Here, a limit of 64 has been chosen to achieve a scroll step equal to 1. + build_command_buffers_when(drawer.slider_int("Stipple factor", &gui_settings.stipple_factor, 1, 64)); + drawer.text(uint16_to_hex_string("Stipple pattern: ", gui_settings.stipple_pattern).c_str()); + + for (int i = 0; i < 16; ++i) + { + ImGui::PushID(i); + if (drawer.checkbox("", &(gui_settings.stipple_pattern_arr[i]))) + { + gui_settings.stipple_pattern = array_to_uint16(gui_settings.stipple_pattern_arr); + + build_command_buffers(); + } + ImGui::PopID(); + if (i % 8 != 7) + ImGui::SameLine(); + } + } +} + +uint16_t DynamicLineRasterization::array_to_uint16(const std::array &array) +{ + uint16_t result = 0; + for (int i = 0; i < 16; ++i) + if (array[i]) + result |= (1 << i); + return result; +} + +bool DynamicLineRasterization::resize(const uint32_t width, const uint32_t height) +{ + ApiVulkanSample::resize(width, height); + update_uniform_buffers(); + return true; +} + +std::unique_ptr create_dynamic_line_rasterization() +{ + return std::make_unique(); +} diff --git a/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.h b/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.h new file mode 100644 index 000000000..ce2af2d71 --- /dev/null +++ b/samples/extensions/dynamic_line_rasterization/dynamic_line_rasterization.h @@ -0,0 +1,91 @@ +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#pragma once + +#include "api_vulkan_sample.h" + +class DynamicLineRasterization : public ApiVulkanSample +{ + public: + DynamicLineRasterization(); + virtual ~DynamicLineRasterization(); + + void render(float delta_time) override; + void build_command_buffers() override; + bool prepare(const vkb::ApplicationOptions &options) override; + void request_gpu_features(vkb::PhysicalDevice &gpu) override; + void on_update_ui_overlay(vkb::Drawer &drawer) override; + bool resize(const uint32_t width, const uint32_t height) override; + + private: + struct CameraUbo + { + alignas(16) glm::mat4 projection; + alignas(16) glm::mat4 view; + alignas(16) glm::mat4 model; + }; + + struct + { + VkPipeline grid; + VkPipeline object; + } pipelines; + + VkDescriptorSet descriptor_set; + VkDescriptorSetLayout descriptor_set_layout; + VkPipelineLayout pipeline_layout; + VkDescriptorPool descriptor_pool; + + std::unique_ptr camera_ubo; + std::unique_ptr vertex_buffer; + std::unique_ptr cube_index_buffer, edges_index_buffer; + + glm::vec4 fill_color, edge_color; + + uint32_t cube_index_count, edges_index_count; + + struct + { + bool fill_enabled = true; + bool grid_enabled = true; + + int selected_rasterization_mode = 0; + std::vector rasterization_mode_names = {"DEFAULT", "RECT", "BRESENHAM", "SMOOTH"}; + + bool stipple_enabled = true; + float line_width = 1.0f; + + int32_t stipple_factor = 1; + uint16_t stipple_pattern; + + std::array stipple_pattern_arr = {}; + } gui_settings; + + void draw(); + void prepare_scene(); + void create_pipelines(); + void create_descriptor_set(); + void create_descriptor_set_layout(); + void setup_descriptor_pool(); + void prepare_uniform_buffers(); + void update_uniform_buffers(); + + static uint16_t array_to_uint16(const std::array &array); +}; + +std::unique_ptr create_dynamic_line_rasterization(); diff --git a/samples/extensions/dynamic_line_rasterization/screenshot.png b/samples/extensions/dynamic_line_rasterization/screenshot.png new file mode 100644 index 0000000000000000000000000000000000000000..a99a4ef9e32939fc10c21e07a9a627267a346117 GIT binary patch literal 107422 zcmb@u1yt4j)-Ak61w=p;0R;g;0i^}$k`SazO1cH8ouK)3tk`Tf`!$(6P5E#O*1Y{71>+mH~7b+6` z*9&5iD+t6DHv@itDPey8hh`S0dImmSA) zTcW34_q}rED{CEk5Bb=`*ug@Kj^hSt`B1qj%c&bdw69KnS~pI!bdZHzkG;=S5Me*{ zBu{ZkE6G=Bu=i7nw|086Z1tCI#o5Y6rHp>1AR(80@Ll##Ubzy+QM(aD5|vMN-2knTbi+k( zJQ&5(yYXzn3x|n@95?nS5}l%l(=>l9v@#B?6NQ@8o7HHZsmsA zDjM}Q)6*@=RC_@tRN@(Hj!v2CuN+=Nd z@TJskrtl|uFa7Wx#p3g9W%~rQeQ6Zy)UX8ZULkn-HTyZQ&!>mKG5ssXiKC3xgs^YN z32i%#(WzILMhwDhn5UYhfDxUAh2L9kVybN$h3B@a{!iK4e7NOVL%7AnCQ}A69_gBKtaPHpA&tESexQj+(fzw)hl!$&u<=M~!Lb+^p z_(87ctrVF=y^h!3-qM||GNQuLtY+`I^w3om7f0Aa!td=ZbW;vw8_dj@nT`by=6?8| zb(-w_V`96^+gg%qL|wnuW+*W3?4J9eGchG$pqEB5 z-zG13O5>N5p3Z4-M}KdHWfj|8Q&X1jqt*G@R)+j-zE6dPQYMPLu5(TC5ojASy#Yf94nfLCHQq!5*ZwdG`%bu+;4#v4G?#L2E>pG92JfOrLj5tEEi%%RVv#$Y z(ndC;g9S`$YdKB+&d+pxN{d3YwR`M4pUqYu@R3k|KBG8pqkAneT{4blVKL*0t{Ff? zQ(zbZr-U((ed87(se5i^Rnp-})65M1t$?F6*$ghOC|@5PBjX4+b0|tp z6CP{!B9c4vDlBaE!9iy}J`8?Z#LLSR8(XBQNlmL@v!=rN7aIXIl=1Ii4^zk^U$@@g zNdBYvL*wTO7eU1I6rVtR(<32=v4_mc_Qy5`+~;z3c0IP^ry+{!>eGvdBfmb6DF~a* z`Y1l0QdB<2_G4jUa@p-ucTGt~YimtBCwja3ut|?WH#1X*@WF!X1Mi2$HtqO%QmLhO zc~jxfe)#wtk%^w6nbg-)VPL!y@;}K?v9z?*BNr9+rg?@_HrxHk+-6VRuhQHNi}Yvf zoyMj!>xQCt?sZ>6KmAh9$nX#fII~`f<@%8#sYFR>M3KUbp?7$Y+uIlQyJs=|8imBC z>qt)@J^E*x8#V7w_v9+^83otQO}!se)8UPl^|aStA1?7t7MuL@rx*JNJ@ec>eCGN2 z9^N;%ezLJW!@=?BjAc4KT0U?45*mtz5n^Kz$jq$w&frw@V7?Q%6JC zf8n61{|fIOBvp+6^QSzBB7|JxF5sTY$dj^2y!zhP4!`ahP*E8;JBP-TY8KoeaUa~I zyB&{5he{If>FFwyo^gW&-6n-a;!DI+V~yIQg@n-aI9grfc8$D54D{DQL1aI)a7C5w z2mCE?4-sqGk?O8;5sY4Eie%3+j~sS)#naL(XlYlv6OvX|@5{*ev6xniYlowu(U6hJ z5IZO4nv7;_PM>?b>UYl3s*R{9{)rCiFqzgy*F8 z8I!Sy=(|sT#xzgAnwx(uDp_v_B$wH=kJ3R76TSDjACK-$M*7)dT706w&K4Qb--`PJ zF{1hrgW%`*6|?#%uN*)9&Q^^3Kkh%{^XbH6VY;!m^1YzJ&rg7QO{=!RZAeB&tvxcy z=^!`LS9IHU{6>G)BX|1XkXfMsTzR=Zu^16?dcE@Uz^Aff(j^S5E5UH6+>TAJrKH~E z&%D%moR&l3p3Be2s;7rBkTc-CH>l21Tl>vclE&DG+R(^I_|;niJUl$?ywall{Nmw) zp1+58CLUfT*IiIE{@!_av)R-kbK~dN(0m3ZM@Iwa)8rj1bInPfD`t9nZ9OSQ;^K4T zrd9bd(f1xm;1w0gzy9Rw`%WPTx3BN1hi5(#5-JJ$)vL?_6ee%6=I76Go;Wx%DrV9q zzo@}zEP5AVVNq0bDsQ`&8JwMcGTz1@I{1e^G4V|tR{}RfVNntW#*IZQG^-Mm@6ivc z0vRnQUKIQL=T=6$^c}`YR68s8{`vWgF-V12)ZxrY5AUY)-{qh9<~nAOn!kM67DTa| zBC)%&k<4LxSICdl{kdA%Z*tKuk*)XXUl_E7CN~I#78kEf)TvfRiefi5^{)*j<(qx~ z{X2@>B-^NBHMWRSY=hc1uRz+oOK1E7OENW9w+vEl-N+y@&k^=*yOISpO)4nILq`tgyV>gWUDKbh%40~VrmxoR#ibG9;{dM0v zex~wnWoi5c>}$0-yFa?~nwyyk*%PtuJxR|pK3pv+zOoVL+W*d=MNyebGKSK`Bs;+S ztEKs(L`-|liHxa9_QX6zjfv{0`;FZSo7+#H`g?nSPD+}XtY4y04%Y9~OG{_y?QOV| z(`bY3)+lq}xSZ@785uQv2Pf&dd3#_Gq!V^|qcNY2m_H)le=yo4WdS z_3{3pp{9d_u6@0pmGsOk`RwmYlirzbZR7QCyn4|0eop5J}@j`3*c zc2e@0EuGim#6y>pQj`z21WOSS@kd8^Ha6~9$zEQpjItJuixQpOcJ!L`0-Kw5h;LEc z$#YAW8R+!$53vY;qnupU*RNGlk_Kr9=cYqD>fNWt@nhdQOx6j>3`+!XFBeiw)-1x&FI|FG%*KjY?pm}8Ifr!wg} zkZ8SyOSXQnp?U2euKj<-$Nwi%{U6T$KmE?x>XnV|1fiQ~Xh=v%V;Q2SJ3W#V;*mSd zOw@|G-@bj*7%n#MZGNWRBo@bg>bkUi=gu7l`>h$ZoY4A7r_DyZ1VO(aS8rVIc^&gu z^))3GRmQh(wR#2PamzDitW*lwI(dpHm=6}r$$tO-jfRHCz`*d@gGxSAN?v}f%5mSz zqCizu^>}yLc6+Y<$B!SeyOZg18Hq_r`|IQ9p){L&d;Pt=l{Twyl$7q1G_q^$DIcXzQrs7 zu^KEfZ8;Zo4+Jv0=vd3W7^OKtNdd2GOg_T>vKE45d--ufWMxDwuO!UCwtp z+_IH6wzl4lX0zOQetBE99DN+?2rqZ{$B!T1y?eKzq2X5~ldnV^r~Q^2$}QN&!d+F> zQTX=qKn^ya&DuyrU?79}RKra?+L@UdPNxI2^|2~iTG~L5TZHVvk&$ntq|8o_c4?H0 z7nYZ0m4e8{MMOo13k_ry6%_>q1vND_wX~*o7JGJgcLRud&Ne3b4W%{^SR3j=0C((NChpSJYKD~bZy1BVI&&h_HxAzSaoxXHA0|SG(xj7?d$M#63 z5|h#2U0w9go(YSHusa_;cg6_~4TaDvwOv<}mX;O~Y3D($T@?+_3iua=At4G%{}MCj`1 zVDSn1;V0$f#D#~yLY0=4?dk4Ts&cS4HZ~R%^o)%i`*izJVPRovUx6Lx7#!1q6E_Y4 z{H1v0b75807@vKJ8Tg6&4Q#bayRR`ZSqTXVDJlEyEaQ-6s&aBnj~?;M$@M``KQeOC zjCxk*KbV-9LPA0o z(d;?LO2%D=|NIG$h=8D1CnhH5j&q%D#YwLnP^M3) z+S##s`SRthTenb9ydaA5^YefG61#f!Dg+^91H5x`GC5_aTA2k|P>Yzbu(X0gNnReC z*~ANYsj0=r$CCy$6ENz1R-|@0+F9J5?@SYg%S(<_ke3&f<9kL=4{!lBgUtQy+qdps zUdzkN^K)~_DJfCm;qIQEhK7coo}S!r=ZuU!y}iNplQXlkT&`y-Pwt8c3qy?DxQ2A( z6`<~V+SkyKn4ix;M%Fkqs-A+ff$23Ue^$HvA|<3es!q^Gxzj5tlye%RdH z1SD#HgmDQDF^e9zC%!%sYW?{W(v8>+83PavpU1^XTpS}rPEj!&)|{ESvc4{vgnpmF zy-q_wNa)Fl(%wM}V$>6lzuInv$zjFt6IN5>kVJnXjjOOdZ{Tc%P$dIq#lTyAb| z5KAUwm1!v{12N{Jd5U61%#RFY_fG5UNrRqbVPWAc z^%|F7;m-g);QP}SWj!g9JmymZuU-L~(vr*o9>h6q3G3>T75U@2)qP9H=d{~s38Luh z>kFe+;)^@&%TSnZ4zRVdQYm~Z9p&k~)k5Lo>g(eZ&$paYa~4@{%g)XY*=%8Lje&u& zyRq?vhQ`{=jGmEkxIkYLKE>QT@6}^=yN!vNmZ0JDYq^^B$lBV|qG7ZcM4YToo;;za zr{~}p?&vU?`xSA1Vo|S0r&jrAU|?W)`1oLR+K4%7iTLt5-`n0p+B- z{p9R;-HF>3PULkB+{ z9j|WJ)6*mU`h=amJUiPTOGZ`I8PIcNM5U$Gabl{`_qB)n7f2%jzVuUoTnh^eh{m0* zHic{zP5OIRQ81m1V<1c%_t%nmj*gGF51kBGpYZ)GHXephrlzJ|$ot0 zA%}>?fiu7kr8ALh0X0XOIjkA)I z1v`aJBp+hYM~E55k??)W%359MP5=N;&B@6DPN9`|4;|UJacM)*@K4$I?;j-Wcb9NT z66-_a>^ap`RS5}sZ}Zu#{JDr&wG|~phg`S_k_Ol)HEu{{QRt~tQ4tnadtrIyG8VXUZWOpS@82c-bTOU{q+>>EoLJlqj0fgC^@5) ze5er5^VYT~VQr&8U_XBR0X5~(qeqKY@7^s#-CG)aN=YfR?{A`yM@>TmVGzgV7#bZd zMgdh^B7)(is%lj|KP4sQg}Q{&$#@xsmGuZkIk}J)9UEM&+kAkPfa^fUPEJq7#l+x) zb;5g#vXo08hFIXPAXpL;8-|A~N(2v~LcvX!+npENN|WdU06RN##VEdh{aQukBw5bG z-Q8%cGQuHcf2l76m;(^31j|k!G3RHekc}LWB7=ivrKL_siz&W%bZcgJ!`f!=@Wt}D z(hlBgY-|K*S#fc6IXj%caurzx%G-34U*KZR$(AU*0o=HjmX?pNuNNjE``fN>CN2s0 z&;^~Ho#7^Da&Bnlm2zzU-Ck`4+IuJ!A|fJ3fTS-YfCC*I9F&%oZQ50CZCN;|re?Rt zaw)5;pO2JT0+OW2l;7vI76L?tpu@!U|Mcmbj2&kz@DXY0+gvH-c#wS0?tO;DXd%mY z)LZIJ6Bid})caLxIgc5_V7sQ$5?-8{X*Am!;*Q17Uk_LV9LuUBnvD>-mW+%HRu&)s zF*f5Q8JFhG%<^(nTpW+%o>9vV6j1bg^b=dNZ98avJC9cf%B`0}Up(yU?#|24kI(5I z9#)3vC%AsuIMfa=c#sr9)dMhhs3)Nf%ScFgq%_f+CJP5AFE2kdG?bQ>)@#_5!}PZ6 zA+E2Snwo_3^XJcddwL$Z@3hiu08mD?Cy$9GL%>e|{Q4*`iheg9xM6?4ypWKPLu59D zR=#fAfzw%(d|!>L>xUOFXwt+{BY0f7quTZ8@4ZHk3l83ysJm(ylRGkUL1n*7r6wi{ z;NYTp2xv*z@$vClS{5WFCFw9$pB=peR=!r)(9>htpQ!|)&=y8#$`}d-wOy}Z(J>__ z$5cb(q5Zl$9|?pvUU2AREL1=nSPqcF1p&jWDRA1Hd6&`JrCpVaA@llma9rFtP!qS0 z9{~bpR#rxG)N7b^?_dLgjDB(#ZE-e7Z5iMhY6KKm!1JuEEE^k}*D9mcTSNzk^Kn2U zn*9j$4Gr;$iA{BNXD22$0Gm7deH}z0?QWr>nwgos5*99VIdu>h_X`RNg8Xx-y@5LP-f6 z@Zk8kC&P*5`EzRJ;uYACi;D{YOyBR{h6Vfd5rH z9}o3b8V?tbyPi7(%E3nt9^T`$ zqi4c&f=p%Bg^2C>^XFh^QP$yV;y}5@Y%5go<)tO0t5>08f-Q)OE3Abe4q*S5bH7^J z+89|_0Bwr>{Mtt=Y?}j!&;1>g9%6~bavuIp67})%f%H8-+3qYVEZkce+)nPMBbfn? zSYf@48`l{}ulX@Oed!j@iIlYT_smSORYE4{SD>UWE-gvKuv`C56sE>qh8u!5E770J zW(E2ZPvDlnA{f&Yty+VhLM0I+XMFfj1J>Qt)O7py?VGsN3lgG*MMd|~Jz-JTZos*b zxdV{wLcA9h71ev)0-ho%A;E1vb&Vtu>S1pluyW^}g|7hp^Kq_YDUxw}B`)3ZFFiaw zfD_54NXW^`RyZH;0e>lHv%Ne(#5aEQ7K7T=Zdv1sNA&a?o72roN=mB*AKlzaOvhc; zM%Y@iR4Q5;8)xC5R5J^9nXRm?f%9amRf!C!%gZLZG2NGUHC+wGGDMfs7M~u(q@a>f0z10 zzPSo`3Kh+&_)mCPm;w}~svEM7K)cAv$-BF|dj*v?)S=vY{uK$X)sDig;z0WVw94A@ zdgggNvFk}*{lI7F2P$}6D~=&_5^{3JxvMHGFM{IWBNCFv#jfAqB1eoB=m)@Z#;ct> zW4T0g=Ft)VIC^LGV~0)YP*k_Jw!-Mt)hcXg$_f6S)tU02^`mR5l|L^$Xn1W9*Z;qA z2*1g$l;r#v9aWW-q=Pc`^rZL>Pyo(WQCRq#fgxW!G$lnJcAKM8CIj84 z(loReOwXUQx6#nj`uqEv1VL-?Rzt&V@plq4-tzjosgcnM>M1} ze1K5XQ&a35)2Y%4P${A7R8&$r{NR!V^_q*56D|+CF&Ww6@o@>;2?TpAz2*;q9w5{ex^1CQpFvEp0JL`h z0$BDLpUH>z5K0q-AZ!StFz*c;$t438s^)JAyf)bE`W!H(y5OrP3Ndq z=xJ(pE1B@|A&nwHjsR?G0djfSUe?2P(BfiYT~KGD0HKm?FJhMiXe<{)PxgjQ`Mof8 z86w=VzP`FkNLZy&_YsqS0xpx(S0 zL~V0^<_tvuQgr(JX$~-beu|3&t@ZLBRmNFdUOwMyql2Ver{4#_`TqSoAPmF|U~Zsv z1SE^plS-#U3w41gId9>} zN=YekIRzXzvK1;YY(OCX&H~&;0G<F0iZ8{8fViT3+;ioKs=N} zAbr=6k!@I=BeD2iY6YuS*c7o55+>29lu9WpAA=yHTx?{(Xu)AQCq_sZ1BLkgdj#Jr zMa4f*{D3|`Gaf3012AWOASo&N;RQVqGfLFY0RiXJ0X)zpl&Y-NJROT-vm}q}gq9BV z#*N#k*LnkP0^-K7n2y0MlUhJ3MiE;tP-f>u`M=FH|*eC@o>3p|B8U9EK`+P=5RgIe;t=NhXEr07@7etIEh&fa<{< zMfl*s33C|k-Mc!Pn$kd<`xMlR;=qg6tkYotEp0IvGA2*aNJHZV>iy?$zoe%df)EIA zpEd$O1&{(1|7drPB_CJ>$bCSU;2#RJ5iM5dFACpot>tuLq%2>TQs+^!Mt(fDIJ~b z*36B7=BVcecbJlofNVeq2W9nA#S->det@YDv zRV zV)P_Bpe}>t03Xq7(QF(X5&}#L2#QI=r`xYvyT0>7tGKnjJ#JN(0&N<6<__q)1xT<8 zqz?~CNk6ium6snu&2e;61xZRP?+-9e_`IJ#-$FXWig$j1XD{OZ<>N=B@)H1i55DiP zLmZ+4#Z1UQ=2iWI!PeGR(_VC;!Za}N#=p*GTjXq=Y^O%5PTx$`!lnmX9{3JUwlqZhSQTl>*nG+9~M%@==@ z&+9g#xw)B`$3;a=%>}eqLR?m8qC`d808STwdNpbl4PH;-z$YL8IOYL^7Sj^e04Gb# z<#3mnSTY7!VINfX=7GjpiwH42dwY8`v)r<>LrI=fD2hxDI}2+M+BE2gD0~{HI`Ydw z{exPjsjK@kAbG6J(yLAbrH!?9$O-QxXfJpm z8Cfm<2Dt*7)||w|yV%%|4!M~xlj}u}_P&WVh@ud;#f60`d0H=z`*I=hAboarc0g4q z$j{Hr%7PQLo@(%B;qr}-*Me3QG%t={gd8@Rg@rqy$EOBBszAPj5{rq63247q_Yz7M z3o~;^98cWPf*lZcD3Q=RKnKI=xJQlC{@Zc*HwZe5=V`_t9w%t!DUSU9y_HM@rRiX$ zpwr0M7>*b?QMeenTipWbmktGFu1!4TRWhIXRrWciA7bwg3D{ z&%p4zug^G+3=6flw-?B9T9xbW&Q9vS3-~M%V#%SQxT-Qh_8oZNP~(D##sj(=FcF^f z6UJD<%k!{H39nL{29e=9Qk{OBGx`1Pu4liYrvmU0`h0J4c{w8~X&2mvH6x}GPcS>+ zwn0QZrUv4Hp>ZMHx_}Zp-F*N4eNfM!g@g3ts)z^07};$T+8_4Wju;Lx@}A~qFRZPt zSy8^9P<$3uTwPt6qr|U&0k4PeyOf3QQ%Zkh1B36FPLCirRPuGai;MT*`fOK+tj|vD zDNoiuOFc<++t>rf2CyV8{Z%La)#G>m=KuV28 zAkTXthv5ZC)YPO0L?W=8&ibX1&2XIFm{skA%R`umL8*oNgdhT0Fp|U8Ts|z2`C0)0qi=$)p29u!;6PsqA5=V!4asNGzhXM)=>2=2u$n)V`H(6TGE@wP&PzGeLX#q<2q9k)QJKHdV6;uU|9@3tE;P_ z>3{|mctG$|S*Qs@In<-~K^g_R6vbj{6RxbPng!g%UKP3nP>yhLaNv#zi9pHu4R@~z z!RSGa>l-^hrVT4aM^0#eHv@F&!fL4Dz1$_@WdACdJD}7+acuwfYe*j?6iqy$AQ%TpaT?1rUTv)h3YLqa`<2_?!H(t|; zhDkGI4E)#1$`y|LCVa1)oGPWB^bQPw8S(lGpboN|`Tm+J-z!+Mu&{Sv;GNwcG$c9z ziqPS_{&XhTZf0fmT3+7eaEq)82fHY?c;>_qL`X>UCDV8B-T^&bDoR;d zS&14~l@$tai%xk(MbA_4FtrK>1C`lZT3dh9loS^B_Vs0@rY>yg`z|cJ{qRBqwA#6l z={@Z-)wZ8Mv3wh$BnFBaqg=ZNG11V}H06CcrT3mlyC%Vb#pqA2b~C1d!Q!40sPyow zzzPApQ&>=zK0+J3Hd+CxXJuuDj)6gdkAES~w6G``0GkCP3t}0_vSAPoHa6H)I9OO+ zZXuz}reiDf^DH_)zW~F4YXqlYadEM+xmh+S`VlFqUVEgjfq^~H2Z(!Cc6O*Fp9hg3TEKW4N8p@MqJI(B9tO5PeWK z!Bk-Y2m}uAK=4?&!e$jTb{I)12Nw@CDG*Abk&2B~gQF4!eF_dtp?d8im=?0h zVtK{IAX39ZOs#0Z6^er~8Au}tbY%2<5AUKu{U|XSWGJu2Q+LWU9%hDumnNG64eZ?P zESR^SV_#P4w&9YIB>|irA0FQ3Q-cx!aSF}V_kx~0Me3zhEn$Gw6p5IjqIU{!-=+dD z&&o0er!`qT3YShTGc8RRFC#Uz)Mhmjav9oP2v^gQQr=ZNfikE^Ld4F`L3_c*4g$R% z7Y|P|^0@&B1CY)a%4spGT8+yofZifMoOnWfyeB5%5a@E?4T0pEBpPPZzXg@wyK(9} zzn6o(J;Xb#9Sn8>kdAlz6k^)b)^bb0TQAdUL<4Omow&8FO){3V46-^illor;@Nbjr zn9J8!6P}%&T~*}_ZY|dhr8{A5P&vdTB-W^F_jY#fdJAy<*-OJ3+MspsE~tjHEA%`_{+k|>NyjWu`D+u<2JNJ=Xq+D zwzdO(eO%^K?k^rbbO#$5WVCJw3%>94K#a-AW}PeA;jST_?syCI3W4tn9dfc5^cp%k zi8CZnG@uy3v6xui9JK+Sczliw{N-5 z$I!d9A8yafKE0@iVYJ4wD}b!85x=H4LBG*5G!z+!$vWnh@tT@~BTyBA8bm}!9)Y0* z42`J766^sGDAZjJXCZMe6`bhH``#G4UJK;!4CoYi-5`iZjhcIS-694e2CDnv;UV-z z0YqF&nZ?j9hE^4;&f>M#upoA3buTtev&+;G2Z?5fBh8Ca~X38TSC7 zyqIYL?Xy=%U0t1qCJQ=9^XaClxP3b?F+dvth6}RkYZ17-2CYqCh%f}Qs0Bu2^4Xsf z)3~&>MQGW(;hGv(NI;-Vf!ET?>cR83T|pF*f&oMz^p#v((r~dK^b~;F zKp+g&Yf@4uyy~Amd9n~ogv4JC!a2$WJXTwdydyA{X9LGBe9Rz5{%#TNj#ie0+SMD{8~JU|5WlTfYOT z^+I<5lc2Q~?baESpY6&f8AdOO`ojFie$X!^P zon7ho@0wmQ^gyOTy19#sD?)Bj2HHh5sPxq*Ur6{BspTCU96*x}v4B}4STE*s4NPx@ zsk^jj&?Da>f=}LSCU(W*!zAW*TJFz+ZvErZ9wUTD%j1E;bAJA5D7%nBCN(F4IvU^+Lrr{{0z(`F{rym- zdY*zc0XhT}UpAK$JJ63H#yB#>`S}yzoX1^`v!tGY=0OtGZewZ50h?+ zasby;0}~P;xE}4TM!{Q+jKJi_`Kax9ce`S~P7CY<107vPRu(V-0v}w+hFObY+mA6Z zgiPBi4AP)gLOO0t)B)v#h*K{yA@r*MUNAKB5%3n2K$$JvF7VcNW?HcG6hXEPiHH!c zI?sfH3u>a(q6}q|JdBOOp_ok|Mk}F{hpi+7NI;r^-v&9M1WnM)A`E)KMug5^T;dyI zqxxE{>h$&@HB3k=*+Z&9AV5|D_5eHVf<{pN0GejdE?QfCB=QOibr>Qb@Op(n$%fS7 zj;pcT)B;%^Amc*Y1henkbV+eovkno5&9C};?U|onvB|~W8W~*}=>U`On0L&rwp{tH#dM9sb$;f0SR$;py8x+(N{!AOJJYQd5 zu$f?12*L{nT|n%FfqA%+)H*{zIHdwT7zD6$bc8hkIc#Wbg#2xf;V5yw_13_^TY$7> zaq0o45M;ExybVYkka94D0Y>vD7`a30fa(BVt*EGIxhT-Xa4|hEE|7{xKxM<`gKh(I zKlO^)#kc?nBbgZ)?_kggCRd>C1S9+H+b4bSn29`C#NY|S2H^#I!^Y<3*Ad<&5O5$R zAVur9|DAAzqrczu)udjNwoKSzn{}{e?5oHtXQ% zcn&f?XiebtT15;)-d5UMZca5yNJv1R;J7*E0fwrZ8^4#nDYVR_&-s1=os8wQC%xmn zuu*i`0)WUH<&U0rFDf#}5oi|3-Z@GIp4siAqocs2ffTE*PYR$y5%}^TG8WohpsI`J z9iWGWg@o*`k3)rv6S?i%xI@{G^x^%RO)7G7&ukKyYkTtsB)CslQ*(3VbaXJ5CfS*j zESLx+Y`6rlA^|iYa4Q47;n+bigSP`MYcyY16#SMCAL-W3%W|Lz%`o}LD(-mJs0=-8l@2NPw}3k$DQ6<)o%=v>K~qcbux zW@l#L`)>1>53nH|ZXc`0y#S*b%VP3}2T*w|KClEJK3vWY^(pfWWlOWOO`twP`-KUp zhlv>)6y&%y0}pEQ8ynNeJp>pBT|2qW6>>sGMg|;;M}nVW{N2%58~QN;(r(}s3&9}4 z;QGEitpX{Ogx~WIj7T>(|Fy)!KMZoOqZg1R@Xb(=}Z#MlMT@E2}!{nV)Z=t(`Cnx6X2%||5 zqpq+EkUW9zB*8u)n8BAmg)tYflK@L$iUX2_j9@z%ZjsBd_er`v)c0r4oC|oeiW1&hV3_A89 z=l~Z(uED_<3NYx_VAInIs&vNjjQ95H=uk8-goNn?p@0Rv_Z6MLei=b!f~*JK{i2Dgb_VKYA26H-qYtQQsi{xSlws*`0~eA0 z18!>P+E#~3xz!@{gn-w*&43oJP#St^fKfqGY_mR)0CWf-5F2&=`bxl=m|QG8*KG1f zc=$a~4?&Ft^bLYf45H3~H^=oQb4~dRqjrw#PS;9PQ(ry4(_M!{($Lif8fEAk`ZRuk ztR1s8NRSAzIT$`k3jnrt2;~>^0qctZjEkv(!4(j#Yn)}%!ZctJ;8NINdO_Tb|BB^u z1os?HfWrMVFfYi!Hwg#ga9uC~xCw0S9HJyG7^bo@2QruZVR#1g_>d4XwlBjGkQ6)X zFp?Y|8o5%F4$?nIwAgn6vrLnyBV~sHa0ZC{4Su;uf9GQ+vx4?1_pf!9fUg; zXj?#Z;C_n+rAneey*Dy3sRmL2T{|>#j*gDdhzBFJ*THPv#_{>=5bN$10q>W z)j-YzCl{DG71c3J@>Gniq&p*^_^Ci^0T&HT(k^&Uxw&8P`+zOPmz3;7J%)M-AqM3e z-V#dtqHeh-`hC(zk7R+isi=JZ2Zy;65#d-{kPBFRP2kx4^X-43^P(1)TVH@4e z5IgyUD{6-?|9N=qPCblJ_^uFrmb9rG7$^bj3CdugH#EsW9|EOmflPWiRZOg}t4m3J z2?^<2j?pqONCPYEu>Tz&-$yVZlxd3^C(WXxqlYr>PC{`U?CdyQ&!UMq*Si2~E}C+H z8fGrZ)BdiJ5wZ2!o#c{l{NS>J;TQ>`ICQtr>b)ereQJ^Om^;oCM!CQwu_vVsbA(}8 zC8fdI+83V@H43R*sh7PP7UKN)Ix;dh@F8c!AukN-m%;IX6%EGP`PGyl^!MXCf50Ar z*nU5bnV?G@~9PIg=9M_(!KxlykL8^{47I*#ybuQ1RUFZCd zZLhlAf|&z|U8L@H$KzSve`XOdAJv}ok8Q^>g6wP-aQP<_qjem$fB*R3XXPix<0)jn z**<4Um`0ps=?RM_{C!AX{l=sJ&#&T(|AL2${>}m*{(Z#$e|lW!-)Dn%uqyDDE&P3D zr#8!#4*Hk+=*~|HILLIXs%Cd~7Xz~V{Blf2-qKL796f;tvdBc+wrG?cY-rMZQhcv{ zjC4ccSc8WQ+MzXnDS*nzI6kaQrE8ouIj?XVcWekj%X7 zx-Z_eFXrW;)~<~%$()~9c3nI>Z}Z)BoF$8Gd8!dlNy#0nI6B%tC@3^-al*9P`<+Sl z$!r*A+q|cjg3_C6AMeFFxliiVVm#)2e|iO+gYoHWVSLpSk520h^L<<_7ghZ%ZXDb8 z*~lv;W#xeN8rRUMs0~|$iq+g_#mAHtcDKhZE*J2A?ACkY&e+o-L@6s<84An+ z^QWioUjpQFY$AEawYFy7#aye!;W<;>#Kj9_u{FUf!}$93)}u$`+sew0D~d;X+6O*= zQ&!5s^>(K3bhEAML~+XeaNO*oBo9+xKlGEg`4Tg;su&q*B%^aV&9&?_zi;~yxMO8} zLS6ri?jM9rlGr)b$UQh%zTA(&?$}R9XWGz{l5M+|>a-UTTvXKF=vy5?bT~h+9K)VO zeY?DZf{abQMgldj#-efh=WPRfdop9=xD})ANAO%bJC(wbj=pQ5Ar6U-nM(9>3 z-*ystmLI7rT}5%w*9j-$d|PQcp1ZR0b#|7&=XYTkU5W)tb905sct>DpKTI7^a5&a> z#JFQssk>~pbjZgsdS$C?pE-GDD0t!BMW>^*5Pl_fbfC@N2Jf4*W3|?swkIupcKtdx z8=Icjtu{NmuIoCSoS)AR>KU1szNP-;W`E;ΞwbH~WaM+2joTFT)lK^7Dd<(! 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

#3M@)0D)GQI7&3rivF%xUeN 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 O5aSD%&Q~GPasLPFvJBh+ literal 0 HcmV?d00001 diff --git a/shaders/dynamic_line_rasterization/base.frag b/shaders/dynamic_line_rasterization/base.frag new file mode 100644 index 000000000..c0f52d12e --- /dev/null +++ b/shaders/dynamic_line_rasterization/base.frag @@ -0,0 +1,27 @@ +#version 450 +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +layout (location = 0) out vec4 outFragColor; +layout( push_constant ) uniform Push_Constants{ + vec4 color; +} pushConstant; + + +void main() { + outFragColor = pushConstant.color; +} diff --git a/shaders/dynamic_line_rasterization/base.vert b/shaders/dynamic_line_rasterization/base.vert new file mode 100644 index 000000000..327223fea --- /dev/null +++ b/shaders/dynamic_line_rasterization/base.vert @@ -0,0 +1,35 @@ +#version 450 +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +layout (location = 0) in vec3 inPos; + +layout (binding = 0) uniform Ubo +{ + mat4 projection; + mat4 view; + mat4 model; +} ubo; + + +out gl_PerVertex { + vec4 gl_Position; +}; + +void main() { + gl_Position = ubo.projection * ubo.view * ubo.model * vec4(inPos.xyz - vec3(0.0f, 1.0f, 0.0f), 1.0f); +} diff --git a/shaders/dynamic_line_rasterization/grid.frag b/shaders/dynamic_line_rasterization/grid.frag new file mode 100644 index 000000000..3936be542 --- /dev/null +++ b/shaders/dynamic_line_rasterization/grid.frag @@ -0,0 +1,58 @@ +#version 450 +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +layout(location = 0) in vec3 nearPoint; +layout(location = 1) in vec3 farPoint; +layout(location = 2) in mat4 view; +layout(location = 6) in mat4 projection; +layout(location = 0) out vec4 outColor; + +vec4 grid(vec3 pos) { + vec2 coord = pos.xz; + vec2 derivative = fwidth(coord); + vec2 grid = abs(fract(coord - 0.5) - 0.5) / derivative; + float line = min(grid.x, grid.y); + float minimumz = min(derivative.y, 1); + float minimumx = min(derivative.x, 1); + vec4 color = vec4(0.5, 0.5, 0.5, 1.0 - min(line, 1.0)); + + if(abs(pos.x) < minimumx) + color.y = 1; + if(abs(pos.z) < minimumz) + color.x = 1; + return color; +} + +float fadeFactor(vec3 pos) { + float z = (projection * view * vec4(pos.xyz, 1.0)).z; + // Empirical values are used to determine when to cut off the grid before moire patterns become visible. + return z * 6 - 0.5; +} + +void main() { + float t = -nearPoint.y / (farPoint.y - nearPoint.y); + vec3 pos = nearPoint + t * (farPoint - nearPoint); + + // Display only the lower plane + if(t < 1) { + vec4 gridColor = grid(pos); + outColor = vec4(gridColor.xyz, gridColor.w * fadeFactor(pos)); + } + else + outColor = vec4(0.0, 0.0, 0.0, 0.0); +} diff --git a/shaders/dynamic_line_rasterization/grid.vert b/shaders/dynamic_line_rasterization/grid.vert new file mode 100644 index 000000000..e3dd87d00 --- /dev/null +++ b/shaders/dynamic_line_rasterization/grid.vert @@ -0,0 +1,55 @@ +#version 450 +/* Copyright (c) 2023, Mobica Limited + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +layout(binding = 0) uniform Ubo { + mat4 projection; + mat4 view; + mat4 model; +} ubo; + +layout(location = 0) out vec3 nearPoint; +layout(location = 1) out vec3 farPoint; +layout(location = 2) out mat4 view; +layout(location = 6) out mat4 projection; + + +vec3 unprojectPoint(float x, float y, float z, mat4 viewProjectionInverse) { + vec4 clipSpacePos = vec4(x, y, z, 1.0); + vec4 eyeSpacePos = viewProjectionInverse * clipSpacePos; + return eyeSpacePos.xyz / eyeSpacePos.w; +} + +vec3 gridPlane[6] = vec3[]( + vec3(1, 1, 0), vec3(-1, -1, 0), vec3(-1, 1, 0), + vec3(-1, -1, 0), vec3(1, 1, 0), vec3(1, -1, 0) +); + +void main() { + vec3 pos = gridPlane[gl_VertexIndex].xyz; + mat4 viewProjectionInverse = inverse(ubo.projection * ubo.view); + + nearPoint = unprojectPoint(pos.x, pos.y, 0.0, viewProjectionInverse); + farPoint = unprojectPoint(pos.x, pos.y, 1.0, viewProjectionInverse); + + view = ubo.view; + projection = ubo.projection; + + gl_Position = vec4(pos, 1.0); + + +} From 15976fa3740ff8b6c60a070780771746d9b6ebd2 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Andreas=20S=C3=BC=C3=9Fenbach?= Date: Mon, 11 Sep 2023 17:09:04 +0200 Subject: [PATCH 12/13] Change vkb::core::HPPDebug and derived classes from a facade over vkb::core::Debug to self-contained classes using vulkan.hpp (#788) --- framework/CMakeLists.txt | 2 +- framework/core/hpp_debug.cpp | 115 ++++++++++ framework/core/hpp_debug.h | 291 ++++++++----------------- framework/core/hpp_vulkan_resource.cpp | 41 ---- framework/core/hpp_vulkan_resource.h | 11 +- 5 files changed, 206 insertions(+), 254 deletions(-) create mode 100644 framework/core/hpp_debug.cpp delete mode 100644 framework/core/hpp_vulkan_resource.cpp diff --git a/framework/CMakeLists.txt b/framework/CMakeLists.txt index 7caebac8a..594cc822c 100644 --- a/framework/CMakeLists.txt +++ b/framework/CMakeLists.txt @@ -304,6 +304,7 @@ set(CORE_FILES core/hpp_buffer.cpp core/hpp_command_buffer.cpp core/hpp_command_pool.cpp + core/hpp_debug.cpp core/hpp_device.cpp core/hpp_image.cpp core/hpp_image_view.cpp @@ -313,7 +314,6 @@ set(CORE_FILES core/hpp_queue.cpp core/hpp_sampler.cpp core/hpp_swapchain.cpp - core/hpp_vulkan_resource.cpp ) set(PLATFORM_FILES diff --git a/framework/core/hpp_debug.cpp b/framework/core/hpp_debug.cpp new file mode 100644 index 000000000..76eb3f6d8 --- /dev/null +++ b/framework/core/hpp_debug.cpp @@ -0,0 +1,115 @@ +/* Copyright (c) 2023, NVIDIA CORPORATION. All rights reserved. + * + * SPDX-License-Identifier: Apache-2.0 + * + * Licensed under the Apache License, Version 2.0 the "License"; + * you may not use this file except in compliance with the License. + * You may obtain a copy of the License at + * + * http://www.apache.org/licenses/LICENSE-2.0 + * + * Unless required by applicable law or agreed to in writing, software + * distributed under the License is distributed on an "AS IS" BASIS, + * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + * See the License for the specific language governing permissions and + * limitations under the License. + */ + +#include "hpp_debug.h" +#include "core/hpp_command_buffer.h" +#include "core/hpp_device.h" + +namespace vkb +{ +namespace core +{ +void HPPDebugUtilsExtDebugUtils::set_debug_name(vk::Device device, vk::ObjectType object_type, uint64_t object_handle, const char *name) const +{ + vk::DebugUtilsObjectNameInfoEXT name_info(object_type, object_handle, name); + device.setDebugUtilsObjectNameEXT(name_info); +} + +void HPPDebugUtilsExtDebugUtils::set_debug_tag( + vk::Device device, vk::ObjectType object_type, uint64_t object_handle, uint64_t tag_name, const void *tag_data, size_t tag_data_size) const +{ + vk::DebugUtilsObjectTagInfoEXT tag_info(object_type, object_handle, tag_name, tag_data_size, tag_data); + device.setDebugUtilsObjectTagEXT(tag_info); +} + +void HPPDebugUtilsExtDebugUtils::cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const +{ + vk::DebugUtilsLabelEXT label_info(name, reinterpret_cast const &>(color)); + command_buffer.beginDebugUtilsLabelEXT(label_info); +} + +void HPPDebugUtilsExtDebugUtils::cmd_end_label(vk::CommandBuffer command_buffer) const +{ + command_buffer.endDebugUtilsLabelEXT(); +} + +void HPPDebugUtilsExtDebugUtils::cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const +{ + vk::DebugUtilsLabelEXT label_info(name, reinterpret_cast const &>(color)); + command_buffer.insertDebugUtilsLabelEXT(label_info); +} + +void HPPDebugMarkerExtDebugUtils::set_debug_name(vk::Device device, vk::ObjectType object_type, uint64_t object_handle, const char *name) const +{ + vk::DebugMarkerObjectNameInfoEXT name_info(vk::debugReportObjectType(object_type), object_handle, name); + device.debugMarkerSetObjectNameEXT(name_info); +} + +void HPPDebugMarkerExtDebugUtils::set_debug_tag( + vk::Device device, vk::ObjectType object_type, uint64_t object_handle, uint64_t tag_name, const void *tag_data, size_t tag_data_size) const +{ + vk::DebugMarkerObjectTagInfoEXT tag_info(vk::debugReportObjectType(object_type), object_handle, tag_name, tag_data_size, tag_data); + device.debugMarkerSetObjectTagEXT(tag_info); +} + +void HPPDebugMarkerExtDebugUtils::cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const +{ + vk::DebugMarkerMarkerInfoEXT marker_info(name, reinterpret_cast const &>(color)); + command_buffer.debugMarkerBeginEXT(marker_info); +} + +void HPPDebugMarkerExtDebugUtils::cmd_end_label(vk::CommandBuffer command_buffer) const +{ + command_buffer.debugMarkerEndEXT(); +} + +void HPPDebugMarkerExtDebugUtils::cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const +{ + vk::DebugMarkerMarkerInfoEXT marker_info(name, reinterpret_cast const &>(color)); + command_buffer.debugMarkerInsertEXT(marker_info); +} + +HPPScopedDebugLabel::HPPScopedDebugLabel(const HPPDebugUtils &debug_utils, + vk::CommandBuffer command_buffer, + std::string const &name, + glm::vec4 const &color) : + debug_utils{&debug_utils}, command_buffer{VK_NULL_HANDLE} +{ + if (!name.empty()) + { + assert(command_buffer); + this->command_buffer = command_buffer; + + debug_utils.cmd_begin_label(command_buffer, name.c_str(), color); + } +} + +HPPScopedDebugLabel::HPPScopedDebugLabel(const vkb::core::HPPCommandBuffer &command_buffer, std::string const &name, glm::vec4 const &color) : + HPPScopedDebugLabel{command_buffer.get_device().get_debug_utils(), command_buffer.get_handle(), name, color} +{ +} + +HPPScopedDebugLabel::~HPPScopedDebugLabel() +{ + if (command_buffer) + { + debug_utils->cmd_end_label(command_buffer); + } +} + +} // namespace core +} // namespace vkb \ No newline at end of file diff --git a/framework/core/hpp_debug.h b/framework/core/hpp_debug.h index 945b9f5aa..5a385d714 100644 --- a/framework/core/hpp_debug.h +++ b/framework/core/hpp_debug.h @@ -17,8 +17,7 @@ #pragma once -#include - +#include #include namespace vkb @@ -28,243 +27,123 @@ namespace core class HPPCommandBuffer; /** - * @brief facade class around vkb::DebugUtils, providing a vulkan.hpp-based interface - * - * See vkb::DebugUtils for documentation + * @brief An interface over platform-specific debug extensions. */ -class HPPDebugUtils : private vkb::DebugUtils +class HPPDebugUtils { public: - void set_debug_name(VkDevice device, - VkObjectType object_type, - uint64_t object_handle, - const char *name) const override - { - set_debug_name( - static_cast(device), static_cast(object_type), object_handle, name); - } - - void set_debug_tag(VkDevice device, - VkObjectType object_type, - uint64_t object_handle, - uint64_t tag_name, - const void *tag_data, - size_t tag_data_size) const override - { - set_debug_tag(static_cast(device), - static_cast(object_type), - object_handle, - tag_name, - tag_data, - tag_data_size); - } - - void cmd_begin_label(VkCommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - cmd_begin_label(static_cast(command_buffer), name, color); - } - - void cmd_end_label(VkCommandBuffer command_buffer) const override - { - cmd_end_label(static_cast(command_buffer)); - } - - void cmd_insert_label(VkCommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - cmd_insert_label(static_cast(command_buffer), name, color); - } - - virtual void set_debug_name(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - const char *name) const = 0; - virtual void set_debug_tag(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - uint64_t tag_name, - const void *tag_data, - size_t tag_data_size) const = 0; - virtual void cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color = {}) const = 0; - virtual void cmd_end_label(vk::CommandBuffer command_buffer) const = 0; - virtual void cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color = {}) const = 0; + virtual ~HPPDebugUtils() = default; + + /** + * @brief Sets the debug name for a Vulkan object. + */ + virtual void set_debug_name(vk::Device device, vk::ObjectType object_type, uint64_t object_handle, const char *name) const = 0; + + /** + * @brief Tags the given Vulkan object with some data. + */ + virtual void set_debug_tag( + vk::Device device, vk::ObjectType object_type, uint64_t object_handle, uint64_t tag_name, const void *tag_data, size_t tag_data_size) const = 0; + + /** + * @brief Inserts a command to begin a new debug label/marker scope. + */ + virtual void cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color = {}) const = 0; + + /** + * @brief Inserts a command to end the current debug label/marker scope. + */ + virtual void cmd_end_label(vk::CommandBuffer command_buffer) const = 0; + + /** + * @brief Inserts a (non-scoped) debug label/marker in the command buffer. + */ + virtual void cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color = {}) const = 0; }; /** - * @brief facade class around vkb::DebugUtilsExtDebugUtils, providing a vulkan.hpp-based interface - * - * See vkb::DebugUtilsExtDebugUtils for documentation + * @brief HPPDebugUtils implemented on top of VK_EXT_debug_utils. */ class HPPDebugUtilsExtDebugUtils final : public vkb::core::HPPDebugUtils { public: - virtual void set_debug_name(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - const char *name) const override - { - reinterpret_cast(this)->set_debug_name( - static_cast(device), static_cast(object_type), object_handle, name); - } - - virtual void set_debug_tag(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - uint64_t tag_name, - const void *tag_data, - size_t tag_data_size) const override - { - reinterpret_cast(this)->set_debug_tag( - static_cast(device), - static_cast(object_type), - object_handle, - tag_name, - tag_data, - tag_data_size); - } - - virtual void - cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_begin_label( - static_cast(command_buffer), name, color); - } - - virtual void cmd_end_label(vk::CommandBuffer command_buffer) const override - { - reinterpret_cast(this)->cmd_end_label( - static_cast(command_buffer)); - } - - virtual void - cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_insert_label( - static_cast(command_buffer), name, color); - } + ~HPPDebugUtilsExtDebugUtils() override = default; + + void set_debug_name(vk::Device device, vk::ObjectType object_type, uint64_t object_handle, const char *name) const override; + + void set_debug_tag( + vk::Device device, vk::ObjectType object_type, uint64_t object_handle, uint64_t tag_name, const void *tag_data, size_t tag_data_size) const override; + + void cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const override; + + void cmd_end_label(vk::CommandBuffer command_buffer) const override; + + void cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const override; }; /** - * @brief facade class around vkb::DebugMarkerExtDebugUtils, providing a vulkan.hpp-based interface - * - * See vkb::DebugMarkerExtDebugUtils for documentation + * @brief HPPDebugUtils implemented on top of VK_EXT_debug_marker. */ class HPPDebugMarkerExtDebugUtils final : public vkb::core::HPPDebugUtils { public: - virtual void set_debug_name(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - const char *name) const override - { - reinterpret_cast(this)->set_debug_name( - static_cast(device), static_cast(object_type), object_handle, name); - } - - virtual void set_debug_tag(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - uint64_t tag_name, - const void *tag_data, - size_t tag_data_size) const override - { - reinterpret_cast(this)->set_debug_tag( - static_cast(device), - static_cast(object_type), - object_handle, - tag_name, - tag_data, - tag_data_size); - } - - virtual void - cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_begin_label( - static_cast(command_buffer), name, color); - } - - virtual void cmd_end_label(vk::CommandBuffer command_buffer) const override - { - reinterpret_cast(this)->cmd_end_label( - static_cast(command_buffer)); - } - - virtual void - cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_insert_label( - static_cast(command_buffer), name, color); - } + ~HPPDebugMarkerExtDebugUtils() override = default; + + void set_debug_name(vk::Device device, vk::ObjectType object_type, uint64_t object_handle, const char *name) const override; + + void set_debug_tag( + vk::Device device, vk::ObjectType object_type, uint64_t object_handle, uint64_t tag_name, const void *tag_data, size_t tag_data_size) const override; + + void cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const override; + + void cmd_end_label(vk::CommandBuffer command_buffer) const override; + + void cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 const &color) const override; }; /** - * @brief facade class around vkb::DummyDebugUtils, providing a vulkan.hpp-based interface - * - * See vkb::DummyDebugUtils for documentation + * @brief No-op HPPDebugUtils. */ class HPPDummyDebugUtils final : public vkb::core::HPPDebugUtils { public: - virtual void set_debug_name(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - const char *name) const override - { - reinterpret_cast(this)->set_debug_name( - static_cast(device), static_cast(object_type), object_handle, name); - } - - virtual void set_debug_tag(vk::Device device, - vk::ObjectType object_type, - uint64_t object_handle, - uint64_t tag_name, - const void *tag_data, - size_t tag_data_size) const override - { - reinterpret_cast(this)->set_debug_tag(static_cast(device), - static_cast(object_type), - object_handle, - tag_name, - tag_data, - tag_data_size); - } - - virtual void - cmd_begin_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_begin_label( - static_cast(command_buffer), name, color); - } - - virtual void cmd_end_label(vk::CommandBuffer command_buffer) const override - { - reinterpret_cast(this)->cmd_end_label( - static_cast(command_buffer)); - } - - virtual void - cmd_insert_label(vk::CommandBuffer command_buffer, const char *name, glm::vec4 color) const override - { - reinterpret_cast(this)->cmd_insert_label( - static_cast(command_buffer), name, color); - } + ~HPPDummyDebugUtils() override = default; + + inline void set_debug_name(vk::Device, vk::ObjectType, uint64_t, const char *) const override + {} + + inline void set_debug_tag(vk::Device, vk::ObjectType, uint64_t, uint64_t, const void *, size_t) const override + {} + + inline void cmd_begin_label(vk::CommandBuffer, const char *, glm::vec4 const &) const override + {} + + inline void cmd_end_label(vk::CommandBuffer) const override + {} + + inline void cmd_insert_label(vk::CommandBuffer, const char *, glm::vec4 const &) const override + {} }; /** - * @brief facade class around vkb::DummyDebugUtils, providing a vulkan.hpp-based interface - * - * See vkb::DummyDebugUtils for documentation + * @brief A RAII debug label. + * If any of EXT_debug_utils or EXT_debug_marker is available, this: + * - Begins a debug label / marker on construction + * - Ends it on destruction */ class HPPScopedDebugLabel final { public: - HPPScopedDebugLabel(const vkb::core::HPPCommandBuffer &command_buffer, const char *name, glm::vec4 color = {}) : - scopedDebugLabel(reinterpret_cast(command_buffer), name, color) - {} + HPPScopedDebugLabel(const vkb::core::HPPDebugUtils &debug_utils, vk::CommandBuffer command_buffer, std::string const &name, glm::vec4 const &color = {}); + + HPPScopedDebugLabel(const vkb::core::HPPCommandBuffer &command_buffer, std::string const &name, glm::vec4 const &color = {}); + + ~HPPScopedDebugLabel(); private: - ScopedDebugLabel scopedDebugLabel; + const vkb::core::HPPDebugUtils *debug_utils; + vk::CommandBuffer command_buffer; }; } // namespace core -} // namespace vkb +} // namespace vkb \ No newline at end of file diff --git a/framework/core/hpp_vulkan_resource.cpp b/framework/core/hpp_vulkan_resource.cpp deleted file mode 100644 index d8945d093..000000000 --- a/framework/core/hpp_vulkan_resource.cpp +++ /dev/null @@ -1,41 +0,0 @@ -/* Copyright (c) 2022, NVIDIA CORPORATION. All rights reserved. - * - * SPDX-License-Identifier: Apache-2.0 - * - * Licensed under the Apache License, Version 2.0 the "License"; - * you may not use this file except in compliance with the License. - * You may obtain a copy of the License at - * - * http://www.apache.org/licenses/LICENSE-2.0 - * - * Unless required by applicable law or agreed to in writing, software - * distributed under the License is distributed on an "AS IS" BASIS, - * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. - * See the License for the specific language governing permissions and - * limitations under the License. - */ - -#include - -#include - -namespace vkb -{ -namespace core -{ -namespace detail -{ -void set_debug_name(const HPPDevice *device, vk::ObjectType object_type, uint64_t handle, const char *debug_name) -{ - if (!debug_name || *debug_name == '\0' || !device) - { - // Can't set name, or no point in setting an empty name - return; - } - - device->get_debug_utils().set_debug_name(device->get_handle(), object_type, handle, debug_name); -} - -} // namespace detail -} // namespace core -} // namespace vkb \ No newline at end of file diff --git a/framework/core/hpp_vulkan_resource.h b/framework/core/hpp_vulkan_resource.h index 39e27b326..59cb79a27 100644 --- a/framework/core/hpp_vulkan_resource.h +++ b/framework/core/hpp_vulkan_resource.h @@ -26,11 +26,6 @@ namespace core { class HPPDevice; -namespace detail -{ -void set_debug_name(const HPPDevice *device, vk::ObjectType object_type, uint64_t handle, const char *debug_name); -} - /// Inherit this for any Vulkan object with a handle of type `HPPHandle`. /// /// This allows the derived class to store a Vulkan handle, and also a pointer to the parent vkb::core::Device. @@ -114,7 +109,11 @@ class HPPVulkanResource inline void set_debug_name(const std::string &name) { debug_name = name; - detail::set_debug_name(device, HPPHandle::objectType, get_handle_u64(), debug_name.c_str()); + + if (device && !debug_name.empty()) + { + device->get_debug_utils().set_debug_name(device->get_handle(), HPPHandle::objectType, get_handle_u64(), debug_name.c_str()); + } } private: From 5e4e3d20b3a86f7814e026042341cf34ef530edd Mon Sep 17 00:00:00 2001 From: Steven Winston Date: Mon, 11 Sep 2023 08:11:15 -0700 Subject: [PATCH 13/13] Antora (#787) * Add VK_EXT_shader_object sample * add the ability to include samples in the antora project. This allows a debug build to copy the files to the correct place for antora to build the site. * fix antora * Try using a flag to run antora build. --------- Co-authored-by: Coleman Jonas --- CMakeLists.txt | 2 + antora/CMakeLists.txt | 53 +++++++++++++++++ antora/antora.yml | 13 +++++ antora/modules/ROOT/nav.adoc | 107 +++++++++++++++++++++++++++++++++++ 4 files changed, 175 insertions(+) create mode 100644 antora/CMakeLists.txt create mode 100644 antora/antora.yml create mode 100644 antora/modules/ROOT/nav.adoc diff --git a/CMakeLists.txt b/CMakeLists.txt index a052a8437..ca650bcce 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -87,3 +87,5 @@ endif() # Add vulkan app (runs all samples) add_subdirectory(app) + +add_subdirectory(antora) diff --git a/antora/CMakeLists.txt b/antora/CMakeLists.txt new file mode 100644 index 000000000..50c19bec5 --- /dev/null +++ b/antora/CMakeLists.txt @@ -0,0 +1,53 @@ +# Copyright 2022-2023 The Khronos Group Inc. +# SPDX-License-Identifier: Apache-2.0 + +# Configure Vulkan Guide Antora tree with transformed markup files. +# Branch selection may come later. For now it is the current branch. + +function(gatherAntoraAssets) + set(DIRS_TO_SEARCH + app + assets + components + docs + framework + samples + scripts + shaders + tests + ) + set(PAGES_DIR_SEARCH) + set(IMAGES_DIR_SEARCH) + foreach (DIR ${DIRS_TO_SEARCH}) + list(APPEND PAGES_DIR_SEARCH ${CMAKE_SOURCE_DIR}/${DIR}/*.adoc) + list(APPEND IMAGES_DIR_SEARCH ${CMAKE_SOURCE_DIR}/${DIR}/*.jpg ${CMAKE_SOURCE_DIR}/${DIR}/*.png ${CMAKE_SOURCE_DIR}/${DIR}/*.gif) + endforeach () + file(GLOB PAGES ${CMAKE_SOURCE_DIR}/*.adoc) + file(GLOB IMAGES ${CMAKE_SOURCE_DIR}/*.jpg ${CMAKE_SOURCE_DIR}/*.png ${CMAKE_SOURCE_DIR}/*.gif) + file(GLOB_RECURSE PAGES_R ${PAGES_DIR_SEARCH}) + file(GLOB_RECURSE IMAGES_R ${IMAGES_DIR_SEARCH}) + + foreach(page ${PAGES}) + file(COPY ${page} DESTINATION ${CMAKE_CURRENT_LIST_DIR}/modules/ROOT/pages) + endforeach () + + foreach(image ${IMAGES}) + file(COPY ${image} DESTINATION ${CMAKE_CURRENT_LIST_DIR}/modules/ROOT/images) + endforeach () + + foreach (page ${PAGES_R}) + file(RELATIVE_PATH relpage ${CMAKE_SOURCE_DIR} ${page}) + get_filename_component(directory ${relpage} DIRECTORY) + file(COPY ${page} DESTINATION ${CMAKE_CURRENT_LIST_DIR}/modules/ROOT/pages/${directory}) + endforeach () + + foreach(image ${IMAGES_R}) + file(RELATIVE_PATH relpage ${CMAKE_SOURCE_DIR} ${image}) + get_filename_component(directory ${relpage} DIRECTORY) + file(COPY ${image} DESTINATION ${CMAKE_CURRENT_LIST_DIR}/modules/ROOT/images/${directory}) + endforeach () +endfunction() + +if(VKB_GENERATE_ANTORA_SITE) + gatherAntoraAssets() +endif() \ No newline at end of file diff --git a/antora/antora.yml b/antora/antora.yml new file mode 100644 index 000000000..63585d2f5 --- /dev/null +++ b/antora/antora.yml @@ -0,0 +1,13 @@ +# Copyright 2022-2023 The Khronos Group Inc. +# SPDX-License-Identifier: Apache-2.0 + +name: samples +title: Vulkan Samples +version: latest +start_page: README.adoc +asciidoc: + attributes: + source-language: asciidoc@ + table-caption: false +nav: + - modules/ROOT/nav.adoc diff --git a/antora/modules/ROOT/nav.adoc b/antora/modules/ROOT/nav.adoc new file mode 100644 index 000000000..cb9ca365f --- /dev/null +++ b/antora/modules/ROOT/nav.adoc @@ -0,0 +1,107 @@ +//// +- Copyright (c) 2023, Holochip Inc +- +- SPDX-License-Identifier: Apache-2.0 +- +- Licensed under the Apache License, Version 2.0 the "License"; +- you may not use this file except in compliance with the License. +- You may obtain a copy of the License at +- +- http://www.apache.org/licenses/LICENSE-2.0 +- +- Unless required by applicable law or agreed to in writing, software +- distributed under the License is distributed on an "AS IS" BASIS, +- WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +- See the License for the specific language governing permissions and +- limitations under the License. +- +//// +* xref:README.adoc[Readme] +* xref:vulkan_basics.adoc[Vulkan basics] +* xref:framework/README.adoc[Sample framework] +* xref:api/README.adoc[Api usage samples] +** xref:api/compute_nbody/README.adoc[Compute N-body❕] +*** xref:api/hpp_compute_nbody/README.adoc[hpp compute nbody❕] +** xref:api/dynamic_uniform_buffers/README.adoc[Dynamic uniform buffers❕] +*** xref:api/hpp_dynamic_uniform_buffers/README.adoc[hpp dynamic uniform buffers❕] +** xref:api/hdr/README.adoc[HDR❕] +*** xref:api/hpp_hdr/README.adoc[hpp hdr❕] +** xref:api/hello_triangle/README.adoc[Hello Triangle❕] +*** xref:api/hpp_hello_triangle/README.adoc[hpp hello triangle❕] +** xref:api/hlsl_shaders/README.adoc[HLSL Shaders] +*** xref:api/hpp_hlsl_shaders/README.adoc[hpp hlsl shaders] +** xref:api/instancing/README.adoc[Instancing❕] +*** xref:api/hpp_instancing/README.adoc[hpp instancing❕] +** xref:api/separate_image_sampler/README.adoc[Separate image sampler] +*** xref:api/hpp_separate_image_sampler/README.adoc[hpp separate image sampler] +** xref:api/terrain_tessellation/README.adoc[Terrain tessellation❕] +*** xref:api/hpp_terrain_tessellation/README.adoc[hpp terrain tessellation❕] +** xref:api/texture_loading/README.adoc[Texture loading❕] +*** xref:api/hpp_texture_loading/README.adoc[hpp texture loading❕] +** xref:api/texture_mipmap_generation/README.adoc[Texture mipmap generation] +*** xref:api/hpp_texture_mipmap_generation/README.adoc[hpp texture mipmap generation] +** xref:api/timestamp_queries/README.adoc[Timestamp queries] +*** xref:api/hpp_timestamp_queries/README.adoc[hpp timestamp queries] +* xref:extensions/README.adoc[Extension usage samples] +** xref:extensions/buffer_device_address/README.adoc[Buffer device address] +** xref:extensions/calibrated_timestamps/README.adoc[Calibrated timestamps] +** xref:extensions/conditional_rendering/README.adoc[Conditional rendering] +** xref:extensions/conservative_rasterization/README.adoc[Conservative rasterization❕] +** xref:extensions/debug_utils/README.adoc[Debug utils] +** xref:extensions/descriptor_buffer_basic/README.adoc[Descriptor buffer basic] +** xref:extensions/descriptor_indexing/README.adoc[Descriptor indexing] +** xref:extensions/dynamic_rendering/README.adoc[Dynamic rendering] +** xref:extensions/extended_dynamic_state2/README.adoc[Extended dynamic state2] +** xref:extensions/fragment_shader_barycentric/README.adoc[Fragment shader barycentric] +** xref:extensions/fragment_shading_rate/README.adoc[Fragment shading rate❕] +** xref:extensions/fragment_shading_rate_dynamic/README.adoc[Fragment shading rate dynamic] +** xref:extensions/full_screen_exclusive/README.adoc[Full screen exclusive] +** xref:extensions/graphics_pipeline_library/README.adoc[Graphics pipeline library] +** xref:extensions/logic_op_dynamic_state/README.adoc[Logic op dynamic state] +** xref:extensions/memory_budget/README.adoc[Memory budget] +** xref:extensions/mesh_shader_culling/README.adoc[Mesh shader culling] +** xref:extensions/mesh_shading/README.adoc[Mesh shading❕] +** xref:extensions/open_cl_interop/README.adoc[OpenCL interop (Cross-vendor)] +** xref:extensions/open_cl_interop_arm/README.adoc[OpenCl interop (Arm)] +** xref:extensions/open_gl_interop/README.adoc[OpenGL interop❕] +** xref:extensions/portability/README.adoc[Portability] +** xref:extensions/push_descriptors/README.adoc[Push descriptors❕] +** xref:extensions/raytracing_basic/README.adoc[Raytracing basic❕] +** xref:extensions/raytracing_extended/README.adoc[Raytracing extended] +** xref:extensions/ray_queries/README.adoc[Ray queries❕] +** xref:extensions/ray_tracing_reflection/README.adoc[Ray tracing reflection] +** xref:extensions/synchronization_2/README.adoc[Synchronization 2❕] +** xref:extensions/timeline_semaphore/README.adoc[Timeline semaphore] +** xref:extensions/vertex_dynamic_state/README.adoc[Vertex dynamic state] +* xref:performance/README.adoc[Performance samples] +** xref:performance/16bit_arithmetic/README.adoc[16bit arithmetic] +** xref:performance/16bit_storage_input_output/README.adoc[16bit storage input output] +** xref:performance/afbc/README.adoc[AFBC] +** xref:performance/async_compute/README.adoc[Async compute] +** xref:performance/command_buffer_usage/README.adoc[Command buffer usage] +** xref:performance/constant_data/README.adoc[Constant data] +** xref:performance/descriptor_management/README.adoc[Descriptor management] +** xref:performance/layout_transitions/README.adoc[Layout transitions] +** xref:performance/msaa/README.adoc[MSAA] +** xref:performance/multithreading_render_passes/README.adoc[Multithreading render passes] +** xref:performance/multi_draw_indirect/README.adoc[Multi draw indirect] +** xref:performance/pipeline_barriers/README.adoc[Pipeline barriers] +** xref:performance/pipeline_cache/README.adoc[Pipeline cache] +*** xref:performance/hpp_pipeline_cache/README.adoc[hpp pipeline cache] +** xref:performance/render_passes/README.adoc[Render passes] +** xref:performance/specialization_constants/README.adoc[Specialization constants] +** xref:performance/subpasses/README.adoc[Subpasses] +** xref:performance/surface_rotation/README.adoc[Surface rotation] +** xref:performance/swapchain_images/README.adoc[Swapchain images] +*** xref:performance/hpp_swapchain_images/README.adoc[hpp swapchain images] +** xref:performance/texture_compression_basisu/README.adoc[Texture compression basisu] +** xref:performance/texture_compression_comparison/README.adoc[Texture compression comparison❕] +** xref:performance/wait_idle/README.adoc[Wait idle] +* xref:tooling/README.adoc[Tooling samples] +** xref:tooling/profiles/README.adoc[Profiles] +* xref:docs/README.adoc[General documentation] +** xref:docs/build.adoc[Build guide] +** xref:docs/create_sample.adoc[Creating a new sample] +** xref:docs/debug_graphs.adoc[Debug graphics] +** xref:docs/memory_limits.adoc[Memory limits] +** xref:docs/misc.adoc[Miscellaneous] \ No newline at end of file