summaryrefslogtreecommitdiffstats
path: root/parser.y
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-06-26 18:55:22 -0700
committerKaz Kylheku <kaz@kylheku.com>2015-06-26 18:55:22 -0700
commit9bf17146994d7586c62f7f52be525ee6f6b08329 (patch)
tree73c7499b2865f71639c9733b97ad4024fd57e8f0 /parser.y
parent526ff444979c6829ec52cc619136747354e57630 (diff)
downloadtxr-9bf17146994d7586c62f7f52be525ee6f6b08329.tar.gz
txr-9bf17146994d7586c62f7f52be525ee6f6b08329.tar.bz2
txr-9bf17146994d7586c62f7f52be525ee6f6b08329.zip
First round of quasiliteral-related fixes.
* parser.l: Do not try to recognize floating-point literals in QSPECIAL state; that is not possible because @134.3 in a quasiliteral parses as a METANUM followed by ".3". On the other hand, recognize METANUM literals in QSPECIAL state, so that @@123 scans. Recognize @ as a token in QSPECIAL state, so @@abc will scan. When transitioning from QSILIT and QWLIT states to QSPECIAL upon scanning @, return a @ token, which is now parsed in the grammar. * parser.y (quasi_meta_helper): New static function. (q_var): Do not handle SYMTOK any more, only the braced variable syntax. SYMTOK is handled as a n_expr. Braced vars are handled with explicit '@' token, which is now produced by the scanner when it shifts from QSILIT to QSPECIAL. (quasi_item): No longer necessary to recognize various forms here such as quotes and splices. Just recognize a n_expr, preceded by '@'.
Diffstat (limited to 'parser.y')
-rw-r--r--parser.y41
1 files changed, 23 insertions, 18 deletions
diff --git a/parser.y b/parser.y
index 01e7295e..12324f51 100644
--- a/parser.y
+++ b/parser.y
@@ -58,6 +58,7 @@ static val rlrec(parser_t *, val form, val line);
static wchar_t char_from_name(const wchar_t *name);
static val make_expr(parser_t *, val sym, val rest, val lineno);
static val check_for_include(val spec_rev);
+static val quasi_meta_helper(val obj);
#if YYBISON
union YYSTYPE;
@@ -685,12 +686,10 @@ o_var : SYMTOK { $$ = list(var_s, symhlpr($1, nil), nao);
yybadtok(yychar, lit("variable spec")); }
;
-q_var : SYMTOK { $$ = list(var_s, symhlpr($1, nil), nao);
+q_var : '@' '{' n_expr n_exprs_opt '}'
+ { $$ = list(var_s, $3, $4, nao);
rl($$, num(parser->lineno)); }
- | '{' n_expr n_exprs_opt '}'
- { $$ = list(var_s, $2, $3, nao);
- rl($$, num(parser->lineno)); }
- | SYMTOK error { $$ = nil;
+ | '@' '{' error { $$ = nil;
yybadtok(yychar, lit("variable spec")); }
;
@@ -959,19 +958,7 @@ quasi_item : litchars { $$ = lit_char_helper($1); }
| q_var { $$ = $1; }
| METANUM { $$ = cons(var_s, cons($1, nil));
rl($$, num(parser->lineno)); }
- | list { $$ = rlcp(cons(expr_s, $1), $1); }
- | '\'' n_expr { $$ = rl(rlcp(cons(expr_s, list(quote_s, $2, nao)), $2),
- num(parser->lineno));
- rlcp_tree($$, $$); }
- | '^' n_expr { $$ = rl(rlcp(cons(expr_s, list(sys_qquote_s, $2, nao)), $2),
- num(parser->lineno));
- rlcp_tree($$, $$); }
- | ',' n_expr { $$ = rl(rlcp(cons(expr_s, list(sys_unquote_s, $2, nao)), $2),
- num(parser->lineno));
- rlcp_tree($$, $$); }
- | SPLICE n_expr { $$ = rl(rlcp(cons(expr_s, list(sys_splice_s, $2, nao)), $2),
- num(parser->lineno));
- rlcp_tree($$, $$); }
+ | '@' n_expr { $$ = quasi_meta_helper($2); }
;
litchars : LITCHAR { $$ = rl(cons(chr($1), nil), num(parser->lineno)); }
@@ -1357,6 +1344,24 @@ static val check_for_include(val spec_rev)
return spec_rev;
}
+static val quasi_meta_helper(val obj)
+{
+ if (integerp(obj) || symbolp(obj))
+ goto var;
+
+ if (atom(obj) || length(obj) != two)
+ goto expr;
+
+ if (first(obj) == var_s && integerp(second(obj)))
+ goto var;
+
+var:
+ return rlcp_tree(cons(var_s, cons(obj, nil)), obj);
+
+expr:
+ return rlcp(cons(expr_s, obj), obj);
+}
+
#ifndef YYEOF
#define YYEOF 0
#endif