summaryrefslogtreecommitdiffstats
path: root/parser.y
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2016-10-18 05:42:58 -0700
committerKaz Kylheku <kaz@kylheku.com>2016-10-18 05:42:58 -0700
commit37636d9b1ceae173635d18043e21f446dfbd2490 (patch)
tree87bb18340363df6aa405aa7e8dea45b3229239ec /parser.y
parentf91ef728d1149d7a849d7c818b3fdc03c61847ad (diff)
downloadtxr-37636d9b1ceae173635d18043e21f446dfbd2490.tar.gz
txr-37636d9b1ceae173635d18043e21f446dfbd2490.tar.bz2
txr-37636d9b1ceae173635d18043e21f446dfbd2490.zip
Adding notation for cycles and shared structure.
This commit implements the parse-side support for handling a notation that exists in ANSI Common Lisp for specifying objects with cycles and shared substructure. * parser.h (struct parser): New members, circ_ref_hash and circ_count. (circref_s, parser_resolve_circ, parser_circ_def, parser_circ_ref): Declared. * parser.c (circref_s): New symbol variable. (parser_mark): Visit the new circ_ref_hash member of the parser structure. (parser_common_init): Initialize new members circ_ref_hash and circ_count of parser structure. (patch_ref, circ_backpatch): New static functions. (parser_resolve_circ, parser_circ_def, parser_circ_ref): New functions. (circref): New static function. (parse_init): Initialize circref_s as sys:circref symbol. Register sys:circref function. * parser.l (grammar): Scan #<num>= and #<num># notation as tokens, extracting their numeric value. * parser.y (HASH_N_EQUALS, HASH_N_HASH): New token types. (i_expr, n_expr): Adding phrases for hash-equalsign and hash-hash syntax. (yybadtoken): Handle new token types in switch. (parse_once): Call parser_resolve_circ after parsing to rewrite any remaining #<num># references in the structure to the objects they denote. (parse): Reset new struct parse members to initial state. Call parser_resolve_circ after parsing to rewrite any remaining #<num># references.
Diffstat (limited to 'parser.y')
-rw-r--r--parser.y19
1 files changed, 18 insertions, 1 deletions
diff --git a/parser.y b/parser.y
index 8ae40e82..c3bc60e5 100644
--- a/parser.y
+++ b/parser.y
@@ -108,6 +108,7 @@ int yyparse(scanner_t *, parser_t *);
%token <lineno> SECRET_ESCAPE_R SECRET_ESCAPE_E SECRET_ESCAPE_I
%token <val> NUMBER METANUM
+%token <val> HASH_N_EQUALS HASH_N_HASH
%token <chr> REGCHAR REGTOKEN LITCHAR SPLICE CONSDOT LAMBDOT
@@ -136,7 +137,7 @@ int yyparse(scanner_t *, parser_t *);
%right SYMTOK '{' '}'
%right ALL SOME NONE MAYBE CASES CHOOSE AND OR END COLLECT UNTIL COLL
%right OUTPUT REPEAT REP FIRST LAST EMPTY DEFINE IF ELIF ELSE
-%right SPACE TEXT NUMBER METANUM
+%right SPACE TEXT NUMBER METANUM HASH_N_EQUALS HASH_N_HASH
%nonassoc '[' ']' '(' ')'
%left '-' ',' '\'' '^' SPLICE '@'
%left '|' '/'
@@ -934,6 +935,10 @@ i_expr : SYMTOK { $$ = symhlpr($1, t); }
num(parser->lineno)); }
| SPLICE i_expr { $$ = rl(rlcp(list(sys_splice_s, $2, nao), $2),
num(parser->lineno)); }
+ | HASH_N_EQUALS { parser_circ_def(parser, $1, unique_s); }
+ i_expr { parser_circ_def(parser, $1, $3);
+ $$ = $3; }
+ | HASH_N_HASH { $$ = parser_circ_ref(parser, $1); }
;
n_expr : SYMTOK { $$ = symhlpr($1, t); }
@@ -972,6 +977,10 @@ n_expr : SYMTOK { $$ = symhlpr($1, t); }
or2($1, $3)),
num(parser->lineno));
} }
+ | HASH_N_EQUALS { parser_circ_def(parser, $1, unique_s); }
+ n_expr { parser_circ_def(parser, $1, $3);
+ $$ = $3; }
+ | HASH_N_HASH { $$ = parser_circ_ref(parser, $1); }
;
n_exprs_opt : n_exprs { $$ = $1; }
@@ -1664,6 +1673,8 @@ void yybadtoken(parser_t *parser, int tok, val context)
case HASH_SLASH: problem = lit("#/"); break;
case HASH_H: problem = lit("#H"); break;
case HASH_S: problem = lit("#S"); break;
+ case HASH_N_EQUALS: problem = lit("#<n>="); break;
+ case HASH_N_HASH: problem = lit("#<n>#"); break;
case WORDS: problem = lit("#\""); break;
case WSPLICE: problem = lit("#*\""); break;
case QWORDS: problem = lit("#`"); break;
@@ -1710,6 +1721,8 @@ int parse_once(val stream, val name, parser_t *parser)
res = yyparse(parser->scanner, parser);
+ parser_resolve_circ(parser);
+
uw_catch(esym, eobj) {
yyerrorf(parser->scanner, lit("exception during parse"), nao);
uw_throw(esym, eobj);
@@ -1733,6 +1746,8 @@ int parse(parser_t *parser, val name, enum prime_parser prim)
parser->errors = 0;
parser->prepared_msg = nil;
+ parser->circ_ref_hash = nil;
+ parser->circ_count = 0;
parser->syntax_tree = nil;
prime_parser(parser, name, prim);
@@ -1743,6 +1758,8 @@ int parse(parser_t *parser, val name, enum prime_parser prim)
prime_parser_post(parser, prim);
+ parser_resolve_circ(parser);
+
uw_catch(esym, eobj) {
yyerrorf(parser->scanner, lit("exception during parse"), nao);
uw_throw(esym, eobj);