summaryrefslogtreecommitdiffstats
path: root/lib.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2021-04-28 20:26:59 -0700
committerKaz Kylheku <kaz@kylheku.com>2021-04-28 20:26:59 -0700
commite0cc6c8c21b2442cd5b1f8f51afadca6f3649d70 (patch)
treeaf3f3e710be36d2a2a49dca0fcf3547341171fa3 /lib.c
parent5410fe471273922ee53c579238e3d00bd28b6789 (diff)
downloadtxr-e0cc6c8c21b2442cd5b1f8f51afadca6f3649d70.tar.gz
txr-e0cc6c8c21b2442cd5b1f8f51afadca6f3649d70.tar.bz2
txr-e0cc6c8c21b2442cd5b1f8f51afadca6f3649d70.zip
lib: document gc problem related to seq-begin.
If seq-begin is used on an object that supports the iter-begin method, there is a gc problem. This does not seem worth fixing for the following reasons. 1. The seq-begin function is marked obsolescent. I removed its one and only internal use in the previous commit, so it won't be called unless application code uses it. 2. Objects supporting the iter-begin function are clearly developed as part of the new iteration protocol. It makes no sense to be newly developing such an object, along with new code which applies seq-begin to it. There is likely zero code in the wild which uses either of these mechanisms. * lib.c (seq_iter_get_oop, seq_iter_peek_oop, seq_iter_get_fast_oop, seq_iter_peek_fast_oop): Add comments documenting the issue.
Diffstat (limited to 'lib.c')
-rw-r--r--lib.c20
1 files changed, 20 insertions, 0 deletions
diff --git a/lib.c b/lib.c
index f60500e6..95db82e4 100644
--- a/lib.c
+++ b/lib.c
@@ -571,6 +571,20 @@ static int seq_iter_get_oop(seq_iter_t *it, val *pval)
{
val iter = it->ui.iter;
+ /* The assignments to ui.iter and ui.next are wrong if the it structure is
+ * embedded inside a heap-allocated iterator object. The object could be a
+ * gen 1 object, whereas the value being assigned could be a gen 0.
+ *
+ * The only way this can happen is if the obsolescent seq-begin function is
+ * used on an object that supports the iter-begin method. The seq-begin
+ * constructor is the only function which creates a heap-allocated iterator
+ * which is initialized via seq_iter_init_with_info, which binds this
+ * seq_iter_get_oop function and its sisters.
+ *
+ * The other heap-allocated iterator type is iter-begin. iter-begin applies
+ * its own handling for OOP iterators; it doesn't set up seq_iter_t for
+ * objects of that type, and so these functions are not used.
+ */
if (it->ul.next != nao) {
val iter_step_meth = get_special_required_slot(iter, iter_step_m);
*pval = it->ul.next;
@@ -597,6 +611,8 @@ static int seq_iter_peek_oop(seq_iter_t *it, val *pval)
{
val iter = it->ui.iter;
+ /* See comment in seq_iter_get_oop */
+
if (it->ul.next != nao) {
*pval = it->ul.next;
return 1;
@@ -629,6 +645,8 @@ static int seq_iter_get_fast_oop(seq_iter_t *it, val *pval)
*pval = item;
}
+ /* See comment in seq_iter_get_oop */
+
it->ui.iter = funcall1(iter_step_meth, iter);
it->ul.next = nao;
return 1;
@@ -646,6 +664,8 @@ static int seq_iter_peek_fast_oop(seq_iter_t *it, val *pval)
return 1;
}
+ /* See comment in seq_iter_get_oop */
+
if (iter) {
val iter_item_meth = get_special_required_slot(iter, iter_item_m);
it->ul.next = *pval = funcall1(iter_item_meth, iter);