XQuartz: RandR: Better handle switching betwen RandR modes that share CG modes

Signed-off-by: Jeremy Huddleston <jeremyhu@apple.com>
This commit is contained in:
Jeremy Huddleston 2010-07-26 11:45:59 -07:00
parent 026a47e212
commit 721edc69c3
2 changed files with 123 additions and 37 deletions

View File

@ -57,9 +57,12 @@ static Bool ignore_next_fake_mode_update = FALSE;
#define DEFAULT_REFRESH 60 #define DEFAULT_REFRESH 60
#define kDisplayModeUsableFlags (kDisplayModeValidFlag | kDisplayModeSafeFlag) #define kDisplayModeUsableFlags (kDisplayModeValidFlag | kDisplayModeSafeFlag)
typedef Bool (*QuartzModeCallback) #define CALLBACK_SUCCESS 0
(ScreenPtr, CGDirectDisplayID, QuartzModeInfoPtr, void *); #define CALLBACK_CONTINUE 1
#define CALLBACK_ERROR -1
typedef int (*QuartzModeCallback)
(ScreenPtr, CGDirectDisplayID, QuartzModeInfoPtr, void *);
#if MAC_OS_X_VERSION_MIN_REQUIRED < 1060 #if MAC_OS_X_VERSION_MIN_REQUIRED < 1060
@ -97,20 +100,22 @@ static void QuartzRandRGetModeInfo (CFDictionaryRef modeRef,
pMode->ref = NULL; pMode->ref = NULL;
} }
static Bool QuartzRandRGetCurrentModeInfo (CGDirectDisplayID screenId, static Bool QuartzRandRCopyCurrentModeInfo (CGDirectDisplayID screenId,
QuartzModeInfoPtr pMode) { QuartzModeInfoPtr pMode) {
CFDictionaryRef curModeRef = CGDisplayCurrentMode(screenId); CFDictionaryRef curModeRef = CGDisplayCurrentMode(screenId);
if (!curModeRef) if (!curModeRef)
return FALSE; return FALSE;
QuartzRandRGetModeInfo(curModeRef, pMode); QuartzRandRGetModeInfo(curModeRef, pMode);
pMode->ref = (void *)curModeRef;
CFRetain(pMode->ref);
return TRUE; return TRUE;
} }
static Bool QuartzRandRSetMode (CGDirectDisplayID screenId, static Bool QuartzRandRSetMode (CGDirectDisplayID screenId,
QuartzModeInfoPtr pMode) { QuartzModeInfoPtr pMode) {
CFDictionaryRef modeRef = (CFDictionaryRef) pMode->ref; CFDictionaryRef modeRef = (CFDictionaryRef) pMode->ref;
return (CGDisplaySwitchToMode(screenId, modeRef) != kCGErrorSuccess); return (CGDisplaySwitchToMode(screenId, modeRef) == kCGErrorSuccess);
} }
static Bool QuartzRandREnumerateModes (ScreenPtr pScreen, static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
@ -122,6 +127,28 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
CFArrayRef modes; CFArrayRef modes;
QuartzModeInfo modeInfo; QuartzModeInfo modeInfo;
int i; int i;
BOOL retval = TRUE;
QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen);
switch(callback(pScreen, screenId, &pQuartzScreen->rootlessMode, data)) {
case CALLBACK_SUCCESS:
return TRUE;
case CALLBACK_ERROR:
return FALSE;
case CALLBACK_CONTINUE:
default:
break;
}
switch(callback(pScreen, screenId, &pQuartzScreen->fullScreenMode, data)) {
case CALLBACK_SUCCESS:
return TRUE;
case CALLBACK_ERROR:
return FALSE;
case CALLBACK_CONTINUE:
default:
break;
}
curModeRef = CGDisplayCurrentMode(screenId); curModeRef = CGDisplayCurrentMode(screenId);
if (!curModeRef) if (!curModeRef)
@ -132,6 +159,7 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
if (!modes) if (!modes)
return FALSE; return FALSE;
for (i = 0; i < CFArrayGetCount(modes); i++) { for (i = 0; i < CFArrayGetCount(modes); i++) {
int cb;
modeRef = (CFDictionaryRef) CFArrayGetValueAtIndex(modes, i); modeRef = (CFDictionaryRef) CFArrayGetValueAtIndex(modes, i);
/* Skip modes that are not usable on the current display or have a /* Skip modes that are not usable on the current display or have a
@ -143,11 +171,16 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
continue; continue;
QuartzRandRGetModeInfo(modeRef, &modeInfo); QuartzRandRGetModeInfo(modeRef, &modeInfo);
modeInfo.ref = modeRef; modeInfo.ref = (void *)modeRef;
if (!callback(pScreen, screenId, &modeInfo, data)) cb = callback(pScreen, screenId, &modeInfo, data);
if (cb == CALLBACK_SUCCESS)
break;
if (cb == CALLBACK_ERROR) {
retval = FALSE;
break; break;
} }
return TRUE; }
return retval;
} }
#else /* we have the new CG APIs from Snow Leopard */ #else /* we have the new CG APIs from Snow Leopard */
@ -162,14 +195,14 @@ static void QuartzRandRGetModeInfo (CGDisplayModeRef modeRef,
pMode->ref = NULL; pMode->ref = NULL;
} }
static Bool QuartzRandRGetCurrentModeInfo (CGDirectDisplayID screenId, static Bool QuartzRandRCopyCurrentModeInfo (CGDirectDisplayID screenId,
QuartzModeInfoPtr pMode) { QuartzModeInfoPtr pMode) {
CGDisplayModeRef curModeRef = CGDisplayCopyDisplayMode(screenId); CGDisplayModeRef curModeRef = CGDisplayCopyDisplayMode(screenId);
if (!curModeRef) if (!curModeRef)
return FALSE; return FALSE;
QuartzRandRGetModeInfo(curModeRef, pMode); QuartzRandRGetModeInfo(curModeRef, pMode);
CGDisplayModeRelease(curModeRef); pMode->ref = curModeRef;
return TRUE; return TRUE;
} }
@ -179,7 +212,7 @@ static Bool QuartzRandRSetMode (CGDirectDisplayID screenId,
if (!modeRef) if (!modeRef)
return FALSE; return FALSE;
return (CGDisplaySetDisplayMode(screenId, modeRef, NULL) != kCGErrorSuccess); return (CGDisplaySetDisplayMode(screenId, modeRef, NULL) == kCGErrorSuccess);
} }
static Bool QuartzRandREnumerateModes (ScreenPtr pScreen, static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
@ -192,6 +225,29 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
CFArrayRef modes; CFArrayRef modes;
QuartzModeInfo modeInfo; QuartzModeInfo modeInfo;
int i; int i;
Bool retval = TRUE;
QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen);
switch(callback(pScreen, screenId, &pQuartzScreen->rootlessMode, data)) {
case CALLBACK_SUCCESS:
return TRUE;
case CALLBACK_ERROR:
return FALSE;
case CALLBACK_CONTINUE:
default:
break;
}
switch(callback(pScreen, screenId, &pQuartzScreen->fullScreenMode, data)) {
case CALLBACK_SUCCESS:
return TRUE;
case CALLBACK_ERROR:
return FALSE;
case CALLBACK_CONTINUE:
default:
break;
}
curModeRef = CGDisplayCopyDisplayMode(screenId); curModeRef = CGDisplayCopyDisplayMode(screenId);
if (!curModeRef) if (!curModeRef)
@ -205,6 +261,7 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
return FALSE; return FALSE;
} }
for (i = 0; i < CFArrayGetCount(modes); i++) { for (i = 0; i < CFArrayGetCount(modes); i++) {
int cb;
modeRef = (CGDisplayModeRef) CFArrayGetValueAtIndex(modes, i); modeRef = (CGDisplayModeRef) CFArrayGetValueAtIndex(modes, i);
/* Skip modes that are not usable on the current display or have a /* Skip modes that are not usable on the current display or have a
@ -220,13 +277,19 @@ static Bool QuartzRandREnumerateModes (ScreenPtr pScreen,
QuartzRandRGetModeInfo(modeRef, &modeInfo); QuartzRandRGetModeInfo(modeRef, &modeInfo);
modeInfo.ref = modeRef; modeInfo.ref = modeRef;
if (!callback(pScreen, screenId, &modeInfo, data)) cb = callback(pScreen, screenId, &modeInfo, data);
if (cb == CALLBACK_SUCCESS)
break;
if (cb == CALLBACK_ERROR) {
retval = FALSE;
break; break;
} }
}
CFRelease(modes); CFRelease(modes);
CFRelease(curPixelEnc); CFRelease(curPixelEnc);
return TRUE; return retval;
} }
#endif /* Snow Leopard CoreGraphics APIs */ #endif /* Snow Leopard CoreGraphics APIs */
@ -246,6 +309,7 @@ static Bool QuartzRandRRegisterMode (ScreenPtr pScreen,
RRScreenSizePtr pSize = RRRegisterSize(pScreen, RRScreenSizePtr pSize = RRRegisterSize(pScreen,
pMode->width, pMode->height, pScreen->mmWidth, pScreen->mmHeight); pMode->width, pMode->height, pScreen->mmWidth, pScreen->mmHeight);
if (pSize) { if (pSize) {
//DEBUG_LOG("registering: %d x %d @ %d %s\n", (int)pMode->width, (int)pMode->height, (int)pMode->refresh, isCurrentMode ? "*" : "");
RRRegisterRate(pScreen, pSize, pMode->refresh); RRRegisterRate(pScreen, pSize, pMode->refresh);
if (isCurrentMode) if (isCurrentMode)
@ -256,14 +320,18 @@ static Bool QuartzRandRRegisterMode (ScreenPtr pScreen,
return FALSE; return FALSE;
} }
static Bool QuartzRandRRegisterModeCallback (ScreenPtr pScreen, static int QuartzRandRRegisterModeCallback (ScreenPtr pScreen,
CGDirectDisplayID screenId, CGDirectDisplayID screenId,
QuartzModeInfoPtr pMode, QuartzModeInfoPtr pMode,
void *data __unused) { void *data __unused) {
return QuartzRandRRegisterMode(pScreen, pMode); if(QuartzRandRRegisterMode(pScreen, pMode)) {
return CALLBACK_CONTINUE;
} else {
return CALLBACK_ERROR;
}
} }
static Bool QuartzRandRSetModeCallback (ScreenPtr pScreen, static int QuartzRandRSetModeCallback (ScreenPtr pScreen,
CGDirectDisplayID screenId, CGDirectDisplayID screenId,
QuartzModeInfoPtr pMode, QuartzModeInfoPtr pMode,
void *data) { void *data) {
@ -271,19 +339,27 @@ static Bool QuartzRandRSetModeCallback (ScreenPtr pScreen,
QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen); QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen);
if (!QuartzRandRModesEqual(pMode, pReqMode)) if (!QuartzRandRModesEqual(pMode, pReqMode))
return TRUE; /* continue enumeration */ return CALLBACK_CONTINUE; /* continue enumeration */
if (pReqMode->ref == pQuartzScreen->currentMode.ref) { DEBUG_LOG("Found a match for requested RandR resolution (%dx%d@%d).\n", (int)pMode->width, (int)pMode->height, (int)pMode->refresh);
if (pQuartzScreen->currentMode.ref && CFEqual(pMode->ref, pQuartzScreen->currentMode.ref)) {
DEBUG_LOG("Requested RandR resolution matches current CG mode\n"); DEBUG_LOG("Requested RandR resolution matches current CG mode\n");
return FALSE; /* We don't need to do anything in CG */ return CALLBACK_SUCCESS; /* We don't need to do anything in CG */
} }
if (QuartzRandRSetMode(screenId, pMode) == FALSE) { if (QuartzRandRSetMode(screenId, pMode)) {
if(pQuartzScreen->currentMode.ref)
CFRelease(pQuartzScreen->currentMode.ref);
pQuartzScreen->currentMode = *pMode;
CFRetain(pQuartzScreen->currentMode.ref);
ignore_next_fake_mode_update = TRUE; ignore_next_fake_mode_update = TRUE;
return FALSE; return CALLBACK_SUCCESS;
} else { } else {
return TRUE; DEBUG_LOG("Error while requesting CG resolution change.\n");
}; return CALLBACK_ERROR;
}
} }
static Bool QuartzRandRGetInfo (ScreenPtr pScreen, Rotation *rotations) { static Bool QuartzRandRGetInfo (ScreenPtr pScreen, Rotation *rotations) {
@ -295,17 +371,17 @@ static Bool QuartzRandRGetInfo (ScreenPtr pScreen, Rotation *rotations) {
if (pQuartzScreen->displayCount == 0) if (pQuartzScreen->displayCount == 0)
return FALSE; return FALSE;
QuartzRandRRegisterMode(pScreen, &pQuartzScreen->rootlessMode);
QuartzRandRRegisterMode(pScreen, &pQuartzScreen->fullScreenMode);
if (pQuartzScreen->displayCount > 1) { if (pQuartzScreen->displayCount > 1) {
/* RandR operations are not well-defined for an X11 screen spanning /* RandR operations are not well-defined for an X11 screen spanning
multiple CG displays. Create two entries for the current virtual multiple CG displays. Create two entries for the current virtual
resolution including/excluding the menu bar. */ resolution including/excluding the menu bar. */
QuartzRandRRegisterMode(pScreen, &pQuartzScreen->rootlessMode);
QuartzRandRRegisterMode(pScreen, &pQuartzScreen->fullScreenMode);
return TRUE; return TRUE;
} }
screenId = pQuartzScreen->displayIDs[0];
screenId = pQuartzScreen->displayIDs[0];
return QuartzRandREnumerateModes(pScreen, screenId, QuartzRandRRegisterModeCallback, NULL); return QuartzRandREnumerateModes(pScreen, screenId, QuartzRandRRegisterModeCallback, NULL);
} }
@ -323,9 +399,14 @@ static Bool QuartzRandRSetConfig (ScreenPtr pScreen,
/* If the client requested the fake rootless mode, switch to rootless. /* If the client requested the fake rootless mode, switch to rootless.
* Otherwise, force fullscreen mode. * Otherwise, force fullscreen mode.
* TODO: Refactor all this fullscreen/rootless crap as it is spaghetti
* has redundancies.
*/ */
QuartzSetFullscreen(reqMode.refresh != FAKE_REFRESH_ROOTLESS);
QuartzSetRootless(reqMode.refresh == FAKE_REFRESH_ROOTLESS); QuartzSetRootless(reqMode.refresh == FAKE_REFRESH_ROOTLESS);
QuartzSetFullscreen(reqMode.refresh != FAKE_REFRESH_ROOTLESS);
if(reqMode.refresh != FAKE_REFRESH_ROOTLESS &&
reqMode.refresh != FAKE_REFRESH_FULLSCREEN)
QuartzShow();
if (pQuartzScreen->displayCount == 0) if (pQuartzScreen->displayCount == 0)
return FALSE; return FALSE;
@ -342,12 +423,10 @@ static Bool QuartzRandRSetConfig (ScreenPtr pScreen,
screenId = pQuartzScreen->displayIDs[0]; screenId = pQuartzScreen->displayIDs[0];
if (QuartzRandREnumerateModes(pScreen, screenId, QuartzRandRSetModeCallback, &reqMode)) { if (QuartzRandREnumerateModes(pScreen, screenId, QuartzRandRSetModeCallback, &reqMode)) {
pQuartzScreen->currentMode = reqMode;
return TRUE; return TRUE;
} }
DEBUG_LOG("Unable to find a matching config: %d x %d @ %d\n", (int)reqMode.width, (int)reqMode.height, (int)reqMode.refresh); DEBUG_LOG("Unable to find a matching config: %d x %d @ %d\n", (int)reqMode.width, (int)reqMode.height, (int)reqMode.refresh);
return FALSE; return FALSE;
} }
@ -355,9 +434,16 @@ static Bool _QuartzRandRUpdateFakeModes (ScreenPtr pScreen) {
QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen); QuartzScreenPtr pQuartzScreen = QUARTZ_PRIV(pScreen);
if (pQuartzScreen->displayCount == 1) { if (pQuartzScreen->displayCount == 1) {
if (!QuartzRandRGetCurrentModeInfo(pQuartzScreen->displayIDs[0], if(pQuartzScreen->fullScreenMode.ref)
CFRelease(pQuartzScreen->fullScreenMode.ref);
if(pQuartzScreen->currentMode.ref)
CFRelease(pQuartzScreen->currentMode.ref);
if (!QuartzRandRCopyCurrentModeInfo(pQuartzScreen->displayIDs[0],
&pQuartzScreen->fullScreenMode)) &pQuartzScreen->fullScreenMode))
return FALSE; return FALSE;
CFRetain(pQuartzScreen->fullScreenMode.ref); /* This extra retain is for currentMode's copy */
} else { } else {
pQuartzScreen->fullScreenMode.width = pScreen->width; pQuartzScreen->fullScreenMode.width = pScreen->width;
pQuartzScreen->fullScreenMode.height = pScreen->height; pQuartzScreen->fullScreenMode.height = pScreen->height;

View File

@ -34,7 +34,7 @@
typedef struct { typedef struct {
size_t width, height; size_t width, height;
int refresh; int refresh;
const void *ref; void *ref; /* CGDisplayModeRef or CFDictionaryRef */
} QuartzModeInfo, *QuartzModeInfoPtr; } QuartzModeInfo, *QuartzModeInfoPtr;
// Quartz specific per screen storage structure // Quartz specific per screen storage structure