Flutter Engine
 
Loading...
Searching...
No Matches
embedder_surface_vulkan.cc
Go to the documentation of this file.
1// Copyright 2013 The Flutter Authors. All rights reserved.
2// Use of this source code is governed by a BSD-style license that can be
3// found in the LICENSE file.
4
6
7#include <utility>
8
14#include "include/gpu/ganesh/GrDirectContext.h"
15#include "include/gpu/vk/VulkanBackendContext.h"
16#include "include/gpu/vk/VulkanExtensions.h"
17#include "third_party/skia/include/gpu/ganesh/vk/GrVkDirectContext.h"
18
19namespace flutter {
20
22 uint32_t version,
23 VkInstance instance,
24 size_t instance_extension_count,
25 const char** instance_extensions,
26 size_t device_extension_count,
27 const char** device_extensions,
28 VkPhysicalDevice physical_device,
29 VkDevice device,
30 uint32_t queue_family_index,
31 VkQueue queue,
32 const VulkanDispatchTable& vulkan_dispatch_table,
33 std::shared_ptr<EmbedderExternalViewEmbedder> external_view_embedder)
34 : vk_(fml::MakeRefCounted<vulkan::VulkanProcTable>(
35 vulkan_dispatch_table.get_instance_proc_address)),
36 device_(*vk_,
37 vulkan::VulkanHandle<VkPhysicalDevice>{physical_device},
41 vulkan_dispatch_table_(vulkan_dispatch_table),
42 external_view_embedder_(std::move(external_view_embedder)) {
43 // Make sure all required members of the dispatch table are checked.
44 if (!vulkan_dispatch_table_.get_instance_proc_address ||
45 !vulkan_dispatch_table_.get_next_image ||
46 !vulkan_dispatch_table_.present_image) {
47 return;
48 }
49
50 bool success = vk_->SetupInstanceProcAddresses(
52 if (!success) {
53 FML_LOG(ERROR) << "Could not setup instance proc addresses.";
54 return;
55 }
56 success =
57 vk_->SetupDeviceProcAddresses(vulkan::VulkanHandle<VkDevice>{device});
58 if (!success) {
59 FML_LOG(ERROR) << "Could not setup device proc addresses.";
60 return;
61 }
62 if (!vk_->IsValid()) {
63 FML_LOG(ERROR) << "VulkanProcTable invalid.";
64 return;
65 }
66
67 main_context_ = CreateGrContext(instance, version, instance_extension_count,
68 instance_extensions, device_extension_count,
69 device_extensions, ContextType::kRender);
70 // TODO(96954): Add a second (optional) queue+family index to the Embedder API
71 // to allow embedders to specify a dedicated transfer queue for
72 // use by the resource context. Queue families with graphics
73 // capability can always be used for memory transferring, but it
74 // would be advantageous to use a dedicated transter queue here.
75 resource_context_ = CreateGrContext(
76 instance, version, instance_extension_count, instance_extensions,
77 device_extension_count, device_extensions, ContextType::kResource);
78
79 valid_ = main_context_ && resource_context_;
80}
81
83 if (main_context_) {
84 main_context_->releaseResourcesAndAbandonContext();
85 }
86 if (resource_context_) {
87 resource_context_->releaseResourcesAndAbandonContext();
88 }
89}
90
91// |GPUSurfaceVulkanDelegate|
95
96// |GPUSurfaceVulkanDelegate|
100
101// |GPUSurfaceVulkanDelegate|
102bool EmbedderSurfaceVulkan::PresentImage(VkImage image, VkFormat format) {
103 return vulkan_dispatch_table_.present_image(image, format);
104}
105
106// |EmbedderSurface|
107bool EmbedderSurfaceVulkan::IsValid() const {
108 return valid_;
109}
110
111// |EmbedderSurface|
112std::unique_ptr<Surface> EmbedderSurfaceVulkan::CreateGPUSurface() {
113 const bool render_to_surface = !external_view_embedder_;
114 return std::make_unique<GPUSurfaceVulkan>(this, main_context_,
115 render_to_surface);
116}
117
118// |EmbedderSurface|
119sk_sp<GrDirectContext> EmbedderSurfaceVulkan::CreateResourceContext() const {
120 return resource_context_;
121}
122
123sk_sp<GrDirectContext> EmbedderSurfaceVulkan::CreateGrContext(
124 VkInstance instance,
125 uint32_t version,
126 size_t instance_extension_count,
127 const char** instance_extensions,
128 size_t device_extension_count,
129 const char** device_extensions,
130 ContextType context_type) const {
131 VkPhysicalDeviceFeatures features;
132 if (!device_.GetPhysicalDeviceFeatures(&features)) {
133 FML_LOG(ERROR) << "Failed to get physical device features.";
134
135 return nullptr;
136 }
137
138 auto get_proc = CreateSkiaGetProc(vk_);
139 if (get_proc == nullptr) {
140 FML_LOG(ERROR) << "Failed to create Vulkan getProc for Skia.";
141 return nullptr;
142 }
143
144 skgpu::VulkanExtensions extensions;
145
146 skgpu::VulkanBackendContext backend_context = {};
147 backend_context.fInstance = instance;
148 backend_context.fPhysicalDevice = device_.GetPhysicalDeviceHandle();
149 backend_context.fDevice = device_.GetHandle();
150 backend_context.fQueue = device_.GetQueueHandle();
151 backend_context.fGraphicsQueueIndex = device_.GetGraphicsQueueIndex();
152 backend_context.fMaxAPIVersion = version;
153 backend_context.fDeviceFeatures = &features;
154 backend_context.fVkExtensions = &extensions;
155 backend_context.fGetProc = get_proc;
156
157 uint32_t vulkan_api_version = version;
158 sk_sp<skgpu::VulkanMemoryAllocator> allocator =
160 vulkan_api_version, instance, device_.GetPhysicalDeviceHandle(),
161 device_.GetHandle(), vk_, true);
162
163 backend_context.fMemoryAllocator = allocator;
164
165 extensions.init(backend_context.fGetProc, backend_context.fInstance,
166 backend_context.fPhysicalDevice, instance_extension_count,
167 instance_extensions, device_extension_count,
168 device_extensions);
169
170 GrContextOptions options =
171 MakeDefaultContextOptions(context_type, GrBackendApi::kVulkan);
172 options.fReduceOpsTaskSplitting = GrContextOptions::Enable::kNo;
173 return GrDirectContexts::MakeVulkan(backend_context, options);
174}
175
176} // namespace flutter
EmbedderSurfaceVulkan(uint32_t version, VkInstance instance, size_t instance_extension_count, const char **instance_extensions, size_t device_extension_count, const char **device_extensions, VkPhysicalDevice physical_device, VkDevice device, uint32_t queue_family_index, VkQueue queue, const VulkanDispatchTable &vulkan_dispatch_table, std::shared_ptr< EmbedderExternalViewEmbedder > external_view_embedder)
FlutterVulkanImage AcquireImage(const DlISize &size) override
Called by the engine to fetch a VkImage for writing the next frame.
const vulkan::VulkanProcTable & vk() override
Obtain a reference to the Vulkan implementation's proc table.
bool PresentImage(VkImage image, VkFormat format) override
Called by the engine once a frame has been rendered to the image and it's ready to be bound for furth...
static sk_sp< VulkanMemoryAllocator > Make(uint32_t vulkan_api_version, VkInstance instance, VkPhysicalDevice physicalDevice, VkDevice device, const fml::RefPtr< vulkan::VulkanProcTable > &vk, bool mustUseCoherentHostVisibleMemory)
const VulkanHandle< VkQueue > & GetQueueHandle() const
const VulkanHandle< VkPhysicalDevice > & GetPhysicalDeviceHandle() const
const VulkanHandle< VkDevice > & GetHandle() const
bool GetPhysicalDeviceFeatures(VkPhysicalDeviceFeatures *features) const
uint32_t GetGraphicsQueueIndex() const
FlutterVulkanImage * image
VkPhysicalDevice physical_device
Definition main.cc:67
VkDevice device
Definition main.cc:69
VkInstance instance
Definition main.cc:64
VkQueue queue
Definition main.cc:71
uint32_t queue_family_index
Definition main.cc:70
uint32_t uint32_t * format
#define FML_LOG(severity)
Definition logging.h:101
it will be possible to load the file into Perfetto s trace viewer use test Running tests that layout and measure text will not yield consistent results across various platforms Enabling this option will make font resolution default to the Ahem test font on all disable asset Prevents usage of any non test fonts unless they were explicitly Loaded via prefetched default font Indicates whether the embedding started a prefetch of the default font manager before creating the engine run In non interactive keep the shell running after the Dart script has completed enable serial On low power devices with low core running concurrent GC tasks on threads can cause them to contend with the UI thread which could potentially lead to jank This option turns off all concurrent GC activities domain network JSON encoded network policy per domain This overrides the DisallowInsecureConnections switch Embedder can specify whether to allow or disallow insecure connections at a domain level old gen heap size
GrContextOptions MakeDefaultContextOptions(ContextType type, std::optional< GrBackendApi > api)
Initializes GrContextOptions with values suitable for Flutter. The options can be further tweaked bef...
@ kRender
The context is used to render to a texture or renderbuffer.
skgpu::VulkanGetProc CreateSkiaGetProc(const fml::RefPtr< vulkan::VulkanProcTable > &vk)
std::function< bool(VkImage image, VkFormat format)> present_image
std::function< FlutterVulkanImage(const DlISize &frame_size)> get_next_image