1*0a9064fbSMasahiro Yamada %option nostdinit noyywrap never-interactive full ecs 2*0a9064fbSMasahiro Yamada %option 8bit nodefault perf-report perf-report 3*0a9064fbSMasahiro Yamada %option noinput 4*0a9064fbSMasahiro Yamada %x COMMAND HELP STRING PARAM 5*0a9064fbSMasahiro Yamada %{ 6*0a9064fbSMasahiro Yamada /* 7*0a9064fbSMasahiro Yamada * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org> 8*0a9064fbSMasahiro Yamada * Released under the terms of the GNU GPL v2.0. 9*0a9064fbSMasahiro Yamada */ 10*0a9064fbSMasahiro Yamada 11*0a9064fbSMasahiro Yamada #include <limits.h> 12*0a9064fbSMasahiro Yamada #include <stdio.h> 13*0a9064fbSMasahiro Yamada #include <stdlib.h> 14*0a9064fbSMasahiro Yamada #include <string.h> 15*0a9064fbSMasahiro Yamada #include <unistd.h> 16*0a9064fbSMasahiro Yamada 17*0a9064fbSMasahiro Yamada #include "lkc.h" 18*0a9064fbSMasahiro Yamada 19*0a9064fbSMasahiro Yamada #define START_STRSIZE 16 20*0a9064fbSMasahiro Yamada 21*0a9064fbSMasahiro Yamada static struct { 22*0a9064fbSMasahiro Yamada struct file *file; 23*0a9064fbSMasahiro Yamada int lineno; 24*0a9064fbSMasahiro Yamada } current_pos; 25*0a9064fbSMasahiro Yamada 26*0a9064fbSMasahiro Yamada static char *text; 27*0a9064fbSMasahiro Yamada static int text_size, text_asize; 28*0a9064fbSMasahiro Yamada 29*0a9064fbSMasahiro Yamada struct buffer { 30*0a9064fbSMasahiro Yamada struct buffer *parent; 31*0a9064fbSMasahiro Yamada YY_BUFFER_STATE state; 32*0a9064fbSMasahiro Yamada }; 33*0a9064fbSMasahiro Yamada 34*0a9064fbSMasahiro Yamada struct buffer *current_buf; 35*0a9064fbSMasahiro Yamada 36*0a9064fbSMasahiro Yamada static int last_ts, first_ts; 37*0a9064fbSMasahiro Yamada 38*0a9064fbSMasahiro Yamada static void zconf_endhelp(void); 39*0a9064fbSMasahiro Yamada static void zconf_endfile(void); 40*0a9064fbSMasahiro Yamada 41*0a9064fbSMasahiro Yamada static void new_string(void) 42*0a9064fbSMasahiro Yamada { 43*0a9064fbSMasahiro Yamada text = xmalloc(START_STRSIZE); 44*0a9064fbSMasahiro Yamada text_asize = START_STRSIZE; 45*0a9064fbSMasahiro Yamada text_size = 0; 46*0a9064fbSMasahiro Yamada *text = 0; 47*0a9064fbSMasahiro Yamada } 48*0a9064fbSMasahiro Yamada 49*0a9064fbSMasahiro Yamada static void append_string(const char *str, int size) 50*0a9064fbSMasahiro Yamada { 51*0a9064fbSMasahiro Yamada int new_size = text_size + size + 1; 52*0a9064fbSMasahiro Yamada if (new_size > text_asize) { 53*0a9064fbSMasahiro Yamada new_size += START_STRSIZE - 1; 54*0a9064fbSMasahiro Yamada new_size &= -START_STRSIZE; 55*0a9064fbSMasahiro Yamada text = realloc(text, new_size); 56*0a9064fbSMasahiro Yamada text_asize = new_size; 57*0a9064fbSMasahiro Yamada } 58*0a9064fbSMasahiro Yamada memcpy(text + text_size, str, size); 59*0a9064fbSMasahiro Yamada text_size += size; 60*0a9064fbSMasahiro Yamada text[text_size] = 0; 61*0a9064fbSMasahiro Yamada } 62*0a9064fbSMasahiro Yamada 63*0a9064fbSMasahiro Yamada static void alloc_string(const char *str, int size) 64*0a9064fbSMasahiro Yamada { 65*0a9064fbSMasahiro Yamada text = xmalloc(size + 1); 66*0a9064fbSMasahiro Yamada memcpy(text, str, size); 67*0a9064fbSMasahiro Yamada text[size] = 0; 68*0a9064fbSMasahiro Yamada } 69*0a9064fbSMasahiro Yamada %} 70*0a9064fbSMasahiro Yamada 71*0a9064fbSMasahiro Yamada n [A-Za-z0-9_] 72*0a9064fbSMasahiro Yamada 73*0a9064fbSMasahiro Yamada %% 74*0a9064fbSMasahiro Yamada int str = 0; 75*0a9064fbSMasahiro Yamada int ts, i; 76*0a9064fbSMasahiro Yamada 77*0a9064fbSMasahiro Yamada [ \t]*#.*\n | 78*0a9064fbSMasahiro Yamada [ \t]*\n { 79*0a9064fbSMasahiro Yamada current_file->lineno++; 80*0a9064fbSMasahiro Yamada return T_EOL; 81*0a9064fbSMasahiro Yamada } 82*0a9064fbSMasahiro Yamada [ \t]*#.* 83*0a9064fbSMasahiro Yamada 84*0a9064fbSMasahiro Yamada 85*0a9064fbSMasahiro Yamada [ \t]+ { 86*0a9064fbSMasahiro Yamada BEGIN(COMMAND); 87*0a9064fbSMasahiro Yamada } 88*0a9064fbSMasahiro Yamada 89*0a9064fbSMasahiro Yamada . { 90*0a9064fbSMasahiro Yamada unput(yytext[0]); 91*0a9064fbSMasahiro Yamada BEGIN(COMMAND); 92*0a9064fbSMasahiro Yamada } 93*0a9064fbSMasahiro Yamada 94*0a9064fbSMasahiro Yamada 95*0a9064fbSMasahiro Yamada <COMMAND>{ 96*0a9064fbSMasahiro Yamada {n}+ { 97*0a9064fbSMasahiro Yamada const struct kconf_id *id = kconf_id_lookup(yytext, yyleng); 98*0a9064fbSMasahiro Yamada BEGIN(PARAM); 99*0a9064fbSMasahiro Yamada current_pos.file = current_file; 100*0a9064fbSMasahiro Yamada current_pos.lineno = current_file->lineno; 101*0a9064fbSMasahiro Yamada if (id && id->flags & TF_COMMAND) { 102*0a9064fbSMasahiro Yamada zconflval.id = id; 103*0a9064fbSMasahiro Yamada return id->token; 104*0a9064fbSMasahiro Yamada } 105*0a9064fbSMasahiro Yamada alloc_string(yytext, yyleng); 106*0a9064fbSMasahiro Yamada zconflval.string = text; 107*0a9064fbSMasahiro Yamada return T_WORD; 108*0a9064fbSMasahiro Yamada } 109*0a9064fbSMasahiro Yamada . 110*0a9064fbSMasahiro Yamada \n { 111*0a9064fbSMasahiro Yamada BEGIN(INITIAL); 112*0a9064fbSMasahiro Yamada current_file->lineno++; 113*0a9064fbSMasahiro Yamada return T_EOL; 114*0a9064fbSMasahiro Yamada } 115*0a9064fbSMasahiro Yamada } 116*0a9064fbSMasahiro Yamada 117*0a9064fbSMasahiro Yamada <PARAM>{ 118*0a9064fbSMasahiro Yamada "&&" return T_AND; 119*0a9064fbSMasahiro Yamada "||" return T_OR; 120*0a9064fbSMasahiro Yamada "(" return T_OPEN_PAREN; 121*0a9064fbSMasahiro Yamada ")" return T_CLOSE_PAREN; 122*0a9064fbSMasahiro Yamada "!" return T_NOT; 123*0a9064fbSMasahiro Yamada "=" return T_EQUAL; 124*0a9064fbSMasahiro Yamada "!=" return T_UNEQUAL; 125*0a9064fbSMasahiro Yamada \"|\' { 126*0a9064fbSMasahiro Yamada str = yytext[0]; 127*0a9064fbSMasahiro Yamada new_string(); 128*0a9064fbSMasahiro Yamada BEGIN(STRING); 129*0a9064fbSMasahiro Yamada } 130*0a9064fbSMasahiro Yamada \n BEGIN(INITIAL); current_file->lineno++; return T_EOL; 131*0a9064fbSMasahiro Yamada --- /* ignore */ 132*0a9064fbSMasahiro Yamada ({n}|[-/.])+ { 133*0a9064fbSMasahiro Yamada const struct kconf_id *id = kconf_id_lookup(yytext, yyleng); 134*0a9064fbSMasahiro Yamada if (id && id->flags & TF_PARAM) { 135*0a9064fbSMasahiro Yamada zconflval.id = id; 136*0a9064fbSMasahiro Yamada return id->token; 137*0a9064fbSMasahiro Yamada } 138*0a9064fbSMasahiro Yamada alloc_string(yytext, yyleng); 139*0a9064fbSMasahiro Yamada zconflval.string = text; 140*0a9064fbSMasahiro Yamada return T_WORD; 141*0a9064fbSMasahiro Yamada } 142*0a9064fbSMasahiro Yamada #.* /* comment */ 143*0a9064fbSMasahiro Yamada \\\n current_file->lineno++; 144*0a9064fbSMasahiro Yamada . 145*0a9064fbSMasahiro Yamada <<EOF>> { 146*0a9064fbSMasahiro Yamada BEGIN(INITIAL); 147*0a9064fbSMasahiro Yamada } 148*0a9064fbSMasahiro Yamada } 149*0a9064fbSMasahiro Yamada 150*0a9064fbSMasahiro Yamada <STRING>{ 151*0a9064fbSMasahiro Yamada [^'"\\\n]+/\n { 152*0a9064fbSMasahiro Yamada append_string(yytext, yyleng); 153*0a9064fbSMasahiro Yamada zconflval.string = text; 154*0a9064fbSMasahiro Yamada return T_WORD_QUOTE; 155*0a9064fbSMasahiro Yamada } 156*0a9064fbSMasahiro Yamada [^'"\\\n]+ { 157*0a9064fbSMasahiro Yamada append_string(yytext, yyleng); 158*0a9064fbSMasahiro Yamada } 159*0a9064fbSMasahiro Yamada \\.?/\n { 160*0a9064fbSMasahiro Yamada append_string(yytext + 1, yyleng - 1); 161*0a9064fbSMasahiro Yamada zconflval.string = text; 162*0a9064fbSMasahiro Yamada return T_WORD_QUOTE; 163*0a9064fbSMasahiro Yamada } 164*0a9064fbSMasahiro Yamada \\.? { 165*0a9064fbSMasahiro Yamada append_string(yytext + 1, yyleng - 1); 166*0a9064fbSMasahiro Yamada } 167*0a9064fbSMasahiro Yamada \'|\" { 168*0a9064fbSMasahiro Yamada if (str == yytext[0]) { 169*0a9064fbSMasahiro Yamada BEGIN(PARAM); 170*0a9064fbSMasahiro Yamada zconflval.string = text; 171*0a9064fbSMasahiro Yamada return T_WORD_QUOTE; 172*0a9064fbSMasahiro Yamada } else 173*0a9064fbSMasahiro Yamada append_string(yytext, 1); 174*0a9064fbSMasahiro Yamada } 175*0a9064fbSMasahiro Yamada \n { 176*0a9064fbSMasahiro Yamada printf("%s:%d:warning: multi-line strings not supported\n", zconf_curname(), zconf_lineno()); 177*0a9064fbSMasahiro Yamada current_file->lineno++; 178*0a9064fbSMasahiro Yamada BEGIN(INITIAL); 179*0a9064fbSMasahiro Yamada return T_EOL; 180*0a9064fbSMasahiro Yamada } 181*0a9064fbSMasahiro Yamada <<EOF>> { 182*0a9064fbSMasahiro Yamada BEGIN(INITIAL); 183*0a9064fbSMasahiro Yamada } 184*0a9064fbSMasahiro Yamada } 185*0a9064fbSMasahiro Yamada 186*0a9064fbSMasahiro Yamada <HELP>{ 187*0a9064fbSMasahiro Yamada [ \t]+ { 188*0a9064fbSMasahiro Yamada ts = 0; 189*0a9064fbSMasahiro Yamada for (i = 0; i < yyleng; i++) { 190*0a9064fbSMasahiro Yamada if (yytext[i] == '\t') 191*0a9064fbSMasahiro Yamada ts = (ts & ~7) + 8; 192*0a9064fbSMasahiro Yamada else 193*0a9064fbSMasahiro Yamada ts++; 194*0a9064fbSMasahiro Yamada } 195*0a9064fbSMasahiro Yamada last_ts = ts; 196*0a9064fbSMasahiro Yamada if (first_ts) { 197*0a9064fbSMasahiro Yamada if (ts < first_ts) { 198*0a9064fbSMasahiro Yamada zconf_endhelp(); 199*0a9064fbSMasahiro Yamada return T_HELPTEXT; 200*0a9064fbSMasahiro Yamada } 201*0a9064fbSMasahiro Yamada ts -= first_ts; 202*0a9064fbSMasahiro Yamada while (ts > 8) { 203*0a9064fbSMasahiro Yamada append_string(" ", 8); 204*0a9064fbSMasahiro Yamada ts -= 8; 205*0a9064fbSMasahiro Yamada } 206*0a9064fbSMasahiro Yamada append_string(" ", ts); 207*0a9064fbSMasahiro Yamada } 208*0a9064fbSMasahiro Yamada } 209*0a9064fbSMasahiro Yamada [ \t]*\n/[^ \t\n] { 210*0a9064fbSMasahiro Yamada current_file->lineno++; 211*0a9064fbSMasahiro Yamada zconf_endhelp(); 212*0a9064fbSMasahiro Yamada return T_HELPTEXT; 213*0a9064fbSMasahiro Yamada } 214*0a9064fbSMasahiro Yamada [ \t]*\n { 215*0a9064fbSMasahiro Yamada current_file->lineno++; 216*0a9064fbSMasahiro Yamada append_string("\n", 1); 217*0a9064fbSMasahiro Yamada } 218*0a9064fbSMasahiro Yamada [^ \t\n].* { 219*0a9064fbSMasahiro Yamada while (yyleng) { 220*0a9064fbSMasahiro Yamada if ((yytext[yyleng-1] != ' ') && (yytext[yyleng-1] != '\t')) 221*0a9064fbSMasahiro Yamada break; 222*0a9064fbSMasahiro Yamada yyleng--; 223*0a9064fbSMasahiro Yamada } 224*0a9064fbSMasahiro Yamada append_string(yytext, yyleng); 225*0a9064fbSMasahiro Yamada if (!first_ts) 226*0a9064fbSMasahiro Yamada first_ts = last_ts; 227*0a9064fbSMasahiro Yamada } 228*0a9064fbSMasahiro Yamada <<EOF>> { 229*0a9064fbSMasahiro Yamada zconf_endhelp(); 230*0a9064fbSMasahiro Yamada return T_HELPTEXT; 231*0a9064fbSMasahiro Yamada } 232*0a9064fbSMasahiro Yamada } 233*0a9064fbSMasahiro Yamada 234*0a9064fbSMasahiro Yamada <<EOF>> { 235*0a9064fbSMasahiro Yamada if (current_file) { 236*0a9064fbSMasahiro Yamada zconf_endfile(); 237*0a9064fbSMasahiro Yamada return T_EOL; 238*0a9064fbSMasahiro Yamada } 239*0a9064fbSMasahiro Yamada fclose(yyin); 240*0a9064fbSMasahiro Yamada yyterminate(); 241*0a9064fbSMasahiro Yamada } 242*0a9064fbSMasahiro Yamada 243*0a9064fbSMasahiro Yamada %% 244*0a9064fbSMasahiro Yamada void zconf_starthelp(void) 245*0a9064fbSMasahiro Yamada { 246*0a9064fbSMasahiro Yamada new_string(); 247*0a9064fbSMasahiro Yamada last_ts = first_ts = 0; 248*0a9064fbSMasahiro Yamada BEGIN(HELP); 249*0a9064fbSMasahiro Yamada } 250*0a9064fbSMasahiro Yamada 251*0a9064fbSMasahiro Yamada static void zconf_endhelp(void) 252*0a9064fbSMasahiro Yamada { 253*0a9064fbSMasahiro Yamada zconflval.string = text; 254*0a9064fbSMasahiro Yamada BEGIN(INITIAL); 255*0a9064fbSMasahiro Yamada } 256*0a9064fbSMasahiro Yamada 257*0a9064fbSMasahiro Yamada 258*0a9064fbSMasahiro Yamada /* 259*0a9064fbSMasahiro Yamada * Try to open specified file with following names: 260*0a9064fbSMasahiro Yamada * ./name 261*0a9064fbSMasahiro Yamada * $(srctree)/name 262*0a9064fbSMasahiro Yamada * The latter is used when srctree is separate from objtree 263*0a9064fbSMasahiro Yamada * when compiling the kernel. 264*0a9064fbSMasahiro Yamada * Return NULL if file is not found. 265*0a9064fbSMasahiro Yamada */ 266*0a9064fbSMasahiro Yamada FILE *zconf_fopen(const char *name) 267*0a9064fbSMasahiro Yamada { 268*0a9064fbSMasahiro Yamada char *env, fullname[PATH_MAX+1]; 269*0a9064fbSMasahiro Yamada FILE *f; 270*0a9064fbSMasahiro Yamada 271*0a9064fbSMasahiro Yamada f = fopen(name, "r"); 272*0a9064fbSMasahiro Yamada if (!f && name != NULL && name[0] != '/') { 273*0a9064fbSMasahiro Yamada env = getenv(SRCTREE); 274*0a9064fbSMasahiro Yamada if (env) { 275*0a9064fbSMasahiro Yamada sprintf(fullname, "%s/%s", env, name); 276*0a9064fbSMasahiro Yamada f = fopen(fullname, "r"); 277*0a9064fbSMasahiro Yamada } 278*0a9064fbSMasahiro Yamada } 279*0a9064fbSMasahiro Yamada return f; 280*0a9064fbSMasahiro Yamada } 281*0a9064fbSMasahiro Yamada 282*0a9064fbSMasahiro Yamada void zconf_initscan(const char *name) 283*0a9064fbSMasahiro Yamada { 284*0a9064fbSMasahiro Yamada yyin = zconf_fopen(name); 285*0a9064fbSMasahiro Yamada if (!yyin) { 286*0a9064fbSMasahiro Yamada printf("can't find file %s\n", name); 287*0a9064fbSMasahiro Yamada exit(1); 288*0a9064fbSMasahiro Yamada } 289*0a9064fbSMasahiro Yamada 290*0a9064fbSMasahiro Yamada current_buf = xmalloc(sizeof(*current_buf)); 291*0a9064fbSMasahiro Yamada memset(current_buf, 0, sizeof(*current_buf)); 292*0a9064fbSMasahiro Yamada 293*0a9064fbSMasahiro Yamada current_file = file_lookup(name); 294*0a9064fbSMasahiro Yamada current_file->lineno = 1; 295*0a9064fbSMasahiro Yamada } 296*0a9064fbSMasahiro Yamada 297*0a9064fbSMasahiro Yamada void zconf_nextfile(const char *name) 298*0a9064fbSMasahiro Yamada { 299*0a9064fbSMasahiro Yamada struct file *iter; 300*0a9064fbSMasahiro Yamada struct file *file = file_lookup(name); 301*0a9064fbSMasahiro Yamada struct buffer *buf = xmalloc(sizeof(*buf)); 302*0a9064fbSMasahiro Yamada memset(buf, 0, sizeof(*buf)); 303*0a9064fbSMasahiro Yamada 304*0a9064fbSMasahiro Yamada current_buf->state = YY_CURRENT_BUFFER; 305*0a9064fbSMasahiro Yamada yyin = zconf_fopen(file->name); 306*0a9064fbSMasahiro Yamada if (!yyin) { 307*0a9064fbSMasahiro Yamada printf("%s:%d: can't open file \"%s\"\n", 308*0a9064fbSMasahiro Yamada zconf_curname(), zconf_lineno(), file->name); 309*0a9064fbSMasahiro Yamada exit(1); 310*0a9064fbSMasahiro Yamada } 311*0a9064fbSMasahiro Yamada yy_switch_to_buffer(yy_create_buffer(yyin, YY_BUF_SIZE)); 312*0a9064fbSMasahiro Yamada buf->parent = current_buf; 313*0a9064fbSMasahiro Yamada current_buf = buf; 314*0a9064fbSMasahiro Yamada 315*0a9064fbSMasahiro Yamada for (iter = current_file->parent; iter; iter = iter->parent ) { 316*0a9064fbSMasahiro Yamada if (!strcmp(current_file->name,iter->name) ) { 317*0a9064fbSMasahiro Yamada printf("%s:%d: recursive inclusion detected. " 318*0a9064fbSMasahiro Yamada "Inclusion path:\n current file : '%s'\n", 319*0a9064fbSMasahiro Yamada zconf_curname(), zconf_lineno(), 320*0a9064fbSMasahiro Yamada zconf_curname()); 321*0a9064fbSMasahiro Yamada iter = current_file->parent; 322*0a9064fbSMasahiro Yamada while (iter && \ 323*0a9064fbSMasahiro Yamada strcmp(iter->name,current_file->name)) { 324*0a9064fbSMasahiro Yamada printf(" included from: '%s:%d'\n", 325*0a9064fbSMasahiro Yamada iter->name, iter->lineno-1); 326*0a9064fbSMasahiro Yamada iter = iter->parent; 327*0a9064fbSMasahiro Yamada } 328*0a9064fbSMasahiro Yamada if (iter) 329*0a9064fbSMasahiro Yamada printf(" included from: '%s:%d'\n", 330*0a9064fbSMasahiro Yamada iter->name, iter->lineno+1); 331*0a9064fbSMasahiro Yamada exit(1); 332*0a9064fbSMasahiro Yamada } 333*0a9064fbSMasahiro Yamada } 334*0a9064fbSMasahiro Yamada file->lineno = 1; 335*0a9064fbSMasahiro Yamada file->parent = current_file; 336*0a9064fbSMasahiro Yamada current_file = file; 337*0a9064fbSMasahiro Yamada } 338*0a9064fbSMasahiro Yamada 339*0a9064fbSMasahiro Yamada static void zconf_endfile(void) 340*0a9064fbSMasahiro Yamada { 341*0a9064fbSMasahiro Yamada struct buffer *parent; 342*0a9064fbSMasahiro Yamada 343*0a9064fbSMasahiro Yamada current_file = current_file->parent; 344*0a9064fbSMasahiro Yamada 345*0a9064fbSMasahiro Yamada parent = current_buf->parent; 346*0a9064fbSMasahiro Yamada if (parent) { 347*0a9064fbSMasahiro Yamada fclose(yyin); 348*0a9064fbSMasahiro Yamada yy_delete_buffer(YY_CURRENT_BUFFER); 349*0a9064fbSMasahiro Yamada yy_switch_to_buffer(parent->state); 350*0a9064fbSMasahiro Yamada } 351*0a9064fbSMasahiro Yamada free(current_buf); 352*0a9064fbSMasahiro Yamada current_buf = parent; 353*0a9064fbSMasahiro Yamada } 354*0a9064fbSMasahiro Yamada 355*0a9064fbSMasahiro Yamada int zconf_lineno(void) 356*0a9064fbSMasahiro Yamada { 357*0a9064fbSMasahiro Yamada return current_pos.lineno; 358*0a9064fbSMasahiro Yamada } 359*0a9064fbSMasahiro Yamada 360*0a9064fbSMasahiro Yamada const char *zconf_curname(void) 361*0a9064fbSMasahiro Yamada { 362*0a9064fbSMasahiro Yamada return current_pos.file ? current_pos.file->name : "<none>"; 363*0a9064fbSMasahiro Yamada } 364