403 lines
11 KiB
C
403 lines
11 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).
|
||
|
*/
|
||
|
|
||
|
|
||
|
/* View/edit this file with tab stops set to 4 */
|
||
|
|
||
|
#ifdef HAVE_XORG_CONFIG_H
|
||
|
#include <xorg-config.h>
|
||
|
#endif
|
||
|
|
||
|
#include "xf86Parser.h"
|
||
|
#include "xf86tokens.h"
|
||
|
#include "Configint.h"
|
||
|
|
||
|
extern LexRec val;
|
||
|
|
||
|
static
|
||
|
xf86ConfigSymTabRec DeviceTab[] =
|
||
|
{
|
||
|
{ENDSECTION, "endsection"},
|
||
|
{IDENTIFIER, "identifier"},
|
||
|
{VENDOR, "vendorname"},
|
||
|
{BOARD, "boardname"},
|
||
|
{CHIPSET, "chipset"},
|
||
|
{RAMDAC, "ramdac"},
|
||
|
{DACSPEED, "dacspeed"},
|
||
|
{CLOCKS, "clocks"},
|
||
|
{OPTION, "option"},
|
||
|
{VIDEORAM, "videoram"},
|
||
|
{BIOSBASE, "biosbase"},
|
||
|
{MEMBASE, "membase"},
|
||
|
{IOBASE, "iobase"},
|
||
|
{CLOCKCHIP, "clockchip"},
|
||
|
{CHIPID, "chipid"},
|
||
|
{CHIPREV, "chiprev"},
|
||
|
{CARD, "card"},
|
||
|
{DRIVER, "driver"},
|
||
|
{BUSID, "busid"},
|
||
|
{TEXTCLOCKFRQ, "textclockfreq"},
|
||
|
{IRQ, "irq"},
|
||
|
{SCREEN, "screen"},
|
||
|
{-1, ""},
|
||
|
};
|
||
|
|
||
|
#define CLEANUP xf86freeDeviceList
|
||
|
|
||
|
XF86ConfDevicePtr
|
||
|
xf86parseDeviceSection (void)
|
||
|
{
|
||
|
int i;
|
||
|
int has_ident = FALSE;
|
||
|
int token;
|
||
|
parsePrologue (XF86ConfDevicePtr, XF86ConfDeviceRec)
|
||
|
|
||
|
/* Zero is a valid value for these */
|
||
|
ptr->dev_chipid = -1;
|
||
|
ptr->dev_chiprev = -1;
|
||
|
ptr->dev_irq = -1;
|
||
|
while ((token = xf86getToken (DeviceTab)) != ENDSECTION)
|
||
|
{
|
||
|
switch (token)
|
||
|
{
|
||
|
case COMMENT:
|
||
|
ptr->dev_comment = xf86addComment(ptr->dev_comment, val.str);
|
||
|
break;
|
||
|
case IDENTIFIER:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Identifier");
|
||
|
if (has_ident == TRUE)
|
||
|
Error (MULTIPLE_MSG, "Identifier");
|
||
|
ptr->dev_identifier = val.str;
|
||
|
has_ident = TRUE;
|
||
|
break;
|
||
|
case VENDOR:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Vendor");
|
||
|
ptr->dev_vendor = val.str;
|
||
|
break;
|
||
|
case BOARD:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Board");
|
||
|
ptr->dev_board = val.str;
|
||
|
break;
|
||
|
case CHIPSET:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Chipset");
|
||
|
ptr->dev_chipset = val.str;
|
||
|
break;
|
||
|
case CARD:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Card");
|
||
|
ptr->dev_card = val.str;
|
||
|
break;
|
||
|
case DRIVER:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Driver");
|
||
|
ptr->dev_driver = val.str;
|
||
|
break;
|
||
|
case RAMDAC:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "Ramdac");
|
||
|
ptr->dev_ramdac = val.str;
|
||
|
break;
|
||
|
case DACSPEED:
|
||
|
for (i = 0; i < CONF_MAXDACSPEEDS; i++)
|
||
|
ptr->dev_dacSpeeds[i] = 0;
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
{
|
||
|
Error (DACSPEED_MSG, CONF_MAXDACSPEEDS);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
ptr->dev_dacSpeeds[0] = (int) (val.realnum * 1000.0 + 0.5);
|
||
|
for (i = 1; i < CONF_MAXDACSPEEDS; i++)
|
||
|
{
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) == NUMBER)
|
||
|
ptr->dev_dacSpeeds[i] = (int)
|
||
|
(val.realnum * 1000.0 + 0.5);
|
||
|
else
|
||
|
{
|
||
|
xf86unGetToken (token);
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
break;
|
||
|
case VIDEORAM:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "VideoRam");
|
||
|
ptr->dev_videoram = val.num;
|
||
|
break;
|
||
|
case BIOSBASE:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "BIOSBase");
|
||
|
ptr->dev_bios_base = val.num;
|
||
|
break;
|
||
|
case MEMBASE:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "MemBase");
|
||
|
ptr->dev_mem_base = val.num;
|
||
|
break;
|
||
|
case IOBASE:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "IOBase");
|
||
|
ptr->dev_io_base = val.num;
|
||
|
break;
|
||
|
case CLOCKCHIP:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "ClockChip");
|
||
|
ptr->dev_clockchip = val.str;
|
||
|
break;
|
||
|
case CHIPID:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "ChipID");
|
||
|
ptr->dev_chipid = val.num;
|
||
|
break;
|
||
|
case CHIPREV:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "ChipRev");
|
||
|
ptr->dev_chiprev = val.num;
|
||
|
break;
|
||
|
|
||
|
case CLOCKS:
|
||
|
token = xf86getSubToken(&(ptr->dev_comment));
|
||
|
for( i = ptr->dev_clocks;
|
||
|
token == NUMBER && i < CONF_MAXCLOCKS; i++ ) {
|
||
|
ptr->dev_clock[i] = (int)(val.realnum * 1000.0 + 0.5);
|
||
|
token = xf86getSubToken(&(ptr->dev_comment));
|
||
|
}
|
||
|
ptr->dev_clocks = i;
|
||
|
xf86unGetToken (token);
|
||
|
break;
|
||
|
case TEXTCLOCKFRQ:
|
||
|
if ((token = xf86getSubToken(&(ptr->dev_comment))) != NUMBER)
|
||
|
Error (NUMBER_MSG, "TextClockFreq");
|
||
|
ptr->dev_textclockfreq = (int)(val.realnum * 1000.0 + 0.5);
|
||
|
break;
|
||
|
case OPTION:
|
||
|
ptr->dev_option_lst = xf86parseOption(ptr->dev_option_lst);
|
||
|
break;
|
||
|
case BUSID:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != STRING)
|
||
|
Error (QUOTE_MSG, "BusID");
|
||
|
ptr->dev_busid = val.str;
|
||
|
break;
|
||
|
case IRQ:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (QUOTE_MSG, "IRQ");
|
||
|
ptr->dev_irq = val.num;
|
||
|
break;
|
||
|
case SCREEN:
|
||
|
if (xf86getSubToken (&(ptr->dev_comment)) != NUMBER)
|
||
|
Error (NUMBER_MSG, "Screen");
|
||
|
ptr->dev_screen = val.num;
|
||
|
break;
|
||
|
case EOF_TOKEN:
|
||
|
Error (UNEXPECTED_EOF_MSG, NULL);
|
||
|
break;
|
||
|
default:
|
||
|
Error (INVALID_KEYWORD_MSG, xf86tokenString ());
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!has_ident)
|
||
|
Error (NO_IDENT_MSG, NULL);
|
||
|
|
||
|
#ifdef DEBUG
|
||
|
printf ("Device section parsed\n");
|
||
|
#endif
|
||
|
|
||
|
return ptr;
|
||
|
}
|
||
|
|
||
|
#undef CLEANUP
|
||
|
|
||
|
void
|
||
|
xf86printDeviceSection (FILE * cf, XF86ConfDevicePtr ptr)
|
||
|
{
|
||
|
int i;
|
||
|
|
||
|
while (ptr)
|
||
|
{
|
||
|
fprintf (cf, "Section \"Device\"\n");
|
||
|
if (ptr->dev_comment)
|
||
|
fprintf (cf, "%s", ptr->dev_comment);
|
||
|
if (ptr->dev_identifier)
|
||
|
fprintf (cf, "\tIdentifier \"%s\"\n", ptr->dev_identifier);
|
||
|
if (ptr->dev_driver)
|
||
|
fprintf (cf, "\tDriver \"%s\"\n", ptr->dev_driver);
|
||
|
if (ptr->dev_vendor)
|
||
|
fprintf (cf, "\tVendorName \"%s\"\n", ptr->dev_vendor);
|
||
|
if (ptr->dev_board)
|
||
|
fprintf (cf, "\tBoardName \"%s\"\n", ptr->dev_board);
|
||
|
if (ptr->dev_chipset)
|
||
|
fprintf (cf, "\tChipSet \"%s\"\n", ptr->dev_chipset);
|
||
|
if (ptr->dev_card)
|
||
|
fprintf (cf, "\tCard \"%s\"\n", ptr->dev_card);
|
||
|
if (ptr->dev_ramdac)
|
||
|
fprintf (cf, "\tRamDac \"%s\"\n", ptr->dev_ramdac);
|
||
|
if (ptr->dev_dacSpeeds[0] > 0 ) {
|
||
|
fprintf (cf, "\tDacSpeed ");
|
||
|
for (i = 0; i < CONF_MAXDACSPEEDS
|
||
|
&& ptr->dev_dacSpeeds[i] > 0; i++ )
|
||
|
fprintf (cf, "%g ", (double) (ptr->dev_dacSpeeds[i])/ 1000.0 );
|
||
|
fprintf (cf, "\n");
|
||
|
}
|
||
|
if (ptr->dev_videoram)
|
||
|
fprintf (cf, "\tVideoRam %d\n", ptr->dev_videoram);
|
||
|
if (ptr->dev_bios_base)
|
||
|
fprintf (cf, "\tBiosBase 0x%lx\n", ptr->dev_bios_base);
|
||
|
if (ptr->dev_mem_base)
|
||
|
fprintf (cf, "\tMemBase 0x%lx\n", ptr->dev_mem_base);
|
||
|
if (ptr->dev_io_base)
|
||
|
fprintf (cf, "\tIOBase 0x%lx\n", ptr->dev_io_base);
|
||
|
if (ptr->dev_clockchip)
|
||
|
fprintf (cf, "\tClockChip \"%s\"\n", ptr->dev_clockchip);
|
||
|
if (ptr->dev_chipid != -1)
|
||
|
fprintf (cf, "\tChipId 0x%x\n", ptr->dev_chipid);
|
||
|
if (ptr->dev_chiprev != -1)
|
||
|
fprintf (cf, "\tChipRev 0x%x\n", ptr->dev_chiprev);
|
||
|
|
||
|
xf86printOptionList(cf, ptr->dev_option_lst, 1);
|
||
|
if (ptr->dev_clocks > 0 ) {
|
||
|
fprintf (cf, "\tClocks ");
|
||
|
for (i = 0; i < ptr->dev_clocks; i++ )
|
||
|
fprintf (cf, "%.1f ", (double)ptr->dev_clock[i] / 1000.0 );
|
||
|
fprintf (cf, "\n");
|
||
|
}
|
||
|
if (ptr->dev_textclockfreq) {
|
||
|
fprintf (cf, "\tTextClockFreq %.1f\n",
|
||
|
(double)ptr->dev_textclockfreq / 1000.0);
|
||
|
}
|
||
|
if (ptr->dev_busid)
|
||
|
fprintf (cf, "\tBusID \"%s\"\n", ptr->dev_busid);
|
||
|
if (ptr->dev_screen > 0)
|
||
|
fprintf (cf, "\tScreen %d\n", ptr->dev_screen);
|
||
|
if (ptr->dev_irq >= 0)
|
||
|
fprintf (cf, "\tIRQ %d\n", ptr->dev_irq);
|
||
|
fprintf (cf, "EndSection\n\n");
|
||
|
ptr = ptr->list.next;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
void
|
||
|
xf86freeDeviceList (XF86ConfDevicePtr ptr)
|
||
|
{
|
||
|
XF86ConfDevicePtr prev;
|
||
|
|
||
|
while (ptr)
|
||
|
{
|
||
|
TestFree (ptr->dev_identifier);
|
||
|
TestFree (ptr->dev_vendor);
|
||
|
TestFree (ptr->dev_board);
|
||
|
TestFree (ptr->dev_chipset);
|
||
|
TestFree (ptr->dev_card);
|
||
|
TestFree (ptr->dev_driver);
|
||
|
TestFree (ptr->dev_ramdac);
|
||
|
TestFree (ptr->dev_clockchip);
|
||
|
TestFree (ptr->dev_comment);
|
||
|
xf86optionListFree (ptr->dev_option_lst);
|
||
|
|
||
|
prev = ptr;
|
||
|
ptr = ptr->list.next;
|
||
|
xf86conffree (prev);
|
||
|
}
|
||
|
}
|
||
|
|
||
|
int
|
||
|
xf86validateDevice (XF86ConfigPtr p)
|
||
|
{
|
||
|
XF86ConfDevicePtr device = p->conf_device_lst;
|
||
|
|
||
|
if (!device) {
|
||
|
xf86validationError ("At least one Device section is required.");
|
||
|
return (FALSE);
|
||
|
}
|
||
|
|
||
|
while (device) {
|
||
|
if (!device->dev_driver) {
|
||
|
xf86validationError (UNDEFINED_DRIVER_MSG, device->dev_identifier);
|
||
|
return (FALSE);
|
||
|
}
|
||
|
device = device->list.next;
|
||
|
}
|
||
|
return (TRUE);
|
||
|
}
|
||
|
|
||
|
XF86ConfDevicePtr
|
||
|
xf86findDevice (const char *ident, XF86ConfDevicePtr p)
|
||
|
{
|
||
|
while (p)
|
||
|
{
|
||
|
if (xf86nameCompare (ident, p->dev_identifier) == 0)
|
||
|
return (p);
|
||
|
|
||
|
p = p->list.next;
|
||
|
}
|
||
|
return (NULL);
|
||
|
}
|
||
|
|
||
|
char *
|
||
|
xf86configStrdup (const char *s)
|
||
|
{
|
||
|
char *tmp;
|
||
|
if (!s) return NULL;
|
||
|
tmp = xf86confmalloc (sizeof (char) * (strlen (s) + 1));
|
||
|
if (tmp)
|
||
|
strcpy (tmp, s);
|
||
|
return (tmp);
|
||
|
}
|