You cannot select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
464 lines
12 KiB
C
464 lines
12 KiB
C
/*
|
|
* Copyright (c) 1999-2003 by The XFree86 Project, Inc.
|
|
*
|
|
* Permission is hereby granted, free of charge, to any person obtaining a
|
|
* copy of this software and associated documentation files (the "Software"),
|
|
* to deal in the Software without restriction, including without limitation
|
|
* the rights to use, copy, modify, merge, publish, distribute, sublicense,
|
|
* and/or sell copies of the Software, and to permit persons to whom the
|
|
* Software is furnished to do so, subject to the following conditions:
|
|
*
|
|
* The above copyright notice and this permission notice shall be included in
|
|
* all copies or substantial portions of the Software.
|
|
*
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL
|
|
* THE COPYRIGHT HOLDER(S) OR AUTHOR(S) BE LIABLE FOR ANY CLAIM, DAMAGES OR
|
|
* OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE,
|
|
* ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
*
|
|
* Except as contained in this notice, the name of the copyright holder(s)
|
|
* and author(s) shall not be used in advertising or otherwise to promote
|
|
* the sale, use or other dealings in this Software without prior written
|
|
* authorization from the copyright holder(s) and author(s).
|
|
*/
|
|
|
|
/*
|
|
* This file contains the VidMode functions required by the extension.
|
|
* These have been added to avoid the need for the higher level extension
|
|
* code to access the private XFree86 data structures directly. Wherever
|
|
* possible this code uses the functions in xf86Mode.c to do the work,
|
|
* so that two version of code that do similar things don't have to be
|
|
* maintained.
|
|
*/
|
|
|
|
#ifdef HAVE_XORG_CONFIG_H
|
|
#include <xorg-config.h>
|
|
#endif
|
|
|
|
#include <X11/X.h>
|
|
#include "os.h"
|
|
#include "xf86.h"
|
|
#include "xf86Priv.h"
|
|
#include "extinit.h"
|
|
|
|
#ifdef XF86VIDMODE
|
|
#include "vidmodestr.h"
|
|
#include "xf86Privstr.h"
|
|
#include "xf86Extensions.h"
|
|
#include "xf86cmap.h"
|
|
|
|
static vidMonitorValue
|
|
xf86VidModeGetMonitorValue(ScreenPtr pScreen, int valtyp, int indx)
|
|
{
|
|
vidMonitorValue ret = { NULL, };
|
|
MonPtr monitor;
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
monitor = pScrn->monitor;
|
|
|
|
switch (valtyp) {
|
|
case VIDMODE_MON_VENDOR:
|
|
ret.ptr = monitor->vendor;
|
|
break;
|
|
case VIDMODE_MON_MODEL:
|
|
ret.ptr = monitor->model;
|
|
break;
|
|
case VIDMODE_MON_NHSYNC:
|
|
ret.i = monitor->nHsync;
|
|
break;
|
|
case VIDMODE_MON_NVREFRESH:
|
|
ret.i = monitor->nVrefresh;
|
|
break;
|
|
case VIDMODE_MON_HSYNC_LO:
|
|
ret.f = (100.0 * monitor->hsync[indx].lo);
|
|
break;
|
|
case VIDMODE_MON_HSYNC_HI:
|
|
ret.f = (100.0 * monitor->hsync[indx].hi);
|
|
break;
|
|
case VIDMODE_MON_VREFRESH_LO:
|
|
ret.f = (100.0 * monitor->vrefresh[indx].lo);
|
|
break;
|
|
case VIDMODE_MON_VREFRESH_HI:
|
|
ret.f = (100.0 * monitor->vrefresh[indx].hi);
|
|
break;
|
|
}
|
|
return ret;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetCurrentModeline(ScreenPtr pScreen, DisplayModePtr *mode, int *dotClock)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
|
|
if (pScrn->currentMode) {
|
|
*mode = pScrn->currentMode;
|
|
*dotClock = pScrn->currentMode->Clock;
|
|
|
|
return TRUE;
|
|
}
|
|
return FALSE;
|
|
}
|
|
|
|
static int
|
|
xf86VidModeGetDotClock(ScreenPtr pScreen, int Clock)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
if ((pScrn->progClock) || (Clock >= MAXCLOCKS))
|
|
return Clock;
|
|
else
|
|
return pScrn->clock[Clock];
|
|
}
|
|
|
|
static int
|
|
xf86VidModeGetNumOfClocks(ScreenPtr pScreen, Bool *progClock)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
if (pScrn->progClock) {
|
|
*progClock = TRUE;
|
|
return 0;
|
|
}
|
|
else {
|
|
*progClock = FALSE;
|
|
return pScrn->numClocks;
|
|
}
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetClocks(ScreenPtr pScreen, int *Clocks)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
int i;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
|
|
if (pScrn->progClock)
|
|
return FALSE;
|
|
|
|
for (i = 0; i < pScrn->numClocks; i++)
|
|
*Clocks++ = pScrn->clock[i];
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetNextModeline(ScreenPtr pScreen, DisplayModePtr *mode, int *dotClock)
|
|
{
|
|
VidModePtr pVidMode;
|
|
DisplayModePtr p;
|
|
|
|
pVidMode = VidModeGetPtr(pScreen);
|
|
|
|
for (p = pVidMode->Next; p != NULL && p != pVidMode->First; p = p->next) {
|
|
if (p->status == MODE_OK) {
|
|
pVidMode->Next = p->next;
|
|
*mode = p;
|
|
*dotClock = xf86VidModeGetDotClock(pScreen, p->Clock);
|
|
return TRUE;
|
|
}
|
|
}
|
|
|
|
return FALSE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetFirstModeline(ScreenPtr pScreen, DisplayModePtr *mode, int *dotClock)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
VidModePtr pVidMode;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
if (pScrn->modes == NULL)
|
|
return FALSE;
|
|
|
|
pVidMode = VidModeGetPtr(pScreen);
|
|
pVidMode->First = pScrn->modes;
|
|
pVidMode->Next = pVidMode->First->next;
|
|
|
|
if (pVidMode->First->status == MODE_OK) {
|
|
*mode = pVidMode->First;
|
|
*dotClock = xf86VidModeGetDotClock(pScreen, pVidMode->First->Clock);
|
|
return TRUE;
|
|
}
|
|
|
|
return xf86VidModeGetNextModeline(pScreen, mode, dotClock);
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeDeleteModeline(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
if (mode == NULL)
|
|
return FALSE;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
xf86DeleteMode(&(pScrn->modes), mode);
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeZoomViewport(ScreenPtr pScreen, int zoom)
|
|
{
|
|
xf86ZoomViewport(pScreen, zoom);
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeSetViewPort(ScreenPtr pScreen, int x, int y)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
pScrn->frameX0 = min(max(x, 0),
|
|
pScrn->virtualX - pScrn->currentMode->HDisplay);
|
|
pScrn->frameX1 = pScrn->frameX0 + pScrn->currentMode->HDisplay - 1;
|
|
pScrn->frameY0 = min(max(y, 0),
|
|
pScrn->virtualY - pScrn->currentMode->VDisplay);
|
|
pScrn->frameY1 = pScrn->frameY0 + pScrn->currentMode->VDisplay - 1;
|
|
if (pScrn->AdjustFrame != NULL)
|
|
(pScrn->AdjustFrame) (pScrn, pScrn->frameX0, pScrn->frameY0);
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetViewPort(ScreenPtr pScreen, int *x, int *y)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
*x = pScrn->frameX0;
|
|
*y = pScrn->frameY0;
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeSwitchMode(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
DisplayModePtr pTmpMode;
|
|
Bool retval;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
/* save in case we fail */
|
|
pTmpMode = pScrn->currentMode;
|
|
/* Force a mode switch */
|
|
pScrn->currentMode = NULL;
|
|
retval = xf86SwitchMode(pScrn->pScreen, mode);
|
|
/* we failed: restore it */
|
|
if (retval == FALSE)
|
|
pScrn->currentMode = pTmpMode;
|
|
return retval;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeLockZoom(ScreenPtr pScreen, Bool lock)
|
|
{
|
|
if (xf86Info.dontZoom)
|
|
return FALSE;
|
|
|
|
xf86LockZoom(pScreen, lock);
|
|
return TRUE;
|
|
}
|
|
|
|
static ModeStatus
|
|
xf86VidModeCheckModeForMonitor(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
if (mode == NULL)
|
|
return MODE_ERROR;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
|
|
return xf86CheckModeForMonitor(mode, pScrn->monitor);
|
|
}
|
|
|
|
static ModeStatus
|
|
xf86VidModeCheckModeForDriver(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
if (mode == NULL)
|
|
return MODE_ERROR;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
|
|
return xf86CheckModeForDriver(pScrn, mode, 0);
|
|
}
|
|
|
|
static void
|
|
xf86VidModeSetCrtcForMode(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
DisplayModePtr ScreenModes;
|
|
|
|
if (mode == NULL)
|
|
return;
|
|
|
|
/* Ugly hack so that the xf86Mode.c function can be used without change */
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
ScreenModes = pScrn->modes;
|
|
pScrn->modes = mode;
|
|
|
|
xf86SetCrtcForModes(pScrn, pScrn->adjustFlags);
|
|
pScrn->modes = ScreenModes;
|
|
return;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeAddModeline(ScreenPtr pScreen, DisplayModePtr mode)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
if (mode == NULL)
|
|
return FALSE;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
|
|
mode->name = strdup(""); /* freed by deletemode */
|
|
mode->status = MODE_OK;
|
|
mode->next = pScrn->modes->next;
|
|
mode->prev = pScrn->modes;
|
|
pScrn->modes->next = mode;
|
|
if (mode->next != NULL)
|
|
mode->next->prev = mode;
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
static int
|
|
xf86VidModeGetNumOfModes(ScreenPtr pScreen)
|
|
{
|
|
DisplayModePtr mode = NULL;
|
|
int dotClock = 0, nummodes = 0;
|
|
|
|
if (!xf86VidModeGetFirstModeline(pScreen, &mode, &dotClock))
|
|
return nummodes;
|
|
|
|
do {
|
|
nummodes++;
|
|
if (!xf86VidModeGetNextModeline(pScreen, &mode, &dotClock))
|
|
return nummodes;
|
|
} while (TRUE);
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeSetGamma(ScreenPtr pScreen, float red, float green, float blue)
|
|
{
|
|
Gamma gamma;
|
|
|
|
gamma.red = red;
|
|
gamma.green = green;
|
|
gamma.blue = blue;
|
|
if (xf86ChangeGamma(pScreen, gamma) != Success)
|
|
return FALSE;
|
|
else
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetGamma(ScreenPtr pScreen, float *red, float *green, float *blue)
|
|
{
|
|
ScrnInfoPtr pScrn;
|
|
|
|
pScrn = xf86ScreenToScrn(pScreen);
|
|
*red = pScrn->gamma.red;
|
|
*green = pScrn->gamma.green;
|
|
*blue = pScrn->gamma.blue;
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeSetGammaRamp(ScreenPtr pScreen, int size, CARD16 *r, CARD16 *g, CARD16 *b)
|
|
{
|
|
xf86ChangeGammaRamp(pScreen, size, r, g, b);
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeGetGammaRamp(ScreenPtr pScreen, int size, CARD16 *r, CARD16 *g, CARD16 *b)
|
|
{
|
|
xf86GetGammaRamp(pScreen, size, r, g, b);
|
|
return TRUE;
|
|
}
|
|
|
|
static Bool
|
|
xf86VidModeInit(ScreenPtr pScreen)
|
|
{
|
|
VidModePtr pVidMode;
|
|
|
|
if (!xf86GetVidModeEnabled()) {
|
|
DebugF("!xf86GetVidModeEnabled()\n");
|
|
return FALSE;
|
|
}
|
|
|
|
pVidMode = VidModeInit(pScreen);
|
|
if (!pVidMode)
|
|
return FALSE;
|
|
|
|
pVidMode->Flags = 0;
|
|
pVidMode->Next = NULL;
|
|
|
|
pVidMode->GetMonitorValue = xf86VidModeGetMonitorValue;
|
|
pVidMode->GetCurrentModeline = xf86VidModeGetCurrentModeline;
|
|
pVidMode->GetFirstModeline = xf86VidModeGetFirstModeline;
|
|
pVidMode->GetNextModeline = xf86VidModeGetNextModeline;
|
|
pVidMode->DeleteModeline = xf86VidModeDeleteModeline;
|
|
pVidMode->ZoomViewport = xf86VidModeZoomViewport;
|
|
pVidMode->GetViewPort = xf86VidModeGetViewPort;
|
|
pVidMode->SetViewPort = xf86VidModeSetViewPort;
|
|
pVidMode->SwitchMode = xf86VidModeSwitchMode;
|
|
pVidMode->LockZoom = xf86VidModeLockZoom;
|
|
pVidMode->GetNumOfClocks = xf86VidModeGetNumOfClocks;
|
|
pVidMode->GetClocks = xf86VidModeGetClocks;
|
|
pVidMode->CheckModeForMonitor = xf86VidModeCheckModeForMonitor;
|
|
pVidMode->CheckModeForDriver = xf86VidModeCheckModeForDriver;
|
|
pVidMode->SetCrtcForMode = xf86VidModeSetCrtcForMode;
|
|
pVidMode->AddModeline = xf86VidModeAddModeline;
|
|
pVidMode->GetDotClock = xf86VidModeGetDotClock;
|
|
pVidMode->GetNumOfModes = xf86VidModeGetNumOfModes;
|
|
pVidMode->SetGamma = xf86VidModeSetGamma;
|
|
pVidMode->GetGamma = xf86VidModeGetGamma;
|
|
pVidMode->SetGammaRamp = xf86VidModeSetGammaRamp;
|
|
pVidMode->GetGammaRamp = xf86VidModeGetGammaRamp;
|
|
pVidMode->GetGammaRampSize = xf86GetGammaRampSize; /* use xf86cmap API directly */
|
|
|
|
return TRUE;
|
|
}
|
|
|
|
void
|
|
XFree86VidModeExtensionInit(void)
|
|
{
|
|
int i;
|
|
Bool enabled = FALSE;
|
|
|
|
DebugF("XFree86VidModeExtensionInit");
|
|
|
|
/* This means that the DDX doesn't want the vidmode extension enabled */
|
|
if (!xf86GetVidModeEnabled())
|
|
return;
|
|
|
|
for (i = 0; i < screenInfo.numScreens; i++) {
|
|
if (xf86VidModeInit (screenInfo.screens[i]))
|
|
enabled = TRUE;
|
|
}
|
|
/* This means that the DDX doesn't want the vidmode extension enabled */
|
|
if (!enabled)
|
|
return;
|
|
|
|
VidModeAddExtension(xf86GetVidModeAllowNonLocal());
|
|
}
|
|
|
|
#endif /* XF86VIDMODE */
|