526 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
			
		
		
	
	
			526 lines
		
	
	
		
			12 KiB
		
	
	
	
		
			C
		
	
	
	
| /* $XFree86: xc/programs/Xserver/hw/xfree86/parser/Flags.c,v 1.23 2003/08/24 17:37:07 dawes Exp $ */
 | |
| /* 
 | |
|  * 
 | |
|  * 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"
 | |
| #include <math.h>
 | |
| 
 | |
| extern LexRec val;
 | |
| 
 | |
| static xf86ConfigSymTabRec ServerFlagsTab[] =
 | |
| {
 | |
| 	{ENDSECTION, "endsection"},
 | |
| 	{NOTRAPSIGNALS, "notrapsignals"},
 | |
| 	{DONTZAP, "dontzap"},
 | |
| 	{DONTZOOM, "dontzoom"},
 | |
| 	{DISABLEVIDMODE, "disablevidmodeextension"},
 | |
| 	{ALLOWNONLOCAL, "allownonlocalxvidtune"},
 | |
| 	{DISABLEMODINDEV, "disablemodindev"},
 | |
| 	{MODINDEVALLOWNONLOCAL, "allownonlocalmodindev"},
 | |
| 	{ALLOWMOUSEOPENFAIL, "allowmouseopenfail"},
 | |
| 	{OPTION, "option"},
 | |
| 	{BLANKTIME, "blanktime"},
 | |
| 	{STANDBYTIME, "standbytime"},
 | |
| 	{SUSPENDTIME, "suspendtime"},
 | |
| 	{OFFTIME, "offtime"},
 | |
| 	{DEFAULTLAYOUT, "defaultserverlayout"},
 | |
| 	{-1, ""},
 | |
| };
 | |
| 
 | |
| #define CLEANUP xf86freeFlags
 | |
| 
 | |
| XF86ConfFlagsPtr
 | |
| xf86parseFlagsSection (void)
 | |
| {
 | |
| 	int token;
 | |
| 	parsePrologue (XF86ConfFlagsPtr, XF86ConfFlagsRec)
 | |
| 
 | |
| 	while ((token = xf86getToken (ServerFlagsTab)) != ENDSECTION)
 | |
| 	{
 | |
| 		int hasvalue = FALSE;
 | |
| 		int strvalue = FALSE;
 | |
| 		int tokentype;
 | |
| 		switch (token)
 | |
| 		{
 | |
| 		case COMMENT:
 | |
| 			ptr->flg_comment = xf86addComment(ptr->flg_comment, val.str);
 | |
| 			break;
 | |
| 			/* 
 | |
| 			 * these old keywords are turned into standard generic options.
 | |
| 			 * we fall through here on purpose
 | |
| 			 */
 | |
| 		case DEFAULTLAYOUT:
 | |
| 			strvalue = TRUE;
 | |
| 		case BLANKTIME:
 | |
| 		case STANDBYTIME:
 | |
| 		case SUSPENDTIME:
 | |
| 		case OFFTIME:
 | |
| 			hasvalue = TRUE;
 | |
| 		case NOTRAPSIGNALS:
 | |
| 		case DONTZAP:
 | |
| 		case DONTZOOM:
 | |
| 		case DISABLEVIDMODE:
 | |
| 		case ALLOWNONLOCAL:
 | |
| 		case DISABLEMODINDEV:
 | |
| 		case MODINDEVALLOWNONLOCAL:
 | |
| 		case ALLOWMOUSEOPENFAIL:
 | |
| 			{
 | |
| 				int i = 0;
 | |
| 				while (ServerFlagsTab[i].token != -1)
 | |
| 				{
 | |
| 					char *tmp;
 | |
| 
 | |
| 					if (ServerFlagsTab[i].token == token)
 | |
| 					{
 | |
| 						char *valstr = NULL;
 | |
| 						/* can't use strdup because it calls malloc */
 | |
| 						tmp = xf86configStrdup (ServerFlagsTab[i].name);
 | |
| 						if (hasvalue)
 | |
| 						{
 | |
| 							tokentype = xf86getSubToken(&(ptr->flg_comment));
 | |
| 							if (strvalue) {
 | |
| 								if (tokentype != STRING)
 | |
| 									Error (QUOTE_MSG, tmp);
 | |
| 								valstr = val.str;
 | |
| 							} else {
 | |
| 								if (tokentype != NUMBER)
 | |
| 									Error (NUMBER_MSG, tmp);
 | |
| 								valstr = xf86confmalloc(16);
 | |
| 								if (valstr)
 | |
| 									sprintf(valstr, "%d", val.num);
 | |
| 							}
 | |
| 						}
 | |
| 						ptr->flg_option_lst = xf86addNewOption
 | |
| 							(ptr->flg_option_lst, tmp, valstr);
 | |
| 					}
 | |
| 					i++;
 | |
| 				}
 | |
| 			}
 | |
