picrin/extlib/benz/cont.c

292 lines
5.9 KiB
C
Raw Normal View History

2014-08-25 00:38:09 -04:00
/**
* See Copyright Notice in picrin.h
*/
#include "picrin.h"
2014-09-18 09:33:20 -04:00
void
2015-06-08 08:04:04 -04:00
pic_wind(pic_state *pic, pic_checkpoint *here, pic_checkpoint *there)
2014-09-18 09:33:20 -04:00
{
if (here == there)
return;
if (here->depth < there->depth) {
pic_wind(pic, here, there->prev);
pic_apply0(pic, there->in);
}
else {
pic_apply0(pic, there->out);
pic_wind(pic, here->prev, there);
}
}
pic_value
pic_dynamic_wind(pic_state *pic, struct pic_proc *in, struct pic_proc *thunk, struct pic_proc *out)
{
2015-06-08 08:04:04 -04:00
pic_checkpoint *here;
2014-09-18 09:33:20 -04:00
pic_value val;
if (in != NULL) {
pic_apply0(pic, in); /* enter */
}
2015-06-08 08:04:04 -04:00
here = pic->cp;
pic->cp = pic_malloc(pic, sizeof(pic_checkpoint));
pic->cp->prev = here;
pic->cp->depth = here->depth + 1;
pic->cp->in = in;
pic->cp->out = out;
2014-09-18 09:33:20 -04:00
val = pic_apply0(pic, thunk);
2015-06-08 08:04:04 -04:00
pic->cp = here;
2014-09-18 09:33:20 -04:00
if (out != NULL) {
pic_apply0(pic, out); /* exit */
}
return val;
}
2014-09-24 07:06:14 -04:00
void
2015-06-01 17:32:43 -04:00
pic_save_point(pic_state *pic, struct pic_cont *cont)
2014-09-19 03:41:49 -04:00
{
2015-06-01 17:32:43 -04:00
cont->jmp.prev = pic->jmp;
pic->jmp = &cont->jmp;
2014-09-19 03:41:49 -04:00
/* save runtime context */
2015-06-08 08:04:04 -04:00
cont->cp = pic->cp;
2015-06-01 17:32:43 -04:00
cont->sp_offset = pic->sp - pic->stbase;
cont->ci_offset = pic->ci - pic->cibase;
cont->xp_offset = pic->xp - pic->xpbase;
cont->arena_idx = pic->arena_idx;
cont->ip = pic->ip;
cont->results = pic_undef_value();
2014-09-19 03:41:49 -04:00
}
void
2015-06-01 17:32:43 -04:00
pic_load_point(pic_state *pic, struct pic_cont *cont)
2014-09-19 03:41:49 -04:00
{
pic_jmpbuf *jmp;
for (jmp = pic->jmp; jmp != NULL; jmp = jmp->prev) {
2015-06-01 17:32:43 -04:00
if (jmp == &cont->jmp) {
break;
}
}
if (jmp == NULL) {
2014-09-19 03:41:49 -04:00
pic_errorf(pic, "calling dead escape continuation");
}
2015-06-08 08:04:04 -04:00
pic_wind(pic, pic->cp, cont->cp);
2014-09-19 03:41:49 -04:00
/* load runtime context */
2015-06-08 08:04:04 -04:00
pic->cp = cont->cp;
2015-06-01 17:32:43 -04:00
pic->sp = pic->stbase + cont->sp_offset;
pic->ci = pic->cibase + cont->ci_offset;
pic->xp = pic->xpbase + cont->xp_offset;
pic->arena_idx = cont->arena_idx;
pic->ip = cont->ip;
2014-09-19 03:41:49 -04:00
}
2015-01-07 16:11:48 -05:00
static pic_value
2015-06-01 17:32:43 -04:00
cont_call(pic_state *pic)
2014-09-19 03:41:49 -04:00
{
2014-09-27 07:17:02 -04:00
size_t argc;
2014-09-19 03:41:49 -04:00
pic_value *argv;
struct pic_data *e;
pic_get_args(pic, "*", &argc, &argv);
2015-06-01 06:47:40 -04:00
e = pic_data_ptr(pic_proc_env_ref(pic, pic_get_proc(pic), "escape"));
((struct pic_cont *)e->data)->results = pic_list_by_array(pic, argc, argv);
2014-09-19 03:41:49 -04:00
2014-09-24 02:34:46 -04:00
pic_load_point(pic, e->data);
PIC_LONGJMP(pic, ((struct pic_cont *)e->data)->jmp.buf, 1);
2015-05-27 11:41:55 -04:00
PIC_UNREACHABLE();
2014-09-19 03:41:49 -04:00
}
2014-09-24 02:34:46 -04:00
struct pic_proc *
2015-06-01 17:32:43 -04:00
pic_make_cont(pic_state *pic, struct pic_cont *cont)
2014-09-24 01:57:49 -04:00
{
2015-06-01 17:32:43 -04:00
static const pic_data_type cont_type = { "cont", pic_free, NULL };
struct pic_proc *c;
2014-09-24 01:57:49 -04:00
struct pic_data *e;
2015-06-01 17:32:43 -04:00
c = pic_make_proc(pic, cont_call, "<cont-procedure>");
2014-09-24 01:57:49 -04:00
2015-06-01 17:32:43 -04:00
e = pic_data_alloc(pic, &cont_type, cont);
2014-09-24 01:57:49 -04:00
/* save the escape continuation in proc */
2015-06-01 06:47:40 -04:00
pic_proc_env_set(pic, c, "escape", pic_obj_value(e));
2014-09-24 01:57:49 -04:00
2015-06-01 17:32:43 -04:00
return c;
2014-09-24 01:57:49 -04:00
}
2014-09-19 03:41:49 -04:00
pic_value
pic_callcc(pic_state *pic, struct pic_proc *proc)
2014-09-19 03:41:49 -04:00
{
2015-06-01 17:32:43 -04:00
struct pic_cont *cont = pic_malloc(pic, sizeof(struct pic_cont));
2014-09-19 03:41:49 -04:00
2015-06-01 17:32:43 -04:00
pic_save_point(pic, cont);
2014-09-19 03:41:49 -04:00
2015-06-01 17:32:43 -04:00
if (PIC_SETJMP(pic, cont->jmp.buf)) {
pic->jmp = pic->jmp->prev;
2015-06-01 17:32:43 -04:00
return pic_values_by_list(pic, cont->results);
2014-09-19 03:41:49 -04:00
}
else {
pic_value val;
2015-06-01 17:32:43 -04:00
val = pic_apply1(pic, proc, pic_obj_value(pic_make_cont(pic, cont)));
2014-09-19 03:41:49 -04:00
pic->jmp = pic->jmp->prev;
2014-09-19 03:41:49 -04:00
return val;
}
}
2014-08-25 00:38:09 -04:00
pic_value
pic_values0(pic_state *pic)
{
return pic_values_by_list(pic, pic_nil_value());
}
pic_value
pic_values1(pic_state *pic, pic_value arg1)
{
return pic_values_by_list(pic, pic_list1(pic, arg1));
}
pic_value
pic_values2(pic_state *pic, pic_value arg1, pic_value arg2)
{
return pic_values_by_list(pic, pic_list2(pic, arg1, arg2));
}
pic_value
pic_values3(pic_state *pic, pic_value arg1, pic_value arg2, pic_value arg3)
{
return pic_values_by_list(pic, pic_list3(pic, arg1, arg2, arg3));
}
pic_value
pic_values4(pic_state *pic, pic_value arg1, pic_value arg2, pic_value arg3, pic_value arg4)
{
return pic_values_by_list(pic, pic_list4(pic, arg1, arg2, arg3, arg4));
}
pic_value
pic_values5(pic_state *pic, pic_value arg1, pic_value arg2, pic_value arg3, pic_value arg4, pic_value arg5)
{
return pic_values_by_list(pic, pic_list5(pic, arg1, arg2, arg3, arg4, arg5));
}
pic_value
2014-09-27 07:21:54 -04:00
pic_values_by_array(pic_state *pic, size_t argc, pic_value *argv)
2014-08-25 00:38:09 -04:00
{
2014-09-27 07:21:54 -04:00
size_t i;
2014-08-25 00:38:09 -04:00
for (i = 0; i < argc; ++i) {
pic->sp[i] = argv[i];
}
2014-09-27 07:21:54 -04:00
pic->ci->retc = (int)argc;
2014-08-25 00:38:09 -04:00
return argc == 0 ? pic_none_value() : pic->sp[0];
}
pic_value
pic_values_by_list(pic_state *pic, pic_value list)
{
2015-01-22 05:28:31 -05:00
pic_value v, it;
int i;
2014-08-25 00:38:09 -04:00
i = 0;
2015-01-22 05:28:31 -05:00
pic_for_each (v, list, it) {
2014-08-25 00:38:09 -04:00
pic->sp[i++] = v;
}
pic->ci->retc = i;
return pic_nil_p(list) ? pic_none_value() : pic->sp[0];
}
2014-09-27 07:21:54 -04:00
size_t
pic_receive(pic_state *pic, size_t n, pic_value *argv)
2014-08-25 00:38:09 -04:00
{
pic_callinfo *ci;
2014-09-27 07:21:54 -04:00
size_t i, retc;
2014-08-25 00:38:09 -04:00
/* take info from discarded frame */
ci = pic->ci + 1;
2014-09-27 07:21:54 -04:00
retc = (size_t)ci->retc;
2014-08-25 00:38:09 -04:00
for (i = 0; i < retc && i < n; ++i) {
argv[i] = ci->fp[i];
}
return retc;
}
static pic_value
pic_cont_callcc(pic_state *pic)
{
struct pic_proc *cb;
pic_get_args(pic, "l", &cb);
return pic_callcc(pic, cb);
2014-08-25 00:38:09 -04:00
}
static pic_value
pic_cont_dynamic_wind(pic_state *pic)
{
struct pic_proc *in, *thunk, *out;
pic_get_args(pic, "lll", &in, &thunk, &out);
return pic_dynamic_wind(pic, in, thunk, out);
}
static pic_value
pic_cont_values(pic_state *pic)
{
2014-09-27 07:17:02 -04:00
size_t argc;
2014-08-25 00:38:09 -04:00
pic_value *argv;
pic_get_args(pic, "*", &argc, &argv);
return pic_values_by_array(pic, argc, argv);
}
static pic_value
pic_cont_call_with_values(pic_state *pic)
{
struct pic_proc *producer, *consumer;
2014-09-27 07:21:54 -04:00
size_t argc;
2014-08-25 00:38:09 -04:00
pic_value args[256];
pic_get_args(pic, "ll", &producer, &consumer);
pic_apply(pic, producer, pic_nil_value());
argc = pic_receive(pic, 256, args);
return pic_apply_trampoline(pic, consumer, pic_list_by_array(pic, argc, args));
}
void
pic_init_cont(pic_state *pic)
{
2015-06-04 00:53:41 -04:00
void pic_defun_vm(pic_state *, const char *, pic_sym *, pic_func_t);
2014-08-25 00:38:09 -04:00
pic_defun(pic, "call-with-current-continuation", pic_cont_callcc);
2014-09-08 06:38:33 -04:00
pic_defun(pic, "call/cc", pic_cont_callcc);
2014-08-25 00:38:09 -04:00
pic_defun(pic, "dynamic-wind", pic_cont_dynamic_wind);
2015-06-04 00:53:41 -04:00
pic_defun_vm(pic, "values", pic->rVALUES, pic_cont_values);
pic_defun_vm(pic, "call-with-values", pic->rCALL_WITH_VALUES, pic_cont_call_with_values);
2014-08-25 00:38:09 -04:00
}