blob: 7e4bfd8e803955e0d631ad6bcba646ec626e5c2d [file] [log] [blame]
/*
* Copyright (C) 2011 The Android Open Source Project
*
* 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 <android/log.h>
#include <driver.h>
#include <gmock/gmock.h>
#include <gtest/gtest.h>
#include <media/NdkImageReader.h>
#include <vulkan/vulkan.h>
#define LOGI(...) \
__android_log_print(ANDROID_LOG_INFO, "libvulkan_test", __VA_ARGS__)
#define LOGE(...) \
__android_log_print(ANDROID_LOG_ERROR, "libvulkan_test", __VA_ARGS__)
#define VK_CHECK(result) ASSERT_EQ(VK_SUCCESS, result)
namespace android {
namespace libvulkantest {
class AImageReaderVulkanSwapchainTest : public ::testing::Test {
public:
AImageReaderVulkanSwapchainTest() {}
AImageReader* mReader = nullptr;
ANativeWindow* mWindow = nullptr;
VkInstance mVkInstance = VK_NULL_HANDLE;
VkPhysicalDevice mPhysicalDev = VK_NULL_HANDLE;
VkDevice mDevice = VK_NULL_HANDLE;
VkSurfaceKHR mSurface = VK_NULL_HANDLE;
VkQueue mPresentQueue = VK_NULL_HANDLE;
uint32_t mPresentQueueFamily = UINT32_MAX;
VkSwapchainKHR mSwapchain = VK_NULL_HANDLE;
void SetUp() override {}
void TearDown() override {}
// ------------------------------------------------------
// Helper methods
// ------------------------------------------------------
void createVulkanInstance(std::vector<const char*>& layers) {
const char* extensions[] = {
VK_KHR_SURFACE_EXTENSION_NAME,
VK_KHR_ANDROID_SURFACE_EXTENSION_NAME,
VK_KHR_GET_SURFACE_CAPABILITIES_2_EXTENSION_NAME,
VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME,
};
VkApplicationInfo appInfo{};
appInfo.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO;
appInfo.pApplicationName = "AImageReader Vulkan Swapchain Test";
appInfo.applicationVersion = 1;
appInfo.pEngineName = "TestEngine";
appInfo.engineVersion = 1;
appInfo.apiVersion = VK_API_VERSION_1_0;
VkInstanceCreateInfo instInfo{};
instInfo.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO;
instInfo.pApplicationInfo = &appInfo;
instInfo.enabledExtensionCount =
sizeof(extensions) / sizeof(extensions[0]);
instInfo.ppEnabledExtensionNames = extensions;
instInfo.enabledLayerCount = layers.size();
instInfo.ppEnabledLayerNames = layers.data();
VkResult res = vkCreateInstance(&instInfo, nullptr, &mVkInstance);
VK_CHECK(res);
LOGE("Vulkan instance created");
}
void createAImageReader(int width, int height, int format, int maxImages) {
media_status_t status =
AImageReader_new(width, height, format, maxImages, &mReader);
ASSERT_EQ(AMEDIA_OK, status) << "Failed to create AImageReader";
ASSERT_NE(nullptr, mReader) << "AImageReader is null";
// Optionally set a listener
AImageReader_ImageListener listener{};
listener.context = this;
listener.onImageAvailable =
&AImageReaderVulkanSwapchainTest::onImageAvailable;
AImageReader_setImageListener(mReader, &listener);
LOGI("AImageReader created with %dx%d, format=%d", width, height,
format);
}
void getANativeWindowFromReader() {
ASSERT_NE(nullptr, mReader);
media_status_t status = AImageReader_getWindow(mReader, &mWindow);
ASSERT_EQ(AMEDIA_OK, status)
<< "Failed to get ANativeWindow from AImageReader";
ASSERT_NE(nullptr, mWindow) << "ANativeWindow is null";
LOGI("ANativeWindow obtained from AImageReader");
}
void createVulkanSurface() {
ASSERT_NE((VkInstance)VK_NULL_HANDLE, mVkInstance);
ASSERT_NE((ANativeWindow*)nullptr, mWindow);
VkAndroidSurfaceCreateInfoKHR surfaceCreateInfo{};
surfaceCreateInfo.sType =
VK_STRUCTURE_TYPE_ANDROID_SURFACE_CREATE_INFO_KHR;
surfaceCreateInfo.window = mWindow;
VkResult res = vkCreateAndroidSurfaceKHR(
mVkInstance, &surfaceCreateInfo, nullptr, &mSurface);
VK_CHECK(res);
LOGI("Vulkan surface created from ANativeWindow");
}
void pickPhysicalDeviceAndQueueFamily() {
ASSERT_NE((VkInstance)VK_NULL_HANDLE, mVkInstance);
uint32_t deviceCount = 0;
vkEnumeratePhysicalDevices(mVkInstance, &deviceCount, nullptr);
ASSERT_GT(deviceCount, 0U) << "No Vulkan physical devices found!";
std::vector<VkPhysicalDevice> devices(deviceCount);
vkEnumeratePhysicalDevices(mVkInstance, &deviceCount, devices.data());
for (auto& dev : devices) {
uint32_t queueFamilyCount = 0;
vkGetPhysicalDeviceQueueFamilyProperties(dev, &queueFamilyCount,
nullptr);
std::vector<VkQueueFamilyProperties> queueProps(queueFamilyCount);
vkGetPhysicalDeviceQueueFamilyProperties(dev, &queueFamilyCount,
queueProps.data());
for (uint32_t i = 0; i < queueFamilyCount; i++) {
VkBool32 support = VK_FALSE;
vkGetPhysicalDeviceSurfaceSupportKHR(dev, i, mSurface,
&support);
if (support == VK_TRUE) {
// Found a queue family that can present
mPhysicalDev = dev;
mPresentQueueFamily = i;
LOGI(
"Physical device found with queue family %u supporting "
"present",
i);
return;
}
}
}
FAIL()
<< "No physical device found that supports present to the surface!";
}
void createDeviceAndGetQueue(std::vector<const char*>& layers,
std::vector<const char*> inExtensions = {}) {
ASSERT_NE((void*)VK_NULL_HANDLE, mPhysicalDev);
ASSERT_NE(UINT32_MAX, mPresentQueueFamily);
float queuePriority = 1.0f;
VkDeviceQueueCreateInfo queueInfo{};
queueInfo.sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO;
queueInfo.queueFamilyIndex = mPresentQueueFamily;
queueInfo.queueCount = 1;
queueInfo.pQueuePriorities = &queuePriority;
VkDeviceCreateInfo deviceInfo{};
deviceInfo.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO;
deviceInfo.queueCreateInfoCount = 1;
deviceInfo.pQueueCreateInfos = &queueInfo;
deviceInfo.enabledLayerCount = layers.size();
deviceInfo.ppEnabledLayerNames = layers.data();
std::vector<const char*> extensions = {
VK_KHR_SWAPCHAIN_EXTENSION_NAME,
};
for (auto extension : inExtensions) {
extensions.push_back(extension);
}
deviceInfo.enabledExtensionCount = extensions.size();
deviceInfo.ppEnabledExtensionNames = extensions.data();
VkResult res =
vkCreateDevice(mPhysicalDev, &deviceInfo, nullptr, &mDevice);
VK_CHECK(res);
LOGI("Logical device created");
vkGetDeviceQueue(mDevice, mPresentQueueFamily, 0, &mPresentQueue);
ASSERT_NE((VkQueue)VK_NULL_HANDLE, mPresentQueue);
LOGI("Acquired present-capable queue");
}
void createSwapchain() {
ASSERT_NE((VkDevice)VK_NULL_HANDLE, mDevice);
ASSERT_NE((VkSurfaceKHR)VK_NULL_HANDLE, mSurface);
VkSurfaceCapabilitiesKHR surfaceCaps{};
VK_CHECK(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(
mPhysicalDev, mSurface, &surfaceCaps));
uint32_t formatCount = 0;
vkGetPhysicalDeviceSurfaceFormatsKHR(mPhysicalDev, mSurface,
&formatCount, nullptr);
ASSERT_GT(formatCount, 0U);
std::vector<VkSurfaceFormatKHR> formats(formatCount);
vkGetPhysicalDeviceSurfaceFormatsKHR(mPhysicalDev, mSurface,
&formatCount, formats.data());
VkSurfaceFormatKHR chosenFormat = formats[0];
LOGI("Chosen surface format: %d", chosenFormat.format);
uint32_t presentModeCount = 0;
vkGetPhysicalDeviceSurfacePresentModesKHR(mPhysicalDev, mSurface,
&presentModeCount, nullptr);
ASSERT_GT(presentModeCount, 0U);
std::vector<VkPresentModeKHR> presentModes(presentModeCount);
vkGetPhysicalDeviceSurfacePresentModesKHR(
mPhysicalDev, mSurface, &presentModeCount, presentModes.data());
VkPresentModeKHR chosenPresentMode = VK_PRESENT_MODE_FIFO_KHR;
for (auto mode : presentModes) {
if (mode == VK_PRESENT_MODE_FIFO_KHR) {
chosenPresentMode = mode;
break;
}
}
LOGI("Chosen present mode: %d", chosenPresentMode);
VkExtent2D swapchainExtent{};
if (surfaceCaps.currentExtent.width == 0xFFFFFFFF) {
swapchainExtent.width = 640; // fallback
swapchainExtent.height = 480; // fallback
} else {
swapchainExtent = surfaceCaps.currentExtent;
}
LOGI("Swapchain extent: %d x %d", swapchainExtent.width,
swapchainExtent.height);
uint32_t desiredImageCount = surfaceCaps.minImageCount + 1;
if (surfaceCaps.maxImageCount > 0 &&
desiredImageCount > surfaceCaps.maxImageCount) {
desiredImageCount = surfaceCaps.maxImageCount;
}
VkSwapchainCreateInfoKHR swapchainInfo{};
swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
swapchainInfo.surface = mSurface;
swapchainInfo.minImageCount = desiredImageCount;
swapchainInfo.imageFormat = chosenFormat.format;
swapchainInfo.imageColorSpace = chosenFormat.colorSpace;
swapchainInfo.imageExtent = swapchainExtent;
swapchainInfo.imageArrayLayers = 1;
swapchainInfo.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT |
VK_IMAGE_USAGE_TRANSFER_SRC_BIT;
swapchainInfo.preTransform = surfaceCaps.currentTransform;
swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
swapchainInfo.presentMode = chosenPresentMode;
swapchainInfo.clipped = VK_TRUE;
swapchainInfo.oldSwapchain = VK_NULL_HANDLE;
uint32_t queueFamilyIndices[] = {mPresentQueueFamily};
swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
swapchainInfo.queueFamilyIndexCount = 1;
swapchainInfo.pQueueFamilyIndices = queueFamilyIndices;
VkResult res =
vkCreateSwapchainKHR(mDevice, &swapchainInfo, nullptr, &mSwapchain);
if (res == VK_SUCCESS) {
LOGI("Swapchain created successfully");
uint32_t swapchainImageCount = 0;
vkGetSwapchainImagesKHR(mDevice, mSwapchain, &swapchainImageCount,
nullptr);
std::vector<VkImage> swapchainImages(swapchainImageCount);
vkGetSwapchainImagesKHR(mDevice, mSwapchain, &swapchainImageCount,
swapchainImages.data());
LOGI("Swapchain has %u images", swapchainImageCount);
} else {
LOGI("Swapchain creation failed");
}
}
// Image available callback (AImageReader)
static void onImageAvailable(void*, AImageReader* reader) {
LOGI("onImageAvailable callback triggered");
AImage* image = nullptr;
media_status_t status = AImageReader_acquireLatestImage(reader, &image);
if (status != AMEDIA_OK || !image) {
LOGE("Failed to acquire latest image");
return;
}
AImage_delete(image);
LOGI("Released acquired image");
}
void cleanUpSwapchainForTest() {
if (mSwapchain != VK_NULL_HANDLE) {
vkDestroySwapchainKHR(mDevice, mSwapchain, nullptr);
mSwapchain = VK_NULL_HANDLE;
}
if (mDevice != VK_NULL_HANDLE) {
vkDestroyDevice(mDevice, nullptr);
mDevice = VK_NULL_HANDLE;
}
if (mSurface != VK_NULL_HANDLE) {
vkDestroySurfaceKHR(mVkInstance, mSurface, nullptr);
mSurface = VK_NULL_HANDLE;
}
if (mVkInstance != VK_NULL_HANDLE) {
vkDestroyInstance(mVkInstance, nullptr);
mVkInstance = VK_NULL_HANDLE;
}
if (mReader) {
// AImageReader_delete(mReader);
mReader = nullptr;
}
// Note: The ANativeWindow from AImageReader is implicitly
// managed by the reader, so we don't explicitly delete it.
mWindow = nullptr;
}
void buildSwapchianForTest(std::vector<const char*>& instanceLayers,
std::vector<const char*>& deviceLayers) {
createVulkanInstance(instanceLayers);
// the "atest libvulkan_test" command will execute this test as a binary
// (not apk) on the device. Consequently we can't render to the screen
// and need to work around this by using AImageReader*
createAImageReader(640, 480, AIMAGE_FORMAT_PRIVATE, 3);
getANativeWindowFromReader();
createVulkanSurface();
pickPhysicalDeviceAndQueueFamily();
createDeviceAndGetQueue(deviceLayers);
createSwapchain();
}
};
TEST_F(AImageReaderVulkanSwapchainTest, TestHelperMethods) {
// Verify that the basic plumbing/helper functions of these tests is
// working. This doesn't directly test any of the layer code. It only
// verifies that we can successfully create a swapchain with an AImageReader
std::vector<const char*> instanceLayers;
std::vector<const char*> deviceLayers;
buildSwapchianForTest(deviceLayers, instanceLayers);
ASSERT_NE(mVkInstance, (VkInstance)VK_NULL_HANDLE);
ASSERT_NE(mPhysicalDev, (VkPhysicalDevice)VK_NULL_HANDLE);
ASSERT_NE(mDevice, (VkDevice)VK_NULL_HANDLE);
ASSERT_NE(mSurface, (VkSurfaceKHR)VK_NULL_HANDLE);
ASSERT_NE(mSwapchain, (VkSwapchainKHR)VK_NULL_HANDLE);
cleanUpSwapchainForTest();
}
// Passing state in these tests requires global state. Wrap each test in an
// anonymous namespace to prevent conflicting names.
namespace {
VKAPI_ATTR VkResult VKAPI_CALL hookedGetPhysicalDeviceImageFormatProperties2KHR(
VkPhysicalDevice,
const VkPhysicalDeviceImageFormatInfo2*,
VkImageFormatProperties2*) {
return VK_ERROR_SURFACE_LOST_KHR;
}
static PFN_vkGetSwapchainGrallocUsage2ANDROID
pfnNextGetSwapchainGrallocUsage2ANDROID = nullptr;
static bool g_grallocCalled = false;
VKAPI_ATTR VkResult VKAPI_CALL hookGetSwapchainGrallocUsage2ANDROID(
VkDevice device,
VkFormat format,
VkImageUsageFlags imageUsage,
VkSwapchainImageUsageFlagsANDROID swapchainImageUsage,
uint64_t* grallocConsumerUsage,
uint64_t* grallocProducerUsage) {
g_grallocCalled = true;
if (pfnNextGetSwapchainGrallocUsage2ANDROID) {
return pfnNextGetSwapchainGrallocUsage2ANDROID(
device, format, imageUsage, swapchainImageUsage,
grallocConsumerUsage, grallocProducerUsage);
}
return VK_ERROR_INITIALIZATION_FAILED;
}
TEST_F(AImageReaderVulkanSwapchainTest, getProducerUsageFallbackTest1) {
// BUG: 379230826
// Verify that getProducerUsage falls back to
// GetSwapchainGrallocUsage*ANDROID if GPDIFP2 fails
std::vector<const char*> instanceLayers = {};
std::vector<const char*> deviceLayers = {};
createVulkanInstance(instanceLayers);
createAImageReader(640, 480, AIMAGE_FORMAT_PRIVATE, 3);
getANativeWindowFromReader();
createVulkanSurface();
pickPhysicalDeviceAndQueueFamily();
createDeviceAndGetQueue(deviceLayers);
auto& pdev = vulkan::driver::GetData(mDevice).driver_physical_device;
auto& pdevDispatchTable = vulkan::driver::GetData(pdev).driver;
auto& deviceDispatchTable = vulkan::driver::GetData(mDevice).driver;
ASSERT_NE(deviceDispatchTable.GetSwapchainGrallocUsage2ANDROID, nullptr);
pdevDispatchTable.GetPhysicalDeviceImageFormatProperties2 =
hookedGetPhysicalDeviceImageFormatProperties2KHR;
deviceDispatchTable.GetSwapchainGrallocUsage2ANDROID =
hookGetSwapchainGrallocUsage2ANDROID;
ASSERT_FALSE(g_grallocCalled);
createSwapchain();
ASSERT_TRUE(g_grallocCalled);
ASSERT_NE(mVkInstance, (VkInstance)VK_NULL_HANDLE);
ASSERT_NE(mPhysicalDev, (VkPhysicalDevice)VK_NULL_HANDLE);
ASSERT_NE(mDevice, (VkDevice)VK_NULL_HANDLE);
ASSERT_NE(mSurface, (VkSurfaceKHR)VK_NULL_HANDLE);
cleanUpSwapchainForTest();
}
} // namespace
// Passing state in these tests requires global state. Wrap each test in an
// anonymous namespace to prevent conflicting names.
namespace {
static bool g_returnNotSupportedOnce = true;
VKAPI_ATTR VkResult VKAPI_CALL
Hook_GetPhysicalDeviceImageFormatProperties2_NotSupportedOnce(
VkPhysicalDevice /*physicalDevice*/,
const VkPhysicalDeviceImageFormatInfo2* /*pImageFormatInfo*/,
VkImageFormatProperties2* /*pImageFormatProperties*/) {
if (g_returnNotSupportedOnce) {
g_returnNotSupportedOnce = false;
return VK_ERROR_FORMAT_NOT_SUPPORTED;
}
return VK_SUCCESS;
}
TEST_F(AImageReaderVulkanSwapchainTest, SurfaceFormats2KHR_IgnoreNotSupported) {
// BUG: 357903074
// Verify that vkGetPhysicalDeviceSurfaceFormats2KHR properly
// ignores VK_ERROR_FORMAT_NOT_SUPPORTED and continues enumerating formats.
std::vector<const char*> instanceLayers;
createVulkanInstance(instanceLayers);
createAImageReader(640, 480, AIMAGE_FORMAT_PRIVATE, 3);
getANativeWindowFromReader();
createVulkanSurface();
pickPhysicalDeviceAndQueueFamily();
auto& pdevDispatchTable = vulkan::driver::GetData(mPhysicalDev).driver;
pdevDispatchTable.GetPhysicalDeviceImageFormatProperties2 =
Hook_GetPhysicalDeviceImageFormatProperties2_NotSupportedOnce;
PFN_vkGetPhysicalDeviceSurfaceFormats2KHR
pfnGetPhysicalDeviceSurfaceFormats2KHR =
reinterpret_cast<PFN_vkGetPhysicalDeviceSurfaceFormats2KHR>(
vkGetInstanceProcAddr(mVkInstance,
"vkGetPhysicalDeviceSurfaceFormats2KHR"));
ASSERT_NE(nullptr, pfnGetPhysicalDeviceSurfaceFormats2KHR)
<< "Could not get pointer to vkGetPhysicalDeviceSurfaceFormats2KHR";
VkPhysicalDeviceSurfaceInfo2KHR surfaceInfo2{};
surfaceInfo2.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SURFACE_INFO_2_KHR;
surfaceInfo2.pNext = nullptr;
surfaceInfo2.surface = mSurface;
uint32_t formatCount = 0;
VkResult res = pfnGetPhysicalDeviceSurfaceFormats2KHR(
mPhysicalDev, &surfaceInfo2, &formatCount, nullptr);
// If the loader never tries a second format, it might fail or 0-out the
// formatCount. The patch ensures it continues to the next format rather
// than bailing out on the first NOT_SUPPORTED.
ASSERT_EQ(VK_SUCCESS, res)
<< "vkGetPhysicalDeviceSurfaceFormats2KHR failed unexpectedly";
ASSERT_GT(formatCount, 0U)
<< "No surface formats found; the loader may have bailed early.";
std::vector<VkSurfaceFormat2KHR> formats(formatCount);
for (auto& f : formats) {
f.sType = VK_STRUCTURE_TYPE_SURFACE_FORMAT_2_KHR;
f.pNext = nullptr;
}
res = pfnGetPhysicalDeviceSurfaceFormats2KHR(mPhysicalDev, &surfaceInfo2,
&formatCount, formats.data());
ASSERT_EQ(VK_SUCCESS, res) << "Failed to retrieve surface formats";
LOGI(
"SurfaceFormats2KHR_IgnoreNotSupported test: found %u formats after "
"ignoring NOT_SUPPORTED",
formatCount);
cleanUpSwapchainForTest();
}
} // namespace
namespace {
TEST_F(AImageReaderVulkanSwapchainTest, MutableFormatSwapchainTest) {
// Test swapchain with mutable format extension
std::vector<const char*> instanceLayers;
std::vector<const char*> deviceLayers;
std::vector<const char*> deviceExtensions = {
VK_KHR_SWAPCHAIN_EXTENSION_NAME,
VK_KHR_SWAPCHAIN_MUTABLE_FORMAT_EXTENSION_NAME,
VK_KHR_MAINTENANCE2_EXTENSION_NAME,
VK_KHR_IMAGE_FORMAT_LIST_EXTENSION_NAME};
createVulkanInstance(instanceLayers);
createAImageReader(640, 480, AIMAGE_FORMAT_PRIVATE, 3);
getANativeWindowFromReader();
createVulkanSurface();
pickPhysicalDeviceAndQueueFamily();
createDeviceAndGetQueue(deviceLayers, deviceExtensions);
ASSERT_NE((VkDevice)VK_NULL_HANDLE, mDevice);
ASSERT_NE((VkSurfaceKHR)VK_NULL_HANDLE, mSurface);
VkSurfaceCapabilitiesKHR surfaceCaps{};
VK_CHECK(vkGetPhysicalDeviceSurfaceCapabilitiesKHR(mPhysicalDev, mSurface,
&surfaceCaps));
uint32_t formatCount = 0;
vkGetPhysicalDeviceSurfaceFormatsKHR(mPhysicalDev, mSurface, &formatCount,
nullptr);
ASSERT_GT(formatCount, 0U);
std::vector<VkSurfaceFormatKHR> formats(formatCount);
vkGetPhysicalDeviceSurfaceFormatsKHR(mPhysicalDev, mSurface, &formatCount,
formats.data());
VkFormat viewFormats[2] = {formats[0].format, formats[0].format};
if (formatCount > 1) {
viewFormats[1] = formats[1].format;
}
VkImageFormatListCreateInfoKHR formatList{};
formatList.sType = VK_STRUCTURE_TYPE_IMAGE_FORMAT_LIST_CREATE_INFO_KHR;
formatList.viewFormatCount = 2;
formatList.pViewFormats = viewFormats;
VkSwapchainCreateInfoKHR swapchainInfo{};
swapchainInfo.sType = VK_STRUCTURE_TYPE_SWAPCHAIN_CREATE_INFO_KHR;
swapchainInfo.pNext = &formatList;
swapchainInfo.surface = mSurface;
swapchainInfo.minImageCount = surfaceCaps.minImageCount + 1;
swapchainInfo.imageFormat = formats[0].format;
swapchainInfo.imageColorSpace = formats[0].colorSpace;
swapchainInfo.imageExtent = surfaceCaps.currentExtent;
swapchainInfo.imageArrayLayers = 1;
swapchainInfo.imageUsage = VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT;
swapchainInfo.preTransform = surfaceCaps.currentTransform;
swapchainInfo.compositeAlpha = VK_COMPOSITE_ALPHA_INHERIT_BIT_KHR;
swapchainInfo.presentMode = VK_PRESENT_MODE_FIFO_KHR;
swapchainInfo.clipped = VK_TRUE;
swapchainInfo.flags = VK_SWAPCHAIN_CREATE_MUTABLE_FORMAT_BIT_KHR;
uint32_t queueFamilyIndices[] = {mPresentQueueFamily};
swapchainInfo.imageSharingMode = VK_SHARING_MODE_EXCLUSIVE;
swapchainInfo.queueFamilyIndexCount = 1;
swapchainInfo.pQueueFamilyIndices = queueFamilyIndices;
VkResult res =
vkCreateSwapchainKHR(mDevice, &swapchainInfo, nullptr, &mSwapchain);
if (res == VK_SUCCESS) {
LOGI("Mutable format swapchain created successfully");
uint32_t imageCount = 0;
vkGetSwapchainImagesKHR(mDevice, mSwapchain, &imageCount, nullptr);
ASSERT_GT(imageCount, 0U);
} else {
LOGI(
"Mutable format swapchain creation failed (extension may not be "
"supported)");
}
cleanUpSwapchainForTest();
}
} // namespace
} // namespace libvulkantest
} // namespace android