summaryrefslogtreecommitdiff
path: root/xf86drm.c
diff options
context:
space:
mode:
authorEmil Velikov <emil.velikov@collabora.com>2016-11-10 17:26:50 +0000
committerEmil Velikov <emil.l.velikov@gmail.com>2016-11-22 13:54:23 +0000
commit37d790f7d449874d0bf199d9ca9871d12b4d599a (patch)
tree1522aa7863dc4e8cc9ff02f690978cb88dfcd419 /xf86drm.c
parent7e0bc3bf1c247e1d53733d0e2e2ada52d29b5327 (diff)
downloadlibdrm-37d790f7d449874d0bf199d9ca9871d12b4d599a.tar.gz
libdrm-37d790f7d449874d0bf199d9ca9871d12b4d599a.tar.bz2
libdrm-37d790f7d449874d0bf199d9ca9871d12b4d599a.zip
xf86drm: introduce drmGetDeviceNameFromFd2
The original version considered only card devices, while this will pick the device/node name regardless - card, control, renderD, other... Current implementation is "linux" specific, in such that it relies on sysfs/uevent file. At the same time this gives us the flexibility to support any nodes even future ones, as long as they're within DRM_MAJOR. Shamelessly copied from mesa, latter by: Gary Wong <gtw@gnu.org> Signed-off-by: Emil Velikov <emil.velikov@collabora.com> Reviewed-by: Nicolai Hähnle <nicolai.haehnle@amd.com>
Diffstat (limited to 'xf86drm.c')
-rw-r--r--xf86drm.c51
1 files changed, 51 insertions, 0 deletions
diff --git a/xf86drm.c b/xf86drm.c
index 9b97bbb6..ed924a74 100644
--- a/xf86drm.c
+++ b/xf86drm.c
@@ -3303,3 +3303,54 @@ free_locals:
free(local_devices);
return ret;
}
+
+char *drmGetDeviceNameFromFd2(int fd)
+{
+#ifdef __linux__
+ struct stat sbuf;
+ char *device_name = NULL;
+ unsigned int maj, min;
+ FILE *f;
+ char buf[512];
+ static const char match[9] = "\nDEVNAME=";
+ int expected = 1;
+
+
+ if (fstat(fd, &sbuf))
+ return NULL;
+
+ maj = major(sbuf.st_rdev);
+ min = minor(sbuf.st_rdev);
+
+ if (maj != DRM_MAJOR || !S_ISCHR(sbuf.st_mode))
+ return NULL;
+
+ snprintf(buf, sizeof(buf), "/sys/dev/char/%d:%d/uevent", maj, min);
+ if (!(f = fopen(buf, "r")))
+ return NULL;
+
+ while (expected < sizeof(match)) {
+ int c = getc(f);
+
+ if (c == EOF) {
+ fclose(f);
+ return NULL;
+ } else if (c == match[expected] )
+ expected++;
+ else
+ expected = 0;
+ }
+
+ strcpy(buf, "/dev/");
+ if (fgets(buf + 5, sizeof(buf) - 5, f)) {
+ buf[strcspn(buf, "\n")] = '\0';
+ device_name = strdup(buf);
+ }
+
+ fclose(f);
+ return device_name;
+#else
+#warning "Missing implementation of drmGetDeviceNameFromFd2"
+ return NULL;
+#endif
+}