Anforderungen  |   Konzepte  |   Entwurf  |   Entwicklung  |   Qualitätssicherung  |   Lebenszyklus  |   Steuerung
 
 
 
 


Quelle  lexer.l   Sprache: unbekannt

 
/* SPDX-License-Identifier: GPL-2.0 */
/*
 * 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 <xalloc.h>
#include "lkc.h"
#include "preprocess.h"

#include "parser.tab.h"

#define YY_DECL  static int yylex1(void)

#define START_STRSIZE 16

/* The Kconfig file currently being parsed.  */
const char *cur_filename;

/*
 * The line number of the current statement. This does not match yylineno.
 * yylineno is used by the lexer, while cur_lineno is used by the parser.
 */
int cur_lineno;

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;
 int yylineno;
 const char *filename;
 int source_lineno;
};

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",
         cur_filename, yylineno, chr);
}
%}

n [A-Za-z0-9_-]

%%
 char open_quote = 0;

#.*   /* 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;
"prompt"  return T_PROMPT;
"range"   return T_RANGE;
"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;
\"|\'   {
    open_quote = 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 (open_quote == 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",
   cur_filename, cur_lineno);
  unput('\n');
  BEGIN(INITIAL);
  yylval.string = text;
  return T_WORD_QUOTE;
 }
 <<EOF>> {
  BEGIN(INITIAL);
  yylval.string = text;
  return T_WORD_QUOTE;
 }
}

<HELP>{
 [ \t]+ {
  int ts, i;

  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",
   cur_filename, yylineno);

 if (current_buf) {
  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 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.
    */
   cur_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.
  * input() returns 0 (not EOF!) when it reachs the end of file.
  */
 while ((c = input()) != 0) {
  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);
 }

 cur_filename = file_lookup(name);
 yylineno = 1;
}

void zconf_nextfile(const char *name)
{
 struct buffer *buf = xmalloc(sizeof(*buf));
 bool recur_include = false;

 buf->state = YY_CURRENT_BUFFER;
 buf->yylineno = yylineno;
 buf->filename = cur_filename;
 buf->source_lineno = cur_lineno;
 buf->parent = current_buf;
 current_buf = buf;
 yyin = zconf_fopen(name);
 if (!yyin) {
  fprintf(stderr, "%s:%d: can't open file \"%s\"\n",
   cur_filename, cur_lineno, name);
  exit(1);
 }
 yy_switch_to_buffer(yy_create_buffer(yyin, YY_BUF_SIZE));

 for (buf = current_buf; buf; buf = buf->parent) {
  if (!strcmp(buf->filename, name))
   recur_include = true;
 }

 if (recur_include) {
  fprintf(stderr,
   "Recursive inclusion detected.\n"
   "Inclusion path:\n"
   "  current file : %s\n", name);

  for (buf = current_buf; buf; buf = buf->parent)
   fprintf(stderr, "  included from: %s:%d\n",
    buf->filename, buf->source_lineno);
  exit(1);
 }

 yylineno = 1;
 cur_filename = file_lookup(name);
}

static void zconf_endfile(void)
{
 struct buffer *tmp;

 fclose(yyin);
 yy_delete_buffer(YY_CURRENT_BUFFER);
 yy_switch_to_buffer(current_buf->state);
 yylineno = current_buf->yylineno;
 cur_filename = current_buf->filename;
 tmp = current_buf;
 current_buf = current_buf->parent;
 free(tmp);
}

[ Dauer der Verarbeitung: 0.13 Sekunden  (vorverarbeitet)  ]

                                                                                                                                                                                                                                                                                                                                                                                                     


Neuigkeiten

     Aktuelles
     Motto des Tages

Software

     Produkte
     Quellcodebibliothek

Aktivitäten

     Artikel über Sicherheit
     Anleitung zur Aktivierung von SSL

Muße

     Gedichte
     Musik
     Bilder

Jenseits des Üblichen ....

Besucherstatistik

Besucherstatistik

Monitoring

Montastic status badge