123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521 |
- /* SPDX-License-Identifier: GPL-2.0-only */
- /*
- * Copyright (C) 2002 Roman Zippel <zippel@linux-m68k.org>
- */
- %option nostdinit noyywrap never-interactive full ecs
- %option 8bit nodefault yylineno
- %x ASSIGN_VAL HELP STRING
- %{
- #include <assert.h>
- #include <limits.h>
- #include <stdio.h>
- #include <stdlib.h>
- #include <string.h>
- #include <glob.h>
- #include <libgen.h>
- #include "lkc.h"
- #include "parser.tab.h"
- #define YY_DECL static int yylex1(void)
- #define START_STRSIZE 16
- static struct {
- struct file *file;
- int lineno;
- } current_pos;
- static int prev_prev_token = T_EOL;
- static int prev_token = T_EOL;
- static char *text;
- static int text_size, text_asize;
- struct buffer {
- struct buffer *parent;
- YY_BUFFER_STATE state;
- };
- static struct buffer *current_buf;
- static int last_ts, first_ts;
- static char *expand_token(const char *in, size_t n);
- static void append_expanded_string(const char *in);
- static void zconf_endhelp(void);
- static void zconf_endfile(void);
- static void new_string(void)
- {
- text = xmalloc(START_STRSIZE);
- text_asize = START_STRSIZE;
- text_size = 0;
- *text = 0;
- }
- static void append_string(const char *str, int size)
- {
- int new_size = text_size + size + 1;
- if (new_size > text_asize) {
- new_size += START_STRSIZE - 1;
- new_size &= -START_STRSIZE;
- text = xrealloc(text, new_size);
- text_asize = new_size;
- }
- memcpy(text + text_size, str, size);
- text_size += size;
- text[text_size] = 0;
- }
- static void alloc_string(const char *str, int size)
- {
- text = xmalloc(size + 1);
- memcpy(text, str, size);
- text[size] = 0;
- }
- static void warn_ignored_character(char chr)
- {
- fprintf(stderr,
- "%s:%d:warning: ignoring unsupported character '%c'\n",
- current_file->name, yylineno, chr);
- }
- %}
- n [A-Za-z0-9_-]
- %%
- int str = 0;
- int ts, i;
- #.* /* ignore comment */
- [ \t]* /* whitespaces */
- \\\n /* escaped new line */
- \n return T_EOL;
- "bool" return T_BOOL;
- "choice" return T_CHOICE;
- "comment" return T_COMMENT;
- "config" return T_CONFIG;
- "def_bool" return T_DEF_BOOL;
- "def_tristate" return T_DEF_TRISTATE;
- "default" return T_DEFAULT;
- "depends" return T_DEPENDS;
- "endchoice" return T_ENDCHOICE;
- "endif" return T_ENDIF;
- "endmenu" return T_ENDMENU;
- "help" return T_HELP;
- "hex" return T_HEX;
- "if" return T_IF;
- "imply" return T_IMPLY;
- "int" return T_INT;
- "mainmenu" return T_MAINMENU;
- "menu" return T_MENU;
- "menuconfig" return T_MENUCONFIG;
- "modules" return T_MODULES;
- "on" return T_ON;
- "optional" return T_OPTIONAL;
- "prompt" return T_PROMPT;
- "range" return T_RANGE;
- "reset" return T_RESET;
- "select" return T_SELECT;
- "source" return T_SOURCE;
- "string" return T_STRING;
- "tristate" return T_TRISTATE;
- "visible" return T_VISIBLE;
- "||" return T_OR;
- "&&" return T_AND;
- "=" return T_EQUAL;
- "!=" return T_UNEQUAL;
- "<" return T_LESS;
- "<=" return T_LESS_EQUAL;
- ">" return T_GREATER;
- ">=" return T_GREATER_EQUAL;
- "!" return T_NOT;
- "(" return T_OPEN_PAREN;
- ")" return T_CLOSE_PAREN;
- ":=" return T_COLON_EQUAL;
- "+=" return T_PLUS_EQUAL;
- \"|\' {
- str = yytext[0];
- new_string();
- BEGIN(STRING);
- }
- ({n}|[/.])+ {
- alloc_string(yytext, yyleng);
- yylval.string = text;
- return T_WORD;
- }
- ({n}|[/.$])+ {
- /* this token includes at least one '$' */
- yylval.string = expand_token(yytext, yyleng);
- if (strlen(yylval.string))
- return T_WORD;
- free(yylval.string);
- }
- . warn_ignored_character(*yytext);
- <ASSIGN_VAL>{
- [^[:blank:]\n]+.* {
- alloc_string(yytext, yyleng);
- yylval.string = text;
- return T_ASSIGN_VAL;
- }
- \n { BEGIN(INITIAL); return T_EOL; }
- .
- }
- <STRING>{
- "$".* append_expanded_string(yytext);
- [^$'"\\\n]+ {
- append_string(yytext, yyleng);
- }
- \\.? {
- append_string(yytext + 1, yyleng - 1);
- }
- \'|\" {
- if (str == yytext[0]) {
- BEGIN(INITIAL);
- yylval.string = text;
- return T_WORD_QUOTE;
- } else
- append_string(yytext, 1);
- }
- \n {
- fprintf(stderr,
- "%s:%d:warning: multi-line strings not supported\n",
- zconf_curname(), zconf_lineno());
- unput('\n');
- BEGIN(INITIAL);
- yylval.string = text;
- return T_WORD_QUOTE;
- }
- <<EOF>> {
- BEGIN(INITIAL);
- yylval.string = text;
- return T_WORD_QUOTE;
- }
- }
- <HELP>{
- [ \t]+ {
- ts = 0;
- for (i = 0; i < yyleng; i++) {
- if (yytext[i] == '\t')
- ts = (ts & ~7) + 8;
- else
- ts++;
- }
- last_ts = ts;
- if (first_ts) {
- if (ts < first_ts) {
- zconf_endhelp();
- return T_HELPTEXT;
- }
- ts -= first_ts;
- while (ts > 8) {
- append_string(" ", 8);
- ts -= 8;
- }
- append_string(" ", ts);
- }
- }
- [ \t]*\n/[^ \t\n] {
- zconf_endhelp();
- return T_HELPTEXT;
- }
- [ \t]*\n {
- append_string("\n", 1);
- }
- [^ \t\n].* {
- while (yyleng) {
- if ((yytext[yyleng-1] != ' ') && (yytext[yyleng-1] != '\t'))
- break;
- yyleng--;
- }
- append_string(yytext, yyleng);
- if (!first_ts)
- first_ts = last_ts;
- }
- <<EOF>> {
- zconf_endhelp();
- return T_HELPTEXT;
- }
- }
- <<EOF>> {
- BEGIN(INITIAL);
- if (prev_token != T_EOL && prev_token != T_HELPTEXT)
- fprintf(stderr, "%s:%d:warning: no new line at end of file\n",
- current_file->name, yylineno);
- if (current_file) {
- zconf_endfile();
- return T_EOL;
- }
- fclose(yyin);
- yyterminate();
- }
- %%
- /* second stage lexer */
- int yylex(void)
- {
- int token;
- repeat:
- token = yylex1();
- if (prev_token == T_EOL || prev_token == T_HELPTEXT) {
- if (token == T_EOL) {
- /* Do not pass unneeded T_EOL to the parser. */
- goto repeat;
- } else {
- /*
- * For the parser, update file/lineno at the first token
- * of each statement. Generally, \n is a statement
- * terminator in Kconfig, but it is not always true
- * because \n could be escaped by a backslash.
- */
- current_pos.file = current_file;
- current_pos.lineno = yylineno;
- }
- }
- if (prev_prev_token == T_EOL && prev_token == T_WORD &&
- (token == T_EQUAL || token == T_COLON_EQUAL || token == T_PLUS_EQUAL))
- BEGIN(ASSIGN_VAL);
- prev_prev_token = prev_token;
- prev_token = token;
- return token;
- }
- static char *expand_token(const char *in, size_t n)
- {
- char *out;
- int c;
- char c2;
- const char *rest, *end;
- new_string();
- append_string(in, n);
- /* get the whole line because we do not know the end of token. */
- while ((c = input()) != EOF) {
- if (c == '\n') {
- unput(c);
- break;
- }
- c2 = c;
- append_string(&c2, 1);
- }
- rest = text;
- out = expand_one_token(&rest);
- /* push back unused characters to the input stream */
- end = rest + strlen(rest);
- while (end > rest)
- unput(*--end);
- free(text);
- return out;
- }
- static void append_expanded_string(const char *str)
- {
- const char *end;
- char *res;
- str++;
- res = expand_dollar(&str);
- /* push back unused characters to the input stream */
- end = str + strlen(str);
- while (end > str)
- unput(*--end);
- append_string(res, strlen(res));
- free(res);
- }
- void zconf_starthelp(void)
- {
- new_string();
- last_ts = first_ts = 0;
- BEGIN(HELP);
- }
- static void zconf_endhelp(void)
- {
- yylval.string = text;
- BEGIN(INITIAL);
- }
- /*
- * Try to open specified file with following names:
- * ./name
- * $(srctree)/name
- * The latter is used when srctree is separate from objtree
- * when compiling the kernel.
- * Return NULL if file is not found.
- */
- FILE *zconf_fopen(const char *name)
- {
- char *env, fullname[PATH_MAX+1];
- FILE *f;
- f = fopen(name, "r");
- if (!f && name != NULL && name[0] != '/') {
- env = getenv(SRCTREE);
- if (env) {
- snprintf(fullname, sizeof(fullname),
- "%s/%s", env, name);
- f = fopen(fullname, "r");
- }
- }
- return f;
- }
- void zconf_initscan(const char *name)
- {
- yyin = zconf_fopen(name);
- if (!yyin) {
- fprintf(stderr, "can't find file %s\n", name);
- exit(1);
- }
- current_buf = xmalloc(sizeof(*current_buf));
- memset(current_buf, 0, sizeof(*current_buf));
- current_file = file_lookup(name);
- yylineno = 1;
- }
- static void __zconf_nextfile(const char *name)
- {
- struct file *iter;
- struct file *file = file_lookup(name);
- struct buffer *buf = xmalloc(sizeof(*buf));
- memset(buf, 0, sizeof(*buf));
- current_buf->state = YY_CURRENT_BUFFER;
- yyin = zconf_fopen(file->name);
- if (!yyin) {
- fprintf(stderr, "%s:%d: can't open file \"%s\"\n",
- zconf_curname(), zconf_lineno(), file->name);
- exit(1);
- }
- yy_switch_to_buffer(yy_create_buffer(yyin, YY_BUF_SIZE));
- buf->parent = current_buf;
- current_buf = buf;
- current_file->lineno = yylineno;
- file->parent = current_file;
- for (iter = current_file; iter; iter = iter->parent) {
- if (!strcmp(iter->name, file->name)) {
- fprintf(stderr,
- "Recursive inclusion detected.\n"
- "Inclusion path:\n"
- " current file : %s\n", file->name);
- iter = file;
- do {
- iter = iter->parent;
- fprintf(stderr, " included from: %s:%d\n",
- iter->name, iter->lineno - 1);
- } while (strcmp(iter->name, file->name));
- exit(1);
- }
- }
- yylineno = 1;
- current_file = file;
- }
- void zconf_nextfile(const char *name)
- {
- glob_t gl;
- int err;
- int i;
- char path[PATH_MAX], *p;
- err = glob(name, GLOB_ERR | GLOB_MARK, NULL, &gl);
- /* ignore wildcard patterns that return no result */
- if (err == GLOB_NOMATCH && strchr(name, '*')) {
- err = 0;
- gl.gl_pathc = 0;
- }
- if (err == GLOB_NOMATCH) {
- p = strdup(current_file->name);
- if (p) {
- snprintf(path, sizeof(path), "%s/%s", dirname(p), name);
- err = glob(path, GLOB_ERR | GLOB_MARK, NULL, &gl);
- free(p);
- }
- }
- if (err) {
- const char *reason = "unknown error";
- switch (err) {
- case GLOB_NOSPACE:
- reason = "out of memory";
- break;
- case GLOB_ABORTED:
- reason = "read error";
- break;
- case GLOB_NOMATCH:
- reason = "No files found";
- break;
- default:
- break;
- }
- printf("%s:%d: glob failed: %s \"%s\"\n", zconf_curname(), zconf_lineno(),
- reason, name);
- exit(1);
- }
- for (i = 0; i < gl.gl_pathc; i++)
- __zconf_nextfile(gl.gl_pathv[i]);
- }
- static void zconf_endfile(void)
- {
- struct buffer *parent;
- current_file = current_file->parent;
- if (current_file)
- yylineno = current_file->lineno;
- parent = current_buf->parent;
- if (parent) {
- fclose(yyin);
- yy_delete_buffer(YY_CURRENT_BUFFER);
- yy_switch_to_buffer(parent->state);
- }
- free(current_buf);
- current_buf = parent;
- }
- int zconf_lineno(void)
- {
- return current_pos.lineno;
- }
- const char *zconf_curname(void)
- {
- return current_pos.file ? current_pos.file->name : "<none>";
- }
|