blob: a8af3f8b847ad705703241f6837964ef15b76ac9 [file] [log] [blame]
// Copyright 2018 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 "ui/ozone/platform/scenic/vulkan_implementation_scenic.h"
#include <lib/zx/channel.h>
#include "base/files/file_path.h"
#include "base/fuchsia/fuchsia_logging.h"
#include "base/native_library.h"
#include "gpu/vulkan/vulkan_function_pointers.h"
#include "gpu/vulkan/vulkan_instance.h"
#include "gpu/vulkan/vulkan_surface.h"
#include "ui/gfx/gpu_fence.h"
#include "ui/ozone/platform/scenic/scenic_window.h"
#include "ui/ozone/platform/scenic/scenic_window_manager.h"
#include "ui/ozone/platform/scenic/vulkan_magma.h"
namespace ui {
VulkanImplementationScenic::VulkanImplementationScenic(
ScenicWindowManager* scenic_window_manager)
: scenic_window_manager_(scenic_window_manager) {}
VulkanImplementationScenic::~VulkanImplementationScenic() = default;
bool VulkanImplementationScenic::InitializeVulkanInstance() {
base::NativeLibraryLoadError error;
base::NativeLibrary handle =
base::LoadNativeLibrary(base::FilePath("libvulkan.so"), &error);
if (!handle) {
LOG(ERROR) << "Failed to load vulkan: " << error.ToString();
return false;
}
gpu::VulkanFunctionPointers* vulkan_function_pointers =
gpu::GetVulkanFunctionPointers();
vulkan_function_pointers->vulkan_loader_library_ = handle;
std::vector<const char*> required_extensions = {
VK_KHR_SURFACE_EXTENSION_NAME, VK_KHR_MAGMA_SURFACE_EXTENSION_NAME,
};
std::vector<const char*> required_layers = {
"VK_LAYER_GOOGLE_image_pipe_swapchain",
};
if (!vulkan_instance_.Initialize(required_extensions, required_layers)) {
vulkan_instance_.Destroy();
return false;
}
vkCreateMagmaSurfaceKHR_ = vkGetInstanceProcAddr(
vulkan_instance_.vk_instance(), "vkCreateMagmaSurfaceKHR");
if (!vkCreateMagmaSurfaceKHR_) {
vulkan_instance_.Destroy();
return false;
}
vkGetPhysicalDeviceMagmaPresentationSupportKHR_ =
vkGetInstanceProcAddr(vulkan_instance_.vk_instance(),
"vkGetPhysicalDeviceMagmaPresentationSupportKHR");
if (!vkGetPhysicalDeviceMagmaPresentationSupportKHR_) {
vulkan_instance_.Destroy();
return false;
}
return true;
}
VkInstance VulkanImplementationScenic::GetVulkanInstance() {
return vulkan_instance_.vk_instance();
}
std::unique_ptr<gpu::VulkanSurface>
VulkanImplementationScenic::CreateViewSurface(gfx::AcceleratedWidget window) {
ScenicWindow* scenic_window = scenic_window_manager_->GetWindow(window);
if (!scenic_window)
return nullptr;
ScenicSession* scenic_session = scenic_window->scenic_session();
fuchsia::images::ImagePipePtr image_pipe;
ScenicSession::ResourceId image_pipe_id =
scenic_session->CreateImagePipe(image_pipe.NewRequest());
scenic_window->SetTexture(image_pipe_id);
scenic_session->ReleaseResource(image_pipe_id);
scenic_session->Present();
VkSurfaceKHR surface;
VkMagmaSurfaceCreateInfoKHR surface_create_info = {};
surface_create_info.sType = VK_STRUCTURE_TYPE_MAGMA_SURFACE_CREATE_INFO_KHR;
surface_create_info.imagePipeHandle =
image_pipe.Unbind().TakeChannel().release();
VkResult result =
reinterpret_cast<PFN_vkCreateMagmaSurfaceKHR>(vkCreateMagmaSurfaceKHR_)(
GetVulkanInstance(), &surface_create_info, nullptr, &surface);
if (result != VK_SUCCESS) {
// This shouldn't fail, and we don't know whether imagePipeHandle was closed
// if it does.
LOG(FATAL) << "vkCreateMagmaSurfaceKHR failed: " << result;
}
return std::make_unique<gpu::VulkanSurface>(GetVulkanInstance(), surface);
}
bool VulkanImplementationScenic::GetPhysicalDevicePresentationSupport(
VkPhysicalDevice physical_device,
const std::vector<VkQueueFamilyProperties>& queue_family_properties,
uint32_t queue_family_index) {
// TODO(spang): vkGetPhysicalDeviceMagmaPresentationSupportKHR returns false
// here. Use it once it is fixed.
NOTIMPLEMENTED();
return true;
}
std::vector<const char*>
VulkanImplementationScenic::GetRequiredDeviceExtensions() {
return {VK_KHR_SWAPCHAIN_EXTENSION_NAME};
}
VkFence VulkanImplementationScenic::CreateVkFenceForGpuFence(
VkDevice vk_device) {
NOTIMPLEMENTED();
return VK_NULL_HANDLE;
}
std::unique_ptr<gfx::GpuFence>
VulkanImplementationScenic::ExportVkFenceToGpuFence(VkDevice vk_device,
VkFence vk_fence) {
NOTIMPLEMENTED();
return nullptr;
}
} // namespace ui