2015-05-24 06:55:12 +02:00
|
|
|
// Copyright 2010 Dolphin Emulator Project
|
2021-07-05 03:22:19 +02:00
|
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
2010-04-22 04:28:34 +00:00
|
|
|
|
2016-05-05 19:43:38 -04:00
|
|
|
#include "UICommon/X11Utils.h"
|
2017-02-24 22:56:33 -05:00
|
|
|
|
|
|
|
#include <algorithm>
|
|
|
|
#include <cstdio>
|
|
|
|
#include <cstdlib>
|
2019-03-02 22:41:50 -06:00
|
|
|
#include <cstring>
|
2010-11-25 02:26:46 +00:00
|
|
|
#include <spawn.h>
|
2017-02-24 22:56:33 -05:00
|
|
|
#include <string>
|
2014-02-19 02:56:29 +01:00
|
|
|
#include <sys/wait.h>
|
2014-02-17 05:18:15 -05:00
|
|
|
#include <unistd.h>
|
2014-02-19 02:56:29 +01:00
|
|
|
|
2019-11-23 19:15:52 -05:00
|
|
|
#include <fmt/format.h>
|
|
|
|
|
2014-06-05 19:29:54 -04:00
|
|
|
#include "Common/Logging/Log.h"
|
2017-02-24 22:56:33 -05:00
|
|
|
#include "Common/StringUtil.h"
|
2019-03-02 22:41:50 -06:00
|
|
|
#include "Core/Config/MainSettings.h"
|
2014-02-22 23:36:30 +01:00
|
|
|
#include "Core/Core.h"
|
2010-11-25 02:26:46 +00:00
|
|
|
|
|
|
|
extern char** environ;
|
|
|
|
|
2010-04-22 04:28:34 +00:00
|
|
|
namespace X11Utils
|
|
|
|
{
|
2015-01-04 17:09:56 +01:00
|
|
|
bool ToggleFullscreen(Display* dpy, Window win)
|
2010-04-22 04:28:34 +00:00
|
|
|
{
|
|
|
|
// Init X event structure for _NET_WM_STATE_FULLSCREEN client message
|
|
|
|
XEvent event;
|
|
|
|
event.xclient.type = ClientMessage;
|
|
|
|
event.xclient.message_type = XInternAtom(dpy, "_NET_WM_STATE", False);
|
|
|
|
event.xclient.window = win;
|
|
|
|
event.xclient.format = 32;
|
2014-08-06 22:16:28 -04:00
|
|
|
event.xclient.data.l[0] = _NET_WM_STATE_TOGGLE;
|
2010-04-22 04:28:34 +00:00
|
|
|
event.xclient.data.l[1] = XInternAtom(dpy, "_NET_WM_STATE_FULLSCREEN", False);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-04-22 04:28:34 +00:00
|
|
|
// Send the event
|
|
|
|
if (!XSendEvent(dpy, DefaultRootWindow(dpy), False,
|
2014-03-11 00:30:55 +13:00
|
|
|
SubstructureRedirectMask | SubstructureNotifyMask, &event))
|
2015-01-04 17:09:56 +01:00
|
|
|
{
|
2020-11-25 21:13:50 -05:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Failed to switch fullscreen/windowed mode.");
|
2015-01-04 17:09:56 +01:00
|
|
|
return false;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2015-01-04 17:09:56 +01:00
|
|
|
return true;
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
|
|
|
|
2021-02-22 14:32:53 +01:00
|
|
|
#ifdef HAVE_XRANDR
|
2010-04-22 04:28:34 +00:00
|
|
|
XRRConfiguration::XRRConfiguration(Display* _dpy, Window _win)
|
2014-03-09 21:14:26 +01:00
|
|
|
: dpy(_dpy), win(_win), screenResources(nullptr), outputInfo(nullptr), crtcInfo(nullptr),
|
2011-02-09 03:12:05 +00:00
|
|
|
fullMode(0), fs_fb_width(0), fs_fb_height(0), fs_fb_width_mm(0), fs_fb_height_mm(0),
|
|
|
|
bValid(true), bIsFullscreen(false)
|
2010-04-22 04:28:34 +00:00
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
int XRRMajorVersion, XRRMinorVersion;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
if (!XRRQueryVersion(dpy, &XRRMajorVersion, &XRRMinorVersion) ||
|
2014-08-30 17:01:19 -04:00
|
|
|
(XRRMajorVersion < 1 || (XRRMajorVersion == 1 && XRRMinorVersion < 3)))
|
2010-07-10 12:35:16 +00:00
|
|
|
{
|
2020-11-25 21:13:50 -05:00
|
|
|
WARN_LOG_FMT(VIDEO, "XRRExtension not supported.");
|
2010-07-10 12:35:16 +00:00
|
|
|
bValid = false;
|
|
|
|
return;
|
|
|
|
}
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
screenResources = XRRGetScreenResourcesCurrent(dpy, win);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
screen = DefaultScreen(dpy);
|
|
|
|
fb_width = DisplayWidth(dpy, screen);
|
|
|
|
fb_height = DisplayHeight(dpy, screen);
|
|
|
|
fb_width_mm = DisplayWidthMM(dpy, screen);
|
|
|
|
fb_height_mm = DisplayHeightMM(dpy, screen);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2020-11-25 21:13:50 -05:00
|
|
|
INFO_LOG_FMT(VIDEO, "XRRExtension-Version {}.{}", XRRMajorVersion, XRRMinorVersion);
|
2010-04-22 04:28:34 +00:00
|
|
|
Update();
|
|
|
|
}
|
|
|
|
|
|
|
|
XRRConfiguration::~XRRConfiguration()
|
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
if (bValid && bIsFullscreen)
|
2010-07-10 12:35:16 +00:00
|
|
|
ToggleDisplayMode(False);
|
2011-02-09 03:12:05 +00:00
|
|
|
if (screenResources)
|
|
|
|
XRRFreeScreenResources(screenResources);
|
|
|
|
if (outputInfo)
|
|
|
|
XRRFreeOutputInfo(outputInfo);
|
|
|
|
if (crtcInfo)
|
|
|
|
XRRFreeCrtcInfo(crtcInfo);
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
void XRRConfiguration::Update()
|
|
|
|
{
|
2019-03-02 22:41:50 -06:00
|
|
|
const std::string fullscreen_display_res = Config::Get(Config::MAIN_FULLSCREEN_DISPLAY_RES);
|
|
|
|
if (fullscreen_display_res == "Auto")
|
2013-05-25 11:43:56 +02:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-07-10 12:35:16 +00:00
|
|
|
if (!bValid)
|
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
if (outputInfo)
|
|
|
|
{
|
|
|
|
XRRFreeOutputInfo(outputInfo);
|
2014-03-09 21:14:26 +01:00
|
|
|
outputInfo = nullptr;
|
2011-02-09 03:12:05 +00:00
|
|
|
}
|
|
|
|
if (crtcInfo)
|
|
|
|
{
|
|
|
|
XRRFreeCrtcInfo(crtcInfo);
|
2014-03-09 21:14:26 +01:00
|
|
|
crtcInfo = nullptr;
|
2011-02-09 03:12:05 +00:00
|
|
|
}
|
|
|
|
fullMode = 0;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2010-04-22 04:28:34 +00:00
|
|
|
// Get the resolution setings for fullscreen mode
|
2011-02-09 03:12:05 +00:00
|
|
|
unsigned int fullWidth, fullHeight;
|
2014-03-09 21:14:26 +01:00
|
|
|
char* output_name = nullptr;
|
2015-01-07 20:20:14 +00:00
|
|
|
char auxFlag = '\0';
|
2019-03-02 22:41:50 -06:00
|
|
|
if (fullscreen_display_res.find(':') == std::string::npos)
|
2011-07-15 02:17:14 +00:00
|
|
|
{
|
|
|
|
fullWidth = fb_width;
|
|
|
|
fullHeight = fb_height;
|
|
|
|
}
|
2011-03-08 22:43:44 +00:00
|
|
|
else
|
2014-08-30 17:01:19 -04:00
|
|
|
{
|
2019-03-02 22:41:50 -06:00
|
|
|
sscanf(fullscreen_display_res.c_str(), "%m[^:]: %ux%u%c", &output_name, &fullWidth, &fullHeight,
|
|
|
|
&auxFlag);
|
2014-08-30 17:01:19 -04:00
|
|
|
}
|
2015-01-07 20:20:14 +00:00
|
|
|
bool want_interlaced = ('i' == auxFlag);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
for (int i = 0; i < screenResources->noutput; i++)
|
|
|
|
{
|
|
|
|
XRROutputInfo* output_info =
|
|
|
|
XRRGetOutputInfo(dpy, screenResources, screenResources->outputs[i]);
|
|
|
|
if (output_info && output_info->crtc && output_info->connection == RR_Connected)
|
|
|
|
{
|
|
|
|
XRRCrtcInfo* crtc_info = XRRGetCrtcInfo(dpy, screenResources, output_info->crtc);
|
|
|
|
if (crtc_info)
|
|
|
|
{
|
2011-03-08 22:43:44 +00:00
|
|
|
if (!output_name || !strcmp(output_name, output_info->name))
|
2011-02-09 03:12:05 +00:00
|
|
|
{
|
2011-03-08 22:43:44 +00:00
|
|
|
// Use the first output for the default setting.
|
|
|
|
if (!output_name)
|
|
|
|
{
|
|
|
|
output_name = strdup(output_info->name);
|
2019-03-02 22:41:50 -06:00
|
|
|
Config::SetBaseOrCurrent(
|
|
|
|
Config::MAIN_FULLSCREEN_DISPLAY_RES,
|
2019-11-23 19:15:52 -05:00
|
|
|
fmt::format("{}: {}x{}", output_info->name, fullWidth, fullHeight));
|
2011-03-08 22:43:44 +00:00
|
|
|
}
|
2011-02-09 03:12:05 +00:00
|
|
|
outputInfo = output_info;
|
|
|
|
crtcInfo = crtc_info;
|
|
|
|
for (int j = 0; j < output_info->nmode && fullMode == 0; j++)
|
|
|
|
{
|
|
|
|
for (int k = 0; k < screenResources->nmode && fullMode == 0; k++)
|
|
|
|
{
|
|
|
|
if (output_info->modes[j] == screenResources->modes[k].id)
|
|
|
|
{
|
|
|
|
if (fullWidth == screenResources->modes[k].width &&
|
2015-01-07 20:20:14 +00:00
|
|
|
fullHeight == screenResources->modes[k].height &&
|
|
|
|
want_interlaced == !!(screenResources->modes[k].modeFlags & RR_Interlace))
|
2011-02-09 03:12:05 +00:00
|
|
|
{
|
|
|
|
fullMode = screenResources->modes[k].id;
|
|
|
|
if (crtcInfo->x + (int)screenResources->modes[k].width > fs_fb_width)
|
|
|
|
fs_fb_width = crtcInfo->x + screenResources->modes[k].width;
|
|
|
|
if (crtcInfo->y + (int)screenResources->modes[k].height > fs_fb_height)
|
|
|
|
fs_fb_height = crtcInfo->y + screenResources->modes[k].height;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (crtc_info->x + (int)crtc_info->width > fs_fb_width)
|
|
|
|
fs_fb_width = crtc_info->x + crtc_info->width;
|
|
|
|
if (crtc_info->y + (int)crtc_info->height > fs_fb_height)
|
|
|
|
fs_fb_height = crtc_info->y + crtc_info->height;
|
|
|
|
}
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
2011-02-09 03:12:05 +00:00
|
|
|
if (crtc_info && crtcInfo != crtc_info)
|
|
|
|
XRRFreeCrtcInfo(crtc_info);
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
2011-02-09 03:12:05 +00:00
|
|
|
if (output_info && outputInfo != output_info)
|
|
|
|
XRRFreeOutputInfo(output_info);
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
2011-02-09 03:12:05 +00:00
|
|
|
fs_fb_width_mm = fs_fb_width * DisplayHeightMM(dpy, screen) / DisplayHeight(dpy, screen);
|
|
|
|
fs_fb_height_mm = fs_fb_height * DisplayHeightMM(dpy, screen) / DisplayHeight(dpy, screen);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
if (output_name)
|
|
|
|
free(output_name);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
if (outputInfo && crtcInfo && fullMode)
|
|
|
|
{
|
2020-11-25 21:13:50 -05:00
|
|
|
INFO_LOG_FMT(VIDEO, "Fullscreen Resolution {}x{}", fullWidth, fullHeight);
|
2011-02-09 03:12:05 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2020-11-25 21:13:50 -05:00
|
|
|
ERROR_LOG_FMT(VIDEO, "Failed to obtain fullscreen size.\n"
|
|
|
|
"Using current desktop resolution for fullscreen.");
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void XRRConfiguration::ToggleDisplayMode(bool bFullscreen)
|
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
if (!bValid || !screenResources || !outputInfo || !crtcInfo || !fullMode)
|
|
|
|
return;
|
|
|
|
if (bFullscreen == bIsFullscreen)
|
2010-07-10 12:35:16 +00:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
XGrabServer(dpy);
|
2010-04-22 04:28:34 +00:00
|
|
|
if (bFullscreen)
|
2011-02-09 03:12:05 +00:00
|
|
|
{
|
|
|
|
XRRSetCrtcConfig(dpy, screenResources, outputInfo->crtc, CurrentTime, crtcInfo->x, crtcInfo->y,
|
|
|
|
fullMode, crtcInfo->rotation, crtcInfo->outputs, crtcInfo->noutput);
|
|
|
|
XRRSetScreenSize(dpy, win, fs_fb_width, fs_fb_height, fs_fb_width_mm, fs_fb_height_mm);
|
|
|
|
bIsFullscreen = true;
|
|
|
|
}
|
2010-04-22 04:28:34 +00:00
|
|
|
else
|
2011-02-09 03:12:05 +00:00
|
|
|
{
|
|
|
|
XRRSetCrtcConfig(dpy, screenResources, outputInfo->crtc, CurrentTime, crtcInfo->x, crtcInfo->y,
|
|
|
|
crtcInfo->mode, crtcInfo->rotation, crtcInfo->outputs, crtcInfo->noutput);
|
|
|
|
XRRSetScreenSize(dpy, win, fb_width, fb_height, fb_width_mm, fb_height_mm);
|
|
|
|
bIsFullscreen = false;
|
|
|
|
}
|
|
|
|
XUngrabServer(dpy);
|
|
|
|
XSync(dpy, false);
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
|
|
|
|
2014-08-06 15:45:40 -04:00
|
|
|
void XRRConfiguration::AddResolutions(std::vector<std::string>& resos)
|
2010-04-22 04:28:34 +00:00
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
if (!bValid || !screenResources)
|
2010-07-10 12:35:16 +00:00
|
|
|
return;
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
// Get all full screen resolutions for the config dialog
|
|
|
|
for (int i = 0; i < screenResources->noutput; i++)
|
2010-04-22 04:28:34 +00:00
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
XRROutputInfo* output_info =
|
|
|
|
XRRGetOutputInfo(dpy, screenResources, screenResources->outputs[i]);
|
2016-06-24 10:43:46 +02:00
|
|
|
|
2011-02-09 03:12:05 +00:00
|
|
|
if (output_info && output_info->crtc && output_info->connection == RR_Connected)
|
|
|
|
{
|
|
|
|
for (int j = 0; j < output_info->nmode; j++)
|
2014-08-30 17:01:19 -04:00
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
for (int k = 0; k < screenResources->nmode; k++)
|
2014-08-30 17:01:19 -04:00
|
|
|
{
|
2011-02-09 03:12:05 +00:00
|
|
|
if (output_info->modes[j] == screenResources->modes[k].id)
|
|
|
|
{
|
2015-01-07 20:20:14 +00:00
|
|
|
bool interlaced = !!(screenResources->modes[k].modeFlags & RR_Interlace);
|
2011-03-08 22:43:44 +00:00
|
|
|
const std::string strRes = std::string(output_info->name) + ": " +
|
2015-01-07 20:20:14 +00:00
|
|
|
std::string(screenResources->modes[k].name) +
|
|
|
|
(interlaced ? "i" : "");
|
2011-02-09 03:12:05 +00:00
|
|
|
// Only add unique resolutions
|
|
|
|
if (std::find(resos.begin(), resos.end(), strRes) == resos.end())
|
|
|
|
{
|
|
|
|
resos.push_back(strRes);
|
|
|
|
}
|
|
|
|
}
|
2014-08-30 17:01:19 -04:00
|
|
|
}
|
|
|
|
}
|
2011-02-09 03:12:05 +00:00
|
|
|
}
|
|
|
|
if (output_info)
|
|
|
|
XRRFreeOutputInfo(output_info);
|
2010-04-22 04:28:34 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#endif
|
2019-05-05 23:48:12 +00:00
|
|
|
} // namespace X11Utils
|