Skip to content

A minimal Vulkan deferred rendering demonstration.

License

Notifications You must be signed in to change notification settings

stripe2933/vk-deferred

Repository files navigation

vk-deferred

Build

Running screenshot

A minimal Vulkan deferred rendering demonstration.

Usage

This project requires support for C++20 modules and the C++23 standard library. The supported compiler is:

  • Clang 18.1.2
  • MSVC 19.40

Additionally, the following build tools are required:

  • CMake 3.30
  • Ninja 1.11

Dependencies

This project depends on:

Dependencies will be automatically fetched using vcpkg.

Build Steps

Tip

This project uses GitHub Runner to ensure build compatibility on Windows (with MSVC), macOS and Linux (with Clang). You can check the workflow files in the .github/workflows folder.

This project, along with its dependency vku, supports vcpkg for dependency management. Follow these steps to build the project:

git clone https://github.com/stripe2933/vk-deferred
cd vk-deferred
cmake --preset=default # Or use your own configuration preset that inherits from the "default" preset.
cmake --build build -t vk-deferred

The executable will be located in the build folder.

Shader compilation

All shaders are located in the shaders folder. They are automatically compiled to SPIR-V and embedded into the binary executable during the build.

About Render Pass and Attachment Images

Since Vulkan allows explicit management of resource usage and memory properties, all attachment images used only in the render pass can be created with vk::ImageUsageFlagBits::eTransientAttachment. Additionally, if the system uses a tile-based GPU, you can use vk::MemoryPropertyFlagBits::eLazilyAllocated to avoid expensive memory allocation for the entire image (only each tile region will be allocated when the image is accessed). Therefore, the memory consumption of the G-buffer in Vulkan is zero-cost (only swapchain images are allocated) on TBGPU, which is hard to be achieved in OpenGL.

The project has a render pass with three subpasses. Each subpass does the following:

  1. G-buffer construction: The G-buffer stores the rendered meshes' position and normal (for simplicity, other fragment information like albedo and specular are not used) in the first and second color attachment images. It also uses the depth-stencil attachment to render depth and stencil=1. G-buffer
  2. Lighting using light volume: In this subpass, spheres corresponding to the light volumes are rendered (See LearnOpenGL for details) only for fragments with stencil=1. The depth-stencil attachment is reused from the previous subpass, so light accumulation occurs only in previously rendered fragments. Since the swapchain attachments are in SRGB format, light cannot be accumulated directly (it should be done in the linear color space), so the B10G11R11UfloatPack32 format is used for the color attachment (also preserving additional HDR info). Light volume rendering Note: Only 20 light volumes are used in the above image to visualize the volume spheres. In the actual implementation, many more light volumes can be used.
  3. Tone mapping: As mentioned earlier, the accumulated result is in HDR format, so it should be tone-mapped to the SRGB format. REC.709 tone mapping is used.

Data Flow Visualization

Data flow

Screenshot from Xcode Metal Debugger.

Note that the G-buffer's position and normal attachment images' loadOp are eDontCare, because only rendered fragments will be referenced in the next subpass due to the stencil test. Similarly, swapchain images have loadOp=eDontCare because all regions will be overwritten using a fullscreen quad.

License

This project is licensed under the MIT License. See the LICENSE file for details.

About

A minimal Vulkan deferred rendering demonstration.

Topics

Resources

License

Stars

Watchers

Forks