summaryrefslogtreecommitdiffstats
path: root/hash.c
diff options
context:
space:
mode:
authorKaz Kylheku <kaz@kylheku.com>2014-01-16 06:23:49 -0800
committerKaz Kylheku <kaz@kylheku.com>2014-01-16 06:23:49 -0800
commitda6ce99db9ac21b417262f5d23e605aee391b70e (patch)
tree7c67c460c2da78d99a7b1ac7cd4db5f26c689679 /hash.c
parent6a3afa9e22cd33e88f0b1527fcff06c6b42c355d (diff)
downloadtxr-da6ce99db9ac21b417262f5d23e605aee391b70e.tar.gz
txr-da6ce99db9ac21b417262f5d23e605aee391b70e.tar.bz2
txr-da6ce99db9ac21b417262f5d23e605aee391b70e.zip
* hash.c (group_by): New function.
* hash.h (group_by): Declared. * eval.c (eval_init): group_by registered as group-by intrinsic. * txr.1: Documented.
Diffstat (limited to 'hash.c')
-rw-r--r--hash.c29
1 files changed, 29 insertions, 0 deletions
diff --git a/hash.c b/hash.c
index 3721baf4..21168c0c 100644
--- a/hash.c
+++ b/hash.c
@@ -684,6 +684,35 @@ val hash_construct(val hashv_args, val pairs)
return hash;
}
+val group_by(val func, val seq, val hashv_args)
+{
+ val hash = hashv(hashv_args);
+
+ if (vectorp(seq)) {
+ cnum i, len;
+
+ for (i = 0, len = c_num(length(seq)); i < len; i++) {
+ val v = vecref(seq, num_fast(i));
+ pushhash(hash, funcall1(func, v), v);
+ }
+ } else {
+ for (; seq; seq = cdr(seq)) {
+ val v = car(seq);
+ pushhash(hash, funcall1(func, v), v);
+ }
+ }
+
+ {
+ val iter = hash_begin(hash);
+ val cell;
+
+ while ((cell = hash_next(iter)) != nil)
+ rplacd(cell, nreverse(cdr(cell)));
+
+ return hash;
+ }
+}
+
static val hash_keys_lazy(val iter, val lcons)
{
val cell = hash_next(iter);