int var_frame_idx = 0;
bc->nsubfunctions = 0;
errors += expand_call_arglist(locations, &b);
- if (bc->parent) {
- // functions should end in a return
- b = BLOCK(b, gen_op_simple(RET));
- } else {
- // the toplevel should YIELD;BACKTRACK; when it finds an answer
- b = BLOCK(b, gen_op_simple(YIELD), gen_op_simple(BACKTRACK));
- }
+ b = BLOCK(b, gen_op_simple(RET));
for (inst* curr = b.first; curr; curr = curr->next) {
if (!curr->next) assert(curr == b.last);
int length = opcode_describe(curr->op)->length;
jv path;
int subexp_nest;
int debug_trace_enabled;
+ int initial_execution;
};
typedef struct {
uint16_t* pc = stack_restore(jq);
assert(pc);
- int backtracking = 0;
+ int backtracking = !jq->initial_execution;
+ jq->initial_execution = 0;
while (1) {
uint16_t opcode = *pc;
pc += offset;
break;
}
-
- case YIELD: {
- jv value = stack_pop(jq);
- stack_save(jq, pc);
- stack_switch(jq);
- return value;
- }
case CALL_BUILTIN: {
int nargs = *pc++;
}
case RET: {
- pc = *frame_current_retaddr(&jq->frame_stk);
- frame_pop(&jq->frame_stk);
+ uint16_t* retaddr = *frame_current_retaddr(&jq->frame_stk);
+ if (retaddr) {
+ // function return
+ pc = retaddr;
+ frame_pop(&jq->frame_stk);
+ } else {
+ // top-level return, yielding value
+ jv value = stack_pop(jq);
+ stack_save(jq, pc - 1);
+ stack_push(jq, jv_null());
+ stack_switch(jq);
+ return value;
+ }
break;
}
+ case ON_BACKTRACK(RET): {
+ // resumed after top-level return
+ goto do_backtrack;
+ }
}
}
}
} else {
new_jq->debug_trace_enabled = 0;
}
+ new_jq->initial_execution = 1;
*jq = new_jq;
}
OP(LOADV, VARIABLE, 1, 1)
OP(STOREV, VARIABLE, 1, 0)
OP(INDEX, NONE, 2, 1)
-OP(YIELD, NONE, 1, 0)
OP(EACH, NONE, 1, 1)
OP(FORK, BRANCH, 0, 0)
OP(JUMP, BRANCH, 0, 0)