Small personal helper library for Vulkan usage in c. Simplify several common tasks, as image or buffers handling, swapchain initialization, etc.
vkh is used by [vkvg] internally and also to speed up vulkan context creation in samples.
git clone https://github.com/jpbruyere/vkhelpers.git
cd vkhelpers
mkdir build
cd build
cmake ..
make && make install
- clone vkh as a subdirectory of your root dir.
- in your main CMakeFile, add
add_subdirectory (vkhelpers)
- add to your TARGET_INCLUDE_DIRECTORIES
${CMAKE_CURRENT_SOURCE_DIR}/vkhelpers/include
and if you want to bypass opaque pointers and be able to address fields of internal structures, add also${CMAKE_CURRENT_SOURCE_DIR}/vkhelpers/src
. - to link vkh staticaly, add to TARGET_LINK_LIBRARIES
vkh_static
orvkh_shared
to link it as a shared library.
#include "vkh.h"
#include "vkh_phyinfo.h"
void init_vulkan () {
const char* layers [] = {"VK_LAYER_KHRONOS_validation"};
const char* exts [] = {"VK_EXT_debug_utils"};
VkhApp app = vkh_app_create ("appname", 1, layers, 1, exts);
VkhPhyInfo is an helper structure that will store common usefull physical device informations, queues flags, memory properties in a single call for all the device present on the machine.
VkhPhyInfo* phys = vkh_app_get_phyinfos (e->app, &phyCount, surf);
Once you have an array of VkhPhyInfo's, you have several functions to inspect available devices:
for (uint i=0; i<phyCount; i++) {
//check VkPhysicalDeviceProperties
VkPhysicalDeviceProperties pdp = vkh_phyinfo_get_properties (phys[i]);
//get VkPhysicalDeviceMemoryProperties
VkPhysicalDeviceMemoryProperties mp = vkh_phyinfo_get_memory_properties (phys[i]);
//get queue properties
VkQueueFamilyProperties* vkh_phyinfo_get_queues_props(phys[i], &qCount);
VkhPhyInfo structure has the array of VkQueueFamilyProperties that has already be parsed two times to detect available queues types. First vkh will try to find a dedicated queue for each queue types (Graphic, Transfer, Compute) and if a type has no dedicated candidate, it will try to find a queue with the requested flag among others. Also if you submit a valid VkSurfaceKHR to vkh_app_get_phyinfos
, presentation support will be queried for all graphic queues. The result of this search may be fetched with:
vkh_phyinfo_get_queue_fam_indices (phy, &presentQ, &graphQ, &transQ, &compQ);
vkh has one function per queue type that use the result of this search. They may be safely called for checking queue availability, it will return false on failure. On success, the queue count of the VkQueueFamilyProperties
of phyinfo will be decreased.
if (vkh_phyinfo_create_presentable_queues (pi, 1, qPriorities, &pQueueInfos[qCount]))
qCount++;
if (vkh_phyinfo_create_compute_queues (pi, 1, qPriorities, &pQueueInfos[qCount]))
qCount++;
if (vkh_phyinfo_create_transfer_queues (pi, 1, qPriorities, &pQueueInfos[qCount]))
qCount++;
To override the vkh default queue selection, create queues with your own family index:
vkh_phyinfo_create_queues (phy, qFam, queueCount, priorities, &qInfo) {
Be aware that the queue count of the vkhinfo structure is decreased to keep track of remaining queues. PhyInfo pointer has to be freed when no longuer in use. Usualy at the end of the vulkan initialization.
vkh_app_free_phyinfos (phyCount, phys);
char const * dex [] = {"VK_KHR_swapchain"};
VkPhysicalDeviceFeatures enabledFeatures = { .fillModeNonSolid = true };
VkDeviceCreateInfo device_info = {
.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
.queueCreateInfoCount = qCount,
.pQueueCreateInfos = (VkDeviceQueueCreateInfo*)&pQueueInfos,
.enabledExtensionCount = enabledExtsCount,
.ppEnabledExtensionNames = dex,
.pEnabledFeatures = &enabledFeatures};
VkhDevice dev = vkh_device_create(e->app, pi, &device_info);
VkhPresenter will help getting rapidly something on screen, it handles the swapchain.
VkhPresenter present = vkh_presenter_create (dev, pi->pQueue, surf, width, height, VK_FORMAT_B8G8R8A8_UNORM, VK_PRESENT_MODE_MAILBOX_KHR);
//create a blitting command buffer per swapchain images with
vkh_presenter_build_blit_cmd (present, vkvg_surface_get_vk_image(surf), width, height);
while (running) {
if (!vkh_presenter_draw (present))
//on draw failed, swapchain is automatically rebuilt
vkh_presenter_build_blit_cmd (present, vkvg_surface_get_vk_image(surf), width, height);
}
TODO
TODO