summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
-rw-r--r--lib.c200
-rw-r--r--lib.h2
-rw-r--r--txr.120
3 files changed, 104 insertions, 118 deletions
diff --git a/lib.c b/lib.c
index 1aae603a..6180d87a 100644
--- a/lib.c
+++ b/lib.c
@@ -450,6 +450,34 @@ void seq_iter_init(val self, seq_iter_t *it, val obj)
}
}
+val seq_getpos(val self, seq_iter_t *it)
+{
+ switch (it->inf.kind) {
+ case SEQ_NIL:
+ case SEQ_LISTLIKE:
+ return it->ui.iter;
+ case SEQ_VECLIKE:
+ return num(it->ui.index);;
+ default:
+ unsup_obj(self, it->inf.obj);
+ }
+}
+
+void seq_setpos(val self, seq_iter_t *it, val pos)
+{
+ switch (it->inf.kind) {
+ case SEQ_NIL:
+ case SEQ_LISTLIKE:
+ it->ui.iter = pos;
+ break;
+ case SEQ_VECLIKE:
+ it->ui.index = c_num(pos);
+ break;
+ default:
+ unsup_obj(self, it->inf.obj);
+ }
+}
+
static void seq_iter_mark(val seq_iter)
{
struct seq_iter *si = coerce(struct seq_iter *, seq_iter->co.handle);
@@ -10778,144 +10806,88 @@ val update(val seq, val fun)
return seq;
}
-static val search_list(val seq, val key, val testfun, val keyfun)
-{
- val siter, kiter;
- val pos = zero;
-
- switch (type(key)) {
- case NIL:
- return pos;
- case CONS:
- case LCONS:
- case LIT:
- case STR:
- case LSTR:
- case VEC:
- /* TODO: optimize me */
- gc_hint(seq);
-
- for (; seq; seq = cdr(seq), pos = plus(pos, one)) {
- for (siter = seq, kiter = key;
- siter && kiter;
- siter = cdr(siter), kiter = cdr(kiter))
- {
- if (!funcall2(testfun,
- funcall1(keyfun, car(siter)),
- funcall1(keyfun, car(kiter))))
- {
- break;
- }
- }
-
- if (!kiter)
- return pos;
-
- if (!siter)
- break;
- }
- break;
- default:
- type_mismatch(lit("search: ~s is not a sequence"), key, nao);
- }
-
- return nil;
-}
-
-val search(val seq, val key, val testfun, val keyfun)
+static val search_common(val self, int from_right,
+ val seq, val key, val testfun, val keyfun)
{
testfun = default_arg(testfun, equal_f);
keyfun = default_arg(keyfun, identity_f);
- seq = nullify(seq);
+ seq_iter_t si, ki;
- switch (type(seq)) {
- case NIL:
+ seq_iter_init(self, &si, seq);
+ seq_iter_init(self, &ki, key);
+
+ if (si.inf.kind == SEQ_NIL) {
return if3(length(key) == zero, zero, nil);
- case CONS:
- case LCONS:
- case LIT:
- case STR:
- case LSTR:
- case VEC:
- case COBJ:
- /* TODO: optimize me */
- return search_list(seq, key, testfun, keyfun);
- default:
- type_mismatch(lit("search: ~s is not a sequence"), seq, nao);
- }
-}
+ } else if (ki.inf.kind == SEQ_HASHLIKE || si.inf.kind == SEQ_HASHLIKE) {
+ type_mismatch(lit("~a: hashes not supported"), self, nao);
+ } else if (ki.inf.kind == SEQ_NIL) {
+ return if3(from_right, length(seq), zero);
+ } else {
+ val selem, kelem;
+ val pos = zero, found = nil;
-val contains(val key, val seq, val testfun, val keyfun)
-{
- return search(seq, key, testfun, keyfun);
-}
+ if (!seq_peek(&ki, &kelem))
+ return if3(from_right, length(seq), zero);
-static val rsearch_list(val seq, val key, val testfun, val keyfun)
-{
- val siter, kiter;
- val pos = zero;
- val found = nil;
+ for (;;) {
+ val did_save = nil;
+ val saved_seq_pos = nil;
+ val saved_key_pos = nil;
+ int more_seq, more_key;
+
+ while ((more_seq = seq_peek(&si, &selem),
+ more_key = seq_peek(&ki, &kelem),
+ more_key && more_seq))
+ {
+ if (!did_save)
+ saved_key_pos = seq_getpos(self, &ki);
- switch (type(key)) {
- case NIL:
- return pos;
- case CONS:
- case LCONS:
- case LIT:
- case STR:
- case LSTR:
- case VEC:
- /* TODO: optimize me */
- gc_hint(seq);
+ seq_geti(&si);
+ seq_geti(&ki);
+
+ if (!did_save) {
+ saved_seq_pos = seq_getpos(self, &si);
+ did_save = t;
+ }
- for (; seq; seq = cdr(seq), pos = plus(pos, one)) {
- for (siter = seq, kiter = key;
- siter && kiter;
- siter = cdr(siter), kiter = cdr(kiter))
- {
if (!funcall2(testfun,
- funcall1(keyfun, car(siter)),
- funcall1(keyfun, car(kiter))))
+ funcall1(keyfun, selem),
+ funcall1(keyfun, kelem)))
{
break;
}
}
- if (!kiter)
- found = pos;
+ if (!more_key) {
+ if (from_right)
+ found = pos;
+ else
+ return pos;
+ }
+
+ if (!more_seq)
+ return found;
- if (!siter)
- break;
+ pos = plus(pos, one);
+ seq_setpos(self, &si, saved_seq_pos);
+ seq_setpos(self, &ki, saved_key_pos);
}
- break;
- default:
- type_mismatch(lit("rsearch: ~s is not a sequence"), key, nao);
}
+}
- return found;
+val search(val seq, val key, val testfun, val keyfun)
+{
+ return search_common(lit("search"), 0, seq, key, testfun, keyfun);
}
val rsearch(val seq, val key, val testfun, val keyfun)
{
- testfun = default_arg(testfun, equal_f);
- keyfun = default_arg(keyfun, identity_f);
- seq = nullify(seq);
+ return search_common(lit("rsearch"), 1, seq, key, testfun, keyfun);
+}
- switch (type(seq)) {
- case NIL:
- return if3(length(key) == zero, zero, nil);
- case CONS:
- case LCONS:
- case LIT:
- case STR:
- case LSTR:
- case VEC:
- case COBJ:
- /* TODO: optimize me */
- return rsearch_list(seq, key, testfun, keyfun);
- default:
- type_mismatch(lit("rsearch: ~s is not a sequence"), seq, nao);
- }
+val contains(val key, val seq, val testfun, val keyfun)
+{
+ return search_common(lit("contains"), 0, seq, key, testfun, keyfun);
}
static val lazy_where_func(val seq_iter, val lcons)
diff --git a/lib.h b/lib.h
index b3541b20..87a5f266 100644
--- a/lib.h
+++ b/lib.h
@@ -551,6 +551,8 @@ void seq_iter_rewind(seq_iter_t *it);
INLINE int seq_get(seq_iter_t *it, val *pval) { return it->get(it, pval); }
INLINE int seq_peek(seq_iter_t *it, val *pval) { return it->peek(it, pval); }
val seq_geti(seq_iter_t *it);
+val seq_getpos(val self, seq_iter_t *it);
+void seq_setpos(val self, seq_iter_t *it, val pos);
val seq_begin(val obj);
val seq_next(val iter, val end_val);
val seq_reset(val iter, val obj);
diff --git a/txr.1 b/txr.1
index 1afe9b7c..5b531ab5 100644
--- a/txr.1
+++ b/txr.1
@@ -29782,12 +29782,11 @@ The arguments
.meta haystack
and
.meta needle
-are sequences: lists, vectors
-or strings, in any combination.
+are sequences. They may not be hash tables.
If
.meta needle
-is not empty, then occurs at some position N within
+is not empty, then it occurs at some position N within
.meta haystack
if
the first element of
@@ -29865,7 +29864,9 @@ The
.code rsearch
function is like
.code search
-except that if
+except for two differences.
+
+Firstly, if
.meta needle
matches
.meta haystack
@@ -29874,6 +29875,17 @@ in multiple places,
returns the right-most matching position rather than
the leftmost.
+Secondly, if
+.meta needle
+is an empty sequence, then
+.code rsearch
+returns the length of
+.codn haystack ,
+thereby effectively declaring that the rightmost match for an empty
+.meta needle
+key occurs at the imaginary position past the element of
+.metn haystack .
+
.coNP Functions @ ref and @ refset
.synb
.mets (ref < seq << index )