c++ vulkan glsl中的非均匀纹理访问

ecfdbz9o  于 2023-04-08  发布在  其他
关注(0)|答案(1)|浏览(100)

我试图写一个计算着色器,光线跟踪的图像,像素的yz平面样本的权利,从图像A,那些在左边的图像B。
我不想对两个图像都进行采样,因此我尝试通过以下操作使用非统一访问:
texture(textures[nonuniformEXT(sampler_id)], vec2(0.5));
并在着色器中启用相关扩展。这将触发以下验证层错误:

Message: Validation Error: [ VUID-VkShaderModuleCreateInfo-pCode-01091 ] Object 0: handle = 0x55a1c21315d0, name = Logical device: AMD RADV RAVEN2, type = VK_OBJECT_TYPE_DEVICE; | MessageID = 0xa7bb8db6 | vkCreateShaderModule(): The SPIR-V Capability (SampledImageArrayNonUniformIndexing) was declared, but none of the requirements were met to use it. The Vulkan spec states: If pCode declares any of the capabilities listed in the SPIR-V Environment appendix, one of the corresponding requirements must be satisfied (https://vulkan.lunarg.com/doc/view/1.2.182.0/linux/1.2-extensions/vkspec.html#VUID-VkShaderModuleCreateInfo-pCode-01091)

如果我读了docs,这似乎是一个硬件功能,但有人说,如果创建正确的扩展对象,我仍然可以有非统一访问。但我不完全确定如何做到这一点。

6yt4nkrj

6yt4nkrj1#

您必须在创建设备时启用该功能。
您可以通过调用vkGetPhysicalDeviceFeatures2存档并沿着pNext链一直到VkPhysicalDeviceVulkan12Features2存档,然后检查shaderSampledImageArrayNonUniformIndexing成员是否指向VK_TRUE,来检查是否支持该功能。
之后,当使用vkCreateDevicearchive创建设备时,在pCreateInfo结构中,在pNext链中,您必须将shaderSampledImageArrayNonUniformIndexing设置为VK_TRUE的VkPhysicalDeviceVulkan12Featuresarchive。

bool checkForNonUniformIndexing(VkPhysicalDevice physicalDevice)
{
    VkPhysicalDeviceFeatures2 features;
    vkGetPhysicalDeviceFeatures2(physicalDevice, &features);

    if(features.sType != VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2)
    {
        return false;
    }

    const VkPhysicalDeviceFeatures2* next = &features;

    do
    {
        // We know the type of the struct based on the `sType` member, but the first 
        // two fields are the same in all of these structs. There may be a more appropriate 
        // generic structure to use, but as long as we don't access any further members
        // we should be mostly fine.
        next = reinterpret_cast<const VkPhysicalDeviceFeatures*>(next->pNext);
        if(next.sType == VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_2_FEATURES)
        {
            const VkPhysicalDeviceVulkan12Features* pVk12Features = reinterpret_cast<const VkPhysicalDeviceVulkan12Features*>(next);
            return next.shaderSampledImageArrayNonUniformIndexing == VK_TRUE;
        }
    } while(next);

    return false;
}

VkDevice* createDevice(VkPhysicalDevice physicalDevice, const VkAllocationCallbacks* pAllocator)
{
    VkPhysicalDeviceVulkan12Features features;
    features.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_VULKAN_1_2_FEATURES;
    features.shaderSampledImageArrayNonUniformIndexing = VK_TRUE;

    VkDeviceCreateInfo createInfo;
    createInfo.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO;
    createInfo.pNext = &features;
    // Setting other create data

    VkDevice device;
    vkCreateDevice(physicalDevice, &createInfo, pAllocator, &device);

    // Error checking

    return device;
}

相关问题