summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lib.c55
-rw-r--r--tests/012/iter.tl19
-rw-r--r--txr.139
3 files changed, 109 insertions, 4 deletions
diff --git a/lib.c b/lib.c
index 6175497f..32ece94c 100644
--- a/lib.c
+++ b/lib.c
@@ -1172,6 +1172,14 @@ val iter_begin(val obj)
}
}
+static val iter_dynamic(struct seq_iter *si_orig)
+{
+ struct seq_iter *si = coerce(struct seq_iter *,
+ chk_copy_obj(coerce(mem_t *, si_orig),
+ sizeof *si));
+ return cobj(coerce(mem_t *, si), seq_iter_cls, &seq_iter_ops);
+}
+
val iter_more(val iter)
{
switch (type(iter)) {
@@ -2447,6 +2455,47 @@ val sub_list(val list, val from, val to)
}
}
+static val sub_iter(val obj, val from, val to)
+{
+ val self = lit("sub");
+ seq_iter_t iter;
+ seq_iter_init(self, &iter, obj);
+ val idx = zero, elem;
+ list_collect_decl (out, ptail);
+
+ if (from == t)
+ return nil;
+
+ if (null_or_missing_p(from))
+ from = zero;
+ else if (minusp(from))
+ goto list;
+
+ if (to == t || null_or_missing_p(to))
+ to = nil;
+ else if (!null_or_missing_p(to) && minusp(to))
+ goto list;
+
+ if (!to) {
+ do {
+ if (ge(idx, from))
+ return iter_dynamic(&iter);
+ idx = succ(idx);
+ } while (seq_get(&iter, &elem));
+ } else {
+ for (; seq_get(&iter, &elem) && lt(idx, to); idx = succ(idx))
+ if (ge(idx, from))
+ ptail = list_collect(ptail, elem);
+ }
+
+ return out;
+
+list:
+ while (seq_get(&iter, &elem))
+ ptail = list_collect(ptail, elem);
+ return sub_list(out, from, to);
+}
+
val replace_list(val list, val items, val from, val to)
{
val self = lit("replace-list");
@@ -7727,6 +7776,7 @@ INLINE val do_generic_funcall(val fun, struct args *args_in)
case LSTR:
case BUF:
carray:
+ default:
bug_unless (args->argc >= ARGS_MIN);
args_normalize_least(args, 3);
@@ -7815,9 +7865,6 @@ INLINE val do_generic_funcall(val fun, struct args *args_in)
fun = method(fun, lambda_s);
break;
}
- /* fallthrough */
- default:
- callerror(fun, lit("object is not callable"));
}
variadic = fun->f.variadic;
@@ -11990,7 +12037,7 @@ val sub(val seq, val from, val to)
}
/* fallthrough */
default:
- type_mismatch(lit("sub: ~s is not a sequence"), seq, nao);
+ return sub_iter(seq, from, to);
}
}
diff --git a/tests/012/iter.tl b/tests/012/iter.tl
index f98107a5..1151d8ca 100644
--- a/tests/012/iter.tl
+++ b/tests/012/iter.tl
@@ -40,3 +40,22 @@
"DC11" "DC12" "DC13" "DC14" "DC15" "DC16" "DC17" "DC18" "DC19"
"DD01" "DD02" "DD03" "DD04" "DD05" "DD06" "DD07" "DD08" "DD09"
"DD11" "DD12" "DD13" "DD14" "DD15" "DD16" "DD17" "DD18" "DD19"))
+
+(mtest
+ [maprod append "A".."F" (drop 1 "00".."15")]
+ ("A01" "A02" "A03" "A04" "A05" "A10" "A11" "A12" "A13" "A14" "A15"
+ "B01" "B02" "B03" "B04" "B05" "B10" "B11" "B12" "B13" "B14" "B15"
+ "C01" "C02" "C03" "C04" "C05" "C10" "C11" "C12" "C13" "C14" "C15"
+ "D01" "D02" "D03" "D04" "D05" "D10" "D11" "D12" "D13" "D14" "D15"
+ "E01" "E02" "E03" "E04" "E05" "E10" "E11" "E12" "E13" "E14" "E15"
+ "F01" "F02" "F03" "F04" "F05" "F10" "F11" "F12" "F13" "F14" "F15")
+ [maprod append "A".."F" ["00".."99" 1..11]]
+ ("A01" "A02" "A03" "A04" "A05" "A06" "A07" "A08" "A09" "A10" "B01"
+ "B02" "B03" "B04" "B05" "B06" "B07" "B08" "B09" "B10" "C01" "C02"
+ "C03" "C04" "C05" "C06" "C07" "C08" "C09" "C10" "D01" "D02" "D03"
+ "D04" "D05" "D06" "D07" "D08" "D09" "D10" "E01" "E02" "E03" "E04"
+ "E05" "E06" "E07" "E08" "E09" "E10" "F01" "F02" "F03" "F04" "F05"
+ "F06" "F07" "F08" "F09" "F10"))
+
+(test
+ [mapcar identity [0..10 3 6]] (3 4 5))
diff --git a/txr.1 b/txr.1
index 61354add..ad0e3da7 100644
--- a/txr.1
+++ b/txr.1
@@ -15533,6 +15533,15 @@ is a list, then the
.meta sequence
form itself must be a place.
+This form is implemented using the
+.code ref
+accessor such that, except for the argument evaluation semantics of the DWIM
+brackets, it is equivalent to using the
+.mono
+.meti (ref < sequence << index )
+.onom
+syntax.
+
.meIP >> [ sequence << from-index..to-below-index ]
Retrieve the specified range of elements.
The range of elements is specified in the
@@ -15564,6 +15573,19 @@ regardless of whether the target is a string, vector or list.
If the target is a string, the replacement sequence must be
a string, or a list or vector of characters.
+The semantics is implemented using the
+.code sub
+accessor, such that the following equivalence holds:
+
+.verb
+ [seq from..to] <--> (sub seq from..to)
+.brev
+
+For this reason,
+.meta sequence
+may be any object that is iterable by
+.codn iter-begin .
+
.meIP >> [ sequence << index-list ]
Elements specified
by
@@ -31512,6 +31534,23 @@ is not an accessor, assigning to the
.code sub
syntax in this case will produce an error.
+The
+.meta sequence
+argument may also be any other object type that is suitable as input to the
+.code iter-begin
+function. In this situation, assigning to
+.code sub
+syntax produces an error. The behavior is complex. In cases where the
+.meta from
+and
+.meta to
+arguments imply that a suffix of
+.meta sequence
+is required, an iterator may be returned which traverses the suffix
+of the sequence. In other cases, a list of the elements selected by
+.code sub
+is returned.
+
If
.meta sequence
is a structure, it must support the