* tests/c++.at (Exception safety): Let the parser support the --debug
option.
On 'p', throw an exception from the %printer.
* data/lalr1.cc (yyparse): Do not display the values we discard, as it
uses %printer, which might have thrown the exception.
/// \brief Reclaim the memory associated to a symbol.
/// \param yymsg Why this token is reclaimed.
/// \brief Reclaim the memory associated to a symbol.
/// \param yymsg Why this token is reclaimed.
+ /// If null, do not display the symbol, just free it.
/// \param yytype The symbol type.
/// \param yyvaluep Its semantic value.
/// \param yylocationp Its location.
/// \param yytype The symbol type.
/// \param yyvaluep Its semantic value.
/// \param yylocationp Its location.
YYUSE (yymsg);
YYUSE (yyvaluep);
YYUSE (yymsg);
YYUSE (yyvaluep);
- YY_SYMBOL_PRINT (yymsg, yytype, yyvaluep, yylocationp);
+ if (yymsg)
+ YY_SYMBOL_PRINT (yymsg, yytype, yyvaluep, yylocationp);
- YYCDEBUG << "Exception caught" << std::endl;
+ YYCDEBUG << "Exception caught: cleaning lookahead and stack"
+ << std::endl;
+ // Do not try to display the values of the reclaimed symbols,
+ // as their printer might throw an exception.
if (yychar != yyempty_)
{
/* Make sure we have latest lookahead translation. See
comments at user semantic actions for why this is
necessary. */
yytoken = yytranslate_ (yychar);
if (yychar != yyempty_)
{
/* Make sure we have latest lookahead translation. See
comments at user semantic actions for why this is
necessary. */
yytoken = yytranslate_ (yychar);
- yydestruct_ ("Cleanup: discarding lookahead", yytoken, &yylval,
- &yylloc);
+ yydestruct_ (YY_NULL, yytoken, &yylval, &yylloc);
}
while (1 < yystate_stack_.height ())
{
}
while (1 < yystate_stack_.height ())
{
- yydestruct_ ("Cleanup: popping",
yystos_[yystate_stack_[0]],
&yysemantic_stack_[0],
&yylocation_stack_[0]);
yystos_[yystate_stack_[0]],
&yysemantic_stack_[0],
&yylocation_stack_[0]);
+ /// A class that counts its number of instances.
struct Object
{
static size_t counter;
struct Object
{
static size_t counter;
+ Object (int v)
+ : val (v)
}
%destructor { delete $$; } <obj>;
}
%destructor { delete $$; } <obj>;
-%printer { yyo << "counter == " << $$->counter; } <obj>;
+%printer
+{
+ yyo << "counter == " << $$->counter;
+ if ($$->val == 'p')
+ throw std::runtime_error ("printer");
+} <obj>;
+%token <obj> 'a' 'p' 's'
-| item list { $$ = $1; delete $2; } /* Right recursion to load the stack. */
+| item list { $$ = $1; delete $2; } // Right recursion to load the stack.
- 'a'
- {
- std::swap ($$, $1);
- }
+ 'a' { std::swap ($$, $1); }
+| 'p' { std::swap ($$, $1); }
| 's'
{
std::swap ($$, $1);
| 's'
{
std::swap ($$, $1);
case 'l':
throw std::runtime_error ("yylex");
default:
case 'l':
throw std::runtime_error ("yylex");
default:
- lvalp->obj = new Object;
+ lvalp->obj = new Object (res);
// Fall through.
case 0:
return res;
// Fall through.
case 0:
return res;
int
main (int argc, const char *argv[])
{
int
main (int argc, const char *argv[])
{
- assert (argc == 2);
- input = argv[1];
+ switch (argc)
+ {
+ case 2:
+ input = argv[1];
+ break;
+ case 3:
+ assert (!strcmp (argv[1], "--debug"));
+ debug = 1;
+ input = argv[2];
+ break;
+ default:
+ abort ();
+ }
+
- debug = !!getenv ("YYDEBUG");
+ debug |= !!getenv ("YYDEBUG");
parser.set_debug_level (debug);
int res = 2;
try
parser.set_debug_level (debug);
int res = 2;
try
[[exception caught: initial-action
]])
[[exception caught: initial-action
]])
+AT_PARSER_CHECK([[./input aaaap]])
+
+AT_PARSER_CHECK([[./input --debug aaaap]], [[2]], [[]], [[stderr]])
+AT_PARSER_CHECK([[grep '^exception caught: printer$' stderr]], [], [ignore])
+
AT_BISON_OPTION_POPDEFS
AT_CLEANUP
AT_BISON_OPTION_POPDEFS
AT_CLEANUP