Change to the 1.3 part of the Xrandr api for fullscreen mode switching on linux to support multiple monitor setups. There is probably some more work to be done, but at least it won't shut off one of your monitors anymore.

git-svn-id: https://dolphin-emu.googlecode.com/svn/trunk@7121 8ced0084-cf51-0410-be5f-012b33b47a6e
This commit is contained in:
Glenn Rice 2011-02-09 03:12:05 +00:00
parent 0c188bfceb
commit edfb62a41f
22 changed files with 8075 additions and 7913 deletions

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

File diff suppressed because it is too large Load Diff

View File

@ -25,6 +25,11 @@
extern char **environ; extern char **environ;
#endif #endif
#if defined(HAVE_WX) && HAVE_WX
#include <string>
#include <algorithm>
#endif
namespace X11Utils namespace X11Utils
{ {
@ -157,35 +162,43 @@ void InhibitScreensaver(Display *dpy, Window win, bool suspend)
XRRConfiguration::XRRConfiguration(Display *_dpy, Window _win) XRRConfiguration::XRRConfiguration(Display *_dpy, Window _win)
: dpy(_dpy) : dpy(_dpy)
, win(_win) , win(_win)
, deskSize(-1), fullSize(-1) , screenResources(NULL), outputInfo(NULL), crtcInfo(NULL)
, bValid(true) , fullMode(0)
, fs_fb_width(0), fs_fb_height(0), fs_fb_width_mm(0), fs_fb_height_mm(0)
, bValid(true), bIsFullscreen(false)
{ {
XRRScreenSize *sizes; int XRRMajorVersion, XRRMinorVersion;
int numSizes;
// XRRSizes is safe to call even if the RANDR extension is not present and numSizes if (!XRRQueryVersion(dpy, &XRRMajorVersion, &XRRMinorVersion) ||
// will be 0 in that case (according to the documentation) (XRRMajorVersion < 1 || (XRRMajorVersion == 1 && XRRMinorVersion < 3)))
sizes = XRRSizes(dpy, DefaultScreen(dpy), &numSizes);
if (!numSizes)
{ {
WARN_LOG(VIDEO, "XRRExtension not supported."); WARN_LOG(VIDEO, "XRRExtension not supported.");
bValid = false; bValid = false;
return; return;
} }
int vidModeMajorVersion, vidModeMinorVersion; screenResources = XRRGetScreenResourcesCurrent(dpy, win);
XRRQueryVersion(dpy, &vidModeMajorVersion, &vidModeMinorVersion);
INFO_LOG(VIDEO, "XRRExtension-Version %d.%d", vidModeMajorVersion, vidModeMinorVersion); 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);
INFO_LOG(VIDEO, "XRRExtension-Version %d.%d", XRRMajorVersion, XRRMinorVersion);
Update(); Update();
} }
XRRConfiguration::~XRRConfiguration() XRRConfiguration::~XRRConfiguration()
{ {
if (bValid) if (bValid && bIsFullscreen)
{
ToggleDisplayMode(False); ToggleDisplayMode(False);
XRRFreeScreenConfigInfo(screenConfig); if (screenResources)
} XRRFreeScreenResources(screenResources);
if (outputInfo)
XRRFreeOutputInfo(outputInfo);
if (crtcInfo)
XRRFreeCrtcInfo(crtcInfo);
} }
void XRRConfiguration::Update() void XRRConfiguration::Update()
@ -193,33 +206,81 @@ void XRRConfiguration::Update()
if (!bValid) if (!bValid)
return; return;
if (outputInfo)
{
XRRFreeOutputInfo(outputInfo);
outputInfo = NULL;
}
if (crtcInfo)
{
XRRFreeCrtcInfo(crtcInfo);
crtcInfo = NULL;
}
fullMode = 0;
// Get the resolution setings for fullscreen mode // Get the resolution setings for fullscreen mode
int fullWidth, fullHeight; unsigned int fullWidth, fullHeight;
char *output_name = NULL;
sscanf(SConfig::GetInstance().m_LocalCoreStartupParameter.strFullscreenResolution.c_str(), sscanf(SConfig::GetInstance().m_LocalCoreStartupParameter.strFullscreenResolution.c_str(),
"%dx%d", &fullWidth, &fullHeight); "%a[^:]: %ux%u", &output_name, &fullWidth, &fullHeight);
XRRScreenSize *sizes; for (int i = 0; i < screenResources->noutput; i++)
int numSizes; {
XRROutputInfo *output_info = XRRGetOutputInfo(dpy, screenResources, screenResources->outputs[i]);
screenConfig = XRRGetScreenInfo(dpy, win); if (output_info && output_info->crtc && output_info->connection == RR_Connected)
{
// Save desktop resolution XRRCrtcInfo *crtc_info = XRRGetCrtcInfo(dpy, screenResources, output_info->crtc);
deskSize = XRRConfigCurrentConfiguration(screenConfig, &screenRotation); if (crtc_info)
// Set the desktop resolution as the default {
fullSize = deskSize; if (!strcmp(output_name, output_info->name))
{
// Find the index of the fullscreen resolution from config outputInfo = output_info;
sizes = XRRConfigSizes(screenConfig, &numSizes); crtcInfo = crtc_info;
if (numSizes > 0 && sizes != NULL) { for (int j = 0; j < output_info->nmode && fullMode == 0; j++)
for (int i = 0; i < numSizes; i++) { {
if ((sizes[i].width == fullWidth) && (sizes[i].height == fullHeight)) { for (int k = 0; k < screenResources->nmode && fullMode == 0; k++)
fullSize = i; {
if (output_info->modes[j] == screenResources->modes[k].id)
{
if (fullWidth == screenResources->modes[k].width &&
fullHeight == screenResources->modes[k].height)
{
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;
} }
} }
INFO_LOG(VIDEO, "Fullscreen Resolution %dx%d",
sizes[fullSize].width, sizes[fullSize].height);
} }
else { }
}
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;
}
}
if (crtc_info && crtcInfo != crtc_info)
XRRFreeCrtcInfo(crtc_info);
}
if (output_info && outputInfo != output_info)
XRRFreeOutputInfo(output_info);
}
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);
if (output_name)
free(output_name);
if (outputInfo && crtcInfo && fullMode)
{
INFO_LOG(VIDEO, "Fullscreen Resolution %dx%d", fullWidth, fullHeight);
}
else
{
ERROR_LOG(VIDEO, "Failed to obtain fullscreen size.\n" ERROR_LOG(VIDEO, "Failed to obtain fullscreen size.\n"
"Using current desktop resolution for fullscreen."); "Using current desktop resolution for fullscreen.");
} }
@ -227,35 +288,64 @@ void XRRConfiguration::Update()
void XRRConfiguration::ToggleDisplayMode(bool bFullscreen) void XRRConfiguration::ToggleDisplayMode(bool bFullscreen)
{ {
if (!bValid) if (!bValid || !screenResources || !outputInfo || !crtcInfo || !fullMode)
return;
if (bFullscreen == bIsFullscreen)
return; return;
XGrabServer(dpy);
if (bFullscreen) if (bFullscreen)
XRRSetScreenConfig(dpy, screenConfig, win, {
fullSize, screenRotation, CurrentTime); 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;
}
else else
XRRSetScreenConfig(dpy, screenConfig, win, {
deskSize, screenRotation, CurrentTime); 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);
} }
#if defined(HAVE_WX) && HAVE_WX #if defined(HAVE_WX) && HAVE_WX
void XRRConfiguration::AddResolutions(wxArrayString& arrayStringFor_FullscreenResolution) void XRRConfiguration::AddResolutions(wxArrayString& arrayStringFor_FullscreenResolution)
{ {
if (!bValid) if (!bValid || !screenResources)
return; return;
int screen; //Get all full screen resolutions for the config dialog
screen = DefaultScreen(dpy); for (int i = 0; i < screenResources->noutput; i++)
//Get all full screen resos for the config dialog
XRRScreenSize *sizes = NULL;
int modeNum = 0;
sizes = XRRSizes(dpy, screen, &modeNum);
if (modeNum > 0 && sizes != NULL)
{ {
for (int i = 0; i < modeNum; i++) XRROutputInfo *output_info =
arrayStringFor_FullscreenResolution.Add(wxString::Format(wxT("%dx%d"), XRRGetOutputInfo(dpy, screenResources, screenResources->outputs[i]);
sizes[i].width, sizes[i].height));
if (output_info && output_info->crtc && output_info->connection == RR_Connected)
{
std::vector<std::string> resos;
for (int j = 0; j < output_info->nmode; j++)
for (int k = 0; k < screenResources->nmode; k++)
if (output_info->modes[j] == screenResources->modes[k].id)
{
std::string strRes(screenResources->modes[k].name);
// Only add unique resolutions
if (std::find(resos.begin(), resos.end(), strRes) == resos.end())
{
resos.push_back(strRes);
arrayStringFor_FullscreenResolution.Add(
wxString::FromUTF8(output_info->name) + wxT(": ") +
wxString::FromUTF8(screenResources->modes[k].name));
}
}
}
if (output_info)
XRRFreeOutputInfo(output_info);
} }
} }
#endif #endif
@ -263,4 +353,3 @@ void XRRConfiguration::AddResolutions(wxArrayString& arrayStringFor_FullscreenRe
#endif #endif
} }

