/* * Copyright 2016 The Android Open Source Project * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * * http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. */ /* NOTE: * This stub HAL is only used internally by the loader when a real HAL * implementation is not present, in order to avoid needing "null HAL" checks * throughout the loader. It does not enumerate any physical devices, and is * only as conformant to the Vulkan and Android HAL interfaces as the loader * needs it to be. Do not use it as an example of a correct implementation; the * code in ../null_driver is better for that. */ #undef LOG_TAG #define LOG_TAG "vkstub" #include <array> #include <bitset> #include <mutex> #include <log/log.h> #include <hardware/hwvulkan.h> #include "stubhal.h" namespace vulkan { namespace stubhal { namespace { const size_t kMaxInstances = 32; static std::mutex g_instance_mutex; static std::bitset<kMaxInstances> g_instance_used(false); static std::array<hwvulkan_dispatch_t, kMaxInstances> g_instances; [[noreturn]] VKAPI_ATTR void NoOp() { LOG_ALWAYS_FATAL("invalid stub function called"); } VKAPI_ATTR VkResult EnumerateInstanceExtensionProperties(const char* /*layer_name*/, uint32_t* count, VkExtensionProperties* /*properties*/) { *count = 0; return VK_SUCCESS; } VKAPI_ATTR VkResult EnumerateInstanceLayerProperties(uint32_t* count, VkLayerProperties* /*properties*/) { *count = 0; return VK_SUCCESS; } VKAPI_ATTR VkResult CreateInstance(const VkInstanceCreateInfo* /*create_info*/, const VkAllocationCallbacks* /*allocator*/, VkInstance* instance) { std::lock_guard<std::mutex> lock(g_instance_mutex); for (size_t i = 0; i < kMaxInstances; i++) { if (!g_instance_used[i]) { g_instance_used[i] = true; g_instances[i].magic = HWVULKAN_DISPATCH_MAGIC; *instance = reinterpret_cast<VkInstance>(&g_instances[i]); return VK_SUCCESS; } } ALOGE("no more instances available (max=%zu)", kMaxInstances); return VK_ERROR_INITIALIZATION_FAILED; } VKAPI_ATTR void DestroyInstance(VkInstance instance, const VkAllocationCallbacks* /*allocator*/) { std::lock_guard<std::mutex> lock(g_instance_mutex); ssize_t idx = reinterpret_cast<hwvulkan_dispatch_t*>(instance) - &g_instances[0]; ALOG_ASSERT(idx >= 0 && static_cast<size_t>(idx) < g_instance_used.size(), "DestroyInstance: invalid instance handle"); g_instance_used[static_cast<size_t>(idx)] = false; } VKAPI_ATTR VkResult EnumeratePhysicalDevices(VkInstance /*instance*/, uint32_t* count, VkPhysicalDevice* /*gpus*/) { *count = 0; return VK_SUCCESS; } VKAPI_ATTR VkResult EnumeratePhysicalDeviceGroups(VkInstance /*instance*/, uint32_t* count, VkPhysicalDeviceGroupProperties* /*properties*/) { *count = 0; return VK_SUCCESS; } VKAPI_ATTR PFN_vkVoidFunction GetInstanceProcAddr(VkInstance instance, const char* name) { if (strcmp(name, "vkCreateInstance") == 0) return reinterpret_cast<PFN_vkVoidFunction>(CreateInstance); if (strcmp(name, "vkDestroyInstance") == 0) return reinterpret_cast<PFN_vkVoidFunction>(DestroyInstance); if (strcmp(name, "vkEnumerateInstanceExtensionProperties") == 0) return reinterpret_cast<PFN_vkVoidFunction>( EnumerateInstanceExtensionProperties); if (strcmp(name, "vkEnumeratePhysicalDevices") == 0) return reinterpret_cast<PFN_vkVoidFunction>(EnumeratePhysicalDevices); if (strcmp(name, "vkEnumeratePhysicalDeviceGroups") == 0) return reinterpret_cast<PFN_vkVoidFunction>( EnumeratePhysicalDeviceGroups); if (strcmp(name, "vkGetInstanceProcAddr") == 0) return reinterpret_cast<PFN_vkVoidFunction>(GetInstanceProcAddr); // Per the spec, return NULL if instance is NULL. if (!instance) return nullptr; // None of the other Vulkan functions should ever be called, as they all // take a VkPhysicalDevice or other object obtained from a physical device. return reinterpret_cast<PFN_vkVoidFunction>(NoOp); } } // anonymous namespace const hwvulkan_device_t kDevice = { .common = { .tag = HARDWARE_DEVICE_TAG, .version = HWVULKAN_DEVICE_API_VERSION_0_1, .module = nullptr, .close = nullptr, }, .EnumerateInstanceExtensionProperties = EnumerateInstanceExtensionProperties, .CreateInstance = CreateInstance, .GetInstanceProcAddr = GetInstanceProcAddr, }; } // namespace stubhal } // namespace vulkan