| 			break;
 | |
| 		case OPTION:
 | |
| 			ptr->flg_option_lst = xf86parseOption(ptr->flg_option_lst);
 | |
| 			break;
 | |
| 
 | |
| 		case EOF_TOKEN:
 | |
| 			Error (UNEXPECTED_EOF_MSG, NULL);
 | |
| 			break;
 | |
| 		default:
 | |
| 			Error (INVALID_KEYWORD_MSG, xf86tokenString ());
 | |
| 			break;
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| #ifdef DEBUG
 | |
| 	printf ("Flags section parsed\n");
 | |
| #endif
 | |
| 
 | |
| 	return ptr;
 | |
| }
 | |
| 
 | |
| #undef CLEANUP
 | |
| 
 | |
| void
 | |
| xf86printServerFlagsSection (FILE * f, XF86ConfFlagsPtr flags)
 | |
| {
 | |
| 	XF86OptionPtr p;
 | |
| 
 | |
| 	if ((!flags) || (!flags->flg_option_lst))
 | |
| 		return;
 | |
| 	p = flags->flg_option_lst;
 | |
| 	fprintf (f, "Section \"ServerFlags\"\n");
 | |
| 	if (flags->flg_comment)
 | |
| 		fprintf (f, "%s", flags->flg_comment);
 | |
| 	xf86printOptionList(f, p, 1);
 | |
| 	fprintf (f, "EndSection\n\n");
 | |
| }
 | |
| 
 | |
| static XF86OptionPtr
 | |
| addNewOption2 (XF86OptionPtr head, char *name, char *val, int used)
 | |
