如何将ncnn代码通过Vulkan-Loader为动态加载Vulkan的形式?

摘要:将ncnn静态链接vulkan改成动态加载vulkan的形式,用到了KhronosGroup组织下的Vulkan-SDK里面的CPP部分。
原本你写的程序是静态链接的系统的vulkan-1.dll,如果系统不存在vulkan-1.dll,则会直接崩溃。 关于将ncnn静态链接vulkan改成动态加载vulkan的形式,然后提供这两个函数 bool ncnn::has_vulkan(); void ncnn::use_vulkan(bool); 请教过ncnn的作者nihui,她对此issue表示不以为意,没有必要,优先级不高。 那就只有自己动手丰衣足食了。 本文的目标是将其改为动态加载的方式,用到了KhronosGroup组织下的Vulkan-SDK里面的CPP部分,即vulkan.hpp 本人的上一篇文章(https://www.cnblogs.com/hyb1/p/17361775.html)说的是如何动态判断是否存在vulkan-1.dll并且再加载的过程, 但是没提到如何针对现有的项目改动,比如说现在的ncnn含有的大量C Style的vulkan函数符号直接调用的代码,如果直接修改,需要将每一个vulkan函数的地方都加一个前缀,类似这样: mVulkanDispatchLoaderDynamic->vkCreateInstance(&instanceCreateInfo, 0, &instance); mVulkanDispatchLoaderDynamic->vkEnumeratePhysicalDevices(g_instance, &physicalDeviceCount, 0); 跨越了很多.cpp和.h文件,修改起来相当麻烦,而且还会遇到全局变量是否有效等等问题。 废话不多说,直接说总结: Vulkan Loader提供了多个宏定义:VULKAN_HPP_STORAGE_API和VULKAN_HPP_STORAGE_SHARED和VULKAN_HPP_STORAGE_SHARED_EXPORT 它们是vulkan.hpp中用来控制vulkan函数的存储类别的宏。 VULKAN_HPP_STORAGE_API: 这个宏用来指定vulkan函数的存储类别,比如__declspec(dllexport)或者__declspec(dllimport)。 这个宏可以在编译时由外部定义,以便于将vulkan函数导出或者导入。 VULKAN_HPP_STORAGE_SHARED: 这个宏用来启用动态链接库的模式,即将vulkan函数作为dll的导出或者导入。 如果定义了这个宏,那么VULKAN_HPP_STORAGE_API会根据VULKAN_HPP_STORAGE_SHARED_EXPORT是否定义来自动设置为__declspec(dllexport)或者__declspec(dllimport)。 如果没有定义这个宏,那么VULKAN_HPP_STORAGE_API会被设置为空。 VULKAN_HPP_STORAGE_SHARED_EXPORT: 这个宏用来控制动态链接库的模式下,vulkan函数是作为dll的导出还是导入。 如果定义了这个宏,那么VULKAN_HPP_STORAGE_API会被设置为__declspec(dllexport),表示vulkan函数是dll的导出。 如果没有定义这个宏,那么VULKAN_HPP_STORAGE_API会被设置为__declspec(dllimport),表示vulkan函数是dll的导入。 如果你是希望通过vk::DispatchLoaderDynamic加载到defaultDispatchLoaderDynamic之后,仍旧按照静态链接的模式来编写代码,但是不是链接到真正的vulkan-1.dll上,而是由vulkan-loader生成的符号,应该这样做(重点来了): 1. 定义VULKAN_HPP_ENABLE_DYNAMIC_LOADER_TOOL和VULKAN_HPP_DISPATCH_LOADER_DYNAMIC两个宏,以启用动态加载vulkan库和函数的功能。 2. 不要定义VULKAN_HPP_STORAGE_SHARED和VULKAN_HPP_STORAGE_SHARED_EXPORT两个宏,以避免使用dll的模式。 3. 使用vk::DynamicLoader类来动态加载vulkan-loader生成的符号,并且获取vkGetInstanceProcAddr函数的地址。 4. 调用VULKAN_HPP_DEFAULT_DISPATCHER.init(vkGetInstanceProcAddr)来初始化默认的函数指针封装对象。
阅读全文