libcubescript/src/cs_vm.hh

100 lines
2.5 KiB
C++
Raw Normal View History

2021-03-23 23:29:32 +01:00
#ifndef LIBCUBESCRIPT_VM_HH
#define LIBCUBESCRIPT_VM_HH
2016-08-12 18:38:43 +02:00
#include <cubescript/cubescript.hh>
2016-08-12 18:38:43 +02:00
2021-03-23 01:11:21 +01:00
#include "cs_std.hh"
#include "cs_ident.hh"
2021-03-24 02:21:27 +01:00
#include "cs_gen.hh"
#include "cs_thread.hh"
2021-03-23 23:32:25 +01:00
namespace cubescript {
2016-08-12 18:38:43 +02:00
2021-03-24 02:21:27 +01:00
struct break_exception {
2016-09-14 23:24:13 +02:00
};
2021-03-24 02:21:27 +01:00
struct continue_exception {
2016-08-12 18:38:43 +02:00
};
struct run_depth_guard {
run_depth_guard() = delete;
run_depth_guard(thread_state &ts);
run_depth_guard(run_depth_guard const &) = delete;
run_depth_guard(run_depth_guard &&) = delete;
~run_depth_guard();
};
struct stack_guard {
thread_state *tsp;
std::size_t oldtop;
stack_guard() = delete;
stack_guard(thread_state &ts):
tsp{&ts}, oldtop{ts.vmstack.size()}
{}
~stack_guard() {
tsp->vmstack.resize(oldtop, any_value{*tsp->pstate});
}
stack_guard(stack_guard const &) = delete;
stack_guard(stack_guard &&) = delete;
};
template<typename F>
static void call_with_args(thread_state &ts, F body) {
if (!ts.callstack) {
2016-09-10 19:54:55 +02:00
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;
2021-03-23 23:29:32 +01:00
ident_link aliaslink = {
ts.callstack->id, ts.callstack,
2021-03-24 02:21:27 +01:00
prevstack ? prevstack->usedargs : ((1 << MAX_ARGUMENTS) - 1),
2016-09-15 04:30:37 +02:00
prevstack ? prevstack->argstack : nullptr
};
ts.callstack = &aliaslink;
2021-03-23 01:11:21 +01:00
call_with_cleanup(std::move(body), [&]() {
2016-09-15 04:30:37 +02:00
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]
);
}
}
});
}
void exec_command(
thread_state &ts, command_impl *id, any_value *args, any_value &res,
std::size_t nargs, bool lookup = false
);
void exec_alias(
thread_state &ts, alias *a, any_value *args, any_value &result,
std::size_t callargs, std::size_t &nargs,
std::size_t offset, std::size_t skip, std::uint32_t op
);
std::uint32_t *vm_exec(
thread_state &ts, std::uint32_t *code, any_value &result
);
2021-03-23 23:32:25 +01:00
} /* namespace cubescript */
2016-08-12 18:38:43 +02:00
2021-03-23 23:29:32 +01:00
#endif /* LIBCUBESCRIPT_VM_HH */