/* * 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 #endif #include #include "os/log_priv.h" #include "os.h" #include "xf86_priv.h" #include "xf86Priv.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(xf86Info.vidModeAllowNonLocal); } #endif /* XF86VIDMODE */