Files
bison/examples/variant.yy
Akim Demaille f6038cb8c3 Use "%define variant" in bench.pl.
* etc/bench.pl.in: No longer use the pseudo directive %variants,
	just use %define variants.
2008-11-07 21:37:56 +01:00

135 lines
2.7 KiB
Plaintext

/* Test file for C++ parsers using variants.
Based on an example by Michiel De Wilde <mdewilde.agilent@gmail.com>. */
%debug
%skeleton "lalr1-fusion.cc"
%defines
%define variant
%code requires // *.hh
{
#include <list>
#include <string>
typedef std::list<std::string> strings_type;
}
%code // *.cc
{
#include <algorithm>
#include <iostream>
#include <iterator>
#include <sstream>
// Prototype of the yylex function providing subsequent tokens.
static yy::parser::token_type yylex(yy::parser::semantic_type* yylval);
// Printing a list of strings.
// Koening look up will look into std, since that's an std::list.
namespace std
{
std::ostream&
operator<<(std::ostream& o, const strings_type& s)
{
std::copy(s.begin(), s.end(),
std::ostream_iterator<strings_type::value_type>(o, "\n"));
return o;
}
}
// Conversion to string.
template <typename T>
inline
std::string
string_cast (const T& t)
{
std::ostringstream o;
o << t;
return o.str();
}
}
%token <std::string> TEXT;
%token <int> NUMBER;
%printer { debug_stream () << $$; } <int> <std::string> <strings_type>;
%token END_OF_FILE 0;
%type <std::string> item;
%type <strings_type> list;
%%
result:
list { std::cout << $1 << std::endl; }
;
list:
/* nothing */ { /* Generates an empty string list */ }
| list item { std::swap ($$, $1); $$.push_back ($2); }
;
item:
TEXT { std::swap ($$, $1); }
| NUMBER { $$ = string_cast ($1); }
;
%%
// The yylex function providing subsequent tokens:
// TEXT "I have three numbers for you:"
// NUMBER 1
// NUMBER 2
// NUMBER 3
// TEXT " and that's all!"
// END_OF_FILE
static
yy::parser::token_type
yylex (yy::parser::semantic_type* yylval)
{
static int stage = 0;
yy::parser::token_type result;
switch (stage)
{
case 0:
yylval->build<std::string> () =
std::string ("I have three numbers for you.");
result = yy::parser::token::TEXT;
break;
case 1:
case 2:
case 3:
yylval->build<int> () = stage;
result = yy::parser::token::NUMBER;
break;
case 4:
yylval->build<std::string> () = std::string ("And that's all!");
result = yy::parser::token::TEXT;
break;
default:
result = yy::parser::token::END_OF_FILE;
break;
}
stage++;
return result;
}
// Mandatory error function
void
yy::parser::error (const yy::parser::location_type& yylloc,
const std::string& message)
{
std::cerr << yylloc << ": " << message << std::endl;
}
int
main (int argc, char *argv[])
{
yy::parser p;
p.set_debug_level (!!getenv ("YYDEBUG"));
p.parse ();
}
// Local Variables:
// mode: C++
// End: