dynamic-wind: work in progress II

This commit is contained in:
ESL 2023-03-30 01:27:28 -04:00
parent 29d212eeff
commit 5dd5ff2c07

38
i.c
View file

@ -688,12 +688,15 @@ define_instruction(appl) {
define_instruction(cwmv) {
obj t = ac, x = spop();
ckx(t); ckx(x);
if (vmcloref(x, 0) == cx_continuation_2Dadapter_2Dcode) {
/* we can run in constant space in some situations */
if (vmcloref(x, 0) == cx_continuation_2Dadapter_2Dcode
&& vmcloref(x, 1) == cx__2Adynamic_2Dstate_2A) {
/* arrange call of t with x as continuation */
int n = vmclolen(x) - 1;
/* [0] adapter_code, [1] dynamic_state */
int n = vmclolen(x) - 2;
assert((cxg_rend - cxg_regs - VM_REGC) > n);
sp = r + VM_REGC; /* stack is empty */
objcpy(sp, &vmcloref(x, 1), n);
objcpy(sp, &vmcloref(x, 2), n);
sp += n; /* contains n elements now */
rd = t; rx = fixnum_obj(0);
ac = fixnum_obj(0);
@ -786,11 +789,13 @@ define_instruction(lck0) {
}
define_instruction(wck) {
obj x = ac, t = spop(); int n; ckx(t); ckx(x);
obj x = ac, t = spop(); ckx(t); ckx(x);
if (vmcloref(x, 0) != cx_continuation_2Dadapter_2Dcode)
failactype("continuation");
/* [0] adapter_code, [1] dynamic_state */
n = vmclolen(x) - 2;
if (vmcloref(x, 1) == cx__2Adynamic_2Dstate_2A) {
/* restore cont stack and invoke t there */
int n = vmclolen(x) - 2;
assert((cxg_rend - cxg_regs - VM_REGC) > n);
sp = r + VM_REGC; /* stack is empty */
objcpy(sp, &vmcloref(x, 2), n);
@ -798,14 +803,26 @@ define_instruction(wck) {
rd = t; rx = fixnum_obj(0);
ac = fixnum_obj(0);
callsubi();
} else {
/* have to arrange call of cont adapter */
spush(x);
spush(cx_callmv_2Dadapter_2Dclosure);
spush(fixnum_obj(0));
/* call the thunk as producer */
rd = t; rx = fixnum_obj(0);
ac = fixnum_obj(0);
callsubi();
}
}
define_instruction(wckr) {
obj x = ac, o = spop(); int n; ckx(x);
obj x = ac, o = spop(); ckx(x);
if (vmcloref(x, 0) != cx_continuation_2Dadapter_2Dcode)
failactype("continuation");
/* [0] adapter_code, [1] dynamic_state */
n = vmclolen(x) - 2;
if (vmcloref(x, 1) == cx__2Adynamic_2Dstate_2A) {
/* restore cont stack and return o there */
int n = vmclolen(x) - 2;
assert((cxg_rend - cxg_regs - VM_REGC) > n);
sp = r + VM_REGC; /* stack is empty */
objcpy(sp, &vmcloref(x, 2), n);
@ -814,6 +831,13 @@ define_instruction(wckr) {
rx = spop();
rd = spop();
retfromi();
} else {
/* have to arrange call of cont adapter */
spush(o);
rd = x; rx = fixnum_obj(0);
ac = fixnum_obj(1);
callsubi();
}
}
define_instruction(kdys) {