View File

@ -36,9 +36,9 @@
// EWMH state actions, see // EWMH state actions, see
// http://freedesktop.org/wiki/Specifications/wm-spec?action=show&redirect=Standards%2Fwm-spec // http://freedesktop.org/wiki/Specifications/wm-spec?action=show&redirect=Standards%2Fwm-spec
#define _NET_WM_STATE_REMOVE 0 /* remove/unset property */ #define _NET_WM_STATE_REMOVE 0 // remove/unset property
#define _NET_WM_STATE_ADD 1 /* add/set property */ #define _NET_WM_STATE_ADD 1 // add/set property
#define _NET_WM_STATE_TOGGLE 2 /* toggle property */ #define _NET_WM_STATE_TOGGLE 2 // toggle property
namespace X11Utils namespace X11Utils
{ {
@ -74,10 +74,15 @@ class XRRConfiguration
private: private:
Display *dpy; Display *dpy;
Window win; Window win;
XRRScreenConfiguration *screenConfig; int screen;
Rotation screenRotation; XRRScreenResources *screenResources;
int deskSize, fullSize; XRROutputInfo *outputInfo;
XRRCrtcInfo *crtcInfo;
RRMode fullMode;
int fb_width, fb_height, fb_width_mm, fb_height_mm;
int fs_fb_width, fs_fb_height, fs_fb_width_mm, fs_fb_height_mm;
bool bValid; bool bValid;
bool bIsFullscreen;
}; };
#endif #endif