diff options
author | Kaz Kylheku <kaz@kylheku.com> | 2020-06-04 07:00:18 -0700 |
---|---|---|
committer | Kaz Kylheku <kaz@kylheku.com> | 2020-06-04 07:00:18 -0700 |
commit | 18f4676d93c4601f76f28470865c2ae589cee095 (patch) | |
tree | 6a67d06d25fcfb82df2f0029d763f95d1e775f7b /eval.c | |
parent | a4e061cc3dfc7b4bfc93ef48386eecb9147a85c2 (diff) | |
download | txr-18f4676d93c4601f76f28470865c2ae589cee095.tar.gz txr-18f4676d93c4601f76f28470865c2ae589cee095.tar.bz2 txr-18f4676d93c4601f76f28470865c2ae589cee095.zip |
mapping: rewrite loop using seq_info and args.
* eval.c (map_common): Do not extract the arguments as a list.
Do not produce a list of iterator objects. Instead, allocate
an array of seq_iter_t objects on the stack using alloca, and
use these for walking the input lists in parallel. Do not cons
a list of the tuples coming from the lists, but rather store
the tuples into a struct args, also on the stack, and invoke
the function with that. Now, the only heap memory we allocate
is the resulting list being accumulated. In the case of mapdo,
no heap allocation takes place. However, if some of the inputs
are hashes, then hash iterators get allocated in seq_iter_init.
(mapcarv, mappendv, mapdov): Pass self argument to map_common,
needed for seq_iter_init.
Diffstat (limited to 'eval.c')
-rw-r--r-- | eval.c | 46 |
1 files changed, 27 insertions, 19 deletions
@@ -5275,7 +5275,7 @@ static val abscond_star(val name, val retval) abort(); } -static val map_common(val fun, struct args *lists, +static val map_common(val self, val fun, struct args *lists, loc (*collect_fn)(loc ptail, val obj), val (*map_fn)(val fun, val seq)) { @@ -5284,26 +5284,34 @@ static val map_common(val fun, struct args *lists, } else if (!args_two_more(lists, 0)) { return map_fn(fun, args_atz(lists, 0)); } else { - val list_of_lists = args_get_list(lists); - val list_of_iters = mapcar_listout(iter_begin_f, list_of_lists); - val list_orig = car(list_of_lists); + cnum i, idx, argc = args_count(lists); + seq_iter_t *iter_array = coerce(seq_iter_t *, + alloca(argc * sizeof *iter_array)); list_collect_decl (out, otail); + for (i = 0, idx = 0; i < argc; i++) + { + val arg = args_get(lists, &idx); + seq_iter_init(self, &iter_array[i], arg); + } + for (;;) { - val iiter, fun_ret; - list_collect_decl (args, atail); - - for (iiter = list_of_iters; iiter; iiter = cdr(iiter)) { - val iter = car(iiter); - if (!iter_more(iter)) { - rcyc_list(list_of_iters); - return collect_fn != 0 ? make_like(out, list_orig) : nil; - } - atail = list_collect(atail, iter_item(iter)); - deref(car_l(iiter)) = iter_step(iter); + val fun_ret; + args_decl(args_fun, max(argc, ARGS_MIN)); + + args_fun->fill = argc; + + for (i = 0; i < argc; i++) { + val elem; + seq_iter_t *iter = &iter_array[i]; + + if (!seq_get(iter, &elem)) + return collect_fn != 0 ? make_like(out, args_at(lists, 0)) : nil; + + args_fun->arg[i] = elem; } - fun_ret = apply(fun, z(args)); + fun_ret = generic_funcall(fun, args_fun); if (collect_fn != 0) otail = collect_fn(otail, fun_ret); @@ -5313,7 +5321,7 @@ static val map_common(val fun, struct args *lists, val mapcarv(val fun, struct args *lists) { - return map_common(fun, lists, list_collect, mapcar); + return map_common(lit("mapcar"), fun, lists, list_collect, mapcar); } val mapcarl(val fun, val list_of_lists) @@ -5324,12 +5332,12 @@ val mapcarl(val fun, val list_of_lists) static val mappendv(val fun, struct args *lists) { - return map_common(fun, lists, list_collect_append, mappend); + return map_common(lit("mappend"), fun, lists, list_collect_append, mappend); } static val mapdov(val fun, struct args *lists) { - return map_common(fun, lists, 0, mapdo); + return map_common(lit("mapdo"), fun, lists, 0, mapdo); } static val lazy_mapcar_func(val env, val lcons) |