1*38e8c45fSAndroid Build Coastguard Worker /*
2*38e8c45fSAndroid Build Coastguard Worker * Copyright 2024 The Android Open Source Project
3*38e8c45fSAndroid Build Coastguard Worker *
4*38e8c45fSAndroid Build Coastguard Worker * Licensed under the Apache License, Version 2.0 (the "License");
5*38e8c45fSAndroid Build Coastguard Worker * you may not use this file except in compliance with the License.
6*38e8c45fSAndroid Build Coastguard Worker * You may obtain a copy of the License at
7*38e8c45fSAndroid Build Coastguard Worker *
8*38e8c45fSAndroid Build Coastguard Worker * http://www.apache.org/licenses/LICENSE-2.0
9*38e8c45fSAndroid Build Coastguard Worker *
10*38e8c45fSAndroid Build Coastguard Worker * Unless required by applicable law or agreed to in writing, software
11*38e8c45fSAndroid Build Coastguard Worker * distributed under the License is distributed on an "AS IS" BASIS,
12*38e8c45fSAndroid Build Coastguard Worker * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13*38e8c45fSAndroid Build Coastguard Worker * See the License for the specific language governing permissions and
14*38e8c45fSAndroid Build Coastguard Worker * limitations under the License.
15*38e8c45fSAndroid Build Coastguard Worker */
16*38e8c45fSAndroid Build Coastguard Worker
17*38e8c45fSAndroid Build Coastguard Worker #undef LOG_TAG
18*38e8c45fSAndroid Build Coastguard Worker #define LOG_TAG "RenderEngine"
19*38e8c45fSAndroid Build Coastguard Worker
20*38e8c45fSAndroid Build Coastguard Worker #include "VulkanInterface.h"
21*38e8c45fSAndroid Build Coastguard Worker
22*38e8c45fSAndroid Build Coastguard Worker #include <include/gpu/GpuTypes.h>
23*38e8c45fSAndroid Build Coastguard Worker #include <include/gpu/vk/VulkanBackendContext.h>
24*38e8c45fSAndroid Build Coastguard Worker
25*38e8c45fSAndroid Build Coastguard Worker #include <log/log_main.h>
26*38e8c45fSAndroid Build Coastguard Worker #include <utils/Timers.h>
27*38e8c45fSAndroid Build Coastguard Worker
28*38e8c45fSAndroid Build Coastguard Worker #include <cinttypes>
29*38e8c45fSAndroid Build Coastguard Worker #include <sstream>
30*38e8c45fSAndroid Build Coastguard Worker
31*38e8c45fSAndroid Build Coastguard Worker namespace android {
32*38e8c45fSAndroid Build Coastguard Worker namespace renderengine {
33*38e8c45fSAndroid Build Coastguard Worker namespace skia {
34*38e8c45fSAndroid Build Coastguard Worker
getGaneshBackendContext()35*38e8c45fSAndroid Build Coastguard Worker VulkanBackendContext VulkanInterface::getGaneshBackendContext() {
36*38e8c45fSAndroid Build Coastguard Worker return this->getGraphiteBackendContext();
37*38e8c45fSAndroid Build Coastguard Worker };
38*38e8c45fSAndroid Build Coastguard Worker
getGraphiteBackendContext()39*38e8c45fSAndroid Build Coastguard Worker VulkanBackendContext VulkanInterface::getGraphiteBackendContext() {
40*38e8c45fSAndroid Build Coastguard Worker VulkanBackendContext backendContext;
41*38e8c45fSAndroid Build Coastguard Worker backendContext.fInstance = mInstance;
42*38e8c45fSAndroid Build Coastguard Worker backendContext.fPhysicalDevice = mPhysicalDevice;
43*38e8c45fSAndroid Build Coastguard Worker backendContext.fDevice = mDevice;
44*38e8c45fSAndroid Build Coastguard Worker backendContext.fQueue = mQueue;
45*38e8c45fSAndroid Build Coastguard Worker backendContext.fGraphicsQueueIndex = mQueueIndex;
46*38e8c45fSAndroid Build Coastguard Worker backendContext.fMaxAPIVersion = mApiVersion;
47*38e8c45fSAndroid Build Coastguard Worker backendContext.fVkExtensions = &mVulkanExtensions;
48*38e8c45fSAndroid Build Coastguard Worker backendContext.fDeviceFeatures2 = mPhysicalDeviceFeatures2;
49*38e8c45fSAndroid Build Coastguard Worker backendContext.fGetProc = mGrGetProc;
50*38e8c45fSAndroid Build Coastguard Worker backendContext.fProtectedContext = mIsProtected ? Protected::kYes : Protected::kNo;
51*38e8c45fSAndroid Build Coastguard Worker backendContext.fDeviceLostContext = this; // VulkanInterface is long-lived
52*38e8c45fSAndroid Build Coastguard Worker backendContext.fDeviceLostProc = onVkDeviceFault;
53*38e8c45fSAndroid Build Coastguard Worker return backendContext;
54*38e8c45fSAndroid Build Coastguard Worker };
55*38e8c45fSAndroid Build Coastguard Worker
createExportableSemaphore()56*38e8c45fSAndroid Build Coastguard Worker VkSemaphore VulkanInterface::createExportableSemaphore() {
57*38e8c45fSAndroid Build Coastguard Worker VkExportSemaphoreCreateInfo exportInfo;
58*38e8c45fSAndroid Build Coastguard Worker exportInfo.sType = VK_STRUCTURE_TYPE_EXPORT_SEMAPHORE_CREATE_INFO;
59*38e8c45fSAndroid Build Coastguard Worker exportInfo.pNext = nullptr;
60*38e8c45fSAndroid Build Coastguard Worker exportInfo.handleTypes = VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT;
61*38e8c45fSAndroid Build Coastguard Worker
62*38e8c45fSAndroid Build Coastguard Worker VkSemaphoreCreateInfo semaphoreInfo;
63*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
64*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.pNext = &exportInfo;
65*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.flags = 0;
66*38e8c45fSAndroid Build Coastguard Worker
67*38e8c45fSAndroid Build Coastguard Worker VkSemaphore semaphore;
68*38e8c45fSAndroid Build Coastguard Worker VkResult err = mFuncs.vkCreateSemaphore(mDevice, &semaphoreInfo, nullptr, &semaphore);
69*38e8c45fSAndroid Build Coastguard Worker if (VK_SUCCESS != err) {
70*38e8c45fSAndroid Build Coastguard Worker ALOGE("%s: failed to create semaphore. err %d\n", __func__, err);
71*38e8c45fSAndroid Build Coastguard Worker return VK_NULL_HANDLE;
72*38e8c45fSAndroid Build Coastguard Worker }
73*38e8c45fSAndroid Build Coastguard Worker
74*38e8c45fSAndroid Build Coastguard Worker return semaphore;
75*38e8c45fSAndroid Build Coastguard Worker }
76*38e8c45fSAndroid Build Coastguard Worker
77*38e8c45fSAndroid Build Coastguard Worker // syncFd cannot be <= 0
importSemaphoreFromSyncFd(int syncFd)78*38e8c45fSAndroid Build Coastguard Worker VkSemaphore VulkanInterface::importSemaphoreFromSyncFd(int syncFd) {
79*38e8c45fSAndroid Build Coastguard Worker VkSemaphoreCreateInfo semaphoreInfo;
80*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_CREATE_INFO;
81*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.pNext = nullptr;
82*38e8c45fSAndroid Build Coastguard Worker semaphoreInfo.flags = 0;
83*38e8c45fSAndroid Build Coastguard Worker
84*38e8c45fSAndroid Build Coastguard Worker VkSemaphore semaphore;
85*38e8c45fSAndroid Build Coastguard Worker VkResult err = mFuncs.vkCreateSemaphore(mDevice, &semaphoreInfo, nullptr, &semaphore);
86*38e8c45fSAndroid Build Coastguard Worker if (VK_SUCCESS != err) {
87*38e8c45fSAndroid Build Coastguard Worker ALOGE("%s: failed to create import semaphore", __func__);
88*38e8c45fSAndroid Build Coastguard Worker return VK_NULL_HANDLE;
89*38e8c45fSAndroid Build Coastguard Worker }
90*38e8c45fSAndroid Build Coastguard Worker
91*38e8c45fSAndroid Build Coastguard Worker VkImportSemaphoreFdInfoKHR importInfo;
92*38e8c45fSAndroid Build Coastguard Worker importInfo.sType = VK_STRUCTURE_TYPE_IMPORT_SEMAPHORE_FD_INFO_KHR;
93*38e8c45fSAndroid Build Coastguard Worker importInfo.pNext = nullptr;
94*38e8c45fSAndroid Build Coastguard Worker importInfo.semaphore = semaphore;
95*38e8c45fSAndroid Build Coastguard Worker importInfo.flags = VK_SEMAPHORE_IMPORT_TEMPORARY_BIT;
96*38e8c45fSAndroid Build Coastguard Worker importInfo.handleType = VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT;
97*38e8c45fSAndroid Build Coastguard Worker importInfo.fd = syncFd;
98*38e8c45fSAndroid Build Coastguard Worker
99*38e8c45fSAndroid Build Coastguard Worker err = mFuncs.vkImportSemaphoreFdKHR(mDevice, &importInfo);
100*38e8c45fSAndroid Build Coastguard Worker if (VK_SUCCESS != err) {
101*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroySemaphore(mDevice, semaphore, nullptr);
102*38e8c45fSAndroid Build Coastguard Worker ALOGE("%s: failed to import semaphore", __func__);
103*38e8c45fSAndroid Build Coastguard Worker return VK_NULL_HANDLE;
104*38e8c45fSAndroid Build Coastguard Worker }
105*38e8c45fSAndroid Build Coastguard Worker
106*38e8c45fSAndroid Build Coastguard Worker return semaphore;
107*38e8c45fSAndroid Build Coastguard Worker }
108*38e8c45fSAndroid Build Coastguard Worker
exportSemaphoreSyncFd(VkSemaphore semaphore)109*38e8c45fSAndroid Build Coastguard Worker int VulkanInterface::exportSemaphoreSyncFd(VkSemaphore semaphore) {
110*38e8c45fSAndroid Build Coastguard Worker int res;
111*38e8c45fSAndroid Build Coastguard Worker
112*38e8c45fSAndroid Build Coastguard Worker VkSemaphoreGetFdInfoKHR getFdInfo;
113*38e8c45fSAndroid Build Coastguard Worker getFdInfo.sType = VK_STRUCTURE_TYPE_SEMAPHORE_GET_FD_INFO_KHR;
114*38e8c45fSAndroid Build Coastguard Worker getFdInfo.pNext = nullptr;
115*38e8c45fSAndroid Build Coastguard Worker getFdInfo.semaphore = semaphore;
116*38e8c45fSAndroid Build Coastguard Worker getFdInfo.handleType = VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT;
117*38e8c45fSAndroid Build Coastguard Worker VkResult err = mFuncs.vkGetSemaphoreFdKHR(mDevice, &getFdInfo, &res);
118*38e8c45fSAndroid Build Coastguard Worker if (VK_SUCCESS != err) {
119*38e8c45fSAndroid Build Coastguard Worker ALOGE("%s: failed to export semaphore, err: %d", __func__, err);
120*38e8c45fSAndroid Build Coastguard Worker return -1;
121*38e8c45fSAndroid Build Coastguard Worker }
122*38e8c45fSAndroid Build Coastguard Worker return res;
123*38e8c45fSAndroid Build Coastguard Worker }
124*38e8c45fSAndroid Build Coastguard Worker
destroySemaphore(VkSemaphore semaphore)125*38e8c45fSAndroid Build Coastguard Worker void VulkanInterface::destroySemaphore(VkSemaphore semaphore) {
126*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroySemaphore(mDevice, semaphore, nullptr);
127*38e8c45fSAndroid Build Coastguard Worker }
128*38e8c45fSAndroid Build Coastguard Worker
onVkDeviceFault(void * callbackContext,const std::string & description,const std::vector<VkDeviceFaultAddressInfoEXT> & addressInfos,const std::vector<VkDeviceFaultVendorInfoEXT> & vendorInfos,const std::vector<std::byte> & vendorBinaryData)129*38e8c45fSAndroid Build Coastguard Worker void VulkanInterface::onVkDeviceFault(void* callbackContext, const std::string& description,
130*38e8c45fSAndroid Build Coastguard Worker const std::vector<VkDeviceFaultAddressInfoEXT>& addressInfos,
131*38e8c45fSAndroid Build Coastguard Worker const std::vector<VkDeviceFaultVendorInfoEXT>& vendorInfos,
132*38e8c45fSAndroid Build Coastguard Worker const std::vector<std::byte>& vendorBinaryData) {
133*38e8c45fSAndroid Build Coastguard Worker VulkanInterface* interface = static_cast<VulkanInterface*>(callbackContext);
134*38e8c45fSAndroid Build Coastguard Worker const std::string protectedStr = interface->mIsProtected ? "protected" : "non-protected";
135*38e8c45fSAndroid Build Coastguard Worker // The final crash string should contain as much differentiating info as possible, up to 1024
136*38e8c45fSAndroid Build Coastguard Worker // bytes. As this final message is constructed, the same information is also dumped to the logs
137*38e8c45fSAndroid Build Coastguard Worker // but in a more verbose format. Building the crash string is unsightly, so the clearer logging
138*38e8c45fSAndroid Build Coastguard Worker // statement is always placed first to give context.
139*38e8c45fSAndroid Build Coastguard Worker ALOGE("VK_ERROR_DEVICE_LOST (%s context): %s", protectedStr.c_str(), description.c_str());
140*38e8c45fSAndroid Build Coastguard Worker std::stringstream crashMsg;
141*38e8c45fSAndroid Build Coastguard Worker crashMsg << "VK_ERROR_DEVICE_LOST (" << protectedStr;
142*38e8c45fSAndroid Build Coastguard Worker
143*38e8c45fSAndroid Build Coastguard Worker if (!addressInfos.empty()) {
144*38e8c45fSAndroid Build Coastguard Worker ALOGE("%zu VkDeviceFaultAddressInfoEXT:", addressInfos.size());
145*38e8c45fSAndroid Build Coastguard Worker crashMsg << ", " << addressInfos.size() << " address info (";
146*38e8c45fSAndroid Build Coastguard Worker for (VkDeviceFaultAddressInfoEXT addressInfo : addressInfos) {
147*38e8c45fSAndroid Build Coastguard Worker ALOGE(" addressType: %d", (int)addressInfo.addressType);
148*38e8c45fSAndroid Build Coastguard Worker ALOGE(" reportedAddress: %" PRIu64, addressInfo.reportedAddress);
149*38e8c45fSAndroid Build Coastguard Worker ALOGE(" addressPrecision: %" PRIu64, addressInfo.addressPrecision);
150*38e8c45fSAndroid Build Coastguard Worker crashMsg << addressInfo.addressType << ":" << addressInfo.reportedAddress << ":"
151*38e8c45fSAndroid Build Coastguard Worker << addressInfo.addressPrecision << ", ";
152*38e8c45fSAndroid Build Coastguard Worker }
153*38e8c45fSAndroid Build Coastguard Worker crashMsg.seekp(-2, crashMsg.cur); // Move back to overwrite trailing ", "
154*38e8c45fSAndroid Build Coastguard Worker crashMsg << ")";
155*38e8c45fSAndroid Build Coastguard Worker }
156*38e8c45fSAndroid Build Coastguard Worker
157*38e8c45fSAndroid Build Coastguard Worker if (!vendorInfos.empty()) {
158*38e8c45fSAndroid Build Coastguard Worker ALOGE("%zu VkDeviceFaultVendorInfoEXT:", vendorInfos.size());
159*38e8c45fSAndroid Build Coastguard Worker crashMsg << ", " << vendorInfos.size() << " vendor info (";
160*38e8c45fSAndroid Build Coastguard Worker for (VkDeviceFaultVendorInfoEXT vendorInfo : vendorInfos) {
161*38e8c45fSAndroid Build Coastguard Worker ALOGE(" description: %s", vendorInfo.description);
162*38e8c45fSAndroid Build Coastguard Worker ALOGE(" vendorFaultCode: %" PRIu64, vendorInfo.vendorFaultCode);
163*38e8c45fSAndroid Build Coastguard Worker ALOGE(" vendorFaultData: %" PRIu64, vendorInfo.vendorFaultData);
164*38e8c45fSAndroid Build Coastguard Worker // Omit descriptions for individual vendor info structs in the crash string, as the
165*38e8c45fSAndroid Build Coastguard Worker // fault code and fault data fields should be enough for clustering, and the verbosity
166*38e8c45fSAndroid Build Coastguard Worker // isn't worth it. Additionally, vendors may just set the general description field of
167*38e8c45fSAndroid Build Coastguard Worker // the overall fault to the description of the first element in this list, and that
168*38e8c45fSAndroid Build Coastguard Worker // overall description will be placed at the end of the crash string.
169*38e8c45fSAndroid Build Coastguard Worker crashMsg << vendorInfo.vendorFaultCode << ":" << vendorInfo.vendorFaultData << ", ";
170*38e8c45fSAndroid Build Coastguard Worker }
171*38e8c45fSAndroid Build Coastguard Worker crashMsg.seekp(-2, crashMsg.cur); // Move back to overwrite trailing ", "
172*38e8c45fSAndroid Build Coastguard Worker crashMsg << ")";
173*38e8c45fSAndroid Build Coastguard Worker }
174*38e8c45fSAndroid Build Coastguard Worker
175*38e8c45fSAndroid Build Coastguard Worker if (!vendorBinaryData.empty()) {
176*38e8c45fSAndroid Build Coastguard Worker // TODO: b/322830575 - Log in base64, or dump directly to a file that gets put in bugreports
177*38e8c45fSAndroid Build Coastguard Worker ALOGE("%zu bytes of vendor-specific binary data (please notify Android's Core Graphics"
178*38e8c45fSAndroid Build Coastguard Worker " Stack team if you observe this message).",
179*38e8c45fSAndroid Build Coastguard Worker vendorBinaryData.size());
180*38e8c45fSAndroid Build Coastguard Worker crashMsg << ", " << vendorBinaryData.size() << " bytes binary";
181*38e8c45fSAndroid Build Coastguard Worker }
182*38e8c45fSAndroid Build Coastguard Worker
183*38e8c45fSAndroid Build Coastguard Worker crashMsg << "): " << description;
184*38e8c45fSAndroid Build Coastguard Worker LOG_ALWAYS_FATAL("%s", crashMsg.str().c_str());
185*38e8c45fSAndroid Build Coastguard Worker };
186*38e8c45fSAndroid Build Coastguard Worker
187*38e8c45fSAndroid Build Coastguard Worker static skgpu::VulkanGetProc sGetProc = [](const char* proc_name,
188*38e8c45fSAndroid Build Coastguard Worker VkInstance instance,
__anonbd5490c30102(const char* proc_name, VkInstance instance, VkDevice device) 189*38e8c45fSAndroid Build Coastguard Worker VkDevice device) {
190*38e8c45fSAndroid Build Coastguard Worker if (device != VK_NULL_HANDLE) {
191*38e8c45fSAndroid Build Coastguard Worker return vkGetDeviceProcAddr(device, proc_name);
192*38e8c45fSAndroid Build Coastguard Worker }
193*38e8c45fSAndroid Build Coastguard Worker return vkGetInstanceProcAddr(instance, proc_name);
194*38e8c45fSAndroid Build Coastguard Worker };
195*38e8c45fSAndroid Build Coastguard Worker
196*38e8c45fSAndroid Build Coastguard Worker #define BAIL(fmt, ...) \
197*38e8c45fSAndroid Build Coastguard Worker { \
198*38e8c45fSAndroid Build Coastguard Worker ALOGE("%s: " fmt ", bailing", __func__, ##__VA_ARGS__); \
199*38e8c45fSAndroid Build Coastguard Worker return; \
200*38e8c45fSAndroid Build Coastguard Worker }
201*38e8c45fSAndroid Build Coastguard Worker
202*38e8c45fSAndroid Build Coastguard Worker #define CHECK_NONNULL(expr) \
203*38e8c45fSAndroid Build Coastguard Worker if ((expr) == nullptr) { \
204*38e8c45fSAndroid Build Coastguard Worker BAIL("[%s] null", #expr); \
205*38e8c45fSAndroid Build Coastguard Worker }
206*38e8c45fSAndroid Build Coastguard Worker
207*38e8c45fSAndroid Build Coastguard Worker #define VK_CHECK(expr) \
208*38e8c45fSAndroid Build Coastguard Worker if ((expr) != VK_SUCCESS) { \
209*38e8c45fSAndroid Build Coastguard Worker BAIL("[%s] failed. err = %d", #expr, expr); \
210*38e8c45fSAndroid Build Coastguard Worker return; \
211*38e8c45fSAndroid Build Coastguard Worker }
212*38e8c45fSAndroid Build Coastguard Worker
213*38e8c45fSAndroid Build Coastguard Worker #define VK_GET_PROC(F) \
214*38e8c45fSAndroid Build Coastguard Worker PFN_vk##F vk##F = (PFN_vk##F)vkGetInstanceProcAddr(VK_NULL_HANDLE, "vk" #F); \
215*38e8c45fSAndroid Build Coastguard Worker CHECK_NONNULL(vk##F)
216*38e8c45fSAndroid Build Coastguard Worker #define VK_GET_INST_PROC(instance, F) \
217*38e8c45fSAndroid Build Coastguard Worker PFN_vk##F vk##F = (PFN_vk##F)vkGetInstanceProcAddr(instance, "vk" #F); \
218*38e8c45fSAndroid Build Coastguard Worker CHECK_NONNULL(vk##F)
219*38e8c45fSAndroid Build Coastguard Worker #define VK_GET_DEV_PROC(device, F) \
220*38e8c45fSAndroid Build Coastguard Worker PFN_vk##F vk##F = (PFN_vk##F)vkGetDeviceProcAddr(device, "vk" #F); \
221*38e8c45fSAndroid Build Coastguard Worker CHECK_NONNULL(vk##F)
222*38e8c45fSAndroid Build Coastguard Worker
init(bool protectedContent)223*38e8c45fSAndroid Build Coastguard Worker void VulkanInterface::init(bool protectedContent) {
224*38e8c45fSAndroid Build Coastguard Worker if (isInitialized()) {
225*38e8c45fSAndroid Build Coastguard Worker ALOGW("Called init on already initialized VulkanInterface");
226*38e8c45fSAndroid Build Coastguard Worker return;
227*38e8c45fSAndroid Build Coastguard Worker }
228*38e8c45fSAndroid Build Coastguard Worker
229*38e8c45fSAndroid Build Coastguard Worker const nsecs_t timeBefore = systemTime();
230*38e8c45fSAndroid Build Coastguard Worker
231*38e8c45fSAndroid Build Coastguard Worker VK_GET_PROC(EnumerateInstanceVersion);
232*38e8c45fSAndroid Build Coastguard Worker uint32_t instanceVersion;
233*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumerateInstanceVersion(&instanceVersion));
234*38e8c45fSAndroid Build Coastguard Worker
235*38e8c45fSAndroid Build Coastguard Worker if (instanceVersion < VK_MAKE_VERSION(1, 1, 0)) {
236*38e8c45fSAndroid Build Coastguard Worker BAIL("Vulkan instance API version %" PRIu32 ".%" PRIu32 ".%" PRIu32 " < 1.1.0",
237*38e8c45fSAndroid Build Coastguard Worker VK_VERSION_MAJOR(instanceVersion), VK_VERSION_MINOR(instanceVersion),
238*38e8c45fSAndroid Build Coastguard Worker VK_VERSION_PATCH(instanceVersion));
239*38e8c45fSAndroid Build Coastguard Worker }
240*38e8c45fSAndroid Build Coastguard Worker
241*38e8c45fSAndroid Build Coastguard Worker const VkApplicationInfo appInfo = {
242*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_APPLICATION_INFO, nullptr, "surfaceflinger", 0, "android platform", 0,
243*38e8c45fSAndroid Build Coastguard Worker VK_MAKE_VERSION(1, 1, 0),
244*38e8c45fSAndroid Build Coastguard Worker };
245*38e8c45fSAndroid Build Coastguard Worker
246*38e8c45fSAndroid Build Coastguard Worker VK_GET_PROC(EnumerateInstanceExtensionProperties);
247*38e8c45fSAndroid Build Coastguard Worker
248*38e8c45fSAndroid Build Coastguard Worker uint32_t extensionCount = 0;
249*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumerateInstanceExtensionProperties(nullptr, &extensionCount, nullptr));
250*38e8c45fSAndroid Build Coastguard Worker std::vector<VkExtensionProperties> instanceExtensions(extensionCount);
251*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumerateInstanceExtensionProperties(nullptr, &extensionCount,
252*38e8c45fSAndroid Build Coastguard Worker instanceExtensions.data()));
253*38e8c45fSAndroid Build Coastguard Worker std::vector<const char*> enabledInstanceExtensionNames;
254*38e8c45fSAndroid Build Coastguard Worker enabledInstanceExtensionNames.reserve(instanceExtensions.size());
255*38e8c45fSAndroid Build Coastguard Worker mInstanceExtensionNames.reserve(instanceExtensions.size());
256*38e8c45fSAndroid Build Coastguard Worker for (const auto& instExt : instanceExtensions) {
257*38e8c45fSAndroid Build Coastguard Worker enabledInstanceExtensionNames.push_back(instExt.extensionName);
258*38e8c45fSAndroid Build Coastguard Worker mInstanceExtensionNames.push_back(instExt.extensionName);
259*38e8c45fSAndroid Build Coastguard Worker }
260*38e8c45fSAndroid Build Coastguard Worker
261*38e8c45fSAndroid Build Coastguard Worker const VkInstanceCreateInfo instanceCreateInfo = {
262*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_INSTANCE_CREATE_INFO,
263*38e8c45fSAndroid Build Coastguard Worker nullptr,
264*38e8c45fSAndroid Build Coastguard Worker 0,
265*38e8c45fSAndroid Build Coastguard Worker &appInfo,
266*38e8c45fSAndroid Build Coastguard Worker 0,
267*38e8c45fSAndroid Build Coastguard Worker nullptr,
268*38e8c45fSAndroid Build Coastguard Worker (uint32_t)enabledInstanceExtensionNames.size(),
269*38e8c45fSAndroid Build Coastguard Worker enabledInstanceExtensionNames.data(),
270*38e8c45fSAndroid Build Coastguard Worker };
271*38e8c45fSAndroid Build Coastguard Worker
272*38e8c45fSAndroid Build Coastguard Worker VK_GET_PROC(CreateInstance);
273*38e8c45fSAndroid Build Coastguard Worker VkInstance instance;
274*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkCreateInstance(&instanceCreateInfo, nullptr, &instance));
275*38e8c45fSAndroid Build Coastguard Worker
276*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, DestroyInstance);
277*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroyInstance = vkDestroyInstance;
278*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, EnumeratePhysicalDevices);
279*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, EnumerateDeviceExtensionProperties);
280*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, GetPhysicalDeviceProperties2);
281*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, GetPhysicalDeviceExternalSemaphoreProperties);
282*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, GetPhysicalDeviceQueueFamilyProperties2);
283*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, GetPhysicalDeviceFeatures2);
284*38e8c45fSAndroid Build Coastguard Worker VK_GET_INST_PROC(instance, CreateDevice);
285*38e8c45fSAndroid Build Coastguard Worker
286*38e8c45fSAndroid Build Coastguard Worker uint32_t physdevCount;
287*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumeratePhysicalDevices(instance, &physdevCount, nullptr));
288*38e8c45fSAndroid Build Coastguard Worker if (physdevCount == 0) {
289*38e8c45fSAndroid Build Coastguard Worker BAIL("Could not find any physical devices");
290*38e8c45fSAndroid Build Coastguard Worker }
291*38e8c45fSAndroid Build Coastguard Worker
292*38e8c45fSAndroid Build Coastguard Worker physdevCount = 1;
293*38e8c45fSAndroid Build Coastguard Worker VkPhysicalDevice physicalDevice;
294*38e8c45fSAndroid Build Coastguard Worker VkResult enumeratePhysDevsErr =
295*38e8c45fSAndroid Build Coastguard Worker vkEnumeratePhysicalDevices(instance, &physdevCount, &physicalDevice);
296*38e8c45fSAndroid Build Coastguard Worker if (enumeratePhysDevsErr != VK_SUCCESS && VK_INCOMPLETE != enumeratePhysDevsErr) {
297*38e8c45fSAndroid Build Coastguard Worker BAIL("vkEnumeratePhysicalDevices failed with non-VK_INCOMPLETE error: %d",
298*38e8c45fSAndroid Build Coastguard Worker enumeratePhysDevsErr);
299*38e8c45fSAndroid Build Coastguard Worker }
300*38e8c45fSAndroid Build Coastguard Worker
301*38e8c45fSAndroid Build Coastguard Worker VkPhysicalDeviceProperties2 physDevProps = {
302*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2,
303*38e8c45fSAndroid Build Coastguard Worker 0,
304*38e8c45fSAndroid Build Coastguard Worker {},
305*38e8c45fSAndroid Build Coastguard Worker };
306*38e8c45fSAndroid Build Coastguard Worker VkPhysicalDeviceProtectedMemoryProperties protMemProps = {
307*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROTECTED_MEMORY_PROPERTIES,
308*38e8c45fSAndroid Build Coastguard Worker 0,
309*38e8c45fSAndroid Build Coastguard Worker {},
310*38e8c45fSAndroid Build Coastguard Worker };
311*38e8c45fSAndroid Build Coastguard Worker
312*38e8c45fSAndroid Build Coastguard Worker if (protectedContent) {
313*38e8c45fSAndroid Build Coastguard Worker physDevProps.pNext = &protMemProps;
314*38e8c45fSAndroid Build Coastguard Worker }
315*38e8c45fSAndroid Build Coastguard Worker
316*38e8c45fSAndroid Build Coastguard Worker vkGetPhysicalDeviceProperties2(physicalDevice, &physDevProps);
317*38e8c45fSAndroid Build Coastguard Worker const uint32_t physicalDeviceApiVersion = physDevProps.properties.apiVersion;
318*38e8c45fSAndroid Build Coastguard Worker if (physicalDeviceApiVersion < VK_MAKE_VERSION(1, 1, 0)) {
319*38e8c45fSAndroid Build Coastguard Worker BAIL("Vulkan physical device API version %" PRIu32 ".%" PRIu32 ".%" PRIu32 " < 1.1.0",
320*38e8c45fSAndroid Build Coastguard Worker VK_VERSION_MAJOR(physicalDeviceApiVersion), VK_VERSION_MINOR(physicalDeviceApiVersion),
321*38e8c45fSAndroid Build Coastguard Worker VK_VERSION_PATCH(physicalDeviceApiVersion));
322*38e8c45fSAndroid Build Coastguard Worker }
323*38e8c45fSAndroid Build Coastguard Worker
324*38e8c45fSAndroid Build Coastguard Worker // Check for syncfd support. Bail if we cannot both import and export them.
325*38e8c45fSAndroid Build Coastguard Worker VkPhysicalDeviceExternalSemaphoreInfo semInfo = {
326*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_EXTERNAL_SEMAPHORE_INFO,
327*38e8c45fSAndroid Build Coastguard Worker nullptr,
328*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
329*38e8c45fSAndroid Build Coastguard Worker };
330*38e8c45fSAndroid Build Coastguard Worker VkExternalSemaphoreProperties semProps = {
331*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_EXTERNAL_SEMAPHORE_PROPERTIES, nullptr, 0, 0, 0,
332*38e8c45fSAndroid Build Coastguard Worker };
333*38e8c45fSAndroid Build Coastguard Worker vkGetPhysicalDeviceExternalSemaphoreProperties(physicalDevice, &semInfo, &semProps);
334*38e8c45fSAndroid Build Coastguard Worker
335*38e8c45fSAndroid Build Coastguard Worker bool sufficientSemaphoreSyncFdSupport = (semProps.exportFromImportedHandleTypes &
336*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT) &&
337*38e8c45fSAndroid Build Coastguard Worker (semProps.compatibleHandleTypes & VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT) &&
338*38e8c45fSAndroid Build Coastguard Worker (semProps.externalSemaphoreFeatures & VK_EXTERNAL_SEMAPHORE_FEATURE_EXPORTABLE_BIT) &&
339*38e8c45fSAndroid Build Coastguard Worker (semProps.externalSemaphoreFeatures & VK_EXTERNAL_SEMAPHORE_FEATURE_IMPORTABLE_BIT);
340*38e8c45fSAndroid Build Coastguard Worker
341*38e8c45fSAndroid Build Coastguard Worker if (!sufficientSemaphoreSyncFdSupport) {
342*38e8c45fSAndroid Build Coastguard Worker BAIL("Vulkan device does not support sufficient external semaphore sync fd features. "
343*38e8c45fSAndroid Build Coastguard Worker "exportFromImportedHandleTypes 0x%x (needed 0x%x) "
344*38e8c45fSAndroid Build Coastguard Worker "compatibleHandleTypes 0x%x (needed 0x%x) "
345*38e8c45fSAndroid Build Coastguard Worker "externalSemaphoreFeatures 0x%x (needed 0x%x) ",
346*38e8c45fSAndroid Build Coastguard Worker semProps.exportFromImportedHandleTypes, VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
347*38e8c45fSAndroid Build Coastguard Worker semProps.compatibleHandleTypes, VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
348*38e8c45fSAndroid Build Coastguard Worker semProps.externalSemaphoreFeatures,
349*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_FEATURE_EXPORTABLE_BIT |
350*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_FEATURE_IMPORTABLE_BIT);
351*38e8c45fSAndroid Build Coastguard Worker } else {
352*38e8c45fSAndroid Build Coastguard Worker ALOGD("Vulkan device supports sufficient external semaphore sync fd features. "
353*38e8c45fSAndroid Build Coastguard Worker "exportFromImportedHandleTypes 0x%x (needed 0x%x) "
354*38e8c45fSAndroid Build Coastguard Worker "compatibleHandleTypes 0x%x (needed 0x%x) "
355*38e8c45fSAndroid Build Coastguard Worker "externalSemaphoreFeatures 0x%x (needed 0x%x) ",
356*38e8c45fSAndroid Build Coastguard Worker semProps.exportFromImportedHandleTypes, VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
357*38e8c45fSAndroid Build Coastguard Worker semProps.compatibleHandleTypes, VK_EXTERNAL_SEMAPHORE_HANDLE_TYPE_SYNC_FD_BIT,
358*38e8c45fSAndroid Build Coastguard Worker semProps.externalSemaphoreFeatures,
359*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_FEATURE_EXPORTABLE_BIT |
360*38e8c45fSAndroid Build Coastguard Worker VK_EXTERNAL_SEMAPHORE_FEATURE_IMPORTABLE_BIT);
361*38e8c45fSAndroid Build Coastguard Worker }
362*38e8c45fSAndroid Build Coastguard Worker
363*38e8c45fSAndroid Build Coastguard Worker uint32_t queueCount;
364*38e8c45fSAndroid Build Coastguard Worker vkGetPhysicalDeviceQueueFamilyProperties2(physicalDevice, &queueCount, nullptr);
365*38e8c45fSAndroid Build Coastguard Worker if (queueCount == 0) {
366*38e8c45fSAndroid Build Coastguard Worker BAIL("Could not find queues for physical device");
367*38e8c45fSAndroid Build Coastguard Worker }
368*38e8c45fSAndroid Build Coastguard Worker
369*38e8c45fSAndroid Build Coastguard Worker std::vector<VkQueueFamilyProperties2> queueProps(queueCount);
370*38e8c45fSAndroid Build Coastguard Worker std::vector<VkQueueFamilyGlobalPriorityPropertiesEXT> queuePriorityProps(queueCount);
371*38e8c45fSAndroid Build Coastguard Worker VkQueueGlobalPriorityKHR queuePriority = VK_QUEUE_GLOBAL_PRIORITY_MEDIUM_KHR;
372*38e8c45fSAndroid Build Coastguard Worker // Even though we don't yet know if the VK_EXT_global_priority extension is available,
373*38e8c45fSAndroid Build Coastguard Worker // we can safely add the request to the pNext chain, and if the extension is not
374*38e8c45fSAndroid Build Coastguard Worker // available, it will be ignored.
375*38e8c45fSAndroid Build Coastguard Worker for (uint32_t i = 0; i < queueCount; ++i) {
376*38e8c45fSAndroid Build Coastguard Worker queuePriorityProps[i].sType = VK_STRUCTURE_TYPE_QUEUE_FAMILY_GLOBAL_PRIORITY_PROPERTIES_EXT;
377*38e8c45fSAndroid Build Coastguard Worker queuePriorityProps[i].pNext = nullptr;
378*38e8c45fSAndroid Build Coastguard Worker queueProps[i].pNext = &queuePriorityProps[i];
379*38e8c45fSAndroid Build Coastguard Worker }
380*38e8c45fSAndroid Build Coastguard Worker vkGetPhysicalDeviceQueueFamilyProperties2(physicalDevice, &queueCount, queueProps.data());
381*38e8c45fSAndroid Build Coastguard Worker
382*38e8c45fSAndroid Build Coastguard Worker int graphicsQueueIndex = -1;
383*38e8c45fSAndroid Build Coastguard Worker for (uint32_t i = 0; i < queueCount; ++i) {
384*38e8c45fSAndroid Build Coastguard Worker // Look at potential answers to the VK_EXT_global_priority query. If answers were
385*38e8c45fSAndroid Build Coastguard Worker // provided, we may adjust the queuePriority.
386*38e8c45fSAndroid Build Coastguard Worker if (queueProps[i].queueFamilyProperties.queueFlags & VK_QUEUE_GRAPHICS_BIT) {
387*38e8c45fSAndroid Build Coastguard Worker for (uint32_t j = 0; j < queuePriorityProps[i].priorityCount; j++) {
388*38e8c45fSAndroid Build Coastguard Worker if (queuePriorityProps[i].priorities[j] > queuePriority) {
389*38e8c45fSAndroid Build Coastguard Worker queuePriority = queuePriorityProps[i].priorities[j];
390*38e8c45fSAndroid Build Coastguard Worker }
391*38e8c45fSAndroid Build Coastguard Worker }
392*38e8c45fSAndroid Build Coastguard Worker if (queuePriority == VK_QUEUE_GLOBAL_PRIORITY_REALTIME_KHR) {
393*38e8c45fSAndroid Build Coastguard Worker mIsRealtimePriority = true;
394*38e8c45fSAndroid Build Coastguard Worker }
395*38e8c45fSAndroid Build Coastguard Worker graphicsQueueIndex = i;
396*38e8c45fSAndroid Build Coastguard Worker break;
397*38e8c45fSAndroid Build Coastguard Worker }
398*38e8c45fSAndroid Build Coastguard Worker }
399*38e8c45fSAndroid Build Coastguard Worker
400*38e8c45fSAndroid Build Coastguard Worker if (graphicsQueueIndex == -1) {
401*38e8c45fSAndroid Build Coastguard Worker BAIL("Could not find a graphics queue family");
402*38e8c45fSAndroid Build Coastguard Worker }
403*38e8c45fSAndroid Build Coastguard Worker
404*38e8c45fSAndroid Build Coastguard Worker uint32_t deviceExtensionCount;
405*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumerateDeviceExtensionProperties(physicalDevice, nullptr, &deviceExtensionCount,
406*38e8c45fSAndroid Build Coastguard Worker nullptr));
407*38e8c45fSAndroid Build Coastguard Worker std::vector<VkExtensionProperties> deviceExtensions(deviceExtensionCount);
408*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkEnumerateDeviceExtensionProperties(physicalDevice, nullptr, &deviceExtensionCount,
409*38e8c45fSAndroid Build Coastguard Worker deviceExtensions.data()));
410*38e8c45fSAndroid Build Coastguard Worker
411*38e8c45fSAndroid Build Coastguard Worker std::vector<const char*> enabledDeviceExtensionNames;
412*38e8c45fSAndroid Build Coastguard Worker enabledDeviceExtensionNames.reserve(deviceExtensions.size());
413*38e8c45fSAndroid Build Coastguard Worker mDeviceExtensionNames.reserve(deviceExtensions.size());
414*38e8c45fSAndroid Build Coastguard Worker for (const auto& devExt : deviceExtensions) {
415*38e8c45fSAndroid Build Coastguard Worker enabledDeviceExtensionNames.push_back(devExt.extensionName);
416*38e8c45fSAndroid Build Coastguard Worker mDeviceExtensionNames.push_back(devExt.extensionName);
417*38e8c45fSAndroid Build Coastguard Worker }
418*38e8c45fSAndroid Build Coastguard Worker
419*38e8c45fSAndroid Build Coastguard Worker mVulkanExtensions.init(sGetProc, instance, physicalDevice, enabledInstanceExtensionNames.size(),
420*38e8c45fSAndroid Build Coastguard Worker enabledInstanceExtensionNames.data(), enabledDeviceExtensionNames.size(),
421*38e8c45fSAndroid Build Coastguard Worker enabledDeviceExtensionNames.data());
422*38e8c45fSAndroid Build Coastguard Worker
423*38e8c45fSAndroid Build Coastguard Worker if (!mVulkanExtensions.hasExtension(VK_KHR_EXTERNAL_SEMAPHORE_FD_EXTENSION_NAME, 1)) {
424*38e8c45fSAndroid Build Coastguard Worker BAIL("Vulkan driver doesn't support external semaphore fd");
425*38e8c45fSAndroid Build Coastguard Worker }
426*38e8c45fSAndroid Build Coastguard Worker
427*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2 = new VkPhysicalDeviceFeatures2;
428*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2->sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FEATURES_2;
429*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2->pNext = nullptr;
430*38e8c45fSAndroid Build Coastguard Worker
431*38e8c45fSAndroid Build Coastguard Worker mSamplerYcbcrConversionFeatures = new VkPhysicalDeviceSamplerYcbcrConversionFeatures;
432*38e8c45fSAndroid Build Coastguard Worker mSamplerYcbcrConversionFeatures->sType =
433*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_SAMPLER_YCBCR_CONVERSION_FEATURES;
434*38e8c45fSAndroid Build Coastguard Worker mSamplerYcbcrConversionFeatures->pNext = nullptr;
435*38e8c45fSAndroid Build Coastguard Worker
436*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2->pNext = mSamplerYcbcrConversionFeatures;
437*38e8c45fSAndroid Build Coastguard Worker void** tailPnext = &mSamplerYcbcrConversionFeatures->pNext;
438*38e8c45fSAndroid Build Coastguard Worker
439*38e8c45fSAndroid Build Coastguard Worker if (protectedContent) {
440*38e8c45fSAndroid Build Coastguard Worker mProtectedMemoryFeatures = new VkPhysicalDeviceProtectedMemoryFeatures;
441*38e8c45fSAndroid Build Coastguard Worker mProtectedMemoryFeatures->sType =
442*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROTECTED_MEMORY_FEATURES;
443*38e8c45fSAndroid Build Coastguard Worker mProtectedMemoryFeatures->pNext = nullptr;
444*38e8c45fSAndroid Build Coastguard Worker *tailPnext = mProtectedMemoryFeatures;
445*38e8c45fSAndroid Build Coastguard Worker tailPnext = &mProtectedMemoryFeatures->pNext;
446*38e8c45fSAndroid Build Coastguard Worker }
447*38e8c45fSAndroid Build Coastguard Worker
448*38e8c45fSAndroid Build Coastguard Worker if (mVulkanExtensions.hasExtension(VK_EXT_DEVICE_FAULT_EXTENSION_NAME, 1)) {
449*38e8c45fSAndroid Build Coastguard Worker mDeviceFaultFeatures = new VkPhysicalDeviceFaultFeaturesEXT;
450*38e8c45fSAndroid Build Coastguard Worker mDeviceFaultFeatures->sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_FAULT_FEATURES_EXT;
451*38e8c45fSAndroid Build Coastguard Worker mDeviceFaultFeatures->pNext = nullptr;
452*38e8c45fSAndroid Build Coastguard Worker *tailPnext = mDeviceFaultFeatures;
453*38e8c45fSAndroid Build Coastguard Worker tailPnext = &mDeviceFaultFeatures->pNext;
454*38e8c45fSAndroid Build Coastguard Worker }
455*38e8c45fSAndroid Build Coastguard Worker
456*38e8c45fSAndroid Build Coastguard Worker vkGetPhysicalDeviceFeatures2(physicalDevice, mPhysicalDeviceFeatures2);
457*38e8c45fSAndroid Build Coastguard Worker // Looks like this would slow things down and we can't depend on it on all platforms
458*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2->features.robustBufferAccess = VK_FALSE;
459*38e8c45fSAndroid Build Coastguard Worker
460*38e8c45fSAndroid Build Coastguard Worker if (protectedContent && !mProtectedMemoryFeatures->protectedMemory) {
461*38e8c45fSAndroid Build Coastguard Worker BAIL("Protected memory not supported");
462*38e8c45fSAndroid Build Coastguard Worker }
463*38e8c45fSAndroid Build Coastguard Worker
464*38e8c45fSAndroid Build Coastguard Worker float queuePriorities[1] = {0.0f};
465*38e8c45fSAndroid Build Coastguard Worker void* queueNextPtr = nullptr;
466*38e8c45fSAndroid Build Coastguard Worker
467*38e8c45fSAndroid Build Coastguard Worker VkDeviceQueueGlobalPriorityCreateInfoEXT queuePriorityCreateInfo = {
468*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_DEVICE_QUEUE_GLOBAL_PRIORITY_CREATE_INFO_EXT,
469*38e8c45fSAndroid Build Coastguard Worker nullptr,
470*38e8c45fSAndroid Build Coastguard Worker // If queue priority is supported, RE should always have realtime priority.
471*38e8c45fSAndroid Build Coastguard Worker queuePriority,
472*38e8c45fSAndroid Build Coastguard Worker };
473*38e8c45fSAndroid Build Coastguard Worker
474*38e8c45fSAndroid Build Coastguard Worker if (mVulkanExtensions.hasExtension(VK_EXT_GLOBAL_PRIORITY_EXTENSION_NAME, 2)) {
475*38e8c45fSAndroid Build Coastguard Worker queueNextPtr = &queuePriorityCreateInfo;
476*38e8c45fSAndroid Build Coastguard Worker }
477*38e8c45fSAndroid Build Coastguard Worker
478*38e8c45fSAndroid Build Coastguard Worker VkDeviceQueueCreateFlags deviceQueueCreateFlags =
479*38e8c45fSAndroid Build Coastguard Worker (VkDeviceQueueCreateFlags)(protectedContent ? VK_DEVICE_QUEUE_CREATE_PROTECTED_BIT : 0);
480*38e8c45fSAndroid Build Coastguard Worker
481*38e8c45fSAndroid Build Coastguard Worker const VkDeviceQueueCreateInfo queueInfo = {
482*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_DEVICE_QUEUE_CREATE_INFO,
483*38e8c45fSAndroid Build Coastguard Worker queueNextPtr,
484*38e8c45fSAndroid Build Coastguard Worker deviceQueueCreateFlags,
485*38e8c45fSAndroid Build Coastguard Worker (uint32_t)graphicsQueueIndex,
486*38e8c45fSAndroid Build Coastguard Worker 1,
487*38e8c45fSAndroid Build Coastguard Worker queuePriorities,
488*38e8c45fSAndroid Build Coastguard Worker };
489*38e8c45fSAndroid Build Coastguard Worker
490*38e8c45fSAndroid Build Coastguard Worker const VkDeviceCreateInfo deviceInfo = {
491*38e8c45fSAndroid Build Coastguard Worker VK_STRUCTURE_TYPE_DEVICE_CREATE_INFO,
492*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2,
493*38e8c45fSAndroid Build Coastguard Worker 0,
494*38e8c45fSAndroid Build Coastguard Worker 1,
495*38e8c45fSAndroid Build Coastguard Worker &queueInfo,
496*38e8c45fSAndroid Build Coastguard Worker 0,
497*38e8c45fSAndroid Build Coastguard Worker nullptr,
498*38e8c45fSAndroid Build Coastguard Worker (uint32_t)enabledDeviceExtensionNames.size(),
499*38e8c45fSAndroid Build Coastguard Worker enabledDeviceExtensionNames.data(),
500*38e8c45fSAndroid Build Coastguard Worker nullptr,
501*38e8c45fSAndroid Build Coastguard Worker };
502*38e8c45fSAndroid Build Coastguard Worker
503*38e8c45fSAndroid Build Coastguard Worker ALOGD("Trying to create Vk device with protectedContent=%d", protectedContent);
504*38e8c45fSAndroid Build Coastguard Worker VkDevice device;
505*38e8c45fSAndroid Build Coastguard Worker VK_CHECK(vkCreateDevice(physicalDevice, &deviceInfo, nullptr, &device));
506*38e8c45fSAndroid Build Coastguard Worker ALOGD("Trying to create Vk device with protectedContent=%d (success)", protectedContent);
507*38e8c45fSAndroid Build Coastguard Worker
508*38e8c45fSAndroid Build Coastguard Worker VkQueue graphicsQueue;
509*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, GetDeviceQueue2);
510*38e8c45fSAndroid Build Coastguard Worker const VkDeviceQueueInfo2 deviceQueueInfo2 = {VK_STRUCTURE_TYPE_DEVICE_QUEUE_INFO_2, nullptr,
511*38e8c45fSAndroid Build Coastguard Worker deviceQueueCreateFlags,
512*38e8c45fSAndroid Build Coastguard Worker (uint32_t)graphicsQueueIndex, 0};
513*38e8c45fSAndroid Build Coastguard Worker vkGetDeviceQueue2(device, &deviceQueueInfo2, &graphicsQueue);
514*38e8c45fSAndroid Build Coastguard Worker
515*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, DeviceWaitIdle);
516*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, DestroyDevice);
517*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDeviceWaitIdle = vkDeviceWaitIdle;
518*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroyDevice = vkDestroyDevice;
519*38e8c45fSAndroid Build Coastguard Worker
520*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, CreateSemaphore);
521*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, ImportSemaphoreFdKHR);
522*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, GetSemaphoreFdKHR);
523*38e8c45fSAndroid Build Coastguard Worker VK_GET_DEV_PROC(device, DestroySemaphore);
524*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkCreateSemaphore = vkCreateSemaphore;
525*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkImportSemaphoreFdKHR = vkImportSemaphoreFdKHR;
526*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkGetSemaphoreFdKHR = vkGetSemaphoreFdKHR;
527*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroySemaphore = vkDestroySemaphore;
528*38e8c45fSAndroid Build Coastguard Worker
529*38e8c45fSAndroid Build Coastguard Worker // At this point, everything's succeeded and we can continue
530*38e8c45fSAndroid Build Coastguard Worker mInitialized = true;
531*38e8c45fSAndroid Build Coastguard Worker mInstance = instance;
532*38e8c45fSAndroid Build Coastguard Worker mPhysicalDevice = physicalDevice;
533*38e8c45fSAndroid Build Coastguard Worker mDevice = device;
534*38e8c45fSAndroid Build Coastguard Worker mQueue = graphicsQueue;
535*38e8c45fSAndroid Build Coastguard Worker mQueueIndex = graphicsQueueIndex;
536*38e8c45fSAndroid Build Coastguard Worker mApiVersion = physicalDeviceApiVersion;
537*38e8c45fSAndroid Build Coastguard Worker // grExtensions already constructed
538*38e8c45fSAndroid Build Coastguard Worker // feature pointers already constructed
539*38e8c45fSAndroid Build Coastguard Worker mGrGetProc = sGetProc;
540*38e8c45fSAndroid Build Coastguard Worker mIsProtected = protectedContent;
541*38e8c45fSAndroid Build Coastguard Worker // mIsRealtimePriority already initialized by constructor
542*38e8c45fSAndroid Build Coastguard Worker // funcs already initialized
543*38e8c45fSAndroid Build Coastguard Worker
544*38e8c45fSAndroid Build Coastguard Worker const nsecs_t timeAfter = systemTime();
545*38e8c45fSAndroid Build Coastguard Worker const float initTimeMs = static_cast<float>(timeAfter - timeBefore) / 1.0E6;
546*38e8c45fSAndroid Build Coastguard Worker ALOGD("%s: Success init Vulkan interface in %f ms", __func__, initTimeMs);
547*38e8c45fSAndroid Build Coastguard Worker }
548*38e8c45fSAndroid Build Coastguard Worker
takeOwnership()549*38e8c45fSAndroid Build Coastguard Worker bool VulkanInterface::takeOwnership() {
550*38e8c45fSAndroid Build Coastguard Worker if (!isInitialized() || mIsOwned) {
551*38e8c45fSAndroid Build Coastguard Worker return false;
552*38e8c45fSAndroid Build Coastguard Worker }
553*38e8c45fSAndroid Build Coastguard Worker mIsOwned = true;
554*38e8c45fSAndroid Build Coastguard Worker return true;
555*38e8c45fSAndroid Build Coastguard Worker }
556*38e8c45fSAndroid Build Coastguard Worker
teardown()557*38e8c45fSAndroid Build Coastguard Worker void VulkanInterface::teardown() {
558*38e8c45fSAndroid Build Coastguard Worker // Core resources that must be destroyed using Vulkan functions.
559*38e8c45fSAndroid Build Coastguard Worker if (mDevice != VK_NULL_HANDLE) {
560*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDeviceWaitIdle(mDevice);
561*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroyDevice(mDevice, nullptr);
562*38e8c45fSAndroid Build Coastguard Worker mDevice = VK_NULL_HANDLE;
563*38e8c45fSAndroid Build Coastguard Worker }
564*38e8c45fSAndroid Build Coastguard Worker if (mInstance != VK_NULL_HANDLE) {
565*38e8c45fSAndroid Build Coastguard Worker mFuncs.vkDestroyInstance(mInstance, nullptr);
566*38e8c45fSAndroid Build Coastguard Worker mInstance = VK_NULL_HANDLE;
567*38e8c45fSAndroid Build Coastguard Worker }
568*38e8c45fSAndroid Build Coastguard Worker
569*38e8c45fSAndroid Build Coastguard Worker // Optional features that can be deleted directly.
570*38e8c45fSAndroid Build Coastguard Worker // TODO: b/293371537 - This section should likely be improved to walk the pNext chain of
571*38e8c45fSAndroid Build Coastguard Worker // mPhysicalDeviceFeatures2 and free everything like HWUI's VulkanManager.
572*38e8c45fSAndroid Build Coastguard Worker if (mProtectedMemoryFeatures) {
573*38e8c45fSAndroid Build Coastguard Worker delete mProtectedMemoryFeatures;
574*38e8c45fSAndroid Build Coastguard Worker mProtectedMemoryFeatures = nullptr;
575*38e8c45fSAndroid Build Coastguard Worker }
576*38e8c45fSAndroid Build Coastguard Worker if (mSamplerYcbcrConversionFeatures) {
577*38e8c45fSAndroid Build Coastguard Worker delete mSamplerYcbcrConversionFeatures;
578*38e8c45fSAndroid Build Coastguard Worker mSamplerYcbcrConversionFeatures = nullptr;
579*38e8c45fSAndroid Build Coastguard Worker }
580*38e8c45fSAndroid Build Coastguard Worker if (mPhysicalDeviceFeatures2) {
581*38e8c45fSAndroid Build Coastguard Worker delete mPhysicalDeviceFeatures2;
582*38e8c45fSAndroid Build Coastguard Worker mPhysicalDeviceFeatures2 = nullptr;
583*38e8c45fSAndroid Build Coastguard Worker }
584*38e8c45fSAndroid Build Coastguard Worker if (mDeviceFaultFeatures) {
585*38e8c45fSAndroid Build Coastguard Worker delete mDeviceFaultFeatures;
586*38e8c45fSAndroid Build Coastguard Worker mDeviceFaultFeatures = nullptr;
587*38e8c45fSAndroid Build Coastguard Worker }
588*38e8c45fSAndroid Build Coastguard Worker
589*38e8c45fSAndroid Build Coastguard Worker // Misc. fields that can be trivially reset without special deletion:
590*38e8c45fSAndroid Build Coastguard Worker mInitialized = false;
591*38e8c45fSAndroid Build Coastguard Worker mIsOwned = false;
592*38e8c45fSAndroid Build Coastguard Worker mPhysicalDevice = VK_NULL_HANDLE; // Implicitly destroyed by destroying mInstance.
593*38e8c45fSAndroid Build Coastguard Worker mQueue = VK_NULL_HANDLE; // Implicitly destroyed by destroying mDevice.
594*38e8c45fSAndroid Build Coastguard Worker mQueueIndex = 0;
595*38e8c45fSAndroid Build Coastguard Worker mApiVersion = 0;
596*38e8c45fSAndroid Build Coastguard Worker mVulkanExtensions = skgpu::VulkanExtensions();
597*38e8c45fSAndroid Build Coastguard Worker mGrGetProc = nullptr;
598*38e8c45fSAndroid Build Coastguard Worker mIsProtected = false;
599*38e8c45fSAndroid Build Coastguard Worker mIsRealtimePriority = false;
600*38e8c45fSAndroid Build Coastguard Worker
601*38e8c45fSAndroid Build Coastguard Worker mFuncs = VulkanFuncs();
602*38e8c45fSAndroid Build Coastguard Worker
603*38e8c45fSAndroid Build Coastguard Worker mInstanceExtensionNames.clear();
604*38e8c45fSAndroid Build Coastguard Worker mDeviceExtensionNames.clear();
605*38e8c45fSAndroid Build Coastguard Worker }
606*38e8c45fSAndroid Build Coastguard Worker
607*38e8c45fSAndroid Build Coastguard Worker } // namespace skia
608*38e8c45fSAndroid Build Coastguard Worker } // namespace renderengine
609*38e8c45fSAndroid Build Coastguard Worker } // namespace android
610