blob: 21eca3dc0d65fb671cce9980b18b2e5011cf2a58 [file] [log] [blame]
// Copyright (c) 2016 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "gpu/vulkan/vulkan_surface.h"
#include <vulkan/vulkan.h>
#include "base/macros.h"
#include "base/stl_util.h"
#include "gpu/vulkan/vulkan_device_queue.h"
#include "gpu/vulkan/vulkan_function_pointers.h"
#include "gpu/vulkan/vulkan_swap_chain.h"
namespace gpu {
namespace {
const VkFormat kPreferredVkFormats32[] = {
VK_FORMAT_B8G8R8A8_UNORM, // FORMAT_BGRA8888,
VK_FORMAT_R8G8B8A8_UNORM, // FORMAT_RGBA8888,
};
const VkFormat kPreferredVkFormats16[] = {
VK_FORMAT_R5G6B5_UNORM_PACK16, // FORMAT_RGB565,
};
} // namespace
VulkanSurface::~VulkanSurface() {
DCHECK_EQ(static_cast<VkSurfaceKHR>(VK_NULL_HANDLE), surface_);
}
VulkanSurface::VulkanSurface(VkInstance vk_instance, VkSurfaceKHR surface)
: vk_instance_(vk_instance), surface_(surface) {
DCHECK_NE(static_cast<VkSurfaceKHR>(VK_NULL_HANDLE), surface_);
}
bool VulkanSurface::Initialize(VulkanDeviceQueue* device_queue,
VulkanSurface::Format format) {
DCHECK(format >= 0 && format < NUM_SURFACE_FORMATS);
DCHECK(device_queue);
device_queue_ = device_queue;
VkResult result = VK_SUCCESS;
VkBool32 present_support;
if (vkGetPhysicalDeviceSurfaceSupportKHR(
device_queue_->GetVulkanPhysicalDevice(),
device_queue_->GetVulkanQueueIndex(), surface_,
&present_support) != VK_SUCCESS) {
DLOG(ERROR) << "vkGetPhysicalDeviceSurfaceSupportKHR() failed: " << result;
return false;
}
if (!present_support) {
DLOG(ERROR) << "Surface not supported by present queue.";
return false;
}
// Get list of supported formats.
uint32_t format_count = 0;
result = vkGetPhysicalDeviceSurfaceFormatsKHR(
device_queue_->GetVulkanPhysicalDevice(), surface_, &format_count,
nullptr);
if (VK_SUCCESS != result) {
DLOG(ERROR) << "vkGetPhysicalDeviceSurfaceFormatsKHR() failed: " << result;
return false;
}
std::vector<VkSurfaceFormatKHR> formats(format_count);
result = vkGetPhysicalDeviceSurfaceFormatsKHR(
device_queue_->GetVulkanPhysicalDevice(), surface_, &format_count,
formats.data());
if (VK_SUCCESS != result) {
DLOG(ERROR) << "vkGetPhysicalDeviceSurfaceFormatsKHR() failed: " << result;
return false;
}
const VkFormat* preferred_formats = (format == FORMAT_RGBA_32)
? kPreferredVkFormats32
: kPreferredVkFormats16;
unsigned int size = (format == FORMAT_RGBA_32)
? base::size(kPreferredVkFormats32)
: base::size(kPreferredVkFormats16);
if (formats.size() == 1 && VK_FORMAT_UNDEFINED == formats[0].format) {
surface_format_.format = preferred_formats[0];
surface_format_.colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
} else {
bool format_set = false;
for (VkSurfaceFormatKHR supported_format : formats) {
unsigned int counter = 0;
while (counter < size && format_set == false) {
if (supported_format.format == preferred_formats[counter]) {
surface_format_ = supported_format;
surface_format_.colorSpace = VK_COLORSPACE_SRGB_NONLINEAR_KHR;
format_set = true;
}
counter++;
}
if (format_set)
break;
}
if (!format_set) {
DLOG(ERROR) << "Format not supported.";
return false;
}
}
// Get Surface Information.
VkSurfaceCapabilitiesKHR surface_caps;
result = vkGetPhysicalDeviceSurfaceCapabilitiesKHR(
device_queue_->GetVulkanPhysicalDevice(), surface_, &surface_caps);
if (VK_SUCCESS != result) {
DLOG(ERROR) << "vkGetPhysicalDeviceSurfaceCapabilitiesKHR() failed: "
<< result;
return false;
}
// These are actual surfaces so the current extent should be defined.
DCHECK_NE(UINT_MAX, surface_caps.currentExtent.width);
DCHECK_NE(UINT_MAX, surface_caps.currentExtent.height);
size_ = gfx::Size(surface_caps.currentExtent.width,
surface_caps.currentExtent.height);
// Create Swapchain.
if (!swap_chain_.Initialize(device_queue_, surface_, surface_caps,
surface_format_)) {
return false;
}
return true;
}
void VulkanSurface::Destroy() {
swap_chain_.Destroy();
vkDestroySurfaceKHR(vk_instance_, surface_, nullptr);
surface_ = VK_NULL_HANDLE;
}
gfx::SwapResult VulkanSurface::SwapBuffers() {
return swap_chain_.SwapBuffers();
}
VulkanSwapChain* VulkanSurface::GetSwapChain() {
return &swap_chain_;
}
void VulkanSurface::Finish() {
vkQueueWaitIdle(device_queue_->GetVulkanQueue());
}
} // namespace gpu