diff --git a/meson_options.txt b/meson_options.txt index a342354acfced..d243997902fbc 100644 --- a/meson_options.txt +++ b/meson_options.txt @@ -181,6 +181,12 @@ option( value : false, description : 'Whether to build the vulkan overlay layer' ) +option( + 'vulkan-device-select-layer', + type : 'boolean', + value : false, + description : 'Whether to build the vulkan device select layer' +) option( 'shared-glapi', type : 'combo', diff --git a/src/vulkan/device-select-layer/VkLayer_MESA_device_select.json b/src/vulkan/device-select-layer/VkLayer_MESA_device_select.json new file mode 100644 index 0000000000000..1d5fffd0135c5 --- /dev/null +++ b/src/vulkan/device-select-layer/VkLayer_MESA_device_select.json @@ -0,0 +1,17 @@ +{ + "file_format_version" : "1.0.0", + "layer" : { + "name": "VK_LAYER_MESA_device_select", + "type": "GLOBAL", + "library_path": "libVkLayer_MESA_device_select.so", + "api_version": "1.1.73", + "implementation_version": "1", + "description": "Linux device selection layer", + "functions": { + "vkNegotiateLoaderLayerInterfaceVersion": "vkNegotiateLoaderLayerInterfaceVersion" + }, + "disable_environment": { + "NODEVICE_SELECT": "1" + } + } +} diff --git a/src/vulkan/device-select-layer/device_select.h b/src/vulkan/device-select-layer/device_select.h new file mode 100644 index 0000000000000..c9483aa0b79c0 --- /dev/null +++ b/src/vulkan/device-select-layer/device_select.h @@ -0,0 +1,47 @@ +/* + * Copyright © 2019 Red Hat + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the next + * paragraph) shall be included in all copies or substantial portions of the + * Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS + * IN THE SOFTWARE. + */ +#ifndef DEVICE_SELECT_H +#define DEVICE_SELECT_H + +#include +#include "xf86drm.h" + +struct device_pci_info { + drmPciDeviceInfo dev_info; + drmPciBusInfo bus_info; + bool has_bus_info; +}; + +#ifdef VK_USE_PLATFORM_XCB_KHR +int device_select_find_xcb_pci_default(struct device_pci_info *devices, uint32_t device_count); +#else +static inline int device_select_find_xcb_pci_default(struct device_pci_info *devices, uint32_t device_count) { return -1; } +#endif + +#ifdef VK_USE_PLATFORM_WAYLAND_KHR +int device_select_find_wayland_pci_default(struct device_pci_info *devices, uint32_t device_count); +#else +static inline int device_select_find_wayland_pci_default(struct device_pci_info *devices, uint32_t device_count) { return -1; } +#endif + +#endif diff --git a/src/vulkan/device-select-layer/device_select_layer.c b/src/vulkan/device-select-layer/device_select_layer.c new file mode 100644 index 0000000000000..7bec8890344c3 --- /dev/null +++ b/src/vulkan/device-select-layer/device_select_layer.c @@ -0,0 +1,483 @@ +/* + * Copyright © 2017 Google + * Copyright © 2019 Red Hat + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the next + * paragraph) shall be included in all copies or substantial portions of the + * Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS + * IN THE SOFTWARE. + */ + +/* Rules for device selection. + * Is there an X or wayland connection open (or DISPLAY set). + * If no - try and find which device was the boot_vga device. + * If yes - try and work out which device is the connection primary, + * DRI_PRIME tagged overrides only work if bus info, =1 will just pick an alternate. + */ + +#include + +#include +#include +#include +#include +#include + +#include "device_select.h" +#include "c99_compat.h" +#include "hash_table.h" +#include "vk_util.h" +#include "c11/threads.h" + +struct instance_info { + PFN_vkDestroyInstance DestroyInstance; + PFN_vkEnumeratePhysicalDevices EnumeratePhysicalDevices; + PFN_vkEnumeratePhysicalDeviceGroups EnumeratePhysicalDeviceGroups; + PFN_vkGetInstanceProcAddr GetInstanceProcAddr; + PFN_GetPhysicalDeviceProcAddr GetPhysicalDeviceProcAddr; + PFN_vkEnumerateDeviceExtensionProperties EnumerateDeviceExtensionProperties; + PFN_vkGetPhysicalDeviceProperties GetPhysicalDeviceProperties; + PFN_vkGetPhysicalDeviceProperties2KHR GetPhysicalDeviceProperties2KHR; + bool has_props2, has_pci_bus; + bool has_wayland, has_xcb; +}; + +static struct hash_table *device_select_instance_ht = NULL; +static mtx_t device_select_mutex; + +static once_flag device_select_is_init = ONCE_FLAG_INIT; + +static void device_select_once_init(void) { + mtx_init(&device_select_mutex, mtx_plain); +} + +static void +device_select_init_instances(void) +{ + call_once(&device_select_is_init, device_select_once_init); + + mtx_lock(&device_select_mutex); + if (!device_select_instance_ht) + device_select_instance_ht = _mesa_hash_table_create(NULL, _mesa_hash_pointer, + _mesa_key_pointer_equal); + mtx_unlock(&device_select_mutex); +} + +static void +device_select_try_free_ht(void) +{ + mtx_lock(&device_select_mutex); + if (device_select_instance_ht) { + if (_mesa_hash_table_num_entries(device_select_instance_ht) == 0) { + _mesa_hash_table_destroy(device_select_instance_ht, NULL); + device_select_instance_ht = NULL; + } + } + mtx_unlock(&device_select_mutex); +} + +static void +device_select_layer_add_instance(VkInstance instance, struct instance_info *info) +{ + device_select_init_instances(); + mtx_lock(&device_select_mutex); + _mesa_hash_table_insert(device_select_instance_ht, instance, info); + mtx_unlock(&device_select_mutex); +} + +static struct instance_info * +device_select_layer_get_instance(VkInstance instance) +{ + struct hash_entry *entry; + struct instance_info *info = NULL; + mtx_lock(&device_select_mutex); + entry = _mesa_hash_table_search(device_select_instance_ht, (void *)instance); + if (entry) + info = (struct instance_info *)entry->data; + mtx_unlock(&device_select_mutex); + return info; +} + +static void +device_select_layer_remove_instance(VkInstance instance) +{ + mtx_lock(&device_select_mutex); + _mesa_hash_table_remove_key(device_select_instance_ht, instance); + mtx_unlock(&device_select_mutex); + device_select_try_free_ht(); +} + +static VkResult device_select_CreateInstance(const VkInstanceCreateInfo *pCreateInfo, + const VkAllocationCallbacks *pAllocator, + VkInstance *pInstance) +{ + VkLayerInstanceCreateInfo *chain_info; + for(chain_info = (VkLayerInstanceCreateInfo*)pCreateInfo->pNext; chain_info; chain_info = (VkLayerInstanceCreateInfo*)chain_info->pNext) + if(chain_info->sType == VK_STRUCTURE_TYPE_LOADER_INSTANCE_CREATE_INFO && chain_info->function == VK_LAYER_LINK_INFO) + break; + + assert(chain_info->u.pLayerInfo); + struct instance_info *info = (struct instance_info *)calloc(1, sizeof(struct instance_info)); + + info->GetInstanceProcAddr = chain_info->u.pLayerInfo->pfnNextGetInstanceProcAddr; + PFN_vkCreateInstance fpCreateInstance = + (PFN_vkCreateInstance)info->GetInstanceProcAddr(NULL, "vkCreateInstance"); + if (fpCreateInstance == NULL) { + return VK_ERROR_INITIALIZATION_FAILED; + } + + chain_info->u.pLayerInfo = chain_info->u.pLayerInfo->pNext; + + VkResult result = fpCreateInstance(pCreateInfo, pAllocator, pInstance); + if (result != VK_SUCCESS) + return result; + + for (unsigned i = 0; i < pCreateInfo->enabledExtensionCount; i++) { + if (!strcmp(pCreateInfo->ppEnabledExtensionNames[i], VK_KHR_GET_PHYSICAL_DEVICE_PROPERTIES_2_EXTENSION_NAME)) + info->has_props2 = true; +#ifdef VK_USE_PLATFORM_WAYLAND_KHR + if (!strcmp(pCreateInfo->ppEnabledExtensionNames[i], VK_KHR_WAYLAND_SURFACE_EXTENSION_NAME)) + info->has_wayland = true; +#endif +#ifdef VK_USE_PLATFORM_XCB_KHR + if (!strcmp(pCreateInfo->ppEnabledExtensionNames[i], VK_KHR_XCB_SURFACE_EXTENSION_NAME)) + info->has_xcb = true; +#endif + } + + info->GetPhysicalDeviceProcAddr = (PFN_GetPhysicalDeviceProcAddr)info->GetInstanceProcAddr(*pInstance, "vk_layerGetPhysicalDeviceProcAddr"); +#define DEVSEL_GET_CB(func) info->func = (PFN_vk##func)info->GetInstanceProcAddr(*pInstance, "vk" #func) + DEVSEL_GET_CB(DestroyInstance); + DEVSEL_GET_CB(EnumeratePhysicalDevices); + DEVSEL_GET_CB(EnumeratePhysicalDeviceGroups); + DEVSEL_GET_CB(GetPhysicalDeviceProperties); + DEVSEL_GET_CB(EnumerateDeviceExtensionProperties); + if (info->has_props2) + DEVSEL_GET_CB(GetPhysicalDeviceProperties2KHR); +#undef DEVSEL_GET_CB + + device_select_layer_add_instance(*pInstance, info); + + return VK_SUCCESS; +} + +static void device_select_DestroyInstance(VkInstance instance, const VkAllocationCallbacks* pAllocator) +{ + struct instance_info *info = device_select_layer_get_instance(instance); + + device_select_layer_remove_instance(instance); + info->DestroyInstance(instance, pAllocator); + free(info); +} + + +static void print_gpu(const struct instance_info *info, unsigned index, VkPhysicalDevice device) +{ + const char *type = ""; + VkPhysicalDevicePCIBusInfoPropertiesEXT ext_pci_properties = (VkPhysicalDevicePCIBusInfoPropertiesEXT) { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PCI_BUS_INFO_PROPERTIES_EXT + }; + VkPhysicalDeviceProperties2KHR properties = (VkPhysicalDeviceProperties2KHR){ + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2_KHR + }; + if (info->has_props2 && info->has_pci_bus) + properties.pNext = &ext_pci_properties; + if (info->GetPhysicalDeviceProperties2KHR) + info->GetPhysicalDeviceProperties2KHR(device, &properties); + else + info->GetPhysicalDeviceProperties(device, &properties.properties); + + switch(properties.properties.deviceType) { + case VK_PHYSICAL_DEVICE_TYPE_OTHER: + default: + type = "other"; + break; + case VK_PHYSICAL_DEVICE_TYPE_INTEGRATED_GPU: + type = "integrated GPU"; + break; + case VK_PHYSICAL_DEVICE_TYPE_DISCRETE_GPU: + type = "discrete GPU"; + break; + case VK_PHYSICAL_DEVICE_TYPE_VIRTUAL_GPU: + type = "virtual GPU"; + break; + case VK_PHYSICAL_DEVICE_TYPE_CPU: + type = "CPU"; + break; + } + fprintf(stderr, " GPU %d: %x:%x \"%s\" %s", index, properties.properties.vendorID, + properties.properties.deviceID, properties.properties.deviceName, type); + if (info->has_pci_bus) + fprintf(stderr, " %04x:%02x:%02x.%x", ext_pci_properties.pciDomain, + ext_pci_properties.pciBus, ext_pci_properties.pciDevice, + ext_pci_properties.pciFunction); + fprintf(stderr, "\n"); +} + +static void fill_drm_device_info(const struct instance_info *info, + struct device_pci_info *drm_device, + VkPhysicalDevice device) +{ + VkPhysicalDevicePCIBusInfoPropertiesEXT ext_pci_properties = (VkPhysicalDevicePCIBusInfoPropertiesEXT) { + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PCI_BUS_INFO_PROPERTIES_EXT + }; + + VkPhysicalDeviceProperties2KHR properties = (VkPhysicalDeviceProperties2KHR){ + .sType = VK_STRUCTURE_TYPE_PHYSICAL_DEVICE_PROPERTIES_2_KHR + }; + + if (info->has_props2 && info->has_pci_bus) + properties.pNext = &ext_pci_properties; + if (info->GetPhysicalDeviceProperties2KHR) + info->GetPhysicalDeviceProperties2KHR(device, &properties); + else + info->GetPhysicalDeviceProperties(device, &properties.properties); + + drm_device->dev_info.vendor_id = properties.properties.vendorID; + drm_device->dev_info.device_id = properties.properties.deviceID; + if (info->has_pci_bus) { + drm_device->has_bus_info = true; + drm_device->bus_info.domain = ext_pci_properties.pciDomain; + drm_device->bus_info.bus = ext_pci_properties.pciBus; + drm_device->bus_info.dev = ext_pci_properties.pciDevice; + drm_device->bus_info.func = ext_pci_properties.pciFunction; + } +} + +static int device_select_find_explicit_default(struct device_pci_info *pci_infos, + uint32_t device_count, + const char *selection) +{ + int default_idx = -1; + unsigned vendor_id, device_id; + int matched = sscanf(selection, "%x:%x", &vendor_id, &device_id); + if (matched != 2) + return default_idx; + + for (unsigned i = 0; i < device_count; ++i) { + if (pci_infos[i].dev_info.vendor_id == vendor_id && + pci_infos[i].dev_info.device_id == device_id) + default_idx = i; + } + return default_idx; +} + +static int device_select_find_dri_prime_tag_default(struct device_pci_info *pci_infos, + uint32_t device_count, + const char *dri_prime) +{ + int default_idx = -1; + for (unsigned i = 0; i < device_count; ++i) { + char *tag = NULL; + if (asprintf(&tag, "pci-%04x_%02x_%02x_%1u", + pci_infos[i].bus_info.domain, + pci_infos[i].bus_info.bus, + pci_infos[i].bus_info.dev, + pci_infos[i].bus_info.func) >= 0) { + if (strcmp(dri_prime, tag)) + default_idx = i; + } + free(tag); + } + return default_idx; +} + +static int device_select_find_boot_vga_default(struct device_pci_info *pci_infos, + uint32_t device_count) +{ + char boot_vga_path[1024]; + int default_idx = -1; + for (unsigned i = 0; i < device_count; ++i) { + /* fallback to probing the pci bus boot_vga device. */ + snprintf(boot_vga_path, 1023, "/sys/bus/pci/devices/%04x:%02x:%02x.%x/boot_vga", pci_infos[i].bus_info.domain, + pci_infos[i].bus_info.bus, pci_infos[i].bus_info.dev, pci_infos[i].bus_info.func); + int fd = open(boot_vga_path, O_RDONLY); + if (fd != -1) { + uint8_t val; + if (read(fd, &val, 1) == 1) { + if (val == '1') + default_idx = i; + } + close(fd); + } + if (default_idx != -1) + break; + } + return default_idx; +} + +static uint32_t get_default_device(const struct instance_info *info, + const char *selection, + uint32_t physical_device_count, + VkPhysicalDevice *pPhysicalDevices) +{ + int default_idx = -1; + const char *dri_prime = getenv("DRI_PRIME"); + bool dri_prime_is_one = false; + + if (dri_prime && !strcmp(dri_prime, "1")) + dri_prime_is_one = true; + + if (dri_prime && !dri_prime_is_one && !info->has_pci_bus) { + fprintf(stderr, "device-select: cannot correctly use DRI_PRIME tag\n"); + } + + struct device_pci_info *pci_infos = (struct device_pci_info *)calloc(physical_device_count, sizeof(struct device_pci_info)); + if (!pci_infos) + return 0; + + for (unsigned i = 0; i < physical_device_count; ++i) { + fill_drm_device_info(info, &pci_infos[i], pPhysicalDevices[i]); + } + + if (selection) + default_idx = device_select_find_explicit_default(pci_infos, physical_device_count, selection); + if (default_idx == -1 && info->has_pci_bus && dri_prime && !dri_prime_is_one) + default_idx = device_select_find_dri_prime_tag_default(pci_infos, physical_device_count, dri_prime); + if (default_idx == -1 && info->has_wayland) + default_idx = device_select_find_wayland_pci_default(pci_infos, physical_device_count); + if (default_idx == -1 && info->has_xcb) + default_idx = device_select_find_xcb_pci_default(pci_infos, physical_device_count); + if (info->has_pci_bus && default_idx == -1) { + default_idx = device_select_find_boot_vga_default(pci_infos, physical_device_count); + } + + /* DRI_PRIME=1 handling - pick any other device than default. */ + if (default_idx != -1 && dri_prime_is_one && physical_device_count > 1) { + if (default_idx == 0) + default_idx = 1; + else if (default_idx == 1) + default_idx = 0; + } + free(pci_infos); + return default_idx == -1 ? 0 : default_idx; +} + +static VkResult device_select_EnumeratePhysicalDevices(VkInstance instance, + uint32_t* pPhysicalDeviceCount, + VkPhysicalDevice *pPhysicalDevices) +{ + struct instance_info *info = device_select_layer_get_instance(instance); + uint32_t physical_device_count = 0; + uint32_t selected_physical_device_count = 0; + const char* selection = getenv("MESA_VK_DEVICE_SELECT"); + VkResult result = info->EnumeratePhysicalDevices(instance, &physical_device_count, NULL); + VK_OUTARRAY_MAKE(out, pPhysicalDevices, pPhysicalDeviceCount); + if (result != VK_SUCCESS) + return result; + + VkPhysicalDevice *physical_devices = (VkPhysicalDevice*)calloc(sizeof(VkPhysicalDevice), physical_device_count); + VkPhysicalDevice *selected_physical_devices = (VkPhysicalDevice*)calloc(sizeof(VkPhysicalDevice), + physical_device_count); + + if (!physical_devices || !selected_physical_devices) { + result = VK_ERROR_OUT_OF_HOST_MEMORY; + goto out; + } + + result = info->EnumeratePhysicalDevices(instance, &physical_device_count, physical_devices); + if (result != VK_SUCCESS) + goto out; + + for (unsigned i = 0; i < physical_device_count; i++) { + uint32_t count; + info->EnumerateDeviceExtensionProperties(physical_devices[i], NULL, &count, NULL); + if (count > 0) { + VkExtensionProperties *extensions = calloc(count, sizeof(VkExtensionProperties)); + if (info->EnumerateDeviceExtensionProperties(physical_devices[i], NULL, &count, extensions) == VK_SUCCESS) { + for (unsigned j = 0; j < count; j++) { + if (!strcmp(extensions[j].extensionName, VK_EXT_PCI_BUS_INFO_EXTENSION_NAME)) + info->has_pci_bus = true; + } + } + free(extensions); + } + } + if (selection && strcmp(selection, "list") == 0) { + fprintf(stderr, "selectable devices:\n"); + for (unsigned i = 0; i < physical_device_count; ++i) + print_gpu(info, i, physical_devices[i]); + exit(0); + } else { + unsigned selected_index = get_default_device(info, selection, physical_device_count, physical_devices); + selected_physical_device_count = physical_device_count; + selected_physical_devices[0] = physical_devices[selected_index]; + for (unsigned i = 0; i < physical_device_count - 1; ++i) { + unsigned this_idx = i < selected_index ? i : i + 1; + selected_physical_devices[i + 1] = physical_devices[this_idx]; + } + } + + if (selected_physical_device_count == 0) { + fprintf(stderr, "WARNING: selected no devices with MESA_VK_DEVICE_SELECT\n"); + } + + assert(result == VK_SUCCESS); + + for (unsigned i = 0; i < selected_physical_device_count; i++) { + vk_outarray_append(&out, ent) { + *ent = selected_physical_devices[i]; + } + } + result = vk_outarray_status(&out); + out: + free(physical_devices); + free(selected_physical_devices); + return result; +} + +static VkResult device_select_EnumeratePhysicalDeviceGroups(VkInstance instance, + uint32_t* pPhysicalDeviceGroupCount, + VkPhysicalDeviceGroupProperties *pPhysicalDeviceGroups) +{ + struct instance_info *info = device_select_layer_get_instance(instance); + VkResult result = info->EnumeratePhysicalDeviceGroups(instance, pPhysicalDeviceGroupCount, pPhysicalDeviceGroups); + return result; +} + +static void (*get_pdevice_proc_addr(VkInstance instance, const char* name))() +{ + struct instance_info *info = device_select_layer_get_instance(instance); + return info->GetPhysicalDeviceProcAddr(instance, name); +} + +static void (*get_instance_proc_addr(VkInstance instance, const char* name))() +{ + if (strcmp(name, "vkCreateInstance") == 0) + return (void(*)())device_select_CreateInstance; + if (strcmp(name, "vkDestroyInstance") == 0) + return (void(*)())device_select_DestroyInstance; + if (strcmp(name, "vkEnumeratePhysicalDevices") == 0) + return (void(*)())device_select_EnumeratePhysicalDevices; + if (strcmp(name, "vkEnumeratePhysicalDeviceGroups") == 0) + return (void(*)())device_select_EnumeratePhysicalDeviceGroups; + + struct instance_info *info = device_select_layer_get_instance(instance); + return info->GetInstanceProcAddr(instance, name); +} + +VK_LAYER_EXPORT VkResult vkNegotiateLoaderLayerInterfaceVersion(VkNegotiateLayerInterface *pVersionStruct) +{ + if (pVersionStruct->loaderLayerInterfaceVersion < 2) + return VK_ERROR_INITIALIZATION_FAILED; + pVersionStruct->loaderLayerInterfaceVersion = 2; + + pVersionStruct->pfnGetInstanceProcAddr = get_instance_proc_addr; + pVersionStruct->pfnGetPhysicalDeviceProcAddr = get_pdevice_proc_addr; + + return VK_SUCCESS; +} diff --git a/src/vulkan/device-select-layer/device_select_wayland.c b/src/vulkan/device-select-layer/device_select_wayland.c new file mode 100644 index 0000000000000..73db275a55a5f --- /dev/null +++ b/src/vulkan/device-select-layer/device_select_wayland.c @@ -0,0 +1,147 @@ +/* + * Copyright © 2019 Red Hat + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the next + * paragraph) shall be included in all copies or substantial portions of the + * Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS + * IN THE SOFTWARE. + */ +#include "macros.h" +#include +#include "wayland-drm-client-protocol.h" +#include "device_select.h" +#include +#include +#include +#include +#include +struct device_select_wayland_info { + struct wl_drm *wl_drm; + drmDevicePtr dev_info; + bool info_is_set; +}; + +static void +device_select_drm_handle_device(void *data, struct wl_drm *drm, const char *device) +{ + struct device_select_wayland_info *info = data; + + int fd = open(device, O_RDWR | O_CLOEXEC); + if (fd == -1) + return; + + int ret = drmGetDevice2(fd, 0, &info->dev_info); + if (ret >= 0) + info->info_is_set = true; + close(fd); + return; +} + +static void +device_select_drm_handle_format(void *data, struct wl_drm *drm, uint32_t format) +{ + +} + +static void +device_select_drm_handle_authenticated(void *data, struct wl_drm *drm) +{ + +} + + +static void +device_select_drm_handle_capabilities(void *data, struct wl_drm *drm, uint32_t value) +{ + +} + + +static const struct wl_drm_listener ds_drm_listener = { + .device = device_select_drm_handle_device, + .format = device_select_drm_handle_format, + .authenticated = device_select_drm_handle_authenticated, + .capabilities = device_select_drm_handle_capabilities +}; + +static void +device_select_registry_global(void *data, struct wl_registry *registry, uint32_t name, + const char *interface, uint32_t version) +{ + struct device_select_wayland_info *info = data; + if (strcmp(interface, "wl_drm") == 0) { + info->wl_drm = wl_registry_bind(registry, name, &wl_drm_interface, MIN2(version, 2)); + wl_drm_add_listener(info->wl_drm, &ds_drm_listener, data); + } +} + +static void +device_select_registry_global_remove_cb(void *data, struct wl_registry *registry, + uint32_t name) +{ + +} + +int device_select_find_wayland_pci_default(struct device_pci_info *devices, uint32_t device_count) +{ + struct wl_display *display; + struct wl_registry *registry = NULL; + unsigned default_idx = -1; + struct device_select_wayland_info info = {}; + + display = wl_display_connect(NULL); + if (!display) + goto out; + + registry = wl_display_get_registry(display); + if (!registry) { + wl_display_disconnect(display); + goto out; + } + + static const struct wl_registry_listener registry_listener = + { device_select_registry_global, device_select_registry_global_remove_cb }; + + wl_registry_add_listener(registry, ®istry_listener, &info); + wl_display_dispatch(display); + wl_display_roundtrip(display); + + + if (info.info_is_set) { + for (unsigned i = 0; i < device_count; i++) { + if (devices[i].has_bus_info) { + if (info.dev_info->businfo.pci->domain == devices[i].bus_info.domain && + info.dev_info->businfo.pci->bus == devices[i].bus_info.bus && + info.dev_info->businfo.pci->dev == devices[i].bus_info.dev && + info.dev_info->businfo.pci->func == devices[i].bus_info.func) + default_idx = i; + } else { + if (info.dev_info->deviceinfo.pci->vendor_id == devices[i].dev_info.vendor_id && + info.dev_info->deviceinfo.pci->device_id == devices[i].dev_info.device_id) + default_idx = i; + } + if (default_idx != -1) + break; + } + } + + wl_drm_destroy(info.wl_drm); + wl_registry_destroy(registry); + wl_display_disconnect(display); + out: + return default_idx; +} diff --git a/src/vulkan/device-select-layer/device_select_x11.c b/src/vulkan/device-select-layer/device_select_x11.c new file mode 100644 index 0000000000000..93b39f269a43c --- /dev/null +++ b/src/vulkan/device-select-layer/device_select_x11.c @@ -0,0 +1,118 @@ +/* + * Copyright © 2019 Red Hat + * + * Permission is hereby granted, free of charge, to any person obtaining a + * copy of this software and associated documentation files (the "Software"), + * to deal in the Software without restriction, including without limitation + * the rights to use, copy, modify, merge, publish, distribute, sublicense, + * and/or sell copies of the Software, and to permit persons to whom the + * Software is furnished to do so, subject to the following conditions: + * + * The above copyright notice and this permission notice (including the next + * paragraph) shall be included in all copies or substantial portions of the + * Software. + * + * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR + * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, + * FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL + * THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER + * LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING + * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS + * IN THE SOFTWARE. + */ + +/* connect to an X server and work out the default device. */ + +#include +#include +#include +#include +#include +#include + +#include "device_select.h" +static int +ds_dri3_open(xcb_connection_t *conn, + xcb_window_t root, + uint32_t provider) +{ + xcb_dri3_open_cookie_t cookie; + xcb_dri3_open_reply_t *reply; + int fd; + + cookie = xcb_dri3_open(conn, + root, + provider); + + reply = xcb_dri3_open_reply(conn, cookie, NULL); + if (!reply) + return -1; + + if (reply->nfd != 1) { + free(reply); + return -1; + } + + fd = xcb_dri3_open_reply_fds(conn, reply)[0]; + free(reply); + fcntl(fd, F_SETFD, fcntl(fd, F_GETFD) | FD_CLOEXEC); + + return fd; +} + +int device_select_find_xcb_pci_default(struct device_pci_info *devices, uint32_t device_count) +{ + const xcb_setup_t *setup; + xcb_screen_iterator_t iter; + int scrn; + xcb_connection_t *conn; + int default_idx = -1; + conn = xcb_connect(NULL, &scrn); + if (!conn) + return -1; + + xcb_query_extension_cookie_t dri3_cookie; + xcb_query_extension_reply_t *dri3_reply; + + dri3_cookie = xcb_query_extension(conn, 4, "DRI3"); + dri3_reply = xcb_query_extension_reply(conn, dri3_cookie, NULL); + + if (!dri3_reply) + goto out; + + if (dri3_reply->present == 0) + goto out; + setup = xcb_get_setup(conn); + iter = xcb_setup_roots_iterator(setup); + + xcb_screen_t *screen = iter.data; + + int dri3_fd = ds_dri3_open(conn, screen->root, 0); + if (dri3_fd == -1) + goto out; + + drmDevicePtr xdev; + int ret = drmGetDevice2(dri3_fd, 0, &xdev); + if (ret < 0) + goto out; + + for (unsigned i = 0; i < device_count; i++) { + if (devices[i].has_bus_info) { + if (xdev->businfo.pci->domain == devices[i].bus_info.domain && + xdev->businfo.pci->bus == devices[i].bus_info.bus && + xdev->businfo.pci->dev == devices[i].bus_info.dev && + xdev->businfo.pci->func == devices[i].bus_info.func) { + default_idx = i; + } + } else { + if (xdev->deviceinfo.pci->vendor_id == devices[i].dev_info.vendor_id && + xdev->deviceinfo.pci->device_id == devices[i].dev_info.device_id) + default_idx = i; + } + if (default_idx != -1) + break; + } +out: + xcb_disconnect(conn); + return default_idx; +} diff --git a/src/vulkan/device-select-layer/meson.build b/src/vulkan/device-select-layer/meson.build new file mode 100644 index 0000000000000..f6dc470156a78 --- /dev/null +++ b/src/vulkan/device-select-layer/meson.build @@ -0,0 +1,54 @@ +# Copyright © 2019 Intel Corporation + +# Permission is hereby granted, free of charge, to any person obtaining a copy +# of this software and associated documentation files (the "Software"), to deal +# in the Software without restriction, including without limitation the rights +# to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +# copies of the Software, and to permit persons to whom the Software is +# furnished to do so, subject to the following conditions: + +# The above copyright notice and this permission notice shall be included in +# all copies or substantial portions of the Software. + +# THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +# IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +# FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +# AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +# LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +# OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +# SOFTWARE. + +vklayer_files = files( + 'device_select_layer.c', +) + +vklayer_deps = [] +vklayer_flags = [] + +if with_platform_x11 + vklayer_files += files('device_select_x11.c') + vklayer_deps += dep_xcb_dri3 + vklayer_flags += [ '-DVK_USE_PLATFORM_XCB_KHR' ] +endif + +if with_platform_wayland + vklayer_files += files('device_select_wayland.c') + vklayer_files += [ wayland_drm_client_protocol_h, wayland_drm_protocol_c ] + vklayer_deps += [dep_wayland_client, dep_wl_protocols] + vklayer_flags += '-DVK_USE_PLATFORM_WAYLAND_KHR' +endif + +vklayer_mesa_device_select = shared_library( + 'VkLayer_MESA_device_select', + vklayer_files, + c_args : [c_vis_args, no_override_init_args, vklayer_flags ], + dependencies : [idep_vulkan_util, idep_mesautil, vklayer_deps, dep_libdrm, dep_dl], + include_directories : [inc_include, inc_util, inc_vulkan_wsi], + link_args : cc.get_supported_link_arguments(['-Wl,-Bsymbolic-functions', '-Wl,-z,relro']), + install : true +) + +install_data( + files('VkLayer_MESA_device_select.json'), + install_dir : join_paths(get_option('datadir'), 'vulkan', 'implicit_layer.d'), +) diff --git a/src/vulkan/meson.build b/src/vulkan/meson.build index e1e8f753b504c..eedbad91cb02f 100644 --- a/src/vulkan/meson.build +++ b/src/vulkan/meson.build @@ -62,3 +62,6 @@ subdir('wsi') if with_vulkan_overlay_layer subdir('overlay-layer') endif +if get_option('vulkan-device-select-layer') + subdir('device-select-layer') +endif