support internal definitions
This commit is contained in:
parent
2fb4720e22
commit
b9fce69c61
|
@ -166,6 +166,22 @@ scope_global_define(pic_state *pic, const char *name)
|
|||
return e->val;
|
||||
}
|
||||
|
||||
static int
|
||||
scope_local_define(pic_state *pic, const char *name, codegen_scope *scope)
|
||||
{
|
||||
struct xh_entry *e;
|
||||
|
||||
e = xh_put(scope->local_tbl, name, scope->argc + scope->localc++);
|
||||
scope->dirty_flags = (int *)pic_realloc(pic, scope->dirty_flags, (e->val) * sizeof(int));
|
||||
return e->val;
|
||||
}
|
||||
|
||||
static bool
|
||||
scope_is_global(codegen_scope *scope)
|
||||
{
|
||||
return scope->up == NULL;
|
||||
}
|
||||
|
||||
static void codegen_call(codegen_state *, pic_value, bool);
|
||||
static struct pic_irep *codegen_lambda(codegen_state *, pic_value);
|
||||
|
||||
|
@ -220,6 +236,7 @@ codegen(codegen_state *state, pic_value obj, bool tailpos)
|
|||
if (sym == pic->sDEFINE) {
|
||||
int idx;
|
||||
pic_value var, val;
|
||||
codegen_scope *s;
|
||||
|
||||
if (pic_length(pic, obj) < 2) {
|
||||
pic_error(pic, "syntax error");
|
||||
|
@ -242,8 +259,9 @@ codegen(codegen_state *state, pic_value obj, bool tailpos)
|
|||
pic_error(pic, "syntax error");
|
||||
}
|
||||
|
||||
s = state->scope;
|
||||
if (scope_is_global(s)) {
|
||||
idx = scope_global_define(pic, pic_symbol_name(pic, pic_sym(var)));
|
||||
|
||||
codegen(state, val, false);
|
||||
irep->code[irep->clen].insn = OP_GSET;
|
||||
irep->code[irep->clen].u.i = idx;
|
||||
|
@ -252,6 +270,17 @@ codegen(codegen_state *state, pic_value obj, bool tailpos)
|
|||
irep->clen++;
|
||||
break;
|
||||
}
|
||||
else {
|
||||
idx = scope_local_define(pic, pic_symbol_name(pic, pic_sym(var)), s);
|
||||
codegen(state, val, false);
|
||||
irep->code[irep->clen].insn = OP_LSET;
|
||||
irep->code[irep->clen].u.i = idx;
|
||||
irep->clen++;
|
||||
irep->code[irep->clen].insn = OP_PUSHFALSE;
|
||||
irep->clen++;
|
||||
break;
|
||||
}
|
||||
}
|
||||
else if (sym == pic->sLAMBDA) {
|
||||
int k = pic->ilen++;
|
||||
irep->code[irep->clen].insn = OP_LAMBDA;
|
||||
|
@ -738,7 +767,6 @@ codegen_lambda(codegen_state *state, pic_value obj)
|
|||
state->scope = new_local_scope(pic, args, state->scope);
|
||||
state->irep = irep = new_irep(pic);
|
||||
irep->argc = state->scope->argc;
|
||||
irep->localc = state->scope->localc;
|
||||
irep->varg = state->scope->varg;
|
||||
{
|
||||
/* body */
|
||||
|
@ -756,6 +784,8 @@ codegen_lambda(codegen_state *state, pic_value obj)
|
|||
irep->code[irep->clen].insn = OP_RET;
|
||||
irep->clen++;
|
||||
|
||||
irep->localc = state->scope->localc;
|
||||
|
||||
/* fixup local references */
|
||||
for (i = 0; i < irep->clen; ++i) {
|
||||
struct pic_code c = irep->code[i];
|
||||
|
|
9
src/vm.c
9
src/vm.c
|
@ -484,6 +484,15 @@ pic_apply(pic_state *pic, struct pic_proc *proc, pic_value argv)
|
|||
}
|
||||
}
|
||||
|
||||
/* prepare local variable area */
|
||||
if (proc->u.irep->localc > 0) {
|
||||
int l = proc->u.irep->localc;
|
||||
if (proc->u.irep->varg) {
|
||||
--l;
|
||||
}
|
||||
pic->sp += l;
|
||||
}
|
||||
|
||||
pc = proc->u.irep->code;
|
||||
pic_gc_arena_restore(pic, ai);
|
||||
JUMP;
|
||||
|
|
Loading…
Reference in New Issue