backend: add preliminary support for driver detection
Signed-off-by: Yuxuan Shui <yshuiv7@gmail.com>
This commit is contained in:
parent
a2d0d5c826
commit
685a583f84
@ -5,6 +5,7 @@
|
|||||||
|
|
||||||
#include <stdbool.h>
|
#include <stdbool.h>
|
||||||
|
|
||||||
|
#include "driver.h"
|
||||||
#include "compiler.h"
|
#include "compiler.h"
|
||||||
#include "kernel.h"
|
#include "kernel.h"
|
||||||
#include "region.h"
|
#include "region.h"
|
||||||
@ -176,6 +177,9 @@ struct backend_operations {
|
|||||||
|
|
||||||
// =========== Hooks ============
|
// =========== Hooks ============
|
||||||
/// Let the backend hook into the event handling queue
|
/// Let the backend hook into the event handling queue
|
||||||
|
// =========== Misc ============
|
||||||
|
/// Return the driver that is been used by the backend
|
||||||
|
enum driver (*detect_driver)(backend_t *backend_data);
|
||||||
};
|
};
|
||||||
|
|
||||||
typedef backend_t *(*backend_init_fn)(session_t *ps)attr_nonnull(1);
|
typedef backend_t *(*backend_init_fn)(session_t *ps)attr_nonnull(1);
|
||||||
|
62
src/backend/driver.c
Normal file
62
src/backend/driver.c
Normal file
@ -0,0 +1,62 @@
|
|||||||
|
// SPDX-License-Identifier: MPL-2.0
|
||||||
|
// Copyright (c) Yuxuan Shui <yshuiv7@gmail.com>
|
||||||
|
|
||||||
|
#include <xcb/randr.h>
|
||||||
|
#include <xcb/xcb.h>
|
||||||
|
|
||||||
|
#include "backend/backend.h"
|
||||||
|
#include "backend/driver.h"
|
||||||
|
#include "common.h"
|
||||||
|
|
||||||
|
enum driver detect_driver(xcb_connection_t *c, backend_t *backend_data, xcb_window_t window) {
|
||||||
|
enum driver ret = 0;
|
||||||
|
// First we try doing backend agnostic detection using RANDR
|
||||||
|
// There's no way to query the X server about what driver is loaded, so RANDR is
|
||||||
|
// our best shot.
|
||||||
|
auto randr_version = xcb_randr_query_version_reply(
|
||||||
|
c, xcb_randr_query_version(c, XCB_RANDR_MAJOR_VERSION, XCB_RANDR_MINOR_VERSION),
|
||||||
|
NULL);
|
||||||
|
if (randr_version &&
|
||||||
|
(randr_version->major_version > 1 || randr_version->minor_version >= 4)) {
|
||||||
|
auto r = xcb_randr_get_providers_reply(
|
||||||
|
c, xcb_randr_get_providers(c, window), NULL);
|
||||||
|
if (r == NULL) {
|
||||||
|
log_warn("Failed to get RANDR providers");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto providers = xcb_randr_get_providers_providers(r);
|
||||||
|
for (auto i = 0; i < xcb_randr_get_providers_providers_length(r); i++) {
|
||||||
|
auto r2 = xcb_randr_get_provider_info_reply(
|
||||||
|
c, xcb_randr_get_provider_info(c, providers[i], r->timestamp), NULL);
|
||||||
|
if (r2 == NULL) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
if (r2->num_outputs == 0) {
|
||||||
|
free(r2);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
auto name = xcb_randr_get_provider_info_name(r2);
|
||||||
|
if (strcasestr(name, "modesetting") != NULL) {
|
||||||
|
ret |= DRIVER_MODESETTING;
|
||||||
|
} else if (strcasestr(name, "Radeon") != NULL) {
|
||||||
|
// Be conservative, add both radeon drivers
|
||||||
|
ret |= DRIVER_AMDGPU | DRIVER_RADEON;
|
||||||
|
} else if (strcasestr(name, "NVIDIA") != NULL) {
|
||||||
|
ret |= DRIVER_NVIDIA;
|
||||||
|
} else if (strcasestr(name, "nouveau") != NULL) {
|
||||||
|
ret |= DRIVER_NOUVEAU;
|
||||||
|
} else if (strcasestr(name, "Intel") != NULL) {
|
||||||
|
ret |= DRIVER_INTEL;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
free(r);
|
||||||
|
}
|
||||||
|
|
||||||
|
// If the backend supports driver detection, use that as well
|
||||||
|
if (backend_data && backend_data->ops->detect_driver) {
|
||||||
|
ret |= backend_data->ops->detect_driver(backend_data);
|
||||||
|
}
|
||||||
|
return ret;
|
||||||
|
}
|
55
src/backend/driver.h
Normal file
55
src/backend/driver.h
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
// SPDX-License-Identifier: MPL-2.0
|
||||||
|
// Copyright (c) Yuxuan Shui <yshuiv7@gmail.com>
|
||||||
|
|
||||||
|
#pragma once
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <xcb/xcb.h>
|
||||||
|
|
||||||
|
struct session;
|
||||||
|
struct backend_base;
|
||||||
|
|
||||||
|
/// A list of possible drivers.
|
||||||
|
/// The driver situation is a bit complicated. There are two drivers we care about: the
|
||||||
|
/// DDX, and the OpenGL driver. They are usually paired, but not always, since there is
|
||||||
|
/// also the generic modesetting driver.
|
||||||
|
/// This enum represents _both_ drivers.
|
||||||
|
enum driver {
|
||||||
|
DRIVER_AMDGPU = 1, // AMDGPU for DDX, radeonsi for OpenGL
|
||||||
|
DRIVER_RADEON = 2, // ATI for DDX, mesa r600 for OpenGL
|
||||||
|
DRIVER_FGLRX = 4,
|
||||||
|
DRIVER_NVIDIA = 8,
|
||||||
|
DRIVER_NOUVEAU = 16,
|
||||||
|
DRIVER_INTEL = 32,
|
||||||
|
DRIVER_MODESETTING = 64,
|
||||||
|
};
|
||||||
|
|
||||||
|
/// Return a list of drivers currently in use by the X server.
|
||||||
|
/// Note, this is a best-effort test, so no guarantee all drivers will be detected.
|
||||||
|
enum driver detect_driver(xcb_connection_t *, struct backend_base *, xcb_window_t);
|
||||||
|
|
||||||
|
// Print driver names to stdout, for diagnostics
|
||||||
|
static inline void print_drivers(enum driver drivers) {
|
||||||
|
if (drivers & DRIVER_AMDGPU) {
|
||||||
|
printf("AMDGPU, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_RADEON) {
|
||||||
|
printf("Radeon, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_FGLRX) {
|
||||||
|
printf("fglrx, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_NVIDIA) {
|
||||||
|
printf("NVIDIA, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_NOUVEAU) {
|
||||||
|
printf("nouveau, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_INTEL) {
|
||||||
|
printf("Intel, ");
|
||||||
|
}
|
||||||
|
if (drivers & DRIVER_MODESETTING) {
|
||||||
|
printf("modesetting, ");
|
||||||
|
}
|
||||||
|
printf("\b\b \n");
|
||||||
|
}
|
@ -1,5 +1,5 @@
|
|||||||
# enable xrender
|
# enable xrender
|
||||||
srcs += [ files('backend_common.c', 'xrender/xrender.c', 'backend.c') ]
|
srcs += [ files('backend_common.c', 'xrender/xrender.c', 'backend.c', 'driver.c') ]
|
||||||
|
|
||||||
# enable opengl
|
# enable opengl
|
||||||
if get_option('opengl')
|
if get_option('opengl')
|
||||||
|
@ -78,6 +78,7 @@
|
|||||||
|
|
||||||
// FIXME This list of includes should get shorter
|
// FIXME This list of includes should get shorter
|
||||||
#include "backend/backend.h"
|
#include "backend/backend.h"
|
||||||
|
#include "backend/driver.h"
|
||||||
#include "compiler.h"
|
#include "compiler.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "kernel.h"
|
#include "kernel.h"
|
||||||
@ -294,6 +295,7 @@ typedef struct session {
|
|||||||
ev_signal int_signal;
|
ev_signal int_signal;
|
||||||
/// backend data
|
/// backend data
|
||||||
backend_t *backend_data;
|
backend_t *backend_data;
|
||||||
|
enum driver drivers;
|
||||||
/// libev mainloop
|
/// libev mainloop
|
||||||
struct ev_loop *loop;
|
struct ev_loop *loop;
|
||||||
|
|
||||||
|
@ -1311,6 +1311,9 @@ static bool redir_start(session_t *ps) {
|
|||||||
|
|
||||||
ps->redirected = true;
|
ps->redirected = true;
|
||||||
|
|
||||||
|
// Re-detect driver since we now have a backend
|
||||||
|
ps->drivers = detect_driver(ps->c, ps->backend_data, ps->root);
|
||||||
|
|
||||||
root_damaged(ps);
|
root_damaged(ps);
|
||||||
|
|
||||||
// Repaint the whole screen
|
// Repaint the whole screen
|
||||||
@ -1985,6 +1988,8 @@ static session_t *session_init(int argc, char **argv, Display *dpy,
|
|||||||
// of OpenGL context.
|
// of OpenGL context.
|
||||||
init_overlay(ps);
|
init_overlay(ps);
|
||||||
|
|
||||||
|
ps->drivers = detect_driver(ps->c, ps->backend_data, ps->root);
|
||||||
|
|
||||||
// Initialize filters, must be preceded by OpenGL context creation
|
// Initialize filters, must be preceded by OpenGL context creation
|
||||||
if (!ps->o.experimental_backends && !init_render(ps)) {
|
if (!ps->o.experimental_backends && !init_render(ps)) {
|
||||||
log_fatal("Failed to initialize the backend");
|
log_fatal("Failed to initialize the backend");
|
||||||
|
@ -4,6 +4,7 @@
|
|||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
#include <xcb/xcb.h>
|
#include <xcb/xcb.h>
|
||||||
|
|
||||||
|
#include "backend/driver.h"
|
||||||
#include "diagnostic.h"
|
#include "diagnostic.h"
|
||||||
#include "config.h"
|
#include "config.h"
|
||||||
#include "common.h"
|
#include "common.h"
|
||||||
@ -21,6 +22,8 @@ void print_diagnostics(session_t *ps, const char *config_file) {
|
|||||||
printf("* Fast Math: Yes\n");
|
printf("* Fast Math: Yes\n");
|
||||||
#endif
|
#endif
|
||||||
printf("* Config file used: %s\n", config_file ?: "None");
|
printf("* Config file used: %s\n", config_file ?: "None");
|
||||||
|
printf("\n### Drivers (inaccurate):\n\n");
|
||||||
|
print_drivers(ps->drivers);
|
||||||
}
|
}
|
||||||
|
|
||||||
// vim: set noet sw=8 ts=8 :
|
// vim: set noet sw=8 ts=8 :
|
||||||
|
Loading…
Reference in New Issue
Block a user