| {
 | |
| 	XF86OptionPtr new, old = NULL;
 | |
| 
 | |
| 	/* Don't allow duplicates */
 | |
|  	if (head != NULL && (old = xf86findOption(head, name)) != NULL)
 | |
|  		new = old;
 | |
|  	else {
 | |
| 		new = xf86confcalloc (1, sizeof (XF86OptionRec));
 | |
|  		new->list.next = NULL;
 | |
|  	}
 | |
|  	new->opt_name = name;
 | |
|  	new->opt_val = val;
 | |
|  	new->opt_used = used;
 | |
| 	
 | |
|   	if (old == NULL)
 | |
| 		return ((XF86OptionPtr) xf86addListItem ((glp) head, (glp) new));
 | |
|  	else 
 | |
|  		return head;
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86addNewOption (XF86OptionPtr head, char *name, char *val)
 | |
| {
 | |
| 	return addNewOption2(head, name, val, 0);
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86freeFlags (XF86ConfFlagsPtr flags)
 | |
| {
 | |
| 	if (flags == NULL)
 | |
| 		return;
 | |
| 	xf86optionListFree (flags->flg_option_lst);
 | |
| 	TestFree(flags->flg_comment);
 | |
| 	xf86conffree (flags);
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86optionListDup (XF86OptionPtr opt)
 | |
| {
 | |
| 	XF86OptionPtr newopt = NULL;
 | |
| 
 | |
| 	while (opt)
 | |
| 	{
 | |
| 		newopt = xf86addNewOption(newopt, xf86configStrdup(opt->opt_name), 
 | |
| 					  xf86configStrdup(opt->opt_val));
 | |
| 		newopt->opt_used = opt->opt_used;
 | |
| 		if (opt->opt_comment)
 | |
| 			newopt->opt_comment = xf86configStrdup(opt->opt_comment);
 | |
| 		opt = opt->list.next;
 | |
| 	}
 | |
| 	return newopt;
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86optionListFree (XF86OptionPtr opt)
 | |
| {
 | |
| 	XF86OptionPtr prev;
 | |
| 
 | |
| 	while (opt)
 | |
| 	{
 | |
| 		TestFree (opt->opt_name);
 | |
| 		TestFree (opt->opt_val);
 | |
| 		TestFree (opt->opt_comment);
 | |
| 		prev = opt;
 | |
| 		opt = opt->list.next;
 | |
| 		xf86conffree (prev);
 | |
| 	}
 | |
| }
 | |
| 
 | |
| char *
 | |
| xf86optionName(XF86OptionPtr opt)
 | |
| {
 | |
| 	if (opt)
 | |
| 		return opt->opt_name;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| char *
 | |
| xf86optionValue(XF86OptionPtr opt)
 | |
| {
 | |
| 	if (opt)
 | |
| 		return opt->opt_val;
 | |
| 	return 0;
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86newOption(char *name, char *value)
 | |
| {
 | |
| 	XF86OptionPtr opt;
 | |
| 
 | |
| 	opt = xf86confcalloc(1, sizeof (XF86OptionRec));
 | |
| 	if (!opt)
 | |
| 		return NULL;
 | |
| 
 | |
| 	opt->opt_used = 0;
 | |
| 	opt->list.next = 0;
 | |
| 	opt->opt_name = name;
 | |
| 	opt->opt_val = value;
 | |
| 
 | |
| 	return opt;
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86nextOption(XF86OptionPtr list)
 | |
| {
 | |
| 	if (!list)
 | |
| 		return NULL;
 | |
| 	return list->list.next;
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * this function searches the given option list for the named option and
 | |
|  * returns a pointer to the option rec if found. If not found, it returns
 | |
|  * NULL
 | |
|  */
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86findOption (XF86OptionPtr list, const char *name)
 | |
| {
 | |
| 	while (list)
 | |
| 	{
 | |
| 		if (xf86nameCompare (list->opt_name, name) == 0)
 | |
| 			return (list);
 | |
| 		list = list->list.next;
 | |
| 	}
 | |
| 	return (NULL);
 | |
| }
 | |
| 
 | |
| /*
 | |
|  * this function searches the given option list for the named option. If
 | |
|  * found and the option has a parameter, a pointer to the parameter is
 | |
|  * returned.  If the option does not have a parameter an empty string is
 | |
|  * returned.  If the option is not found, a NULL is returned.
 | |
|  */
 | |
| 
 | |
| char *
 | |
| xf86findOptionValue (XF86OptionPtr list, const char *name)
 | |
| {
 | |
| 	XF86OptionPtr p = xf86findOption (list, name);
 | |
| 
 | |
| 	if (p)
 | |
| 	{
 | |
| 		if (p->opt_val)
 | |
| 			return (p->opt_val);
 | |
| 		else
 | |
| 			return "";
 | |
| 	}
 | |
| 	return (NULL);
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86optionListCreate( const char **options, int count, int used )
 | |
| {
 | |
| 	XF86OptionPtr p = NULL;
 | |
| 	char *t1, *t2;
 | |
| 	int i;
 | |
| 
 | |
| 	if (count == -1)
 | |
| 	{
 | |
| 		for (count = 0; options[count]; count++)
 | |
| 			;
 | |
| 	}
 | |
| 	if( (count % 2) != 0 )
 | |
| 	{
 | |
| 		fprintf( stderr, "xf86optionListCreate: count must be an even number.\n" );
 | |
| 		return (NULL);
 | |
| 	}
 | |
| 	for (i = 0; i < count; i += 2)
 | |
| 	{
 | |
| 		/* can't use strdup because it calls malloc */
 | |
| 		t1 = xf86confmalloc (sizeof (char) *
 | |
| 				(strlen (options[i]) + 1));
 | |
| 		strcpy (t1, options[i]);
 | |
| 		t2 = xf86confmalloc (sizeof (char) *
 | |
| 				(strlen (options[i + 1]) + 1));
 | |
| 		strcpy (t2, options[i + 1]);
 | |
| 		p = addNewOption2 (p, t1, t2, used);
 | |
| 	}
 | |
| 
 | |
| 	return (p);
 | |
| }
 | |
| 
 | |
| /* the 2 given lists are merged. If an option with the same name is present in
 | |
|  * both, the option from the user list - specified in the second argument -
 | |
|  * is used. The end result is a single valid list of options. Duplicates
 | |
|  * are freed, and the original lists are no longer guaranteed to be complete.
 | |
|  */
 | |
| XF86OptionPtr
 | |
| xf86optionListMerge (XF86OptionPtr head, XF86OptionPtr tail)
 | |
| {
 | |
| 	XF86OptionPtr a, b, ap = NULL, bp = NULL;
 | |
| 
 | |
| 	a = tail;
 | |
| 	b = head;
 | |
| 	while (tail && b) {
 | |
| 		if (xf86nameCompare (a->opt_name, b->opt_name) == 0) {
 | |
| 			if (b == head)
 | |
| 				head = a;
 | |
| 			else
 | |
| 				bp->list.next = a;
 | |
| 			if (a == tail)
 | |
| 				tail = a->list.next;
 | |
| 			else
 | |
| 				ap->list.next = a->list.next;
 | |
| 			a->list.next = b->list.next;
 | |
| 			b->list.next = NULL;
 | |
| 			xf86optionListFree (b);
 | |
| 			b = a->list.next;
 | |
| 			bp = a;
 | |
| 			a = tail;
 | |
| 			ap = NULL;
 | |
| 		} else {
 | |
| 			ap = a;
 | |
| 			if (!(a = a->list.next)) {
 | |
| 				a = tail;
 | |
| 				bp = b;
 | |
| 				b = b->list.next;
 | |
| 				ap = NULL;
 | |
| 			}
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	if (head) {
 | |
| 		for (a = head; a->list.next; a = a->list.next)
 | |
| 			;
 | |
| 		a->list.next = tail;
 | |
| 	} else 
 | |
| 		head = tail;
 | |
| 
 | |
| 	return (head);
 | |
| }
 | |
| 
 | |
| char *
 | |
| xf86uLongToString(unsigned long i)
 | |
| {
 | |
| 	char *s;
 | |
| 	int l;
 | |
| 
 | |
| 	l = (int)(ceil(log10((double)i) + 2.5));
 | |
| 	s = xf86confmalloc(l);
 | |
| 	if (!s)
 | |
| 		return NULL;
 | |
| 	sprintf(s, "%lu", i);
 | |
| 	return s;
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86debugListOptions(XF86OptionPtr Options)
 | |
| {
 | |
| 	while (Options) {
 | |
| 		ErrorF("Option: %s Value: %s\n",Options->opt_name,Options->opt_val);
 | |
| 		Options = Options->list.next;
 | |
| 	}
 | |
| }
 | |
| 
 | |
| XF86OptionPtr
 | |
| xf86parseOption(XF86OptionPtr head)
 | |
| {
 | |
| 	XF86OptionPtr option, cnew, old;
 | |
| 	char *name, *comment = NULL;
 | |
| 	int token;
 | |
| 
 | |
| 	if ((token = xf86getSubToken(&comment)) != STRING) {
 | |
| 		xf86parseError(BAD_OPTION_MSG, NULL);
 | |
| 		if (comment)
 | |
| 			xf86conffree(comment);
 | |
| 		return (head);
 | |
| 	}
 | |
| 
 | |
| 	name = val.str;
 | |
| 	if ((token = xf86getSubToken(&comment)) == STRING) {
 | |
| 		option = xf86newOption(name, val.str);
 | |
| 		option->opt_comment = comment;
 | |
| 		if ((token = xf86getToken(NULL)) == COMMENT)
 | |
| 			option->opt_comment = xf86addComment(option->opt_comment, val.str);
 | |
| 		else
 | |
| 			xf86unGetToken(token);
 | |
| 	}
 | |
| 	else {
 | |
| 		option = xf86newOption(name, NULL);
 | |
| 		option->opt_comment = comment;
 | |
| 		if (token == COMMENT)
 | |
| 			option->opt_comment = xf86addComment(option->opt_comment, val.str);
 | |
| 		else
 | |
| 			xf86unGetToken(token);
 | |
| 	}
 | |
| 
 | |
| 	old = NULL;
 | |
| 
 | |
| 	/* Don't allow duplicates */
 | |
| 	if (head != NULL && (old = xf86findOption(head, name)) != NULL) {
 | |
| 		cnew = old;
 | |
| 		xf86conffree(option->opt_name);
 | |
| 		TestFree(option->opt_val);
 | |
| 		TestFree(option->opt_comment);
 | |
| 		xf86conffree(option);
 | |
| 	}
 | |
| 	else
 | |
| 		cnew = option;
 | |
| 	
 | |
| 	if (old == NULL)
 | |
| 		return ((XF86OptionPtr)xf86addListItem((glp)head, (glp)cnew));
 | |
| 
 | |
| 	return (head);
 | |
| }
 | |
| 
 | |
| void
 | |
| xf86printOptionList(FILE *fp, XF86OptionPtr list, int tabs)
 | |
| {
 | |
| 	int i;
 | |
| 
 | |
| 	if (!list)
 | |
| 		return;
 | |
| 	while (list) {
 | |
| 		for (i = 0; i < tabs; i++)
 | |
| 			fputc('\t', fp);
 | |
| 		if (list->opt_val)
 | |
| 			fprintf(fp, "Option	    \"%s\" \"%s\"", list->opt_name, list->opt_val);
 | |
| 		else
 | |
| 			fprintf(fp, "Option	    \"%s\"", list->opt_name);
 | |
| 		if (list->opt_comment)
 | |
| 			fprintf(fp, "%s", list->opt_comment);
 | |
| 		else
 | |
| 			fputc('\n', fp);
 | |
| 		list = list->list.next;
 | |
| 	}
 | |
| }
 |