blob: 25015e3cf7cc233cb5046614ebbec98a5775d7b5 [file] [log] [blame]
#include <gtest/gtest.h>
#include "CompositorVk.h"
#include <algorithm>
#include <array>
#include <filesystem>
#include <glm/gtx/matrix_transform_2d.hpp>
#include <memory>
#include <optional>
#include "BorrowedImageVk.h"
#include "aemu/base/synchronization/Lock.h"
#include "gfxstream/ImageUtils.h"
#include "tests/VkTestUtils.h"
#include "vulkan/VulkanDispatch.h"
#include "vulkan/vk_util.h"
namespace gfxstream {
namespace vk {
namespace {
static constexpr const bool kDefaultSaveImageIfComparisonFailed = false;
std::string GetTestDataPath(const std::string& basename) {
const std::filesystem::path currentPath = android::base::getProgramDirectory();
return (currentPath / "tests" / "testdata" / basename).string();
}
static constexpr const uint32_t kColorBlack = 0xFF000000;
static constexpr const uint32_t kColorRed = 0xFF0000FF;
static constexpr const uint32_t kColorGreen = 0xFF00FF00;
static constexpr const uint32_t kDefaultImageWidth = 256;
static constexpr const uint32_t kDefaultImageHeight = 256;
class CompositorVkTest : public ::testing::Test {
protected:
using TargetImage = RenderResourceVk<VK_IMAGE_LAYOUT_COLOR_ATTACHMENT_OPTIMAL,
VK_IMAGE_USAGE_COLOR_ATTACHMENT_BIT>;
using SourceImage = RenderTextureVk;
static void SetUpTestCase() { k_vk = vkDispatch(false); }
void SetUp() override {
#if defined(__APPLE__) && defined(__arm64__)
GTEST_SKIP() << "Skipping all test on Apple M2, as they are failing, see b/263494782";
#endif
ASSERT_NE(k_vk, nullptr);
createInstance();
pickPhysicalDevice();
createLogicalDevice();
if (!supportsFeatures(TargetImage::k_vkFormat, VK_FORMAT_FEATURE_COLOR_ATTACHMENT_BIT)) {
GTEST_SKIP() << "Skipping test as format " << TargetImage::k_vkFormat
<< " does not support VK_FORMAT_FEATURE_COLOR_ATTACHMENT_BIT";
}
if (!supportsFeatures(SourceImage::k_vkFormat, VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT)) {
GTEST_SKIP() << "Skipping test as format " << SourceImage::k_vkFormat
<< " does not support VK_FORMAT_FEATURE_SAMPLED_IMAGE_BIT";
}
const VkCommandPoolCreateInfo commandPoolCi = {
.sType = VK_STRUCTURE_TYPE_COMMAND_POOL_CREATE_INFO,
.queueFamilyIndex = m_compositorQueueFamilyIndex,
};
ASSERT_EQ(k_vk->vkCreateCommandPool(m_vkDevice, &commandPoolCi, nullptr, &m_vkCommandPool),
VK_SUCCESS);
k_vk->vkGetDeviceQueue(m_vkDevice, m_compositorQueueFamilyIndex, 0, &m_compositorVkQueue);
ASSERT_NE(m_compositorVkQueue, VK_NULL_HANDLE);
m_compositorVkQueueLock = std::make_shared<android::base::Lock>();
}
void TearDown() override {
#if defined(__APPLE__) && defined(__arm64__)
return;
#endif
k_vk->vkDestroyCommandPool(m_vkDevice, m_vkCommandPool, nullptr);
k_vk->vkDestroyDevice(m_vkDevice, nullptr);
m_vkDevice = VK_NULL_HANDLE;
k_vk->vkDestroyInstance(m_vkInstance, nullptr);
m_vkInstance = VK_NULL_HANDLE;
}
std::unique_ptr<CompositorVk> createCompositor() {
return CompositorVk::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_compositorVkQueueLock, m_compositorQueueFamilyIndex,
/*maxFramesInFlight=*/3);
}
template <typename SourceOrTargetImage>
std::unique_ptr<const SourceOrTargetImage> createImageWithColor(uint32_t sourceWidth,
uint32_t sourceHeight,
uint32_t sourceColor) {
auto source =
SourceOrTargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, sourceWidth, sourceHeight);
if (source == nullptr) {
return nullptr;
}
std::vector<uint32_t> sourcePixels(sourceWidth * sourceHeight, sourceColor);
if (!source->write(sourcePixels)) {
return nullptr;
}
return source;
}
std::unique_ptr<const SourceImage> createSourceImageFromPng(const std::string& filename) {
uint32_t sourceWidth;
uint32_t sourceHeight;
std::vector<uint32_t> sourcePixels;
if (!LoadRGBAFromPng(filename, &sourceWidth, &sourceHeight, &sourcePixels)) {
return nullptr;
}
auto source =
SourceImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, sourceWidth, sourceHeight);
if (source == nullptr) {
return nullptr;
}
if (!source->write(sourcePixels)) {
return nullptr;
}
return source;
}
bool isRGBAPixelNear(uint32_t actualPixel, uint32_t expectedPixel) {
const uint8_t* actualRGBA = reinterpret_cast<const uint8_t*>(&actualPixel);
const uint8_t* expectedRGBA = reinterpret_cast<const uint8_t*>(&expectedPixel);
constexpr const uint32_t kRGBA8888Tolerance = 2;
for (uint32_t channel = 0; channel < 4; channel++) {
const uint8_t actualChannel = actualRGBA[channel];
const uint8_t expectedChannel = expectedRGBA[channel];
if ((std::max(actualChannel, expectedChannel) -
std::min(actualChannel, expectedChannel)) > kRGBA8888Tolerance) {
return false;
}
}
return true;
}
bool compareRGBAPixels(const uint32_t* actualPixels, const uint32_t* expectedPixels,
const uint32_t width, const uint32_t height) {
bool comparisonFailed = false;
uint32_t reportedIncorrectPixels = 0;
constexpr const uint32_t kMaxReportedIncorrectPixels = 10;
for (uint32_t y = 0; y < width; y++) {
for (uint32_t x = 0; x < height; x++) {
const uint32_t actualPixel = actualPixels[y * height + x];
const uint32_t expectedPixel = expectedPixels[y * width + x];
if (!isRGBAPixelNear(actualPixel, expectedPixel)) {
comparisonFailed = true;
if (reportedIncorrectPixels < kMaxReportedIncorrectPixels) {
reportedIncorrectPixels++;
const uint8_t* actualRGBA = reinterpret_cast<const uint8_t*>(&actualPixel);
const uint8_t* expectedRGBA =
reinterpret_cast<const uint8_t*>(&expectedPixel);
ADD_FAILURE()
<< "Pixel comparison failed at (" << x << ", " << y << ") "
<< " with actual "
<< " r:" << static_cast<int>(actualRGBA[0])
<< " g:" << static_cast<int>(actualRGBA[1])
<< " b:" << static_cast<int>(actualRGBA[2])
<< " a:" << static_cast<int>(actualRGBA[3]) << " but expected "
<< " r:" << static_cast<int>(expectedRGBA[0])
<< " g:" << static_cast<int>(expectedRGBA[1])
<< " b:" << static_cast<int>(expectedRGBA[2])
<< " a:" << static_cast<int>(expectedRGBA[3]);
}
}
}
}
return comparisonFailed;
}
void compareImageWithGoldenPng(const TargetImage* target, const std::string& filename,
const bool saveImageIfComparisonFailed) {
const uint32_t targetWidth = target->m_width;
const uint32_t targetHeight = target->m_height;
const auto targetPixelsOpt = target->read();
ASSERT_TRUE(targetPixelsOpt.has_value());
const auto& targetPixels = *targetPixelsOpt;
uint32_t goldenWidth;
uint32_t goldenHeight;
std::vector<uint32_t> goldenPixels;
const bool loadedGolden =
LoadRGBAFromPng(filename, &goldenWidth, &goldenHeight, &goldenPixels);
EXPECT_TRUE(loadedGolden) << "Failed to load golden image from " << filename;
bool comparisonFailed = !loadedGolden;
if (loadedGolden) {
EXPECT_EQ(target->m_width, goldenWidth)
<< "Invalid width comparison with golden image from " << filename;
EXPECT_EQ(target->m_height, goldenHeight)
<< "Invalid height comparison with golden image from " << filename;
if (targetWidth != goldenWidth || targetHeight != goldenHeight) {
comparisonFailed = true;
}
if (!comparisonFailed) {
comparisonFailed = compareRGBAPixels(targetPixels.data(), goldenPixels.data(),
goldenWidth, goldenHeight);
}
}
if (saveImageIfComparisonFailed && comparisonFailed) {
const std::string output = (std::filesystem::temp_directory_path() /
std::filesystem::path(filename).filename())
.string();
SaveRGBAToPng(targetWidth, targetHeight, targetPixels.data(), output);
ADD_FAILURE() << "Saved composition result to " << output;
}
}
template <typename SourceOrTargetImage>
std::unique_ptr<BorrowedImageInfoVk> createBorrowedImageInfo(const SourceOrTargetImage* image) {
static int sImageId = 0;
auto ret = std::make_unique<BorrowedImageInfoVk>();
ret->id = sImageId++;
ret->width = image->m_width;
ret->height = image->m_height;
ret->image = image->m_vkImage;
ret->imageCreateInfo = image->m_vkImageCreateInfo;
ret->imageView = image->m_vkImageView;
ret->preBorrowLayout = SourceOrTargetImage::k_vkImageLayout;
ret->preBorrowQueueFamilyIndex = m_compositorQueueFamilyIndex;
ret->postBorrowLayout = SourceOrTargetImage::k_vkImageLayout;
ret->postBorrowQueueFamilyIndex = m_compositorQueueFamilyIndex;
return ret;
}
void checkImageFilledWith(const TargetImage* image, uint32_t expectedColor) {
auto actualPixelsOpt = image->read();
ASSERT_TRUE(actualPixelsOpt.has_value());
auto& actualPixels = *actualPixelsOpt;
const std::vector<uint32_t> expectedPixels(image->numOfPixels(), expectedColor);
compareRGBAPixels(actualPixels.data(), expectedPixels.data(), image->m_width,
image->m_height);
}
void fillImageWith(const TargetImage* image, uint32_t color) {
const std::vector<uint32_t> pixels(image->numOfPixels(), color);
ASSERT_TRUE(image->write(pixels)) << "Failed to fill image with color:" << color;
checkImageFilledWith(image, color);
}
static const VulkanDispatch* k_vk;
VkInstance m_vkInstance = VK_NULL_HANDLE;
VkPhysicalDevice m_vkPhysicalDevice = VK_NULL_HANDLE;
uint32_t m_compositorQueueFamilyIndex = 0;
VkDevice m_vkDevice = VK_NULL_HANDLE;
VkCommandPool m_vkCommandPool = VK_NULL_HANDLE;
VkQueue m_compositorVkQueue = VK_NULL_HANDLE;
std::shared_ptr<android::base::Lock> m_compositorVkQueueLock;
private:
void createInstance() {
const VkApplicationInfo appInfo = {
.sType = VK_STRUCTURE_TYPE_APPLICATION_INFO,
.pNext = nullptr,
.pApplicationName = "emulator CompositorVk unittest",
.applicationVersion = VK_MAKE_VERSION(1, 0, 0),
.pEngineName = "No Engine",
.engineVersion = VK_MAKE_VERSION(1, 0, 0),
.apiVersion = VK_API_VERSION_1_1,
};
const VkInstanceCreateInfo instanceCi = {
.sType = VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
.pApplicationInfo = &appInfo,
.enabledExtensionCount = 0,
.ppEnabledExtensionNames = nullptr,
};
ASSERT_EQ(k_vk->vkCreateInstance(&instanceCi, nullptr, &m_vkInstance), VK_SUCCESS);
ASSERT_NE(m_vkInstance, VK_NULL_HANDLE);
}
void pickPhysicalDevice() {
uint32_t physicalDeviceCount = 0;
ASSERT_EQ(k_vk->vkEnumeratePhysicalDevices(m_vkInstance, &physicalDeviceCount, nullptr),
VK_SUCCESS);
ASSERT_GT(physicalDeviceCount, 0);
std::vector<VkPhysicalDevice> physicalDevices(physicalDeviceCount);
ASSERT_EQ(k_vk->vkEnumeratePhysicalDevices(m_vkInstance, &physicalDeviceCount,
physicalDevices.data()),
VK_SUCCESS);
for (const auto &device : physicalDevices) {
uint32_t queueFamilyCount = 0;
k_vk->vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount, nullptr);
ASSERT_GT(queueFamilyCount, 0);
std::vector<VkQueueFamilyProperties> queueFamilyProperties(queueFamilyCount);
k_vk->vkGetPhysicalDeviceQueueFamilyProperties(device, &queueFamilyCount,
queueFamilyProperties.data());
uint32_t queueFamilyIndex = 0;
for (; queueFamilyIndex < queueFamilyCount; queueFamilyIndex++) {
if (CompositorVk::queueSupportsComposition(
queueFamilyProperties[queueFamilyIndex])) {
break;
}
}
if (queueFamilyIndex == queueFamilyCount) {
continue;
}
m_compositorQueueFamilyIndex = queueFamilyIndex;
m_vkPhysicalDevice = device;
return;
}
FAIL() << "Can't find a suitable VkPhysicalDevice.";
}
void createLogicalDevice() {
const float queuePriority = 1.0f;
const VkDeviceQueueCreateInfo queueCi = {
.sType = VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
.queueFamilyIndex = m_compositorQueueFamilyIndex,
.queueCount = 1,
.pQueuePriorities = &queuePriority,
};
const VkPhysicalDeviceFeatures2 features = {
.sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2,
.pNext = nullptr,
};
const VkDeviceCreateInfo deviceCi = {
.sType = VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
.pNext = &features,
.queueCreateInfoCount = 1,
.pQueueCreateInfos = &queueCi,
.enabledLayerCount = 0,
.enabledExtensionCount = 0,
.ppEnabledExtensionNames = nullptr,
};
ASSERT_EQ(k_vk->vkCreateDevice(m_vkPhysicalDevice, &deviceCi, nullptr, &m_vkDevice),
VK_SUCCESS);
ASSERT_NE(m_vkDevice, VK_NULL_HANDLE);
}
bool supportsFeatures(VkFormat format, VkFormatFeatureFlags features) {
VkFormatProperties formatProperties = {};
k_vk->vkGetPhysicalDeviceFormatProperties(m_vkPhysicalDevice, format, &formatProperties);
return (formatProperties.optimalTilingFeatures & features) == features;
}
};
const VulkanDispatch* CompositorVkTest::k_vk = nullptr;
TEST_F(CompositorVkTest, QueueSupportsComposition) {
VkQueueFamilyProperties properties = {};
properties.queueFlags &= ~VK_QUEUE_GRAPHICS_BIT;
ASSERT_FALSE(CompositorVk::queueSupportsComposition(properties));
properties.queueFlags |= VK_QUEUE_GRAPHICS_BIT;
ASSERT_TRUE(CompositorVk::queueSupportsComposition(properties));
}
TEST_F(CompositorVkTest, Init) { ASSERT_NE(createCompositor(), nullptr); }
TEST_F(CompositorVkTest, EmptyCompositionShouldDrawABlackFrame) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
std::vector<std::unique_ptr<const TargetImage>> targets;
constexpr const uint32_t kNumImages = 10;
for (uint32_t i = 0; i < kNumImages; i++) {
auto target =
TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, kDefaultImageWidth, kDefaultImageHeight);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorRed);
targets.emplace_back(std::move(target));
}
for (uint32_t i = 0; i < kNumImages; i++) {
const Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(targets[i].get()),
.layers = {}, // Note: this is empty!
};
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
}
for (const auto& target : targets) {
checkImageFilledWith(target.get(), kColorBlack);
}
}
TEST_F(CompositorVkTest, SimpleComposition) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source = createSourceImageFromPng(GetTestDataPath("256x256_android.png"));
ASSERT_NE(source, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, source->m_width, source->m_height);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 64,
.top = 32,
.right = 128,
.bottom = 160,
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(),
GetTestDataPath("256x256_golden_simple_composition.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, BlendPremultiplied) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source =
createSourceImageFromPng(GetTestDataPath("256x256_android_with_transparency.png"));
ASSERT_NE(source, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, source->m_width, source->m_height);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(),
GetTestDataPath("256x256_golden_blend_premultiplied.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, Crop) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source = createSourceImageFromPng(GetTestDataPath("256x256_android.png"));
ASSERT_NE(source, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, source->m_width, source->m_height);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width) / 2.0f,
.bottom = static_cast<float>(source->m_height) / 2.0f,
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath("256x256_golden_crop.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, SolidColor) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, 256, 256);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = nullptr,
.props =
{
.composeMode = HWC2_COMPOSITION_SOLID_COLOR,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.alpha = 0.75f,
.color =
{
.r = 255,
.g = 255,
.b = 0,
.a = 255,
},
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath("256x256_golden_solid_color.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, SolidColorBelow) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source =
createSourceImageFromPng(GetTestDataPath("256x256_android_with_transparency.png"));
ASSERT_NE(source, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, source->m_width, source->m_height);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = nullptr,
.props =
{
.composeMode = HWC2_COMPOSITION_SOLID_COLOR,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.alpha = 1.0f,
.color =
{
.r = 0,
.g = 0,
.b = 255,
.a = 255,
},
},
});
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath("256x256_golden_solid_color_below.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, SolidColorAbove) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source = createSourceImageFromPng(GetTestDataPath("256x256_android.png"));
ASSERT_NE(source, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, source->m_width, source->m_height);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = nullptr,
.props =
{
.composeMode = HWC2_COMPOSITION_SOLID_COLOR,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.alpha = 1.0f,
.color =
{
.r = 0,
.g = 255,
.b = 0,
.a = 127,
},
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath("256x256_golden_solid_color_above.png"),
kDefaultSaveImageIfComparisonFailed);
}
TEST_F(CompositorVkTest, Transformations) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source = createSourceImageFromPng(GetTestDataPath("256x256_android.png"));
ASSERT_NE(source, nullptr);
Compositor::CompositionRequest compositionRequest;
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 32,
.top = 32,
.right = 224,
.bottom = 224,
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
const std::unordered_map<hwc_transform_t, std::string> transformToGolden = {
{HWC_TRANSFORM_NONE, "256x256_golden_transform_none.png"},
{HWC_TRANSFORM_FLIP_H, "256x256_golden_transform_fliph.png"},
{HWC_TRANSFORM_FLIP_V, "256x256_golden_transform_flipv.png"},
{HWC_TRANSFORM_ROT_90, "256x256_golden_transform_rot90.png"},
{HWC_TRANSFORM_ROT_180, "256x256_golden_transform_rot180.png"},
{HWC_TRANSFORM_ROT_270, "256x256_golden_transform_rot270.png"},
{HWC_TRANSFORM_FLIP_H_ROT_90, "256x256_golden_transform_fliphrot90.png"},
{HWC_TRANSFORM_FLIP_V_ROT_90, "256x256_golden_transform_flipvrot90.png"},
};
for (const auto [transform, golden] : transformToGolden) {
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice,
m_compositorVkQueue, m_vkCommandPool, 256, 256);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
compositionRequest.target = createBorrowedImageInfo(target.get());
compositionRequest.layers[0].props.transform = transform;
compositionRequest.layers[0].source = createBorrowedImageInfo(source.get());
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath(golden),
kDefaultSaveImageIfComparisonFailed);
}
}
TEST_F(CompositorVkTest, MultipleTargetsComposition) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
constexpr const uint32_t kNumCompostions = 10;
auto source = createImageWithColor<SourceImage>(256, 256, kColorGreen);
ASSERT_NE(source, nullptr);
std::vector<std::unique_ptr<const TargetImage>> targets;
for (uint32_t i = 0; i < kNumCompostions; i++) {
auto target = createImageWithColor<TargetImage>(256, 256, kColorBlack);
ASSERT_NE(target, nullptr);
targets.emplace_back(std::move(target));
}
Compositor::CompositionRequest compositionRequest = {};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = 0,
.bottom = static_cast<int>(source->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source->m_width),
.bottom = static_cast<float>(source->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
const uint32_t displayFrameWidth = 256 / kNumCompostions;
for (uint32_t i = 0; i < kNumCompostions; i++) {
const auto& target = targets[i];
compositionRequest.target = createBorrowedImageInfo(target.get());
compositionRequest.layers[0].source = createBorrowedImageInfo(source.get()),
compositionRequest.layers[0].props.displayFrame.left = (i + 0) * displayFrameWidth;
compositionRequest.layers[0].props.displayFrame.right = (i + 1) * displayFrameWidth;
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(
target.get(),
GetTestDataPath("256x256_golden_multiple_targets_" + std::to_string(i) + ".png"),
kDefaultSaveImageIfComparisonFailed);
}
}
TEST_F(CompositorVkTest, MultipleLayers) {
auto compositor = createCompositor();
ASSERT_NE(compositor, nullptr);
auto source1 = createSourceImageFromPng(GetTestDataPath("256x256_android.png"));
ASSERT_NE(source1, nullptr);
auto source2 =
createSourceImageFromPng(GetTestDataPath("256x256_android_with_transparency.png"));
ASSERT_NE(source2, nullptr);
auto target = TargetImage::create(*k_vk, m_vkDevice, m_vkPhysicalDevice, m_compositorVkQueue,
m_vkCommandPool, kDefaultImageWidth, kDefaultImageHeight);
ASSERT_NE(target, nullptr);
fillImageWith(target.get(), kColorBlack);
Compositor::CompositionRequest compositionRequest = {
.target = createBorrowedImageInfo(target.get()),
};
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source1.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 32.0,
.top = 32.0,
.right = static_cast<float>(source1->m_width) - 32.0f,
.bottom = static_cast<float>(source1->m_height) - 32.0f,
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_NONE,
},
});
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source2.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width),
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source2->m_width) / 2.0f,
.bottom = static_cast<float>(source2->m_height) / 2.0f,
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_ROT_90,
},
});
compositionRequest.layers.emplace_back(Compositor::CompositionRequestLayer{
.source = createBorrowedImageInfo(source2.get()),
.props =
{
.composeMode = HWC2_COMPOSITION_DEVICE,
.displayFrame =
{
.left = 0,
.top = 0,
.right = static_cast<int>(target->m_width) / 2,
.bottom = static_cast<int>(target->m_height),
},
.crop =
{
.left = 0,
.top = 0,
.right = static_cast<float>(source2->m_width),
.bottom = static_cast<float>(source2->m_height),
},
.blendMode = HWC2_BLEND_MODE_PREMULTIPLIED,
.alpha = 1.0,
.color =
{
.r = 0,
.g = 0,
.b = 0,
.a = 0,
},
.transform = HWC_TRANSFORM_FLIP_V_ROT_90,
},
});
auto compositionCompleteWaitable = compositor->compose(compositionRequest);
compositionCompleteWaitable.wait();
compareImageWithGoldenPng(target.get(), GetTestDataPath("256x256_golden_multiple_layers.png"),
kDefaultSaveImageIfComparisonFailed);
}
} // namespace
} // namespace vk
} // namespace gfxstream