%option reentrant noyywrap
%option extra-type="glcpp_parser_t *"
+%x ST_DEFINE
+%x ST_DEFVAL
+
SPACE [[:space:]]
NONSPACE [^[:space:]]
NEWLINE [\n]
%%
-{HASH}define{HSPACE}* {
+{HASH}undef{HSPACE}* {
+ return UNDEF;
+}
+
+ /* We use the ST_DEFINE and ST_DEFVAL states so that we can
+ * pass a space token, (yes, a token for whitespace!), since
+ * the preprocessor specification requires distinguishing
+ * "#define foo()" from "#define foo ()".
+ */
+{HASH}define{HSPACE}* {
+ BEGIN ST_DEFINE;
return DEFINE;
}
-{HASH}undef{HSPACE}* {
- return UNDEF;
+<ST_DEFINE>{IDENTIFIER} {
+ BEGIN ST_DEFVAL;
+ yylval.str = xtalloc_strdup (yyextra, yytext);
+ return IDENTIFIER;
}
+<ST_DEFVAL>\n {
+ BEGIN INITIAL;
+ return NEWLINE;
+}
+
+<ST_DEFVAL>{HSPACE}+ {
+ BEGIN INITIAL;
+ return SPACE;
+}
-{IDENTIFIER} {
+<ST_DEFVAL>"(" {
+ BEGIN INITIAL;
+ return '(';
+}
+
+{IDENTIFIER} {
yylval.str = xtalloc_strdup (yyextra, yytext);
switch (glcpp_parser_macro_type (yyextra, yylval.str))
{
}
}
-[(),] { return yytext[0]; }
+[(),] {
+ return yytext[0];
+}
{TOKEN} {
yylval.str = xtalloc_strdup (yyextra, yytext);
%parse-param {glcpp_parser_t *parser}
%lex-param {void *scanner}
-%token DEFINE FUNC_MACRO IDENTIFIER NEWLINE OBJ_MACRO TOKEN UNDEF
-%type <str> FUNC_MACRO IDENTIFIER OBJ_MACRO TOKEN string
+%token DEFINE FUNC_MACRO IDENTIFIER NEWLINE OBJ_MACRO SPACE TOKEN UNDEF
+%type <str> FUNC_MACRO IDENTIFIER OBJ_MACRO TOKEN word word_or_symbol
%type <list> argument argument_list parameter_list replacement_list
%%
}
| macro
| directive_with_newline
-| NEWLINE {
- printf ("\n");
- }
+| NEWLINE { printf ("\n"); }
+| '(' { printf ("("); }
+| ')' { printf (")"); }
+| ',' { printf (","); }
;
macro:
/* empty */ {
$$ = _list_create (parser);
}
-| argument string {
+| argument word {
_list_append_item ($1, $2);
talloc_free ($2);
}
;
directive:
- DEFINE IDENTIFIER replacement_list {
- _define_object_macro (parser, $2, $3);
+ DEFINE IDENTIFIER {
+ list_t *list = _list_create (parser);
+ _define_object_macro (parser, $2, list);
+ }
+| DEFINE IDENTIFIER SPACE replacement_list {
+ _define_object_macro (parser, $2, $4);
}
| DEFINE IDENTIFIER '(' parameter_list ')' replacement_list {
_define_function_macro (parser, $2, $4, $6);
/* empty */ {
$$ = _list_create (parser);
}
-| replacement_list string {
+| replacement_list word_or_symbol {
_list_append_item ($1, $2);
talloc_free ($2);
$$ = $1;
}
;
-string:
+word_or_symbol:
+ word { $$ = $1; }
+| '(' { $$ = xtalloc_strdup (parser, "("); }
+| ')' { $$ = xtalloc_strdup (parser, ")"); }
+| ',' { $$ = xtalloc_strdup (parser, ","); }
+;
+
+word:
IDENTIFIER { $$ = $1; }
| FUNC_MACRO { $$ = $1; }
| OBJ_MACRO { $$ = $1; }