#ifndef LIBCUBESCRIPT_GEN_HH #define LIBCUBESCRIPT_GEN_HH #include "cubescript/cubescript.hh" #include #include #include #include #include "cs_std.hh" #include "cs_bcode.hh" #include "cs_ident.hh" #include "cs_thread.hh" namespace cubescript { static constexpr int MAX_ARGUMENTS = 25; static constexpr int MAX_RESULTS = 7; static constexpr int ID_IDX_DUMMY = MAX_ARGUMENTS; static constexpr int ID_IDX_NUMARGS = MAX_ARGUMENTS + 1; static constexpr int ID_IDX_DBGALIAS = MAX_ARGUMENTS + 2; struct codegen_state { state &cs; codegen_state *prevps; bool parsing = true; valbuf code; char const *source, *send; std::size_t current_line; std::string_view src_name; codegen_state() = delete; codegen_state(state &csr): cs{csr}, prevps{csr.p_tstate->cstate}, code{csr.p_tstate->istate}, source{}, send{}, current_line{1}, src_name{} { csr.p_tstate->cstate = this; } ~codegen_state() { done(); } void done() { if (!parsing) { return; } cs.p_tstate->cstate = prevps; parsing = false; } std::string_view get_str(); charbuf get_str_dup(); std::string_view get_word(); void gen_str(std::string_view word) { if (word.size() <= 3) { std::uint32_t op = BC_INST_VAL_INT | BC_RET_STRING; for (size_t i = 0; i < word.size(); ++i) { op |= std::uint32_t( static_cast(word[i]) ) << ((i + 1) * 8); } code.push_back(op); return; } code.push_back(BC_INST_VAL | BC_RET_STRING | (word.size() << 8)); auto it = reinterpret_cast(word.data()); code.append(it, it + (word.size() / sizeof(std::uint32_t))); std::size_t esz = word.size() % sizeof(std::uint32_t); char c[sizeof(std::uint32_t)] = {0}; std::memcpy(c, word.data() + word.size() - esz, esz); std::uint32_t u; std::memcpy(&u, c, sizeof(u)); code.push_back(u); } void gen_str() { code.push_back(BC_INST_VAL_INT | BC_RET_STRING); } void gen_null() { code.push_back(BC_INST_VAL_INT | BC_RET_NULL); } void gen_int(integer_type i = 0) { if (i >= -0x800000 && i <= 0x7FFFFF) { code.push_back(BC_INST_VAL_INT | BC_RET_INT | (i << 8)); } else { std::uint32_t u[bc_store_size] = {0}; std::memcpy(u, &i, sizeof(i)); code.push_back(BC_INST_VAL | BC_RET_INT); code.append(u, u + bc_store_size); } } void gen_int(std::string_view word); void gen_float(float_type f = 0.0f) { if (integer_type(f) == f && f >= -0x800000 && f <= 0x7FFFFF) { code.push_back(BC_INST_VAL_INT | BC_RET_FLOAT | (integer_type(f) << 8)); } else { std::uint32_t u[bc_store_size] = {0}; std::memcpy(u, &f, sizeof(f)); code.push_back(BC_INST_VAL | BC_RET_FLOAT); code.append(u, u + bc_store_size); } } void gen_float(std::string_view word); void gen_ident(ident *id) { code.push_back( ((id->get_index() < MAX_ARGUMENTS) ? BC_INST_IDENT_ARG : BC_INST_IDENT ) | (id->get_index() << 8) ); } void gen_ident() { gen_ident(cs.p_tstate->istate->identmap[ID_IDX_DUMMY]); } void gen_ident(std::string_view word) { gen_ident(cs.new_ident(word)); } void gen_value( int wordtype, std::string_view word = std::string_view(), int line = 0 ); void gen_main(std::string_view s, int ret_type = VAL_ANY); void next_char() { if (source == send) { return; } if (*source == '\n') { ++current_line; } ++source; } char current(size_t ahead = 0) { if (std::size_t(send - source) <= ahead) { return '\0'; } return source[ahead]; } std::string_view read_macro_name(); char skip_until(std::string_view chars); char skip_until(char cf); void skip_comments(); }; } /* namespace cubescript */ #endif /* LIBCUBESCRIPT_GEN_HH */