226 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			226 lines
		
	
	
		
			5.5 KiB
		
	
	
	
		
			C
		
	
	
	
| #ifdef HAVE_XORG_CONFIG_H
 | |
| #include <xorg-config.h>
 | |
| #endif
 | |
| 
 | |
| #ifdef XSERVER_PLATFORM_BUS
 | |
| 
 | |
| #include <xf86drm.h>
 | |
| #include <fcntl.h>
 | |
| #include <unistd.h>
 | |
| #include <errno.h>
 | |
| #include <string.h>
 | |
| 
 | |
| /* Linux platform device support */
 | |
| #include "xf86_OSproc.h"
 | |
| 
 | |
| #include "xf86.h"
 | |
| #include "xf86platformBus.h"
 | |
| #include "xf86Bus.h"
 | |
| 
 | |
| #include "hotplug.h"
 | |
| #include "systemd-logind.h"
 | |
| 
 | |
| static Bool
 | |
| get_drm_info(struct OdevAttributes *attribs, char *path, int delayed_index)
 | |
| {
 | |
|     drmVersionPtr v;
 | |
|     int fd;
 | |
|     int err = 0;
 | |
|     Bool paused, server_fd = FALSE;
 | |
| 
 | |
|     LogMessage(X_INFO, "Platform probe for %s\n", attribs->syspath);
 | |
| 
 | |
|     fd = systemd_logind_take_fd(attribs->major, attribs->minor, path, &paused);
 | |
|     if (fd != -1) {
 | |
|         if (paused) {
 | |
|             LogMessage(X_ERROR,
 | |
|                     "Error systemd-logind returned paused fd for drm node\n");
 | |
|             systemd_logind_release_fd(attribs->major, attribs->minor, -1);
 | |
|             return FALSE;
 | |
|         }
 | |
|         attribs->fd = fd;
 | |
|         server_fd = TRUE;
 | |
|     }
 | |
| 
 | |
|     if (fd == -1)
 | |
|         fd = open(path, O_RDWR | O_CLOEXEC, 0);
 | |
| 
 | |
|     if (fd == -1)
 | |
|         return FALSE;
 | |
| 
 | |
|     /* for a delayed probe we've already added the device */
 | |
|     if (delayed_index == -1) {
 | |
|             xf86_add_platform_device(attribs, FALSE);
 | |
|             delayed_index = xf86_num_platform_devices - 1;
 | |
|     }
 | |
| 
 | |
|     if (server_fd)
 | |
|         xf86_platform_devices[delayed_index].flags |= XF86_PDEV_SERVER_FD;
 | |
| 
 | |
|     v = drmGetVersion(fd);
 | |
|     if (!v) {
 | |
|         xf86Msg(X_ERROR, "%s: failed to query DRM version\n", path);
 | |
|         goto out;
 | |
|     }
 | |
| 
 | |
|     xf86_platform_odev_attributes(delayed_index)->driver = XNFstrdup(v->name);
 | |
|     drmFreeVersion(v);
 | |
| 
 | |
| out:
 | |
|     if (!server_fd)
 | |
|         close(fd);
 | |
|     return (err == 0);
 | |
| }
 | |
| 
 | |
| Bool
 | |
| xf86PlatformDeviceCheckBusID(struct xf86_platform_device *device, const char *busid)
 | |
