135c4bbdfSmrg/* 235c4bbdfSmrg * Copyright (c) 2014 NVIDIA Corporation. All rights reserved. 335c4bbdfSmrg * 435c4bbdfSmrg * Permission is hereby granted, free of charge, to any person 535c4bbdfSmrg * obtaining a copy of this software and associated documentation 635c4bbdfSmrg * files (the "Software"), to deal in the Software without 735c4bbdfSmrg * restriction, including without limitation the rights to use, 835c4bbdfSmrg * copy, modify, merge, publish, distribute, sublicense, and/or sell 935c4bbdfSmrg * copies of the Software, and to permit persons to whom the 1035c4bbdfSmrg * Software is furnished to do so, subject to the following 1135c4bbdfSmrg * conditions: 1235c4bbdfSmrg * 1335c4bbdfSmrg * The above copyright notice and this permission notice shall be 1435c4bbdfSmrg * included in all copies or substantial portions of the Software. 1535c4bbdfSmrg * 1635c4bbdfSmrg * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 1735c4bbdfSmrg * EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES 1835c4bbdfSmrg * OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND 1935c4bbdfSmrg * NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT 2035c4bbdfSmrg * HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, 2135c4bbdfSmrg * WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING 2235c4bbdfSmrg * FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 2335c4bbdfSmrg * OTHER DEALINGS IN THE SOFTWARE. 2435c4bbdfSmrg */ 2535c4bbdfSmrg 2635c4bbdfSmrg#ifdef HAVE_XORG_CONFIG_H 2735c4bbdfSmrg#include <xorg-config.h> 2835c4bbdfSmrg#endif 2935c4bbdfSmrg 3035c4bbdfSmrg#include "os.h" 3135c4bbdfSmrg#include "xf86Parser.h" 3235c4bbdfSmrg#include "xf86tokens.h" 3335c4bbdfSmrg#include "Configint.h" 3435c4bbdfSmrg 351b5d61b8Smrgstatic const xf86ConfigSymTabRec OutputClassTab[] = { 3635c4bbdfSmrg {ENDSECTION, "endsection"}, 3735c4bbdfSmrg {IDENTIFIER, "identifier"}, 3835c4bbdfSmrg {DRIVER, "driver"}, 391b5d61b8Smrg {MODULEPATH, "modulepath"}, 401b5d61b8Smrg {OPTION, "option"}, 4135c4bbdfSmrg {MATCH_DRIVER, "matchdriver"}, 4235c4bbdfSmrg {-1, ""}, 4335c4bbdfSmrg}; 4435c4bbdfSmrg 4535c4bbdfSmrgstatic void 4635c4bbdfSmrgxf86freeOutputClassList(XF86ConfOutputClassPtr ptr) 4735c4bbdfSmrg{ 4835c4bbdfSmrg XF86ConfOutputClassPtr prev; 4935c4bbdfSmrg 5035c4bbdfSmrg while (ptr) { 5135c4bbdfSmrg xf86MatchGroup *group, *next; 5235c4bbdfSmrg char **list; 5335c4bbdfSmrg 5435c4bbdfSmrg TestFree(ptr->identifier); 5535c4bbdfSmrg TestFree(ptr->comment); 5635c4bbdfSmrg TestFree(ptr->driver); 571b5d61b8Smrg TestFree(ptr->modulepath); 5835c4bbdfSmrg 5935c4bbdfSmrg xorg_list_for_each_entry_safe(group, next, &ptr->match_driver, entry) { 6035c4bbdfSmrg xorg_list_del(&group->entry); 6135c4bbdfSmrg for (list = group->values; *list; list++) 6235c4bbdfSmrg free(*list); 6335c4bbdfSmrg free(group); 6435c4bbdfSmrg } 6535c4bbdfSmrg 661b5d61b8Smrg xf86optionListFree(ptr->option_lst); 671b5d61b8Smrg 6835c4bbdfSmrg prev = ptr; 6935c4bbdfSmrg ptr = ptr->list.next; 7035c4bbdfSmrg free(prev); 7135c4bbdfSmrg } 7235c4bbdfSmrg} 7335c4bbdfSmrg 7435c4bbdfSmrg#define CLEANUP xf86freeOutputClassList 7535c4bbdfSmrg 7635c4bbdfSmrg#define TOKEN_SEP "|" 7735c4bbdfSmrg 7835c4bbdfSmrgstatic void 7935c4bbdfSmrgadd_group_entry(struct xorg_list *head, char **values) 8035c4bbdfSmrg{ 8135c4bbdfSmrg xf86MatchGroup *group; 8235c4bbdfSmrg 8335c4bbdfSmrg group = malloc(sizeof(*group)); 8435c4bbdfSmrg if (group) { 8535c4bbdfSmrg group->values = values; 8635c4bbdfSmrg xorg_list_add(&group->entry, head); 8735c4bbdfSmrg } 8835c4bbdfSmrg} 8935c4bbdfSmrg 9035c4bbdfSmrgXF86ConfOutputClassPtr 9135c4bbdfSmrgxf86parseOutputClassSection(void) 9235c4bbdfSmrg{ 9335c4bbdfSmrg int has_ident = FALSE; 9435c4bbdfSmrg int token; 9535c4bbdfSmrg 9635c4bbdfSmrg parsePrologue(XF86ConfOutputClassPtr, XF86ConfOutputClassRec) 9735c4bbdfSmrg 9835c4bbdfSmrg /* Initialize MatchGroup lists */ 9935c4bbdfSmrg xorg_list_init(&ptr->match_driver); 10035c4bbdfSmrg 10135c4bbdfSmrg while ((token = xf86getToken(OutputClassTab)) != ENDSECTION) { 10235c4bbdfSmrg switch (token) { 10335c4bbdfSmrg case COMMENT: 10435c4bbdfSmrg ptr->comment = xf86addComment(ptr->comment, xf86_lex_val.str); 10558cf2af7Smrg free(xf86_lex_val.str); 10658cf2af7Smrg xf86_lex_val.str = NULL; 10735c4bbdfSmrg break; 10835c4bbdfSmrg case IDENTIFIER: 10935c4bbdfSmrg if (xf86getSubToken(&(ptr->comment)) != STRING) 11035c4bbdfSmrg Error(QUOTE_MSG, "Identifier"); 11135c4bbdfSmrg if (has_ident == TRUE) 11235c4bbdfSmrg Error(MULTIPLE_MSG, "Identifier"); 11335c4bbdfSmrg ptr->identifier = xf86_lex_val.str; 11435c4bbdfSmrg has_ident = TRUE; 11535c4bbdfSmrg break; 11635c4bbdfSmrg case DRIVER: 11735c4bbdfSmrg if (xf86getSubToken(&(ptr->comment)) != STRING) 11835c4bbdfSmrg Error(QUOTE_MSG, "Driver"); 11935c4bbdfSmrg else 12035c4bbdfSmrg ptr->driver = xf86_lex_val.str; 12135c4bbdfSmrg break; 1221b5d61b8Smrg case MODULEPATH: 1231b5d61b8Smrg if (xf86getSubToken(&(ptr->comment)) != STRING) 1241b5d61b8Smrg Error(QUOTE_MSG, "ModulePath"); 1251b5d61b8Smrg if (ptr->modulepath) { 1261b5d61b8Smrg char *path; 1271b5d61b8Smrg XNFasprintf(&path, "%s,%s", ptr->modulepath, xf86_lex_val.str); 1281b5d61b8Smrg free(xf86_lex_val.str); 1291b5d61b8Smrg free(ptr->modulepath); 1301b5d61b8Smrg ptr->modulepath = path; 1311b5d61b8Smrg } else { 1321b5d61b8Smrg ptr->modulepath = xf86_lex_val.str; 1331b5d61b8Smrg } 1341b5d61b8Smrg break; 1351b5d61b8Smrg case OPTION: 1361b5d61b8Smrg ptr->option_lst = xf86parseOption(ptr->option_lst); 1371b5d61b8Smrg break; 13835c4bbdfSmrg case MATCH_DRIVER: 13935c4bbdfSmrg if (xf86getSubToken(&(ptr->comment)) != STRING) 14035c4bbdfSmrg Error(QUOTE_MSG, "MatchDriver"); 14135c4bbdfSmrg add_group_entry(&ptr->match_driver, 14235c4bbdfSmrg xstrtokenize(xf86_lex_val.str, TOKEN_SEP)); 14335c4bbdfSmrg free(xf86_lex_val.str); 14435c4bbdfSmrg break; 14535c4bbdfSmrg case EOF_TOKEN: 14635c4bbdfSmrg Error(UNEXPECTED_EOF_MSG); 14735c4bbdfSmrg break; 14835c4bbdfSmrg default: 14935c4bbdfSmrg Error(INVALID_KEYWORD_MSG, xf86tokenString()); 15035c4bbdfSmrg break; 15135c4bbdfSmrg } 15235c4bbdfSmrg } 15335c4bbdfSmrg 15435c4bbdfSmrg if (!has_ident) 15535c4bbdfSmrg Error(NO_IDENT_MSG); 15635c4bbdfSmrg 15735c4bbdfSmrg#ifdef DEBUG 15835c4bbdfSmrg printf("OutputClass section parsed\n"); 15935c4bbdfSmrg#endif 16035c4bbdfSmrg 16135c4bbdfSmrg return ptr; 16235c4bbdfSmrg} 16335c4bbdfSmrgvoid 16435c4bbdfSmrgxf86printOutputClassSection(FILE * cf, XF86ConfOutputClassPtr ptr) 16535c4bbdfSmrg{ 16635c4bbdfSmrg const xf86MatchGroup *group; 16735c4bbdfSmrg char *const *cur; 16835c4bbdfSmrg 16935c4bbdfSmrg while (ptr) { 17035c4bbdfSmrg fprintf(cf, "Section \"OutputClass\"\n"); 17135c4bbdfSmrg if (ptr->comment) 17235c4bbdfSmrg fprintf(cf, "%s", ptr->comment); 17335c4bbdfSmrg if (ptr->identifier) 17435c4bbdfSmrg fprintf(cf, "\tIdentifier \"%s\"\n", ptr->identifier); 17535c4bbdfSmrg if (ptr->driver) 17635c4bbdfSmrg fprintf(cf, "\tDriver \"%s\"\n", ptr->driver); 17735c4bbdfSmrg 17835c4bbdfSmrg xorg_list_for_each_entry(group, &ptr->match_driver, entry) { 17935c4bbdfSmrg fprintf(cf, "\tMatchDriver \""); 18035c4bbdfSmrg for (cur = group->values; *cur; cur++) 18135c4bbdfSmrg fprintf(cf, "%s%s", cur == group->values ? "" : TOKEN_SEP, 18235c4bbdfSmrg *cur); 18335c4bbdfSmrg fprintf(cf, "\"\n"); 18435c4bbdfSmrg } 18535c4bbdfSmrg 18635c4bbdfSmrg fprintf(cf, "EndSection\n\n"); 18735c4bbdfSmrg ptr = ptr->list.next; 18835c4bbdfSmrg } 18935c4bbdfSmrg} 190