diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2016-04-27 06:45:23 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2016-04-27 06:45:23 -0700 |
commit | 7afbcc19bef877560f096c02e63b22f9f650b74e (patch) | |
tree | 088feceb91b034a6695f9dd0eb2e0de126eb06f6 /parser.y | |
parent | c43eb573779da0965ecb84feb23e21b39405bb4b (diff) | |
download | txr-7afbcc19bef877560f096c02e63b22f9f650b74e.tar.gz txr-7afbcc19bef877560f096c02e63b22f9f650b74e.tar.bz2 txr-7afbcc19bef877560f096c02e63b22f9f650b74e.zip |
Diagnose empty clauses better in parallel constructs.
In this change we allow the grammar, as such, to express empty
clauses in the parallel constructs like @(some), @(cases),
@(gather) and so forth. However, we raise an error when these
occur. This results in a cleaner diagnostic behavior.
In the future, empty clauses may be allowed; the semantics has
to be worked out. An empty clause should neither succeed nor
fail; the behavior should be as if it is not there.
The general strategy in this patch is to eliminate the use
of the clauses terminal symbol (completely) and use
clauses_opt everywhere, and then diagnose when that
produces a nil. Some elems are similarly changed to elems_opt
in the horizontal versions of directives.
* parser.y (clauses): Nonterminal symbol completely removed.
(spec): Empty production rule removed, by using clauses_opt
instead of clauses.
(all_clause, some_clause, none_clause,
maybe_clause, cases_clause, choose_clause, gather_clause):
Instead of diagnosing empty clauses via a dedicated
grammar rule, diagnose by looking for nil subtree emerging
from from a clauses phrase.
(gather_parts): Use clauses_opt. Yield nil instead of
a cons if this clauses_opt is nil.
(additional_gather_parts): Simplify grammar and actions
by recursing back to gather_parts. If gather_parts produces
nil, diagnose the empty and/or subclause situation.
(collect_clause): Use clauses_opt and diagnose empty.
Simplify error production, which doesn't have to look
at the lookahead token yychar any more to implement
the empty diagnostic.
(clause_parts): Use clauses_opt instad of clauses. Yield
nil instead of a cons if clause_opt yields nil.
(additional_parts): Grammar and actions simplified by
recursing back to clause_parts. If the recursive
clause_parts produces nil, generate empty and/or
subclause diagnostic.
(elem): Use elems_opt for the various horizontal clauses
and implement empty checks for them. Explicit empty
productions are eliminated.
(clause_parts_h): Changed in way analogous to clause_parts.
(additional_parts_h): Changed analogously to
additional_parts.
(try_clause): Use clauses_opt and catch empties.
Eliminate the second error production.
(catch_clauses_opt): Diagnose empty catch and finally, unless
in compatibility mode with 139 or less. The grammar already
uses clauses_opt here; empty catches and finally had been
allowed. Fixed messages in error productions for catch
and finally to refer to catch and finally rather than try.
Diffstat (limited to 'parser.y')
-rw-r--r-- | parser.y | 205 |
1 files changed, 119 insertions, 86 deletions
@@ -110,7 +110,7 @@ int yyparse(scanner_t *, parser_t *); %token <chr> REGCHAR REGTOKEN LITCHAR SPLICE CONSDOT LAMBDOT -%type <val> spec clauses_rev clauses clauses_opt clause +%type <val> spec clauses_rev clauses_opt clause %type <val> all_clause some_clause none_clause maybe_clause block_clause %type <val> cases_clause choose_clause gather_clause collect_clause until_last %type <val> collect_repeat @@ -146,8 +146,7 @@ int yyparse(scanner_t *, parser_t *); %% -spec : clauses { parser->syntax_tree = $1; } - | /* empty */ { parser->syntax_tree = nil; } +spec : clauses_opt { parser->syntax_tree = $1; } | SECRET_ESCAPE_R regexpr { parser->syntax_tree = $2; end_of_regex(scnr); } | SECRET_ESCAPE_E n_expr { parser->syntax_tree = $2; YYACCEPT; } byacc_fool { internal_error("notreached"); } @@ -180,8 +179,6 @@ byacc_fool : n_expr { internal_error("notreached"); } | { internal_error("notreached"); } ; -clauses : clauses_rev { $$ = nreverse($1); } - clauses_rev : clause { $$ = check_for_include(cons($1, nil)); } | clauses_rev clause { $$ = check_for_include(cons($2, $1)); } ; @@ -208,49 +205,47 @@ clause : all_clause { $$ = cons($1, nil); rlcp($$, $1); } | line { $$ = $1; } ; -all_clause : ALL newl clause_parts { $$ = list(all_s, $3, nao); +all_clause : ALL newl clause_parts { if (nilp($3)) + yyerr("empty all clause"); + $$ = list(all_s, $3, nao); rl($$, num($1)); } | ALL newl error { $$ = nil; yybadtok(yychar, lit("all clause")); } - | ALL newl END newl { $$ = nil; - yyerr("empty all clause"); } - ; some_clause : SOME exprs_opt ')' - newl clause_parts { $$ = list(some_s, $5, $2, nao); + newl clause_parts { if (nilp($5)) + yyerr("empty some clause"); + $$ = list(some_s, $5, $2, nao); rl($$, num($1)); } | SOME exprs_opt ')' newl error { $$ = nil; yybadtok(yychar, lit("some clause")); } - | SOME exprs_opt ')' - newl END newl { $$ = nil; - yyerr("empty some clause"); } ; -none_clause : NONE newl clause_parts { $$ = list(none_s, $3, nao); +none_clause : NONE newl clause_parts { if (nilp($3)) + yyerr("empty none clause"); + $$ = list(none_s, $3, nao); rl($$, num($1)); } | NONE newl error { $$ = nil; yybadtok(yychar, lit("none clause")); } - | NONE newl END newl { $$ = nil; - yyerr("empty none clause"); } ; -maybe_clause : MAYBE newl clause_parts { $$ = list(maybe_s, $3, nao); +maybe_clause : MAYBE newl clause_parts { if (nilp($3)) + yyerr("empty maybe clause"); + $$ = list(maybe_s, $3, nao); rl($$, num($1)); } | MAYBE newl error { $$ = nil; yybadtok(yychar, lit("maybe clause")); } - | MAYBE newl END newl { $$ = nil; - yyerr("empty maybe clause"); } ; -cases_clause : CASES newl clause_parts { $$ = list(cases_s, $3, nao); +cases_clause : CASES newl clause_parts { if (nilp($3)) + yyerr("empty cases clause"); + $$ = list(cases_s, $3, nao); rl($$, num($1)); } | CASES newl error { $$ = nil; yybadtok(yychar, lit("cases clause")); } - | CASES newl END newl { $$ = nil; - yyerr("empty cases clause"); } ; block_clause : BLOCK exprs_opt ')' @@ -270,19 +265,20 @@ block_clause : BLOCK exprs_opt ')' ; choose_clause : CHOOSE exprs_opt ')' - newl clause_parts { $$ = list(choose_s, $5, $2, nao); + newl clause_parts { if (nilp($5)) + yyerr("empty choose clause"); + $$ = list(choose_s, $5, $2, nao); rl($$, num($1)); } | CHOOSE exprs_opt ')' newl error { $$ = nil; yybadtok(yychar, lit("choose clause")); } - | CHOOSE exprs_opt ')' - newl END newl { $$ = nil; - yyerr("empty choose clause"); } ; gather_clause : GATHER exprs_opt ')' newl gather_parts - END newl { $$ = list(gather_s, + END newl { if (nilp($5)) + yyerr("empty gather clause"); + $$ = list(gather_s, append2(mapcar(curry_12_1(func_n2(cons), nil), first($5)), rest($5)), $2, nao); @@ -291,8 +287,12 @@ gather_clause : GATHER exprs_opt ')' | GATHER exprs_opt ')' newl gather_parts until_last exprs_opt ')' newl - clauses - END newl { $$ = list(gather_s, + clauses_opt + END newl { if (nilp($5)) + yyerr("empty gather clause"); + if (nilp($10)) + yyerr("empty until/last clause in gather"); + $$ = list(gather_s, append2(mapcar(curry_12_1(func_n2(cons), nil), first($5)), rest($5)), $2, cons(cdr($6), @@ -302,27 +302,36 @@ gather_clause : GATHER exprs_opt ')' | GATHER exprs_opt ')' newl error { $$ = nil; yybadtok(yychar, lit("gather clause")); } - | GATHER exprs_opt ')' - newl END newl { $$ = nil; - yyerr("empty gather clause"); } ; -gather_parts : clauses additional_gather_parts { $$ = cons($1, $2); } +gather_parts : clauses_opt additional_gather_parts + { $$ = if2($1, cons($1, $2)); } ; -additional_gather_parts : AND newl clauses additional_gather_parts { $$ = cons($3, $4); } - | OR newl clauses additional_parts { $$ = cons($3, $4); } - | /* empty */ { $$ = nil; } +additional_gather_parts : AND newl gather_parts { $$ = $3; + if (nilp($$)) + yyerr("empty and subclause"); } + | OR newl gather_parts { $$ = $3; + if (nilp($$)) + yyerr("empty or subclause"); } + | /* empty */ { $$ = nil; } ; collect_clause : collect_repeat exprs_opt ')' newl - clauses END newl { $$ = list(car($1), + clauses_opt END newl { if (nilp($5)) + yyerr("empty collect clause"); + $$ = list(car($1), $5, nil, $2, nao); rl($$, cdr($1)); } | collect_repeat exprs_opt ')' - newl clauses until_last exprs_opt ')' - newl clauses END newl { $$ = list(car($1), $5, + newl clauses_opt until_last exprs_opt ')' + newl clauses_opt END newl + { if (nilp($5)) + yyerr("empty collect clause"); + if (nilp($10)) + yyerr("empty until/last in collect"); + $$ = list(car($1), $5, cons(cdr($6), cons($7, $10)), $2, nao); @@ -330,12 +339,7 @@ collect_clause : collect_repeat exprs_opt ')' newl rl($10, car($6)); } | collect_repeat exprs_opt ')' newl error { $$ = nil; - if (yychar == UNTIL || - yychar == END || - yychar == LAST) - yyerr("empty collect"); - else - yybadtok(yychar, lit("collect clause")); } + yybadtok(yychar, lit("collect clause")); } ; collect_repeat : COLLECT { $$ = cons(collect_s, num($1)); } @@ -346,12 +350,16 @@ until_last : UNTIL { $$ = cons(num($1), until_s); } | LAST { $$ = cons(num($1), last_s); } ; -clause_parts : clauses additional_parts { $$ = cons($1, $2); } +clause_parts : clauses_opt additional_parts { $$ = if2($1, cons($1, $2)); } ; -additional_parts : END newl { $$ = nil; } - | AND newl clauses additional_parts { $$ = cons($3, $4); } - | OR newl clauses additional_parts { $$ = cons($3, $4); } +additional_parts : END newl { $$ = nil; } + | AND newl clause_parts { $$ = $3; + if (nilp($$)) + yyerr("empty and subclause"); } + | OR newl clause_parts { $$ = $3; + if (nilp($$)) + yyerr("empty or subclause"); } ; if_clause : IF exprs_opt ')' @@ -447,21 +455,33 @@ elem : texts { $$ = rlcp(cons(text_s, $1), $1); $$ = rlcp(cons(sym, expand_meta(rest($1), nil)), $1); } - | COLL exprs_opt ')' elems END { $$ = list(coll_s, $4, nil, $2, nao); + | COLL exprs_opt ')' elems_opt END { if (nilp($4)) + yyerr("empty coll clause"); + $$ = list(coll_s, $4, nil, $2, nao); rl($$, num($1)); } - | COLL exprs_opt ')' elems + | COLL exprs_opt ')' elems_opt until_last exprs_opt ')' - elems END { $$ = list(coll_s, $4, cons(cdr($5), + elems_opt END { if (nilp($4)) + yyerr("empty coll clause"); + if (nilp($8)) + yyerr("empty until/last in coll"); + $$ = list(coll_s, $4, cons(cdr($5), cons($6, $8)), $2, nao); rl($$, num($1)); rl($6, car($5)); } - | REP exprs_opt ')' elems END { $$ = list(rep_s, $4, nil, $2, nao); + | REP exprs_opt ')' elems END { if (nilp($2)) + yyerr("empty rep clause"); + $$ = list(rep_s, $4, nil, $2, nao); rl($$, num($1)); } | REP exprs_opt ')' elems until_last exprs_opt ')' elems END - { $$ = list(rep_s, $4, cons(cdr($5), + { if (nilp($2)) + yyerr("empty rep clause"); + if (nilp($8)) + yyerr("empty until/last in rep"); + $$ = list(rep_s, $4, cons(cdr($5), cons($6, $8)), $2, nao); rl($$, num($1)); @@ -470,32 +490,42 @@ elem : texts { $$ = rlcp(cons(text_s, $1), $1); yybadtok(yychar, lit("coll clause")); } | REP error { $$ = nil; yybadtok(yychar, lit("rep clause")); } - | ALL clause_parts_h { $$ = rl(list(all_s, t, $2, nao), num($1)); } - | ALL END { yyerr("empty all clause"); } + | ALL clause_parts_h { $$ = rl(list(all_s, t, $2, nao), num($1)); + if (nilp($2)) + yyerr("empty all clause"); } | SOME exprs_opt ')' - clause_parts_h { $$ = rl(list(some_s, t, $4, $2, nao), num($1)); } - | SOME exprs_opt ')' END { yyerr("empty some clause"); } - | NONE clause_parts_h { $$ = rl(list(none_s, t, $2, nao), num($1)); } - | NONE END { yyerr("empty none clause"); } - | MAYBE clause_parts_h { $$ = rl(list(maybe_s, t, $2, nao), num($1)); } - | MAYBE END { yyerr("empty maybe clause"); } - | CASES clause_parts_h { $$ = rl(list(cases_s, t, $2, nao), num($1)); } - | CASES END { yyerr("empty cases clause"); } + clause_parts_h { $$ = rl(list(some_s, t, $4, $2, nao), num($1)); + if (nilp($4)) + yyerr("empty some clause"); } + | NONE clause_parts_h { $$ = rl(list(none_s, t, $2, nao), num($1)); + if (nilp($2)) + yyerr("empty none clause"); } + | MAYBE clause_parts_h { $$ = rl(list(maybe_s, t, $2, nao), num($1)); + if (nilp($2)) + yyerr("empty maybe clause"); } + | CASES clause_parts_h { $$ = rl(list(cases_s, t, $2, nao), num($1)); + if (nilp($2)) + yyerr("empty cases clause"); } | CHOOSE exprs_opt ')' clause_parts_h { $$ = list(choose_s, t, $4, $2, nao); - rl($$, num($1)); } - | CHOOSE exprs_opt ')' END { yyerr("empty cases clause"); } + rl($$, num($1)); + if (nilp($4)) + yyerr("empty cases clause"); } | DEFINE exprs ')' elems END { $$ = list(define_s, t, $4, $2, nao); rl($$, num($1)); } ; -clause_parts_h : elems additional_parts_h { $$ = cons($1, $2); } +clause_parts_h : elems_opt additional_parts_h { $$ = if2($1, cons($1, $2)); } ; -additional_parts_h : END { $$ = nil; } - | AND elems additional_parts_h { $$ = cons($2, $3); } - | OR elems additional_parts_h { $$ = cons($2, $3); } +additional_parts_h : END { $$ = nil; } + | AND clause_parts_h { $$ = $2; + if (nilp($$)) + yyerr("empty and subclause"); } + | OR clause_parts_h { $$ = $2; + if (nilp($$)) + yyerr("empty or subclause"); } ; define_clause : DEFINE exprs ')' newl @@ -516,51 +546,54 @@ define_clause : DEFINE exprs ')' newl ; try_clause : TRY newl - clauses + clauses_opt catch_clauses_opt - END newl { $$ = list(try_s, + END newl { if (nilp($3)) + yyerr("empty try clause"); + $$ = list(try_s, flatten(mapcar(func_n1(second), $4)), $3, $4, nao); rl($$, num($1)); } | TRY newl error { $$ = nil; - if (yychar == END || yychar == CATCH || - yychar == FINALLY) - yyerr("empty try clause"); - else - yybadtok(yychar, lit("try clause")); } - | TRY newl - clauses - error { $$ = nil; yybadtok(yychar, lit("try clause")); } ; catch_clauses_opt : CATCH ')' newl clauses_opt - catch_clauses_opt { $$ = cons(list(catch_s, cons(t, nil), + catch_clauses_opt { if ((!opt_compat || opt_compat > 139) + && nilp($4)) + yyerr("empty catch clause"); + $$ = cons(list(catch_s, cons(t, nil), $4, nao), $5); rl($$, num($1)); } | CATCH exprs ')' newl clauses_opt - catch_clauses_opt { $$ = cons(list(catch_s, $2, $5, nao), + catch_clauses_opt { if ((!opt_compat || opt_compat > 139) + && nilp($5)) + yyerr("empty catch clause"); + $$ = cons(list(catch_s, $2, $5, nao), $6); rl($$, num($1)); } | FINALLY newl - clauses_opt { $$ = cons(list(finally_s, nil, + clauses_opt { if ((!opt_compat || opt_compat > 139) + && nilp($3)) + yyerr("empty finally clause"); + $$ = cons(list(finally_s, nil, $3, nao), nil); rl($$, num($1)); } | { $$ = nil; } | CATCH ')' newl error { $$ = nil; - yybadtok(yychar, lit("try clause")); } + yybadtok(yychar, lit("catch clause")); } | CATCH exprs ')' newl error { $$ = nil; - yybadtok(yychar, lit("try clause")); } + yybadtok(yychar, lit("catch clause")); } | FINALLY newl error { $$ = nil; - yybadtok(yychar, lit("try clause")); } + yybadtok(yychar, lit("finally clause")); } ; |