diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2016-10-18 05:41:45 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2016-10-18 05:41:45 -0700 |
commit | f91ef728d1149d7a849d7c818b3fdc03c61847ad (patch) | |
tree | 1114df7103b610538a25a00ea603afc72960daf7 /parser.y | |
parent | b25e08a3328ffa4bf29a8a8651973d88ae79b90c (diff) | |
download | txr-f91ef728d1149d7a849d7c818b3fdc03c61847ad.tar.gz txr-f91ef728d1149d7a849d7c818b3fdc03c61847ad.tar.bz2 txr-f91ef728d1149d7a849d7c818b3fdc03c61847ad.zip |
Detect cycles in rlcp_tree.
This will be required when the parser becomes
capable of creating object graphs with cycles.
* parser.c (parser_callgraph_circ_check): New function.
* parser.h (struct circ_stack): New struct.
(parser_callgraph_circ_check): Declared.
* parser.y (rlcp_tree_rec): New static function.
(rlcp_tree): Reduced to wrapper for rlcp_tree_rec.
Diffstat (limited to 'parser.y')
-rw-r--r-- | parser.y | 19 |
1 files changed, 16 insertions, 3 deletions
@@ -1501,17 +1501,30 @@ val rlrec(parser_t *parser, val form, val line) return form; } -val rlcp_tree(val to, val from) +static val rlcp_tree_rec(val to, val from, struct circ_stack *up) { val ret = to; - for (; consp(to); to = cdr(to)) { + while (consp(to)) { + val a = car(to); + struct circ_stack rlcs = { up, a }; rlcp(to, from); - rlcp_tree(car(to), from); + if (!parser_callgraph_circ_check(up, a)) + break; + rlcp_tree_rec(a, from, &rlcs); + to = cdr(to); + if (!parser_callgraph_circ_check(up, to)) + break; } return ret; } + +val rlcp_tree(val to, val from) +{ + return rlcp_tree_rec(to, from, 0); +} + static wchar_t char_from_name(const wchar_t *name) { static struct { |