mirror of
https://github.com/hyprwm/Hyprland
synced 2025-01-09 01:09:49 +01:00
renderer,config: add custom DRM modeline support (#2254)
This allows specifying custom display resolutions for the DRM backend. This is useful for display overclocking, working around broken EDIDs, etc. To use this feature, specify a modeline instead of a resolution in the config, for example: monitor = DP-1, modeline 1071.101 3840 3848 3880 3920 2160 2263 2271 2277 +hsync -vsync, 0x0, 1 This example is a custom 3840x2160@120Hz mode with tightened timings. I use it because the standard timings don't work with my monitor and GPU combination (M28U with RX580). The syntax is compatible with Sway and Xorg.
This commit is contained in:
parent
e7c2ea9724
commit
2f87e4c2f3
4 changed files with 103 additions and 18 deletions
|
@ -470,6 +470,58 @@ void CConfigManager::handleRawExec(const std::string& command, const std::string
|
||||||
g_pKeybindManager->spawn(args);
|
g_pKeybindManager->spawn(args);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
static bool parseModeLine(const std::string& modeline, drmModeModeInfo& mode) {
|
||||||
|
auto args = CVarList(modeline, 0, 's');
|
||||||
|
|
||||||
|
auto keyword = args[0];
|
||||||
|
std::transform(keyword.begin(), keyword.end(), keyword.begin(), ::tolower);
|
||||||
|
|
||||||
|
if (keyword != "modeline")
|
||||||
|
return false;
|
||||||
|
|
||||||
|
if (args.size() < 10) {
|
||||||
|
Debug::log(ERR, "modeline parse error: expected at least 9 arguments, got %i", args.size() - 1);
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
int argno = 1;
|
||||||
|
|
||||||
|
mode.type = DRM_MODE_TYPE_USERDEF;
|
||||||
|
mode.clock = std::stof(args[argno++]) * 1000;
|
||||||
|
mode.hdisplay = std::stoi(args[argno++]);
|
||||||
|
mode.hsync_start = std::stoi(args[argno++]);
|
||||||
|
mode.hsync_end = std::stoi(args[argno++]);
|
||||||
|
mode.htotal = std::stoi(args[argno++]);
|
||||||
|
mode.vdisplay = std::stoi(args[argno++]);
|
||||||
|
mode.vsync_start = std::stoi(args[argno++]);
|
||||||
|
mode.vsync_end = std::stoi(args[argno++]);
|
||||||
|
mode.vtotal = std::stoi(args[argno++]);
|
||||||
|
mode.vrefresh = mode.clock * 1000.0 * 1000.0 / mode.htotal / mode.vtotal;
|
||||||
|
|
||||||
|
static std::unordered_map<std::string, uint32_t> flagsmap = {
|
||||||
|
{"+hsync", DRM_MODE_FLAG_PHSYNC},
|
||||||
|
{"-hsync", DRM_MODE_FLAG_NHSYNC},
|
||||||
|
{"+vsync", DRM_MODE_FLAG_PVSYNC},
|
||||||
|
{"-vsync", DRM_MODE_FLAG_NVSYNC},
|
||||||
|
};
|
||||||
|
|
||||||
|
for (; argno < static_cast<int>(args.size()); argno++) {
|
||||||
|
auto key = args[argno];
|
||||||
|
std::transform(key.begin(), key.end(), key.begin(), ::tolower);
|
||||||
|
|
||||||
|
auto it = flagsmap.find(key);
|
||||||
|
|
||||||
|
if (it != flagsmap.end())
|
||||||
|
mode.flags |= it->second;
|
||||||
|
else
|
||||||
|
Debug::log(ERR, "invalid flag %s in modeline", it->first.c_str());
|
||||||
|
}
|
||||||
|
|
||||||
|
snprintf(mode.name, sizeof(mode.name), "%dx%d@%d", mode.hdisplay, mode.vdisplay, mode.vrefresh / 1000);
|
||||||
|
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
void CConfigManager::handleMonitor(const std::string& command, const std::string& args) {
|
void CConfigManager::handleMonitor(const std::string& command, const std::string& args) {
|
||||||
|
|
||||||
// get the monitor config
|
// get the monitor config
|
||||||
|
@ -531,6 +583,9 @@ void CConfigManager::handleMonitor(const std::string& command, const std::string
|
||||||
newrule.resolution = Vector2D(-1, -1);
|
newrule.resolution = Vector2D(-1, -1);
|
||||||
} else if (ARGS[1].find("highres") == 0) {
|
} else if (ARGS[1].find("highres") == 0) {
|
||||||
newrule.resolution = Vector2D(-1, -2);
|
newrule.resolution = Vector2D(-1, -2);
|
||||||
|
} else if (parseModeLine(ARGS[1], newrule.drmMode)) {
|
||||||
|
newrule.resolution = Vector2D(newrule.drmMode.hdisplay, newrule.drmMode.vdisplay);
|
||||||
|
newrule.refreshRate = newrule.drmMode.vrefresh / 1000;
|
||||||
} else {
|
} else {
|
||||||
newrule.resolution.x = stoi(ARGS[1].substr(0, ARGS[1].find_first_of('x')));
|
newrule.resolution.x = stoi(ARGS[1].substr(0, ARGS[1].find_first_of('x')));
|
||||||
newrule.resolution.y = stoi(ARGS[1].substr(ARGS[1].find_first_of('x') + 1, ARGS[1].find_first_of('@')));
|
newrule.resolution.y = stoi(ARGS[1].substr(ARGS[1].find_first_of('x') + 1, ARGS[1].find_first_of('@')));
|
||||||
|
|
|
@ -11,6 +11,7 @@
|
||||||
#include <algorithm>
|
#include <algorithm>
|
||||||
#include <regex>
|
#include <regex>
|
||||||
#include <optional>
|
#include <optional>
|
||||||
|
#include <xf86drmMode.h>
|
||||||
#include "../Window.hpp"
|
#include "../Window.hpp"
|
||||||
#include "../helpers/WLClasses.hpp"
|
#include "../helpers/WLClasses.hpp"
|
||||||
|
|
||||||
|
@ -44,6 +45,7 @@ struct SMonitorRule {
|
||||||
wl_output_transform transform = WL_OUTPUT_TRANSFORM_NORMAL;
|
wl_output_transform transform = WL_OUTPUT_TRANSFORM_NORMAL;
|
||||||
std::string mirrorOf = "";
|
std::string mirrorOf = "";
|
||||||
bool enable10bit = false;
|
bool enable10bit = false;
|
||||||
|
drmModeModeInfo drmMode = {};
|
||||||
};
|
};
|
||||||
|
|
||||||
struct SWorkspaceRule {
|
struct SWorkspaceRule {
|
||||||
|
|
|
@ -6,6 +6,7 @@
|
||||||
#include <vector>
|
#include <vector>
|
||||||
#include <array>
|
#include <array>
|
||||||
#include <memory>
|
#include <memory>
|
||||||
|
#include <xf86drmMode.h>
|
||||||
#include "Timer.hpp"
|
#include "Timer.hpp"
|
||||||
|
|
||||||
struct SMonitorRule;
|
struct SMonitorRule;
|
||||||
|
@ -31,6 +32,8 @@ class CMonitor {
|
||||||
Vector2D vecReservedTopLeft = Vector2D(0, 0);
|
Vector2D vecReservedTopLeft = Vector2D(0, 0);
|
||||||
Vector2D vecReservedBottomRight = Vector2D(0, 0);
|
Vector2D vecReservedBottomRight = Vector2D(0, 0);
|
||||||
|
|
||||||
|
drmModeModeInfo customDrmMode = {};
|
||||||
|
|
||||||
// WLR stuff
|
// WLR stuff
|
||||||
wlr_damage_ring damage;
|
wlr_damage_ring damage;
|
||||||
wlr_output* output = nullptr;
|
wlr_output* output = nullptr;
|
||||||
|
|
|
@ -1520,12 +1520,16 @@ bool CHyprRenderer::applyMonitorRule(CMonitor* pMonitor, SMonitorRule* pMonitorR
|
||||||
DELTALESSTHAN(pMonitor->refreshRate, pMonitorRule->refreshRate, 1) && pMonitor->scale == pMonitorRule->scale &&
|
DELTALESSTHAN(pMonitor->refreshRate, pMonitorRule->refreshRate, 1) && pMonitor->scale == pMonitorRule->scale &&
|
||||||
((DELTALESSTHAN(pMonitor->vecPosition.x, pMonitorRule->offset.x, 1) && DELTALESSTHAN(pMonitor->vecPosition.y, pMonitorRule->offset.y, 1)) ||
|
((DELTALESSTHAN(pMonitor->vecPosition.x, pMonitorRule->offset.x, 1) && DELTALESSTHAN(pMonitor->vecPosition.y, pMonitorRule->offset.y, 1)) ||
|
||||||
pMonitorRule->offset == Vector2D(-1, -1)) &&
|
pMonitorRule->offset == Vector2D(-1, -1)) &&
|
||||||
pMonitor->transform == pMonitorRule->transform && pMonitorRule->enable10bit == pMonitor->enabled10bit) {
|
pMonitor->transform == pMonitorRule->transform && pMonitorRule->enable10bit == pMonitor->enabled10bit &&
|
||||||
|
!memcmp(&pMonitor->customDrmMode, &pMonitorRule->drmMode, sizeof(pMonitor->customDrmMode))) {
|
||||||
|
|
||||||
Debug::log(LOG, "Not applying a new rule to %s because it's already applied!", pMonitor->szName.c_str());
|
Debug::log(LOG, "Not applying a new rule to %s because it's already applied!", pMonitor->szName.c_str());
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Needed in case we are switching from a custom modeline to a standard mode
|
||||||
|
pMonitor->customDrmMode = {};
|
||||||
|
|
||||||
if (pMonitorRule->scale > 0.1) {
|
if (pMonitorRule->scale > 0.1) {
|
||||||
wlr_output_set_scale(pMonitor->output, pMonitorRule->scale);
|
wlr_output_set_scale(pMonitor->output, pMonitorRule->scale);
|
||||||
pMonitor->scale = pMonitorRule->scale;
|
pMonitor->scale = pMonitorRule->scale;
|
||||||
|
@ -1540,7 +1544,7 @@ bool CHyprRenderer::applyMonitorRule(CMonitor* pMonitor, SMonitorRule* pMonitorR
|
||||||
|
|
||||||
// loop over modes and choose an appropriate one.
|
// loop over modes and choose an appropriate one.
|
||||||
if (pMonitorRule->resolution != Vector2D() && pMonitorRule->resolution != Vector2D(-1, -1) && pMonitorRule->resolution != Vector2D(-1, -2)) {
|
if (pMonitorRule->resolution != Vector2D() && pMonitorRule->resolution != Vector2D(-1, -1) && pMonitorRule->resolution != Vector2D(-1, -2)) {
|
||||||
if (!wl_list_empty(&pMonitor->output->modes)) {
|
if (!wl_list_empty(&pMonitor->output->modes) && pMonitorRule->drmMode.type != DRM_MODE_TYPE_USERDEF) {
|
||||||
wlr_output_mode* mode;
|
wlr_output_mode* mode;
|
||||||
bool found = false;
|
bool found = false;
|
||||||
|
|
||||||
|
@ -1598,18 +1602,38 @@ bool CHyprRenderer::applyMonitorRule(CMonitor* pMonitor, SMonitorRule* pMonitorR
|
||||||
(float)pMonitorRule->refreshRate);
|
(float)pMonitorRule->refreshRate);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
// custom resolution
|
||||||
|
bool fail = false;
|
||||||
|
|
||||||
|
if (pMonitorRule->drmMode.type == DRM_MODE_TYPE_USERDEF) {
|
||||||
|
if (!wlr_output_is_drm(pMonitor->output)) {
|
||||||
|
Debug::log(ERR, "Tried to set custom modeline on non-DRM output");
|
||||||
|
fail = true;
|
||||||
|
} else {
|
||||||
|
auto* mode = wlr_drm_connector_add_mode(pMonitor->output, &pMonitorRule->drmMode);
|
||||||
|
if (mode) {
|
||||||
|
wlr_output_set_mode(pMonitor->output, mode);
|
||||||
|
pMonitor->customDrmMode = pMonitorRule->drmMode;
|
||||||
|
} else {
|
||||||
|
Debug::log(ERR, "wlr_drm_connector_add_mode failed");
|
||||||
|
fail = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
} else {
|
} else {
|
||||||
wlr_output_set_custom_mode(pMonitor->output, (int)pMonitorRule->resolution.x, (int)pMonitorRule->resolution.y, (int)pMonitorRule->refreshRate * 1000);
|
wlr_output_set_custom_mode(pMonitor->output, (int)pMonitorRule->resolution.x, (int)pMonitorRule->resolution.y, (int)pMonitorRule->refreshRate * 1000);
|
||||||
|
}
|
||||||
|
|
||||||
pMonitor->vecSize = pMonitorRule->resolution;
|
pMonitor->vecSize = pMonitorRule->resolution;
|
||||||
pMonitor->refreshRate = pMonitorRule->refreshRate;
|
pMonitor->refreshRate = pMonitorRule->refreshRate;
|
||||||
|
|
||||||
if (!wlr_output_test(pMonitor->output)) {
|
if (fail || !wlr_output_test(pMonitor->output)) {
|
||||||
Debug::log(ERR, "Custom resolution FAILED, falling back to preferred");
|
Debug::log(ERR, "Custom resolution FAILED, falling back to preferred");
|
||||||
|
|
||||||
const auto PREFERREDMODE = wlr_output_preferred_mode(pMonitor->output);
|
const auto PREFERREDMODE = wlr_output_preferred_mode(pMonitor->output);
|
||||||
|
|
||||||
if (!PREFERREDMODE) {
|
if (!PREFERREDMODE) {
|
||||||
Debug::log(ERR, "Monitor %s has NO PREFERRED MODE, and an INVALID one was requested: %ix%i@%2f", (int)pMonitorRule->resolution.x,
|
Debug::log(ERR, "Monitor %s has NO PREFERRED MODE, and an INVALID one was requested: %ix%i@%2f", pMonitor->output->name, (int)pMonitorRule->resolution.x,
|
||||||
(int)pMonitorRule->resolution.y, (float)pMonitorRule->refreshRate);
|
(int)pMonitorRule->resolution.y, (float)pMonitorRule->refreshRate);
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
@ -1623,6 +1647,7 @@ bool CHyprRenderer::applyMonitorRule(CMonitor* pMonitor, SMonitorRule* pMonitorR
|
||||||
|
|
||||||
pMonitor->refreshRate = PREFERREDMODE->refresh / 1000.f;
|
pMonitor->refreshRate = PREFERREDMODE->refresh / 1000.f;
|
||||||
pMonitor->vecSize = Vector2D(PREFERREDMODE->width, PREFERREDMODE->height);
|
pMonitor->vecSize = Vector2D(PREFERREDMODE->width, PREFERREDMODE->height);
|
||||||
|
pMonitor->customDrmMode = {};
|
||||||
} else {
|
} else {
|
||||||
Debug::log(LOG, "Set a custom mode %ix%i@%2f (mode not found in monitor modes)", (int)pMonitorRule->resolution.x, (int)pMonitorRule->resolution.y,
|
Debug::log(LOG, "Set a custom mode %ix%i@%2f (mode not found in monitor modes)", (int)pMonitorRule->resolution.x, (int)pMonitorRule->resolution.y,
|
||||||
(float)pMonitorRule->refreshRate);
|
(float)pMonitorRule->refreshRate);
|
||||||
|
|
Loading…
Reference in a new issue