RetroArch/gfx/display_servers/dispserv_x11.c

589 lines
16 KiB
C
Raw Normal View History

2018-05-25 10:56:22 +01:00
/* RetroArch - A frontend for libretro.
* Copyright (C) 2010-2014 - Hans-Kristian Arntzen
* Copyright (C) 2011-2017 - Daniel De Matteis
* Copyright (C) 2016-2019 - Brad Parker
2018-05-25 10:56:22 +01:00
*
* RetroArch is free software: you can redistribute it and/or modify it under the terms
* of the GNU General Public License as published by the Free Software Found-
* ation, either version 3 of the License, or (at your option) any later version.
*
* RetroArch is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY;
* without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR
* PURPOSE. See the GNU General Public License for more details.
*
* You should have received a copy of the GNU General Public License along with RetroArch.
* If not, see <http://www.gnu.org/licenses/>.
*/
/* We are targeting XRandR 1.2 here. */
#include <compat/strl.h>
#include <sys/types.h>
#include <unistd.h>
2019-01-30 22:46:29 +00:00
#include <X11/Xlib.h>
2018-05-25 10:56:22 +01:00
#include "../../config.h"
#ifdef HAVE_XRANDR
#include <X11/extensions/Xrandr.h>
#include <X11/extensions/randr.h>
#include <X11/extensions/Xrender.h>
#endif
2018-05-25 10:56:22 +01:00
#include "../video_display_server.h"
#include "../common/x11_common.h"
#include "../../configuration.h"
#include "../video_driver.h" /* needed to set refresh rate in set resolution */
#include "../video_crt_switch.h" /* needed to set aspect for low res in linux */
#ifdef HAVE_XRANDR
static char xrandr[1024] = {0};
static char crt_name[16] = {0};
static int crt_name_id = 0;
2018-09-10 19:36:57 +01:00
static bool crt_en = false;
static unsigned crt_id = 20;
static char orig_output[256] = {0};
static char old_mode[256] = {0};
static char new_mode[256] = {0};
2019-01-30 22:41:13 +00:00
static XRRModeInfo crt_rrmode;
#endif
2018-05-25 10:56:22 +01:00
static bool x11_display_server_using_global_dpy = false;
2018-05-25 10:56:22 +01:00
typedef struct
{
unsigned opacity;
bool decorations;
} dispserv_x11_t;
2019-03-14 16:10:53 +00:00
#ifdef HAVE_XRANDR
static Display* x11_display_server_open_display(void)
{
Display *dpy = g_x11_dpy;
if (dpy)
x11_display_server_using_global_dpy = true;
else
{
/* SDL might use X11 but doesn't use g_x11_dpy, so open it manually */
dpy = XOpenDisplay(0);
x11_display_server_using_global_dpy = false;
}
return dpy;
}
2019-03-14 16:10:53 +00:00
#endif
2019-03-14 16:10:53 +00:00
#ifdef HAVE_XRANDR
static void x11_display_server_close_display(Display *dpy)
{
if (!dpy || x11_display_server_using_global_dpy || dpy == g_x11_dpy)
return;
XCloseDisplay(dpy);
}
2019-03-14 16:10:53 +00:00
#endif
2018-05-25 10:56:22 +01:00
static void* x11_display_server_init(void)
{
dispserv_x11_t *dispserv = (dispserv_x11_t*)calloc(1, sizeof(*dispserv));
if (!dispserv)
return NULL;
return dispserv;
}
static void x11_display_server_destroy(void *data)
{
2018-10-02 16:56:28 -04:00
dispserv_x11_t *dispserv = (dispserv_x11_t*)data;
#ifdef HAVE_XRANDR
2018-10-02 16:56:28 -04:00
if (crt_en)
{
snprintf(xrandr, sizeof(xrandr),
"xrandr --newmode 700x480_59.94 13.849698 700 742 801 867 480 490 496 533 interlace -hsync -vsync");
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --addmode \"%s\" 700x480_59.94",
orig_output);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --output \"%s\" --mode 700x480_59.94",
orig_output);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --delmode \"%s\" \"%s\"",
orig_output, old_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --rmmode \"%s\"",
old_mode);
system(xrandr);
2018-09-10 17:42:33 +01:00
}
#endif
2018-05-25 10:56:22 +01:00
if (dispserv)
free(dispserv);
}
2018-09-17 03:40:17 +02:00
static bool x11_display_server_set_window_opacity(void *data, unsigned opacity)
2018-05-25 10:56:22 +01:00
{
dispserv_x11_t *serv = (dispserv_x11_t*)data;
Atom net_wm_opacity = XInternAtom(g_x11_dpy, "_NET_WM_WINDOW_OPACITY", False);
Atom cardinal = XInternAtom(g_x11_dpy, "CARDINAL", False);
serv->opacity = opacity;
opacity = opacity * ((unsigned)-1 / 100.0);
if (opacity == (unsigned)-1)
XDeleteProperty(g_x11_dpy, g_x11_win, net_wm_opacity);
2018-10-02 16:56:28 -04:00
else
2018-11-24 19:38:01 +01:00
XChangeProperty(g_x11_dpy, g_x11_win, net_wm_opacity, cardinal,
32, PropModeReplace, (const unsigned char*)&opacity, 1);
2018-05-25 10:56:22 +01:00
return true;
}
2018-09-17 03:40:17 +02:00
static bool x11_display_server_set_window_decorations(void *data, bool on)
2018-05-25 10:56:22 +01:00
{
dispserv_x11_t *serv = (dispserv_x11_t*)data;
2018-11-24 19:38:01 +01:00
if (serv)
serv->decorations = on;
2018-05-25 10:56:22 +01:00
2019-02-03 15:49:35 -08:00
/* menu_setting performs a reinit instead to properly apply
2018-11-24 19:38:01 +01:00
* decoration changes */
2018-05-25 10:56:22 +01:00
return true;
}
#ifdef HAVE_XRANDR
2018-09-17 03:40:17 +02:00
static bool x11_display_server_set_resolution(void *data,
unsigned width, unsigned height, int int_hz, float hz, int center, int monitor_index, int xoffset)
2018-05-25 10:56:22 +01:00
{
2019-01-31 15:13:36 +01:00
int screen;
Window window;
XRRScreenResources *res = NULL;
Display *dpy = NULL;
2019-01-31 15:13:36 +01:00
int i = 0;
int hfp = 0;
int hsp = 0;
int hbp = 0;
int vfp = 0;
int vsp = 0;
int vbp = 0;
int hmax = 0;
int vmax = 0;
int pdefault = 8;
int pwidth = 0;
float roundw = 0.0f;
float pixel_clock = 0;
crt_en = true;
crt_name_id += 1;
snprintf(crt_name, sizeof(crt_name), "CRT%d", crt_name_id);
2019-02-03 15:49:35 -08:00
snprintf(old_mode, sizeof(old_mode), "%s", new_mode);
2019-01-31 15:13:36 +01:00
dpy = XOpenDisplay(0);
screen = DefaultScreen(dpy);
window = RootWindow(dpy, screen);
2018-10-02 16:56:28 -04:00
2018-05-25 10:56:22 +01:00
/* set core refresh from hz */
2018-10-02 16:56:28 -04:00
video_monitor_set_refresh_rate(hz);
2018-11-24 19:38:01 +01:00
/* following code is the mode line generator */
pwidth = width;
2018-10-02 16:56:28 -04:00
if (height < 400 && width > 400)
pwidth = width / 2;
2018-10-02 16:56:28 -04:00
roundw = roundf((float)pwidth / (float)height * 100) / 100;
2018-10-02 16:56:28 -04:00
if (height > width)
roundw = roundf((float)height / (float)width * 100) / 100;
2018-05-25 10:56:22 +01:00
if (roundw > 1.35)
roundw = 1.25;
2018-10-02 16:56:28 -04:00
if (roundw < 1.20)
roundw = 1.34;
if (width < 700)
{
hfp = width * 1.055;
hbp = width * roundw - 8;
}
else
{
hfp = (width * 1.055) + (width / 40);
hbp = (width * roundw) + (width /24);
xoffset = xoffset*2;
}
2019-02-22 10:11:06 -05:00
hsp = (width * 1.140) - (xoffset*4);
2018-05-25 10:56:22 +01:00
hmax = hbp;
2018-10-02 16:56:28 -04:00
2018-05-25 10:56:22 +01:00
if (height < 241)
vmax = 261;
if (height < 241 && hz > 56 && hz < 58)
vmax = 280;
2018-10-02 16:56:28 -04:00
if (height < 241 && hz < 55)
2018-05-25 10:56:22 +01:00
vmax = 313;
if (height > 250 && height < 260 && hz > 54)
vmax = 296;
if (height > 250 && height < 260 && hz > 52 && hz < 54)
vmax = 285;
if (height > 250 && height < 260 && hz < 52)
vmax = 313;
if (height > 260 && height < 300)
vmax = 318;
2018-05-25 10:56:22 +01:00
if (height > 400 && hz > 56)
vmax = 533;
2018-05-25 10:56:22 +01:00
if (height > 520 && hz < 57)
vmax = 580;
if (height > 300 && hz < 56)
vmax = 615;
if (height > 500 && hz < 56)
vmax = 624;
2018-10-02 16:56:28 -04:00
if (height > 300)
pdefault = pdefault * 2;
2018-05-25 10:56:22 +01:00
2018-10-02 16:56:28 -04:00
vfp = height + ((vmax - height) / 2) - pdefault;
2018-05-25 10:56:22 +01:00
if (height < 300)
2018-11-24 19:38:01 +01:00
vsp = vfp + 3; /* needs to be 3 for progressive */
if (height > 300)
2018-11-24 19:38:01 +01:00
vsp = vfp + 6; /* needs to be 6 for interlaced */
2018-10-02 16:56:28 -04:00
2018-05-25 10:56:22 +01:00
vbp = vmax;
if (height < 300)
2018-10-02 16:56:28 -04:00
pixel_clock = (hmax * vmax * hz) / 1000000;
2018-05-25 10:56:22 +01:00
if (height > 300)
2018-10-02 16:56:28 -04:00
pixel_clock = ((hmax * vmax * hz) / 1000000) / 2;
2018-11-24 19:38:01 +01:00
/* above code is the modeline generator */
2018-05-25 10:56:22 +01:00
2018-09-10 18:12:44 +01:00
/* create interlaced newmode from modline variables */
if (height < 300)
{
snprintf(xrandr, sizeof(xrandr),
"xrandr --newmode \"%s_%dx%d_%0.2f\" %f %d %d %d %d %d %d %d %d -hsync -vsync",
crt_name, width, height, hz, pixel_clock, width, hfp, hsp, hbp, height, vfp, vsp, vbp);
2018-09-10 18:12:44 +01:00
system(xrandr);
2018-05-25 10:56:22 +01:00
}
2018-05-25 10:56:22 +01:00
/* create interlaced newmode from modline variables */
if (height > 300)
2018-10-02 16:56:28 -04:00
{
snprintf(xrandr, sizeof(xrandr),
"xrandr --newmode \"%s_%dx%d_%0.2f\" %f %d %d %d %d %d %d %d %d interlace -hsync -vsync",
crt_name, width, height, hz, pixel_clock, width, hfp, hsp, hbp, height, vfp, vsp, vbp);
2018-05-25 10:56:22 +01:00
system(xrandr);
}
2018-10-02 16:56:28 -04:00
/* variable for new mode */
snprintf(new_mode, sizeof(new_mode), "%s_%dx%d_%0.2f", crt_name, width, height, hz);
2018-10-02 16:56:28 -04:00
2019-02-03 15:49:35 -08:00
/* need to run loops for DVI0 - DVI-2 and VGA0 - VGA-2 outputs to
2018-11-24 19:38:01 +01:00
* add and delete modes */
crt_rrmode.id = crt_id;
2019-01-30 21:55:33 +00:00
crt_rrmode.width = width;
crt_rrmode.height = height;
crt_rrmode.dotClock = pixel_clock;
crt_rrmode.hSyncStart = hfp;
crt_rrmode.hSyncEnd = hsp;
crt_rrmode.hTotal = hmax;
crt_rrmode.hSkew = 0;
crt_rrmode.vSyncStart = vfp;
crt_rrmode.vSyncEnd = vsp;
crt_rrmode.vTotal = vmax;
crt_rrmode.name = new_mode;
crt_rrmode.nameLength = sizeof(new_mode);
crt_rrmode.modeFlags = 0;
2019-02-03 15:49:35 -08:00
res = XRRGetScreenResources(dpy, window);
2019-02-03 15:49:35 -08:00
2019-01-30 21:55:33 +00:00
if (monitor_index == 0)
2018-10-02 16:56:28 -04:00
{
2019-02-06 09:16:26 -08:00
for (i = 0; i < res->noutput; i++)
2019-02-03 15:49:35 -08:00
{
XRROutputInfo *outputs = XRRGetOutputInfo(dpy, res, res->outputs[i]);
2019-02-01 20:38:50 +00:00
2019-01-30 21:55:33 +00:00
if (outputs->connection == RR_Connected)
{
2019-02-22 10:11:06 -05:00
snprintf(orig_output, sizeof(orig_output), "%s", outputs->name);
2019-02-03 15:49:35 -08:00
snprintf(xrandr, sizeof(xrandr),
"xrandr --addmode \"%s\" \"%s\"",
outputs->name, new_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --output \"%s\" --mode \"%s\"",
outputs->name, new_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --delmode \"%s\" \"%s\"",
outputs->name, old_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --rmmode \"%s\"",
old_mode);
system(xrandr);
2019-01-30 21:55:33 +00:00
}
}
2019-02-22 10:11:06 -05:00
}
else if (monitor_index > 0)
{
XRROutputInfo *outputs = XRRGetOutputInfo(dpy, res, res->outputs[monitor_index]);
2019-02-22 10:11:06 -05:00
if (outputs->connection == RR_Connected)
{
snprintf(orig_output, sizeof(orig_output), "%s", outputs->name);
snprintf(xrandr, sizeof(xrandr),
"xrandr --addmode \"%s\" \"%s\"",
outputs->name, new_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --output \"%s\" --mode \"%s\"",
outputs->name, new_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --delmode \"%s\" \"%s\"",
outputs->name, old_mode);
system(xrandr);
snprintf(xrandr, sizeof(xrandr),
"xrandr --rmmode \"%s\"",
old_mode);
system(xrandr);
2019-02-22 10:11:06 -05:00
}
}
2018-10-02 16:56:28 -04:00
return true;
2018-05-25 10:56:22 +01:00
}
#endif
2018-05-25 10:56:22 +01:00
2018-11-25 17:06:55 +01:00
const char *x11_display_server_get_output_options(void *data)
2018-09-17 03:40:17 +02:00
{
#ifdef HAVE_XRANDR
Display *dpy;
XRRScreenResources *res;
XRROutputInfo *info;
Window root;
int i;
static char s[PATH_MAX_LENGTH];
if (!(dpy = XOpenDisplay(0)))
return NULL;
root = RootWindow(dpy, DefaultScreen(dpy));
if (!(res = XRRGetScreenResources(dpy, root)))
return NULL;
for (i = 0; i < res->noutput; i++)
{
if (!(info = XRRGetOutputInfo(dpy, res, res->outputs[i])))
return NULL;
strlcat(s, info->name, sizeof(s));
if ((i+1) < res->noutput)
strlcat(s, "|", sizeof(s));
}
return s;
#else
2018-09-17 03:40:17 +02:00
/* TODO/FIXME - hardcoded for now; list should be built up dynamically later */
return "HDMI-0|HDMI-1|HDMI-2|HDMI-3|DVI-0|DVI-1|DVI-2|DVI-3|VGA-0|VGA-1|VGA-2|VGA-3|Config";
#endif
2018-09-17 03:40:17 +02:00
}
#ifdef HAVE_XRANDR
static void x11_display_server_set_screen_orientation(enum rotation rotation)
{
int i, j;
XRRScreenResources *screen;
/* switched to using XOpenDisplay() due to deinit order issue with g_x11_dpy when restoring original rotation on exit */
Display *dpy = XOpenDisplay(0);
XRRScreenConfiguration *config = XRRGetScreenInfo(dpy, DefaultRootWindow(dpy));
double dpi = (25.4 * DisplayHeight(dpy, DefaultScreen(dpy))) / DisplayHeightMM(dpy, DefaultScreen(dpy));
XGrabServer(dpy);
screen = XRRGetScreenResources(dpy, DefaultRootWindow(dpy));
for (i = 0; i < screen->noutput; i++)
{
XRROutputInfo *info = XRRGetOutputInfo(dpy, screen, screen->outputs[i]);
if (info->connection != RR_Connected)
{
XRRFreeOutputInfo(info);
continue;
}
for (j = 0; j < info->ncrtc; j++)
{
XRRCrtcInfo *crtc = XRRGetCrtcInfo(dpy, screen, screen->crtcs[j]);
Rotation new_rotation = RR_Rotate_0;
if (crtc->width == 0 || crtc->height == 0)
{
XRRFreeCrtcInfo(crtc);
continue;
}
switch (rotation)
{
case ORIENTATION_NORMAL:
default:
if (crtc->rotations & RR_Rotate_0)
new_rotation = RR_Rotate_0;
break;
case ORIENTATION_VERTICAL:
if (crtc->rotations & RR_Rotate_270)
new_rotation = RR_Rotate_270;
break;
case ORIENTATION_FLIPPED:
if (crtc->rotations & RR_Rotate_180)
new_rotation = RR_Rotate_180;
break;
case ORIENTATION_FLIPPED_ROTATED:
if (crtc->rotations & RR_Rotate_90)
new_rotation = RR_Rotate_90;
break;
}
XRRSetCrtcConfig(dpy, screen, screen->crtcs[j], CurrentTime, 0, 0, None, RR_Rotate_0, NULL, 0);
if ((crtc->rotation & RR_Rotate_0 || crtc->rotation & RR_Rotate_180) && (rotation == ORIENTATION_VERTICAL || rotation == ORIENTATION_FLIPPED_ROTATED))
{
unsigned width = crtc->width;
crtc->width = crtc->height;
crtc->height = width;
}
else if ((crtc->rotation & RR_Rotate_90 || crtc->rotation & RR_Rotate_270) && (rotation == ORIENTATION_NORMAL || rotation == ORIENTATION_FLIPPED))
{
unsigned width = crtc->width;
crtc->width = crtc->height;
crtc->height = width;
}
crtc->rotation = new_rotation;
XRRSetScreenSize(dpy, DefaultRootWindow(dpy), crtc->width, crtc->height, (25.4 * crtc->width) / dpi, (25.4 * crtc->height) / dpi);
XRRSetCrtcConfig(dpy, screen, screen->crtcs[j], CurrentTime, crtc->x, crtc->y, crtc->mode, crtc->rotation, crtc->outputs, crtc->noutput);
XRRFreeCrtcInfo(crtc);
}
XRRFreeOutputInfo(info);
}
XRRFreeScreenResources(screen);
XUngrabServer(dpy);
XSync(dpy, False);
XRRFreeScreenConfigInfo(config);
XCloseDisplay(dpy);
}
static enum rotation x11_display_server_get_screen_orientation(void)
{
int i, j;
Display *dpy = x11_display_server_open_display();
XRRScreenResources *screen = XRRGetScreenResources(dpy, DefaultRootWindow(dpy));
XRRScreenConfiguration *config = XRRGetScreenInfo(dpy, DefaultRootWindow(dpy));
enum rotation rotation = ORIENTATION_NORMAL;
for (i = 0; i < screen->noutput; i++)
{
XRROutputInfo *info = XRRGetOutputInfo(dpy, screen, screen->outputs[i]);
if (info->connection != RR_Connected)
{
XRRFreeOutputInfo(info);
continue;
}
for (j = 0; j < info->ncrtc; j++)
{
XRRCrtcInfo *crtc = XRRGetCrtcInfo(dpy, screen, screen->crtcs[j]);
if (crtc->width == 0 || crtc->height == 0)
{
XRRFreeCrtcInfo(crtc);
continue;
}
switch (crtc->rotation)
{
case RR_Rotate_0:
default:
rotation = ORIENTATION_NORMAL;
break;
case RR_Rotate_270:
rotation = ORIENTATION_VERTICAL;
break;
case RR_Rotate_180:
rotation = ORIENTATION_FLIPPED;
break;
case RR_Rotate_90:
rotation = ORIENTATION_FLIPPED_ROTATED;
break;
}
XRRFreeCrtcInfo(crtc);
}
XRRFreeOutputInfo(info);
}
XRRFreeScreenResources(screen);
XRRFreeScreenConfigInfo(config);
x11_display_server_close_display(dpy);
return rotation;
}
#endif
static uint32_t x11_display_server_get_flags(void *data)
{
uint32_t flags = 0;
#ifdef HAVE_XRANDR
BIT32_SET(flags, DISPSERV_CTX_CRT_SWITCHRES);
#endif
return flags;
}
2018-05-25 10:56:22 +01:00
const video_display_server_t dispserv_x11 = {
x11_display_server_init,
x11_display_server_destroy,
2018-09-17 03:40:17 +02:00
x11_display_server_set_window_opacity,
2018-11-24 02:47:22 +01:00
NULL, /* set_window_progress */
2018-09-17 03:40:17 +02:00
x11_display_server_set_window_decorations,
#ifdef HAVE_XRANDR
2018-09-17 03:40:17 +02:00
x11_display_server_set_resolution,
#else
NULL, /* set_resolution */
#endif
2018-11-24 02:47:22 +01:00
NULL, /* get_resolution_list */
2018-09-17 03:40:17 +02:00
x11_display_server_get_output_options,
#ifdef HAVE_XRANDR
x11_display_server_set_screen_orientation,
x11_display_server_get_screen_orientation,
#else
NULL, /* set_screen_orientation */
NULL, /* get_screen_orientation */
#endif
x11_display_server_get_flags,
2018-05-25 10:56:22 +01:00
"x11"
};