libcubescript/src/cs_vm.hh

60 lines
1.5 KiB
C++

#ifndef LIBCUBESCRIPT_VM_HH
#define LIBCUBESCRIPT_VM_HH
#include <cubescript/cubescript.hh>
#include "cs_std.hh"
#include "cs_ident.hh"
#include "cs_gen.hh"
#include "cs_thread.hh"
namespace cubescript {
struct break_exception {
};
struct continue_exception {
};
template<typename F>
static void call_with_args(thread_state &ts, F body) {
if (!ts.callstack) {
body();
return;
}
valarray<ident_stack, MAX_ARGUMENTS> argstack{*ts.pstate};
int argmask1 = ts.callstack->usedargs;
for (int i = 0; argmask1; argmask1 >>= 1, ++i) {
if (argmask1 & 1) {
static_cast<alias_impl *>(ts.istate->identmap[i])->undo_arg(
argstack[i]
);
}
}
ident_link *prevstack = ts.callstack->next;
ident_link aliaslink = {
ts.callstack->id, ts.callstack,
prevstack ? prevstack->usedargs : ((1 << MAX_ARGUMENTS) - 1),
prevstack ? prevstack->argstack : nullptr
};
ts.callstack = &aliaslink;
call_with_cleanup(std::move(body), [&]() {
if (prevstack) {
prevstack->usedargs = aliaslink.usedargs;
}
ts.callstack = aliaslink.next;
int argmask2 = ts.callstack->usedargs;
for (int i = 0; argmask2; argmask2 >>= 1, ++i) {
if (argmask2 & 1) {
static_cast<alias_impl *>(ts.istate->identmap[i])->redo_arg(
argstack[i]
);
}
}
});
}
} /* namespace cubescript */
#endif /* LIBCUBESCRIPT_VM_HH */