summaryrefslogtreecommitdiffstats
path: root/parser.y
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2015-11-07 10:21:42 -0800
committerKaz Kylheku <kaz@kylheku.com>2015-11-07 10:21:42 -0800
commit660af6ed504bac0258834f6e4c58ad81454dbad8 (patch)
treed888fa89f333c390cde67094b4b65cea2806405c /parser.y
parent7bbb027e111cc4b90248cd9d9da127839d91038a (diff)
downloadtxr-660af6ed504bac0258834f6e4c58ad81454dbad8.tar.gz
txr-660af6ed504bac0258834f6e4c58ad81454dbad8.tar.bz2
txr-660af6ed504bac0258834f6e4c58ad81454dbad8.zip
New iread function.
The read function no longer works like it used to on an interactive terminal because of the support for .. and . syntax on a top-level expression. The iread function is provided which uses a modified syntax that doesn't support these operators on a top-level expression. The parser thus doesn't look one token ahead, and so iread can return immediately. * eval.c (eval_init): Register iread intrinsic function. * parser.c (prime_parser): Only push back the recently seen token when priming for a regular Lisp read. Handle the prime_interactive method by preparing a SECRET_ESCAPE_I token. (lisp_parse_impl): New static function, formed from previous lisp_parse. Takes a boolean argument indicating interactive mode. (prime_parser_post): New function. (lisp_parse): Now a wrapper for lisp_parse_impl which passes a nil to indicate noninteractive read. (iread): New function. * parser.h (enum prime_parser): New member, prime_interactive. (scrub_scanner, iread, prime_parser_post): Declared. * parser.l (prime_scanner): Handle the prime_interactive case the same way as prime_lisp. (scrub_scanner): New function. * parser.y (SECRET_ESCAPE_I): New token type. (i_expr): New nonterminal symbol. Like n_expr, but doesn't support dot or dotdot operators, except in nested subexpressions. (spec): Handle SECRET_ESCAPE_I by way of i_expr. (sym_helper): Before freeing the token lexeme, call scrub_scanner. If the token is registered as the scanner's most recently seen token, the scanner must forget that registration, because it is no longer valid. (parse): Call prime_parser_post. * txr.1: Documented iread.
Diffstat (limited to 'parser.y')
-rw-r--r--parser.y44
1 files changed, 42 insertions, 2 deletions
diff --git a/parser.y b/parser.y
index d63eba4f..cd351bbd 100644
--- a/parser.y
+++ b/parser.y
@@ -102,7 +102,7 @@ int yyparse(scanner_t *, parser_t *);
%token <lineno> ERRTOK /* deliberately not used in grammar */
%token <lineno> HASH_BACKSLASH HASH_SLASH DOTDOT HASH_H HASH_S HASH_R
%token <lineno> WORDS WSPLICE QWORDS QWSPLICE
-%token <lineno> SECRET_ESCAPE_R SECRET_ESCAPE_E
+%token <lineno> SECRET_ESCAPE_R SECRET_ESCAPE_E SECRET_ESCAPE_I
%token <val> NUMBER METANUM
@@ -117,7 +117,7 @@ int yyparse(scanner_t *, parser_t *);
%type <val> if_clause elif_clauses_opt else_clause_opt
%type <val> line elems_opt elems clause_parts_h additional_parts_h
%type <val> text texts elem var var_op modifiers vector hash struct range
-%type <val> list exprs exprs_opt expr n_exprs r_exprs n_expr n_exprs_opt
+%type <val> list exprs exprs_opt expr n_exprs r_exprs i_expr n_expr n_exprs_opt
%type <val> out_clauses out_clauses_opt out_clause
%type <val> repeat_clause repeat_parts_opt o_line
%type <val> o_elems_opt o_elems o_elem o_var q_var rep_elem rep_parts_opt
@@ -149,6 +149,8 @@ spec : clauses { 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"); }
+ | SECRET_ESCAPE_I i_expr { parser->syntax_tree = $2; YYACCEPT; }
+ byacc_fool { internal_error("notreached"); }
| SECRET_ESCAPE_E { if (yychar == YYEOF) {
parser->syntax_tree = nao;
YYACCEPT;
@@ -156,6 +158,13 @@ spec : clauses { parser->syntax_tree = $1; }
yybadtok(yychar, nil);
parser->syntax_tree = nil;
} }
+ | SECRET_ESCAPE_I { if (yychar == YYEOF) {
+ parser->syntax_tree = nao;
+ YYACCEPT;
+ } else {
+ yybadtok(yychar, nil);
+ parser->syntax_tree = nil;
+ } }
| error '\n' { parser->syntax_tree = nil;
if (parser->errors >= 8)
YYABORT;
@@ -820,6 +829,31 @@ r_exprs : n_expr { val exprs = cons($1, nil);
$$ = term_atom_cons; }
;
+i_expr : SYMTOK { $$ = symhlpr($1, t); }
+ | METANUM { $$ = cons(var_s, cons($1, nil));
+ rl($$, num(parser->lineno)); }
+ | NUMBER { $$ = $1; }
+ | list { $$ = $1; }
+ | vector { $$ = $1; }
+ | hash { $$ = $1; }
+ | struct { $$ = $1; }
+ | range { $$ = $1; }
+ | lisp_regex { $$ = $1; }
+ | chrlit { $$ = $1; }
+ | strlit { $$ = $1; }
+ | quasilit { $$ = $1; }
+ | WORDS wordslit { $$ = rl($2, num($1)); }
+ | QWORDS wordsqlit { $$ = rl(cons(quasilist_s, $2), num($1)); }
+ | '\'' i_expr { $$ = rl(rlcp(list(quote_s, $2, nao), $2),
+ num(parser->lineno)); }
+ | '^' i_expr { $$ = rl(rlcp(list(sys_qquote_s, $2, nao), $2),
+ num(parser->lineno)); }
+ | ',' i_expr { $$ = rl(rlcp(list(sys_unquote_s, $2, nao), $2),
+ num(parser->lineno)); }
+ | SPLICE i_expr { $$ = rl(rlcp(list(sys_splice_s, $2, nao), $2),
+ num(parser->lineno)); }
+ ;
+
n_expr : SYMTOK { $$ = symhlpr($1, t); }
| METANUM { $$ = cons(var_s, cons($1, nil));
rl($$, num(parser->lineno)); }
@@ -1107,11 +1141,13 @@ static val sym_helper(parser_t *parser, wchar_t *lexeme, val meta_allowed)
if (colon == lexeme) {
package = keyword_package_var;
sym_name = string(colon + 1);
+ scrub_scanner(parser->scanner, SYMTOK, tokfree);
free(tokfree);
} else if (colon != 0) {
pkg_name = string(lexeme);
package = find_package(pkg_name);
sym_name = string(colon + 1);
+ scrub_scanner(parser->scanner, SYMTOK, tokfree);
free(tokfree);
if (!package) {
yyerrorf(scnr, lit("~a:~a: package ~a not found"), pkg_name, sym_name, pkg_name, nao);
@@ -1119,6 +1155,7 @@ static val sym_helper(parser_t *parser, wchar_t *lexeme, val meta_allowed)
}
} else {
sym_name = string(lexeme);
+ scrub_scanner(parser->scanner, SYMTOK, tokfree);
free(tokfree);
}
@@ -1543,12 +1580,15 @@ int parse(parser_t *parser, val name, enum prime_parser prim)
parser->errors = 0;
parser->prepared_msg = nil;
parser->syntax_tree = nil;
+
prime_parser(parser, name, prim);
uw_catch_begin(cons(error_s, nil), esym, eobj);
res = yyparse(parser->scanner, parser);
+ prime_parser_post(parser, prim);
+
uw_catch(esym, eobj) {
yyerrorf(parser->scanner, lit("exception during parse"), nao);
uw_throw(esym, eobj);