2019-04-19 19:21:38 -04:00
|
|
|
// SPDX-License-Identifier: MPL-2.0
|
|
|
|
// Copyright (c) Yuxuan Shui <yshuiv7@gmail.com>
|
2019-08-09 19:56:04 -04:00
|
|
|
#include <stdlib.h>
|
|
|
|
#include <string.h>
|
2019-04-19 19:21:38 -04:00
|
|
|
|
|
|
|
#include <xcb/randr.h>
|
|
|
|
#include <xcb/xcb.h>
|
|
|
|
|
|
|
|
#include "backend/backend.h"
|
|
|
|
#include "backend/driver.h"
|
|
|
|
#include "common.h"
|
2019-08-09 19:56:04 -04:00
|
|
|
#include "compiler.h"
|
|
|
|
#include "log.h"
|
2019-04-19 19:21:38 -04:00
|
|
|
|
2020-04-22 13:05:04 -04:00
|
|
|
/// Apply driver specified global workarounds. It's safe to call this multiple times.
|
2020-04-22 15:13:18 -04:00
|
|
|
void apply_driver_workarounds(struct session *ps, enum driver driver) {
|
2020-04-22 13:05:04 -04:00
|
|
|
if (driver & DRIVER_NVIDIA) {
|
2020-04-22 15:20:22 -04:00
|
|
|
ps->o.xrender_sync_fence = true;
|
2020-04-22 13:05:04 -04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-12-19 05:29:59 -05:00
|
|
|
enum vblank_scheduler_type choose_vblank_scheduler(enum driver driver) {
|
|
|
|
if (driver & DRIVER_NVIDIA) {
|
|
|
|
return VBLANK_SCHEDULER_SGI_VIDEO_SYNC;
|
|
|
|
}
|
|
|
|
return VBLANK_SCHEDULER_PRESENT;
|
|
|
|
}
|
|
|
|
|
2019-04-19 19:21:38 -04:00
|
|
|
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");
|
2019-09-20 04:04:35 -04:00
|
|
|
free(randr_version);
|
2019-04-19 19:21:38 -04:00
|
|
|
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;
|
|
|
|
}
|
|
|
|
|
2019-04-19 20:09:10 -04:00
|
|
|
auto name_len = xcb_randr_get_provider_info_name_length(r2);
|
|
|
|
assert(name_len >= 0);
|
|
|
|
auto name =
|
|
|
|
strndup(xcb_randr_get_provider_info_name(r2), (size_t)name_len);
|
2019-04-19 19:21:38 -04:00
|
|
|
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;
|
|
|
|
}
|
2019-04-19 20:09:10 -04:00
|
|
|
free(name);
|
2019-09-20 04:07:31 -04:00
|
|
|
free(r2);
|
2019-04-19 19:21:38 -04:00
|
|
|
}
|
|
|
|
free(r);
|
|
|
|
}
|
2019-09-20 04:04:35 -04:00
|
|
|
free(randr_version);
|
2019-04-19 19:21:38 -04:00
|
|
|
|
|
|
|
// 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;
|
|
|
|
}
|