| {
 | |
|     const char *syspath = device->attribs->syspath;
 | |
|     BusType bustype;
 | |
|     const char *id;
 | |
| 
 | |
|     if (!syspath)
 | |
|         return FALSE;
 | |
| 
 | |
|     bustype = StringToBusType(busid, &id);
 | |
|     if (bustype == BUS_PCI) {
 | |
|         struct pci_device *pPci = device->pdev;
 | |
|         if (!pPci)
 | |
|             return FALSE;
 | |
| 
 | |
|         if (xf86ComparePciBusString(busid,
 | |
|                                     ((pPci->domain << 8)
 | |
|                                      | pPci->bus),
 | |
|                                     pPci->dev, pPci->func)) {
 | |
|             return TRUE;
 | |
|         }
 | |
|     }
 | |
|     else if (bustype == BUS_PLATFORM) {
 | |
|         /* match on the minimum string */
 | |
|         int len = strlen(id);
 | |
| 
 | |
|         if (strlen(syspath) < strlen(id))
 | |
|             len = strlen(syspath);
 | |
| 
 | |
|         if (strncmp(id, syspath, len))
 | |
|             return FALSE;
 | |
|         return TRUE;
 | |
|     }
 | |
|     else if (bustype == BUS_USB) {
 | |
|         if (strcasecmp(busid, device->attribs->busid))
 | |
|             return FALSE;
 | |
|         return TRUE;
 | |
|     }
 | |
|     return FALSE;
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86PlatformReprobeDevice(int index, struct OdevAttributes *attribs)
 | |
| {
 | |
|     Bool ret;
 | |
|     char *dpath = attribs->path;
 | |
| 
 | |
|     ret = get_drm_info(attribs, dpath, index);
 | |
|     if (ret == FALSE) {
 | |
|         xf86_remove_platform_device(index);
 | |
|         return;
 | |
|     }
 | |
|     ret = xf86platformAddDevice(xf86PlatformFindHotplugDriver(index), index);
 | |
|     if (ret == -1)
 | |
|         xf86_remove_platform_device(index);
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86PlatformDeviceProbe(struct OdevAttributes *attribs)
 | |
| {
 | |
|     int i;
 | |
|     char *path = attribs->path;
 | |
|     Bool ret;
 | |
| 
 | |
|     if (!path)
 | |
|         goto out_free;
 | |
| 
 | |
|     for (i = 0; i < xf86_num_platform_devices; i++) {
 | |
|         char *dpath = xf86_platform_odev_attributes(i)->path;
 | |
| 
 | |
|         if (dpath && !strcmp(path, dpath))
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if (i != xf86_num_platform_devices)
 | |
|         goto out_free;
 | |
| 
 | |
|     LogMessage(X_INFO, "xfree86: Adding drm device (%s)\n", path);
 | |
| 
 | |
|     if (!xf86VTOwner()) {
 | |
|             /* if we don't currently own the VT then don't probe the device,
 | |
|                just mark it as unowned for later use */
 | |
|             xf86_add_platform_device(attribs, TRUE);
 | |
|             return;
 | |
|     }
 | |
| 
 | |
|     ret = get_drm_info(attribs, path, -1);
 | |
|     if (ret == FALSE)
 | |
|         goto out_free;
 | |
| 
 | |
|     return;
 | |
| 
 | |
| out_free:
 | |
|     config_odev_free_attributes(attribs);
 | |
| }
 | |
| 
 | |
| void NewGPUDeviceRequest(struct OdevAttributes *attribs)
 | |
| {
 | |
|     int old_num = xf86_num_platform_devices;
 | |
|     int ret;
 | |
|     const char *driver_name;
 | |
| 
 | |
|     xf86PlatformDeviceProbe(attribs);
 | |
| 
 | |
|     if (old_num == xf86_num_platform_devices)
 | |
|         return;
 | |
| 
 | |
|     if (xf86_get_platform_device_unowned(xf86_num_platform_devices - 1) == TRUE)
 | |
|         return;
 | |
| 
 | |
|     /* Scan and update PCI devices before adding new platform device */
 | |
|     xf86PlatformScanPciDev();
 | |
|     driver_name = xf86PlatformFindHotplugDriver(xf86_num_platform_devices - 1);
 | |
| 
 | |
|     ret = xf86platformAddDevice(driver_name, xf86_num_platform_devices-1);
 | |
|     if (ret == -1)
 | |
|         xf86_remove_platform_device(xf86_num_platform_devices-1);
 | |
| 
 | |
|     ErrorF("xf86: found device %d\n", xf86_num_platform_devices);
 | |
|     return;
 | |
| }
 | |
| 
 | |
| void DeleteGPUDeviceRequest(struct OdevAttributes *attribs)
 | |
| {
 | |
|     int index;
 | |
|     char *syspath = attribs->syspath;
 | |
| 
 | |
|     if (!syspath)
 | |
|         goto out;
 | |
| 
 | |
|     for (index = 0; index < xf86_num_platform_devices; index++) {
 | |
|         char *dspath = xf86_platform_odev_attributes(index)->syspath;
 | |
|         if (dspath && !strcmp(syspath, dspath))
 | |
|             break;
 | |
|     }
 | |
| 
 | |
|     if (index == xf86_num_platform_devices)
 | |
|         goto out;
 | |
| 
 | |
|     ErrorF("xf86: remove device %d %s\n", index, syspath);
 | |
| 
 | |
|     if (xf86_get_platform_device_unowned(index) == TRUE)
 | |
|             xf86_remove_platform_device(index);
 | |
|     else
 | |
|             xf86platformRemoveDevice(index);
 | |
| out:
 | |
|     config_odev_free_attributes(attribs);
 | |
| }
 | |
| 
 | |
| #endif
 |