xserver-multidpi/hw/xfree86/parser/Monitor.c

862 lines
29 KiB
C

/*
*
* Copyright (c) 1997 Metro Link Incorporated
*
* 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 X CONSORTIUM 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 Metro Link shall not be
* used in advertising or otherwise to promote the sale, use or other dealings
* in this Software without prior written authorization from Metro Link.
*
*/
/*
* Copyright (c) 1997-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).
*/
#ifdef HAVE_XORG_CONFIG_H
#include <xorg-config.h>
#endif
#include "xf86Parser.h"
#include "xf86tokens.h"
#include "Configint.h"
static const xf86ConfigSymTabRec MonitorTab[] = {
{ENDSECTION, "endsection"},
{IDENTIFIER, "identifier"},
{VENDOR, "vendorname"},
{MODEL, "modelname"},
{USEMODES, "usemodes"},
{MODELINE, "modeline"},
{DISPLAYSIZE, "displaysize"},
{HORIZSYNC, "horizsync"},
{VERTREFRESH, "vertrefresh"},
{MODE, "mode"},
{GAMMA, "gamma"},
{OPTION, "option"},
{-1, ""},
};
static const xf86ConfigSymTabRec ModesTab[] = {
{ENDSECTION, "endsection"},
{IDENTIFIER, "identifier"},
{MODELINE, "modeline"},
{MODE, "mode"},
{-1, ""},
};
static const xf86ConfigSymTabRec TimingTab[] = {
{TT_INTERLACE, "interlace"},
{TT_PHSYNC, "+hsync"},
{TT_NHSYNC, "-hsync"},
{TT_PVSYNC, "+vsync"},
{TT_NVSYNC, "-vsync"},
{TT_CSYNC, "composite"},
{TT_PCSYNC, "+csync"},
{TT_NCSYNC, "-csync"},
{TT_DBLSCAN, "doublescan"},
{TT_HSKEW, "hskew"},
{TT_BCAST, "bcast"},
{TT_VSCAN, "vscan"},
{-1, ""},
};
static const xf86ConfigSymTabRec ModeTab[] = {
{DOTCLOCK, "dotclock"},
{HTIMINGS, "htimings"},
{VTIMINGS, "vtimings"},
{FLAGS, "flags"},
{HSKEW, "hskew"},
{BCAST, "bcast"},
{VSCAN, "vscan"},
{ENDMODE, "endmode"},
{-1, ""},
};
#define CLEANUP xf86freeModeLineList
static void
xf86freeModeLineList(XF86ConfModeLinePtr ptr)
{
XF86ConfModeLinePtr prev;
while (ptr) {
TestFree(ptr->ml_identifier);
TestFree(ptr->ml_comment);
prev = ptr;
ptr = ptr->list.next;
free(prev);
}
}
static XF86ConfModeLinePtr
xf86parseModeLine(void)
{
int token;
parsePrologue(XF86ConfModeLinePtr, XF86ConfModeLineRec)
/* Identifier */
if (xf86getSubToken(&(ptr->ml_comment)) != STRING)
Error("ModeLine identifier expected");
ptr->ml_identifier = xf86_lex_val.str;
/* DotClock */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine dotclock expected");
ptr->ml_clock = (int) (xf86_lex_val.realnum * 1000.0 + 0.5);
/* HDisplay */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine Hdisplay expected");
ptr->ml_hdisplay = xf86_lex_val.num;
/* HSyncStart */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine HSyncStart expected");
ptr->ml_hsyncstart = xf86_lex_val.num;
/* HSyncEnd */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine HSyncEnd expected");
ptr->ml_hsyncend = xf86_lex_val.num;
/* HTotal */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine HTotal expected");
ptr->ml_htotal = xf86_lex_val.num;
/* VDisplay */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine Vdisplay expected");
ptr->ml_vdisplay = xf86_lex_val.num;
/* VSyncStart */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine VSyncStart expected");
ptr->ml_vsyncstart = xf86_lex_val.num;
/* VSyncEnd */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine VSyncEnd expected");
ptr->ml_vsyncend = xf86_lex_val.num;
/* VTotal */
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("ModeLine VTotal expected");
ptr->ml_vtotal = xf86_lex_val.num;
token = xf86getSubTokenWithTab(&(ptr->ml_comment), TimingTab);
while ((token == TT_INTERLACE) || (token == TT_PHSYNC) ||
(token == TT_NHSYNC) || (token == TT_PVSYNC) ||
(token == TT_NVSYNC) || (token == TT_CSYNC) ||
(token == TT_PCSYNC) || (token == TT_NCSYNC) ||
(token == TT_DBLSCAN) || (token == TT_HSKEW) ||
(token == TT_VSCAN) || (token == TT_BCAST)) {
switch (token) {
case TT_INTERLACE:
ptr->ml_flags |= XF86CONF_INTERLACE;
break;
case TT_PHSYNC:
ptr->ml_flags |= XF86CONF_PHSYNC;
break;
case TT_NHSYNC:
ptr->ml_flags |= XF86CONF_NHSYNC;
break;
case TT_PVSYNC:
ptr->ml_flags |= XF86CONF_PVSYNC;
break;
case TT_NVSYNC:
ptr->ml_flags |= XF86CONF_NVSYNC;
break;
case TT_CSYNC:
ptr->ml_flags |= XF86CONF_CSYNC;
break;
case TT_PCSYNC:
ptr->ml_flags |= XF86CONF_PCSYNC;
break;
case TT_NCSYNC:
ptr->ml_flags |= XF86CONF_NCSYNC;
break;
case TT_DBLSCAN:
ptr->ml_flags |= XF86CONF_DBLSCAN;
break;
case TT_HSKEW:
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error(NUMBER_MSG, "Hskew");
ptr->ml_hskew = xf86_lex_val.num;
ptr->ml_flags |= XF86CONF_HSKEW;
break;
case TT_BCAST:
ptr->ml_flags |= XF86CONF_BCAST;
break;
case TT_VSCAN:
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error(NUMBER_MSG, "Vscan");
ptr->ml_vscan = xf86_lex_val.num;
ptr->ml_flags |= XF86CONF_VSCAN;
break;
case EOF_TOKEN:
Error(UNEXPECTED_EOF_MSG);
break;
default:
Error(INVALID_KEYWORD_MSG, xf86tokenString());
break;
}
token = xf86getSubTokenWithTab(&(ptr->ml_comment), TimingTab);
}
xf86unGetToken(token);
#ifdef DEBUG
printf("ModeLine parsed\n");
#endif
return ptr;
}
static XF86ConfModeLinePtr
xf86parseVerboseMode(void)
{
int token, token2;
int had_dotclock = 0, had_htimings = 0, had_vtimings = 0;
parsePrologue(XF86ConfModeLinePtr, XF86ConfModeLineRec)
if (xf86getSubToken(&(ptr->ml_comment)) != STRING)
Error("Mode name expected");
ptr->ml_identifier = xf86_lex_val.str;
while ((token = xf86getToken(ModeTab)) != ENDMODE) {
switch (token) {
case COMMENT:
ptr->ml_comment = xf86addComment(ptr->ml_comment, xf86_lex_val.str);
break;
case DOTCLOCK:
if ((token = xf86getSubToken(&(ptr->ml_comment))) != NUMBER)
Error(NUMBER_MSG, "DotClock");
ptr->ml_clock = (int) (xf86_lex_val.realnum * 1000.0 + 0.5);
had_dotclock = 1;
break;
case HTIMINGS:
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_hdisplay = xf86_lex_val.num;
else
Error("Horizontal display expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_hsyncstart = xf86_lex_val.num;
else
Error("Horizontal sync start expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_hsyncend = xf86_lex_val.num;
else
Error("Horizontal sync end expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_htotal = xf86_lex_val.num;
else
Error("Horizontal total expected");
had_htimings = 1;
break;
case VTIMINGS:
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_vdisplay = xf86_lex_val.num;
else
Error("Vertical display expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_vsyncstart = xf86_lex_val.num;
else
Error("Vertical sync start expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_vsyncend = xf86_lex_val.num;
else
Error("Vertical sync end expected");
if (xf86getSubToken(&(ptr->ml_comment)) == NUMBER)
ptr->ml_vtotal = xf86_lex_val.num;
else
Error("Vertical total expected");
had_vtimings = 1;
break;
case FLAGS:
token = xf86getSubToken(&(ptr->ml_comment));
if (token != STRING)
Error(QUOTE_MSG, "Flags");
while (token == STRING) {
token2 = xf86getStringToken(TimingTab);
switch (token2) {
case TT_INTERLACE:
ptr->ml_flags |= XF86CONF_INTERLACE;
break;
case TT_PHSYNC:
ptr->ml_flags |= XF86CONF_PHSYNC;
break;
case TT_NHSYNC:
ptr->ml_flags |= XF86CONF_NHSYNC;
break;
case TT_PVSYNC:
ptr->ml_flags |= XF86CONF_PVSYNC;
break;
case TT_NVSYNC:
ptr->ml_flags |= XF86CONF_NVSYNC;
break;
case TT_CSYNC:
ptr->ml_flags |= XF86CONF_CSYNC;
break;
case TT_PCSYNC:
ptr->ml_flags |= XF86CONF_PCSYNC;
break;
case TT_NCSYNC:
ptr->ml_flags |= XF86CONF_NCSYNC;
break;
case TT_DBLSCAN:
ptr->ml_flags |= XF86CONF_DBLSCAN;
break;
case EOF_TOKEN:
Error(UNEXPECTED_EOF_MSG);
break;
default:
Error("Unknown flag string");
break;
}
token = xf86getSubToken(&(ptr->ml_comment));
}
xf86unGetToken(token);
break;
case HSKEW:
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("Horizontal skew expected");
ptr->ml_flags |= XF86CONF_HSKEW;
ptr->ml_hskew = xf86_lex_val.num;
break;
case VSCAN:
if (xf86getSubToken(&(ptr->ml_comment)) != NUMBER)
Error("Vertical scan count expected");
ptr->ml_flags |= XF86CONF_VSCAN;
ptr->ml_vscan = xf86_lex_val.num;
break;
case EOF_TOKEN:
Error(UNEXPECTED_EOF_MSG);
break;
default:
Error("Unexpected token in verbose \"Mode\" entry\n");
}
}
if (!had_dotclock)
Error("the dotclock is missing");
if (!had_htimings)
Error("the horizontal timings are missing");
if (!had_vtimings)
Error("the vertical timings are missing");
#ifdef DEBUG
printf("Verbose Mode parsed\n");
#endif
return ptr;
}
#undef CLEANUP
#define CLEANUP xf86freeMonitorList
XF86ConfMonitorPtr
xf86parseMonitorSection(void)
{
int has_ident = FALSE;
int token;
parsePrologue(XF86ConfMonitorPtr, XF86ConfMonitorRec)
while ((token = xf86getToken(MonitorTab)) != ENDSECTION) {
switch (token) {
case COMMENT:
ptr->mon_comment = xf86addComment(ptr->mon_comment, xf86_lex_val.str);
break;
case IDENTIFIER:
if (xf86getSubToken(&(ptr->mon_comment)) != STRING)
Error(QUOTE_MSG, "Identifier");
if (has_ident == TRUE)
Error(MULTIPLE_MSG, "Identifier");
ptr->mon_identifier = xf86_lex_val.str;
has_ident = TRUE;
break;
case VENDOR:
if (xf86getSubToken(&(ptr->mon_comment)) != STRING)
Error(QUOTE_MSG, "Vendor");
ptr->mon_vendor = xf86_lex_val.str;
break;
case MODEL:
if (xf86getSubToken(&(ptr->mon_comment)) != STRING)
Error(QUOTE_MSG, "ModelName");
ptr->mon_modelname = xf86_lex_val.str;
break;
case MODE:
HANDLE_LIST(mon_modeline_lst, xf86parseVerboseMode,
XF86ConfModeLinePtr);
break;
case MODELINE:
HANDLE_LIST(mon_modeline_lst, xf86parseModeLine,
XF86ConfModeLinePtr);
break;
case DISPLAYSIZE:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER)
Error(DISPLAYSIZE_MSG);
ptr->mon_width = xf86_lex_val.realnum;
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER)
Error(DISPLAYSIZE_MSG);
ptr->mon_height = xf86_lex_val.realnum;
break;
case HORIZSYNC:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER)
Error(HORIZSYNC_MSG);
do {
if (ptr->mon_n_hsync >= CONF_MAX_HSYNC)
Error("Sorry. Too many horizontal sync intervals.");
ptr->mon_hsync[ptr->mon_n_hsync].lo = xf86_lex_val.realnum;
switch (token = xf86getSubToken(&(ptr->mon_comment))) {
case COMMA:
ptr->mon_hsync[ptr->mon_n_hsync].hi =
ptr->mon_hsync[ptr->mon_n_hsync].lo;
break;
case DASH:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER ||
(float) xf86_lex_val.realnum <
ptr->mon_hsync[ptr->mon_n_hsync].lo)
Error(HORIZSYNC_MSG);
ptr->mon_hsync[ptr->mon_n_hsync].hi = xf86_lex_val.realnum;
if ((token = xf86getSubToken(&(ptr->mon_comment))) == COMMA)
break;
ptr->mon_n_hsync++;
goto HorizDone;
default:
/* We cannot currently know if a '\n' was found,
* or this is a real error
*/
ptr->mon_hsync[ptr->mon_n_hsync].hi =
ptr->mon_hsync[ptr->mon_n_hsync].lo;
ptr->mon_n_hsync++;
goto HorizDone;
}
ptr->mon_n_hsync++;
} while ((token = xf86getSubToken(&(ptr->mon_comment))) == NUMBER);
HorizDone:
xf86unGetToken(token);
break;
case VERTREFRESH:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER)
Error(VERTREFRESH_MSG);
do {
ptr->mon_vrefresh[ptr->mon_n_vrefresh].lo = xf86_lex_val.realnum;
switch (token = xf86getSubToken(&(ptr->mon_comment))) {
case COMMA:
ptr->mon_vrefresh[ptr->mon_n_vrefresh].hi =
ptr->mon_vrefresh[ptr->mon_n_vrefresh].lo;
break;
case DASH:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER ||
(float) xf86_lex_val.realnum <
ptr->mon_vrefresh[ptr->mon_n_vrefresh].lo)
Error(VERTREFRESH_MSG);
ptr->mon_vrefresh[ptr->mon_n_vrefresh].hi = xf86_lex_val.realnum;
if ((token = xf86getSubToken(&(ptr->mon_comment))) == COMMA)
break;
ptr->mon_n_vrefresh++;
goto VertDone;
default:
/* We cannot currently know if a '\n' was found,
* or this is a real error
*/
ptr->mon_vrefresh[ptr->mon_n_vrefresh].hi =
ptr->mon_vrefresh[ptr->mon_n_vrefresh].lo;
ptr->mon_n_vrefresh++;
goto VertDone;
}
if (ptr->mon_n_vrefresh >= CONF_MAX_VREFRESH)
Error("Sorry. Too many vertical refresh intervals.");
ptr->mon_n_vrefresh++;
} while ((token = xf86getSubToken(&(ptr->mon_comment))) == NUMBER);
VertDone:
xf86unGetToken(token);
break;
case GAMMA:
if (xf86getSubToken(&(ptr->mon_comment)) != NUMBER) {
Error(INVALID_GAMMA_MSG);
}
else {
ptr->mon_gamma_red = ptr->mon_gamma_green =
ptr->mon_gamma_blue = xf86_lex_val.realnum;
if (xf86getSubToken(&(ptr->mon_comment)) == NUMBER) {
ptr->mon_gamma_green = xf86_lex_val.realnum;
if (xf86getSubToken(&(ptr->mon_comment)) == NUMBER) {
ptr->mon_gamma_blue = xf86_lex_val.realnum;
}
else {
Error(INVALID_GAMMA_MSG);
}
}
else
xf86unGetToken(token);
}
break;
case OPTION:
ptr->mon_option_lst = xf86parseOption(ptr->mon_option_lst);
break;
case USEMODES:
{
XF86ConfModesLinkPtr mptr;
if ((token = xf86getSubToken(&(ptr->mon_comment))) != STRING)
Error(QUOTE_MSG, "UseModes");
/* add to the end of the list of modes sections
referenced here */
mptr = calloc(1, sizeof(XF86ConfModesLinkRec));
mptr->list.next = NULL;
mptr->ml_modes_str = xf86_lex_val.str;
mptr->ml_modes = NULL;
ptr->mon_modes_sect_lst = (XF86ConfModesLinkPtr)
xf86addListItem((GenericListPtr) ptr->mon_modes_sect_lst,
(GenericListPtr) mptr);
}
break;
case EOF_TOKEN:
Error(UNEXPECTED_EOF_MSG);
break;
default:
xf86parseError(INVALID_KEYWORD_MSG, xf86tokenString());
CLEANUP(ptr);
return NULL;
break;
}
}
if (!has_ident)
Error(NO_IDENT_MSG);
#ifdef DEBUG
printf("Monitor section parsed\n");
#endif
return ptr;
}
#undef CLEANUP
#define CLEANUP xf86freeModesList
XF86ConfModesPtr
xf86parseModesSection(void)
{
int has_ident = FALSE;
int token;
parsePrologue(XF86ConfModesPtr, XF86ConfModesRec)
while ((token = xf86getToken(ModesTab)) != ENDSECTION) {
switch (token) {
case COMMENT:
ptr->modes_comment = xf86addComment(ptr->modes_comment, xf86_lex_val.str);
break;
case IDENTIFIER:
if (xf86getSubToken(&(ptr->modes_comment)) != STRING)
Error(QUOTE_MSG, "Identifier");
if (has_ident == TRUE)
Error(MULTIPLE_MSG, "Identifier");
ptr->modes_identifier = xf86_lex_val.str;
has_ident = TRUE;
break;
case MODE:
HANDLE_LIST(mon_modeline_lst, xf86parseVerboseMode,
XF86ConfModeLinePtr);
break;
case MODELINE:
HANDLE_LIST(mon_modeline_lst, xf86parseModeLine,
XF86ConfModeLinePtr);
break;
default:
xf86parseError(INVALID_KEYWORD_MSG, xf86tokenString());
CLEANUP(ptr);
return NULL;
break;
}
}
if (!has_ident)
Error(NO_IDENT_MSG);
#ifdef DEBUG
printf("Modes section parsed\n");
#endif
return ptr;
}
#undef CLEANUP
void
xf86printMonitorSection(FILE * cf, XF86ConfMonitorPtr ptr)
{
int i;
XF86ConfModeLinePtr mlptr;
XF86ConfModesLinkPtr mptr;
while (ptr) {
mptr = ptr->mon_modes_sect_lst;
fprintf(cf, "Section \"Monitor\"\n");
if (ptr->mon_comment)
fprintf(cf, "%s", ptr->mon_comment);
if (ptr->mon_identifier)
fprintf(cf, "\tIdentifier \"%s\"\n", ptr->mon_identifier);
if (ptr->mon_vendor)
fprintf(cf, "\tVendorName \"%s\"\n", ptr->mon_vendor);
if (ptr->mon_modelname)
fprintf(cf, "\tModelName \"%s\"\n", ptr->mon_modelname);
while (mptr) {
fprintf(cf, "\tUseModes \"%s\"\n", mptr->ml_modes_str);
mptr = mptr->list.next;
}
if (ptr->mon_width)
fprintf(cf, "\tDisplaySize %d\t%d\n",
ptr->mon_width, ptr->mon_height);
for (i = 0; i < ptr->mon_n_hsync; i++) {
fprintf(cf, "\tHorizSync %2.1f - %2.1f\n",
ptr->mon_hsync[i].lo, ptr->mon_hsync[i].hi);
}
for (i = 0; i < ptr->mon_n_vrefresh; i++) {
fprintf(cf, "\tVertRefresh %2.1f - %2.1f\n",
ptr->mon_vrefresh[i].lo, ptr->mon_vrefresh[i].hi);
}
if (ptr->mon_gamma_red) {
if (ptr->mon_gamma_red == ptr->mon_gamma_green
&& ptr->mon_gamma_red == ptr->mon_gamma_blue) {
fprintf(cf, "\tGamma %.4g\n", ptr->mon_gamma_red);
}
else {
fprintf(cf, "\tGamma %.4g %.4g %.4g\n",
ptr->mon_gamma_red,
ptr->mon_gamma_green, ptr->mon_gamma_blue);
}
}
for (mlptr = ptr->mon_modeline_lst; mlptr; mlptr = mlptr->list.next) {
fprintf(cf, "\tModeLine \"%s\" %2.1f ",
mlptr->ml_identifier, mlptr->ml_clock / 1000.0);
fprintf(cf, "%d %d %d %d %d %d %d %d",
mlptr->ml_hdisplay, mlptr->ml_hsyncstart,
mlptr->ml_hsyncend, mlptr->ml_htotal,
mlptr->ml_vdisplay, mlptr->ml_vsyncstart,
mlptr->ml_vsyncend, mlptr->ml_vtotal);
if (mlptr->ml_flags & XF86CONF_PHSYNC)
fprintf(cf, " +hsync");
if (mlptr->ml_flags & XF86CONF_NHSYNC)
fprintf(cf, " -hsync");
if (mlptr->ml_flags & XF86CONF_PVSYNC)
fprintf(cf, " +vsync");
if (mlptr->ml_flags & XF86CONF_NVSYNC)
fprintf(cf, " -vsync");
if (mlptr->ml_flags & XF86CONF_INTERLACE)
fprintf(cf, " interlace");
if (mlptr->ml_flags & XF86CONF_CSYNC)
fprintf(cf, " composite");
if (mlptr->ml_flags & XF86CONF_PCSYNC)
fprintf(cf, " +csync");
if (mlptr->ml_flags & XF86CONF_NCSYNC)
fprintf(cf, " -csync");
if (mlptr->ml_flags & XF86CONF_DBLSCAN)
fprintf(cf, " doublescan");
if (mlptr->ml_flags & XF86CONF_HSKEW)
fprintf(cf, " hskew %d", mlptr->ml_hskew);
if (mlptr->ml_flags & XF86CONF_BCAST)
fprintf(cf, " bcast");
fprintf(cf, "\n");
}
xf86printOptionList(cf, ptr->mon_option_lst, 1);
fprintf(cf, "EndSection\n\n");
ptr = ptr->list.next;
}
}
void
xf86printModesSection(FILE * cf, XF86ConfModesPtr ptr)
{
XF86ConfModeLinePtr mlptr;
while (ptr) {
fprintf(cf, "Section \"Modes\"\n");
if (ptr->modes_comment)
fprintf(cf, "%s", ptr->modes_comment);
if (ptr->modes_identifier)
fprintf(cf, "\tIdentifier \"%s\"\n", ptr->modes_identifier);
for (mlptr = ptr->mon_modeline_lst; mlptr; mlptr = mlptr->list.next) {
fprintf(cf, "\tModeLine \"%s\" %2.1f ",
mlptr->ml_identifier, mlptr->ml_clock / 1000.0);
fprintf(cf, "%d %d %d %d %d %d %d %d",
mlptr->ml_hdisplay, mlptr->ml_hsyncstart,
mlptr->ml_hsyncend, mlptr->ml_htotal,
mlptr->ml_vdisplay, mlptr->ml_vsyncstart,
mlptr->ml_vsyncend, mlptr->ml_vtotal);
if (mlptr->ml_flags & XF86CONF_PHSYNC)
fprintf(cf, " +hsync");
if (mlptr->ml_flags & XF86CONF_NHSYNC)
fprintf(cf, " -hsync");
if (mlptr->ml_flags & XF86CONF_PVSYNC)
fprintf(cf, " +vsync");
if (mlptr->ml_flags & XF86CONF_NVSYNC)
fprintf(cf, " -vsync");
if (mlptr->ml_flags & XF86CONF_INTERLACE)
fprintf(cf, " interlace");
if (mlptr->ml_flags & XF86CONF_CSYNC)
fprintf(cf, " composite");
if (mlptr->ml_flags & XF86CONF_PCSYNC)
fprintf(cf, " +csync");
if (mlptr->ml_flags & XF86CONF_NCSYNC)
fprintf(cf, " -csync");
if (mlptr->ml_flags & XF86CONF_DBLSCAN)
fprintf(cf, " doublescan");
if (mlptr->ml_flags & XF86CONF_HSKEW)
fprintf(cf, " hskew %d", mlptr->ml_hskew);
if (mlptr->ml_flags & XF86CONF_VSCAN)
fprintf(cf, " vscan %d", mlptr->ml_vscan);
if (mlptr->ml_flags & XF86CONF_BCAST)
fprintf(cf, " bcast");
if (mlptr->ml_comment)
fprintf(cf, "%s", mlptr->ml_comment);
else
fprintf(cf, "\n");
}
fprintf(cf, "EndSection\n\n");
ptr = ptr->list.next;
}
}
void
xf86freeMonitorList(XF86ConfMonitorPtr ptr)
{
XF86ConfMonitorPtr prev;
while (ptr) {
TestFree(ptr->mon_identifier);
TestFree(ptr->mon_vendor);
TestFree(ptr->mon_modelname);
TestFree(ptr->mon_comment);
xf86optionListFree(ptr->mon_option_lst);
xf86freeModeLineList(ptr->mon_modeline_lst);
prev = ptr;
ptr = ptr->list.next;
free(prev);
}
}
void
xf86freeModesList(XF86ConfModesPtr ptr)
{
XF86ConfModesPtr prev;
while (ptr) {
TestFree(ptr->modes_identifier);
TestFree(ptr->modes_comment);
xf86freeModeLineList(ptr->mon_modeline_lst);
prev = ptr;
ptr = ptr->list.next;
free(prev);
}
}
XF86ConfMonitorPtr
xf86findMonitor(const char *ident, XF86ConfMonitorPtr p)
{
while (p) {
if (xf86nameCompare(ident, p->mon_identifier) == 0)
return p;
p = p->list.next;
}
return NULL;
}
XF86ConfModesPtr
xf86findModes(const char *ident, XF86ConfModesPtr p)
{
while (p) {
if (xf86nameCompare(ident, p->modes_identifier) == 0)
return p;
p = p->list.next;
}
return NULL;
}
XF86ConfModeLinePtr
xf86findModeLine(const char *ident, XF86ConfModeLinePtr p)
{
while (p) {
if (xf86nameCompare(ident, p->ml_identifier) == 0)
return p;
p = p->list.next;
}
return NULL;
}
int
xf86validateMonitor(XF86ConfigPtr p, XF86ConfScreenPtr screen)
{
XF86ConfMonitorPtr monitor = screen->scrn_monitor;
XF86ConfModesLinkPtr modeslnk = monitor->mon_modes_sect_lst;
XF86ConfModesPtr modes;
while (modeslnk) {
modes = xf86findModes(modeslnk->ml_modes_str, p->conf_modes_lst);
if (!modes) {
xf86validationError(UNDEFINED_MODES_MSG,
modeslnk->ml_modes_str,
screen->scrn_identifier);
return FALSE;
}
modeslnk->ml_modes = modes;
modeslnk = modeslnk->list.next;
}
return TRUE;
}