mirror of
https://git.savannah.gnu.org/git/bison.git
synced 2026-03-10 12:53:03 +00:00
Make it possible to return a symbol_type from yylex.
* data/lalr1.cc (b4_lex_symbol_if): New. (parse): When lex_symbol is defined, expected yylex to return the complete lookahead. * etc/bench.pl.in (generate_grammar_list): Extend to support this yylex interface. (bench_variant_parser): Exercise it.
This commit is contained in:
10
ChangeLog
10
ChangeLog
@@ -1,3 +1,13 @@
|
|||||||
|
2008-11-11 Akim Demaille <demaille@gostai.com>
|
||||||
|
|
||||||
|
Make it possible to return a symbol_type from yylex.
|
||||||
|
* data/lalr1.cc (b4_lex_symbol_if): New.
|
||||||
|
(parse): When lex_symbol is defined, expected yylex to return the
|
||||||
|
complete lookahead.
|
||||||
|
* etc/bench.pl.in (generate_grammar_list): Extend to support this
|
||||||
|
yylex interface.
|
||||||
|
(bench_variant_parser): Exercise it.
|
||||||
|
|
||||||
2008-11-11 Akim Demaille <demaille@gostai.com>
|
2008-11-11 Akim Demaille <demaille@gostai.com>
|
||||||
|
|
||||||
Remove useless bench case.
|
Remove useless bench case.
|
||||||
|
|||||||
@@ -53,6 +53,12 @@ b4_variant_if([
|
|||||||
]) # b4_variant_if
|
]) # b4_variant_if
|
||||||
|
|
||||||
|
|
||||||
|
# b4_lex_symbol_if([IF-YYLEX-RETURNS-A-COMPLETE-SYMBOL], [IF-NOT])
|
||||||
|
# ----------------------------------------------------------------
|
||||||
|
m4_define([b4_lex_symbol_if],
|
||||||
|
[b4_percent_define_ifdef([[lex_symbol]], [$1], [$2])])
|
||||||
|
|
||||||
|
|
||||||
# b4_assert_if([IF-ASSERTIONS-ARE-USED], [IF-NOT])
|
# b4_assert_if([IF-ASSERTIONS-ARE-USED], [IF-NOT])
|
||||||
# ------------------------------------------------
|
# ------------------------------------------------
|
||||||
m4_define([b4_assert_if],
|
m4_define([b4_assert_if],
|
||||||
@@ -1144,14 +1150,15 @@ m4_popdef([b4_at_dollar])])dnl
|
|||||||
/* Read a lookahead token. */
|
/* Read a lookahead token. */
|
||||||
if (yyempty)
|
if (yyempty)
|
||||||
{
|
{
|
||||||
YYCDEBUG << "Reading a token: ";
|
YYCDEBUG << "Reading a token: ";
|
||||||
yyla.type = yytranslate_ (]b4_c_function_call([yylex], [int],
|
]b4_lex_symbol_if(
|
||||||
|
[ yyla = yylex();],
|
||||||
|
[ yyla.type = yytranslate_ (b4_c_function_call([yylex], [int],
|
||||||
[[YYSTYPE*], [&yyla.value]][]dnl
|
[[YYSTYPE*], [&yyla.value]][]dnl
|
||||||
b4_locations_if([, [[location*], [&yyla.location]]])dnl
|
b4_locations_if([, [[location*], [&yyla.location]]])dnl
|
||||||
m4_ifdef([b4_lex_param], [, ]b4_lex_param))[);
|
m4_ifdef([b4_lex_param], [, ]b4_lex_param)));])[
|
||||||
yyempty = false;
|
yyempty = false;
|
||||||
}
|
}
|
||||||
|
|
||||||
YY_SYMBOL_PRINT ("Next token is", yyla);
|
YY_SYMBOL_PRINT ("Next token is", yyla);
|
||||||
|
|
||||||
/* If the proper action on seeing token YYLA.TYPE is to reduce or
|
/* If the proper action on seeing token YYLA.TYPE is to reduce or
|
||||||
|
|||||||
@@ -580,11 +580,13 @@ sub generate_grammar_list ($$@)
|
|||||||
my ($base, $max, @directive) = @_;
|
my ($base, $max, @directive) = @_;
|
||||||
my $directives = directives ($base, @directive);
|
my $directives = directives ($base, @directive);
|
||||||
my $variant = grep { /%define "?variant"?/ } @directive;
|
my $variant = grep { /%define "?variant"?/ } @directive;
|
||||||
|
my $lex_symbol = grep { /%define "?lex_symbol"?/ } @directive;
|
||||||
my $out = new IO::File ">$base.y"
|
my $out = new IO::File ">$base.y"
|
||||||
or die;
|
or die;
|
||||||
print $out <<EOF;
|
print $out <<EOF;
|
||||||
%language "C++"
|
%language "C++"
|
||||||
%defines
|
%defines
|
||||||
|
%locations
|
||||||
$directives
|
$directives
|
||||||
|
|
||||||
%code requires // *.h
|
%code requires // *.h
|
||||||
@@ -598,22 +600,18 @@ $directives
|
|||||||
#include <iostream>
|
#include <iostream>
|
||||||
#include <sstream>
|
#include <sstream>
|
||||||
|
|
||||||
// Prototype of the yylex function providing subsequent tokens.
|
|
||||||
static yy::parser::token_type yylex(yy::parser::semantic_type* yylval);
|
|
||||||
|
|
||||||
#define STAGE_MAX ($max * 10) // max = $max
|
#define STAGE_MAX ($max * 10) // max = $max
|
||||||
|
|
||||||
|
#define USE_LEX_SYMBOL $lex_symbol
|
||||||
#define USE_VARIANTS $variant
|
#define USE_VARIANTS $variant
|
||||||
#if USE_VARIANTS
|
|
||||||
# define IF_VARIANTS(True, False) True
|
|
||||||
#else
|
|
||||||
# define IF_VARIANTS(True, False) False
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef ONE_STAGE_BUILD
|
// Prototype of the yylex function providing subsequent tokens.
|
||||||
# define IF_ONE_STAGE_BUILD(True, False) True
|
static
|
||||||
|
#if USE_LEX_SYMBOL
|
||||||
|
yy::parser::symbol_type yylex();
|
||||||
#else
|
#else
|
||||||
# define IF_ONE_STAGE_BUILD(True, False) False
|
yy::parser::token_type yylex(yy::parser::semantic_type* yylval,
|
||||||
|
yy::parser::location_type* yylloc);
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
// Conversion to string.
|
// Conversion to string.
|
||||||
@@ -627,6 +625,8 @@ static yy::parser::token_type yylex(yy::parser::semantic_type* yylval);
|
|||||||
return o.str ();
|
return o.str ();
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
%token END_OF_FILE 0
|
||||||
EOF
|
EOF
|
||||||
|
|
||||||
if ($variant)
|
if ($variant)
|
||||||
@@ -636,7 +636,6 @@ EOF
|
|||||||
%token <int> NUMBER
|
%token <int> NUMBER
|
||||||
%printer { std::cerr << "Number: " << $$; } <int>
|
%printer { std::cerr << "Number: " << $$; } <int>
|
||||||
%printer { std::cerr << "Text: " << $$; } <std::string>
|
%printer { std::cerr << "Text: " << $$; } <std::string>
|
||||||
%token END_OF_FILE 0
|
|
||||||
%type <std::string> text result
|
%type <std::string> text result
|
||||||
|
|
||||||
%%
|
%%
|
||||||
@@ -660,7 +659,6 @@ EOF
|
|||||||
%token <ival> NUMBER
|
%token <ival> NUMBER
|
||||||
%printer { std::cerr << "Number: " << $$; } <ival>
|
%printer { std::cerr << "Number: " << $$; } <ival>
|
||||||
%printer { std::cerr << "Text: " << *$$; } <sval>
|
%printer { std::cerr << "Text: " << *$$; } <sval>
|
||||||
%token END_OF_FILE 0
|
|
||||||
%type <sval> text result
|
%type <sval> text result
|
||||||
|
|
||||||
%%
|
%%
|
||||||
@@ -678,39 +676,63 @@ EOF
|
|||||||
|
|
||||||
print $out <<'EOF';
|
print $out <<'EOF';
|
||||||
%%
|
%%
|
||||||
|
#
|
||||||
|
|
||||||
static
|
static
|
||||||
yy::parser::token_type
|
#if USE_LEX_SYMBOL
|
||||||
yylex(yy::parser::semantic_type* yylval)
|
yy::parser::symbol_type yylex()
|
||||||
|
#else
|
||||||
|
yy::parser::token_type yylex(yy::parser::semantic_type* yylval,
|
||||||
|
yy::parser::location_type* yylloc)
|
||||||
|
#endif
|
||||||
{
|
{
|
||||||
|
typedef yy::parser::token token;
|
||||||
static int stage = -1;
|
static int stage = -1;
|
||||||
++stage;
|
++stage;
|
||||||
if (stage == STAGE_MAX)
|
if (stage == STAGE_MAX)
|
||||||
return yy::parser::token::END_OF_FILE;
|
{
|
||||||
|
#if USE_LEX_SYMBOL
|
||||||
|
return yy::parser::make_symbol <token::END_OF_FILE> (yy::location());
|
||||||
|
#else
|
||||||
|
*yylloc = yy::location ();
|
||||||
|
return token::END_OF_FILE;
|
||||||
|
#endif
|
||||||
|
}
|
||||||
else if (stage % 2)
|
else if (stage % 2)
|
||||||
{
|
{
|
||||||
#if USE_VARIANTS
|
#if USE_LEX_SYMBOL
|
||||||
# ifdef ONE_STAGE_BUILD
|
return yy::parser::make_symbol <token::NUMBER> (stage, yy::location());
|
||||||
|
#elif defined ONE_STAGE_BUILD
|
||||||
yylval->build(stage);
|
yylval->build(stage);
|
||||||
# else
|
*yylloc = yy::location ();
|
||||||
|
return token::NUMBER;
|
||||||
|
#elif USE_VARIANTS
|
||||||
yylval->build<int>() = stage;
|
yylval->build<int>() = stage;
|
||||||
# endif
|
*yylloc = yy::location ();
|
||||||
|
return token::NUMBER;
|
||||||
#else
|
#else
|
||||||
yylval->ival = stage;
|
yylval->ival = stage;
|
||||||
|
*yylloc = yy::location ();
|
||||||
|
return token::NUMBER;
|
||||||
#endif
|
#endif
|
||||||
return yy::parser::token::NUMBER;
|
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
#if USE_VARIANTS
|
#if USE_LEX_SYMBOL
|
||||||
# ifdef ONE_STAGE_BUILD
|
return yy::parser::make_symbol <token::TEXT> ("A string.", yy::location());
|
||||||
|
#elif defined ONE_STAGE_BUILD
|
||||||
yylval->build(std::string("A string."));
|
yylval->build(std::string("A string."));
|
||||||
# else
|
*yylloc = yy::location ();
|
||||||
|
return token::TEXT;
|
||||||
|
#elif USE_VARIANTS
|
||||||
yylval->build<std::string>() = std::string("A string.");
|
yylval->build<std::string>() = std::string("A string.");
|
||||||
# endif
|
*yylloc = yy::location ();
|
||||||
|
return token::TEXT;
|
||||||
#else
|
#else
|
||||||
yylval->sval = new std::string("A string.");
|
yylval->sval = new std::string("A string.");
|
||||||
|
*yylloc = yy::location ();
|
||||||
|
return token::TEXT;
|
||||||
#endif
|
#endif
|
||||||
return yy::parser::token::TEXT;
|
|
||||||
}
|
}
|
||||||
abort();
|
abort();
|
||||||
}
|
}
|
||||||
@@ -886,7 +908,7 @@ sub bench_push_parser ()
|
|||||||
|
|
||||||
=item C<bench_variant_parser ()>
|
=item C<bench_variant_parser ()>
|
||||||
|
|
||||||
Bench the C++ lalr1.cc parser using Boost.Variants or %union.
|
Bench the C++ lalr1.cc parser using variants or %union.
|
||||||
|
|
||||||
=cut
|
=cut
|
||||||
|
|
||||||
@@ -894,11 +916,10 @@ sub bench_variant_parser ()
|
|||||||
{
|
{
|
||||||
bench ('list',
|
bench ('list',
|
||||||
qw(
|
qw(
|
||||||
[ %debug ]
|
[
|
||||||
&
|
%d variant
|
||||||
[ %d variant
|
|
||||||
&
|
&
|
||||||
[ #d ONE_STAGE_BUILD ]
|
[ #d ONE_STAGE_BUILD | %d lex_symbol ]
|
||||||
]
|
]
|
||||||
)
|
)
|
||||||
);
|
);
|
||||||
|
|||||||
Reference in New Issue
Block a user