summaryrefslogtreecommitdiffstats
path: root/parser.y
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2014-03-25 22:10:31 -0700
committerKaz Kylheku <kaz@kylheku.com>2014-03-25 22:10:31 -0700
commitf4a6c56c8e8841c1991c1bb44546681ccbdb8f3a (patch)
tree943413daf7f15ed66562552b9cb43c8ada7053d0 /parser.y
parent4c6d387b73dca86caed48b1e786c5c1bc2c4716b (diff)
downloadtxr-f4a6c56c8e8841c1991c1bb44546681ccbdb8f3a.tar.gz
txr-f4a6c56c8e8841c1991c1bb44546681ccbdb8f3a.tar.bz2
txr-f4a6c56c8e8841c1991c1bb44546681ccbdb8f3a.zip
* eval.c (me_quasilist): New static function.
(eval_init): Register me_quasilist as quasilist macro expander. * lib.c (quasilist_s): New global variable. (obj_init): quasilist_s initialized. * lib.h (quasilist_s): Declared. * match.c (do_txreval): Handle quasilist syntax. * parser.l (QWLIT): New exclusive state. Extend lexical grammar to transition to QWLIT state upon the #` or #*` sequence which kicks off a word literal, and in that state, piecewise lexically analyze the QLL, mostly by borrowing rules from quasiliterals. * parser.y (QWORDS, QWSPLICE): New tokens. (n_exprs): Integrate splicing form of QLL syntax. (n_expr): Integrate non-splicing form of QLL syntax. (litchars): Propagate line number info. (quasilit): Fix "string literal" wording in error message. * txr.1: Introduced WLL abbreviation for word list literals, cleaned up the text a little, and documented QLL's.
Diffstat (limited to 'parser.y')
-rw-r--r--parser.y27
1 files changed, 22 insertions, 5 deletions
diff --git a/parser.y b/parser.y
index f9931541..ff3ff944 100644
--- a/parser.y
+++ b/parser.y
@@ -79,7 +79,7 @@ static val parsed_spec;
%token <lineno> MOD MODLAST DEFINE TRY CATCH FINALLY
%token <lineno> ERRTOK /* deliberately not used in grammar */
%token <lineno> HASH_BACKSLASH HASH_SLASH DOTDOT HASH_H
-%token <lineno> WORDS WSPLICE
+%token <lineno> WORDS WSPLICE QWORDS QWSPLICE
%token <lineno> SECRET_ESCAPE_R SECRET_ESCAPE_E
%token <val> NUMBER METANUM
@@ -102,7 +102,7 @@ static val parsed_spec;
%type <val> regex lisp_regex regexpr regbranch
%type <val> regterm regtoken regclass regclassterm regrange
%type <val> strlit chrlit quasilit quasi_items quasi_item litchars wordslit
-%type <val> not_a_clause
+%type <val> wordsqlit not_a_clause
%type <chr> regchar
%type <lineno> '(' '[' '@'
@@ -757,6 +757,9 @@ n_exprs : n_expr { $$ = rlcp(cons($1, nil), $1); }
| WSPLICE wordslit { $$ = rl($2, num($1)); }
| WSPLICE wordslit
n_exprs { $$ = nappend2(rl($2, num($1)), $3); }
+ | QWSPLICE wordsqlit { $$ = rl($2, num($1)); }
+ | QWSPLICE wordsqlit
+ n_exprs { $$ = nappend2(rl($2, num($1)), $3); }
;
n_expr : SYMTOK { $$ = sym_helper($1, t); }
@@ -773,6 +776,7 @@ n_expr : SYMTOK { $$ = sym_helper($1, t); }
| strlit { $$ = $1; }
| quasilit { $$ = $1; }
| WORDS wordslit { $$ = rl($2, num($1)); }
+ | QWORDS wordsqlit { $$ = rl(cons(quasilist_s, $2), num($1)); }
| '\'' n_expr { $$ = rlcp(list(quote_s, $2, nao), $2); }
| '^' n_expr { $$ = rlcp(list(sys_qquote_s, $2, nao), $2); }
| ',' n_expr { $$ = rlcp(list(sys_unquote_s, $2, nao), $2); }
@@ -911,7 +915,7 @@ quasilit : '`' '`' { $$ = null_string; }
rlcp($$, $2);
rl($$, num(lineno)); }
| '`' error { $$ = nil;
- yybadtoken(yychar, lit("string literal")); }
+ yybadtoken(yychar, lit("quasistring")); }
;
quasi_items : quasi_item { $$ = cons($1, nil);
@@ -934,13 +938,24 @@ litchars : LITCHAR { $$ = rl(cons(chr($1), nil), num(lineno)); }
wordslit : '"' { $$ = nil; }
| ' ' wordslit { $$ = $2; }
- | '\n' wordslit { $$ = $2; }
| litchars wordslit { val word = lit_char_helper($1);
$$ = rlcp(cons(word, $2), $1); }
| error { $$ = nil;
- yybadtoken(yychar, lit("word literal")); }
+ yybadtoken(yychar, lit("word list")); }
;
+wordsqlit : '`' { $$ = nil; }
+ | ' ' wordsqlit { $$ = $2; }
+ | quasi_items '`' { val qword = cons(quasi_s,
+ o_elems_transform($1));
+ $$ = rlcp(cons(qword, nil), $1); }
+ | quasi_items ' '
+ wordsqlit
+ { val qword = cons(quasi_s,
+ o_elems_transform($1));
+ $$ = rlcp(cons(qword, $3), $1); }
+ ;
+
not_a_clause : ALL { $$ = make_expr(all_s, nil, num(lineno)); }
| SOME { $$ = make_expr(some_s, nil, num(lineno)); }
| NONE { $$ = make_expr(none_s, nil, num(lineno)); }
@@ -1345,6 +1360,8 @@ void yybadtoken(int tok, val context)
case HASH_H: problem = lit("#H"); break;
case WORDS: problem = lit("#\""); break;
case WSPLICE: problem = lit("#*\""); break;
+ case QWORDS: problem = lit("#`"); break;
+ case QWSPLICE: problem = lit("#*`"); break;
}
if (problem != 0)