/*************************************************************************** gbc_trans_subr.c (c) 2000-2017 BenoƮt Minisini This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ***************************************************************************/ #define _TRANS_SUBR_C #include #include #include #include "gb_common.h" #include "gb_error.h" #include "gbc_compile.h" #include "gbc_trans.h" #include "gb_code.h" #include "gb_limit.h" /*#define DEBUG*/ typedef struct { char *name; SUBR_INFO *info; } TRANS_SUBR_INFO; SUBR_INFO *TRANS_find_subr(int index) { SUBR_INFO *info = &COMP_subr_info[index]; if (COMP_version >= 0x03190000) return info; if (info->opcode == 24 && info->optype == 28) index = SUBR_find("#Pi"); else if (info->opcode == 25) THROW("The '&1()' subroutine only exists since version 3.19", info->name); else if (info->opcode == 57) { if (info->optype == 1) index = SUBR_find("#.Flush"); else if (info->optype == 2) index = SUBR_find("#.InputFrom"); else if (info->optype == 3) index = SUBR_find("#.OutputTo"); else if (info->optype == 4) index = SUBR_find("#.ErrorFrom"); } return &COMP_subr_info[index]; } void TRANS_subr(int subr, int nparam) { static TRANS_SUBR_INFO subr_info[] = { { ".Print" }, { ".Input" }, { ".Write" }, { ".WriteBytes" }, { ".Read" }, { ".ReadBytes" }, { ".Open" }, { ".Close" }, { "Seek" }, { ".LineInput" }, { ".Flush" }, { ".Exec" }, { ".Shell" }, { ".Wait" }, { ".Kill" }, { ".Move" }, { ".Mkdir" }, { ".Rmdir" }, { ".Array" }, {".Collection" }, { ".Copy" }, { ".Link" }, { ".Error" }, { ".Lock" }, { ".Unlock" }, { ".LockWait" }, { ".InputFrom" }, { ".OutputTo" }, { ".Debug" }, { ".Sleep" }, { ".Randomize" }, { ".ErrorTo" }, { "Left" }, { "Mid" }, { ".OpenMemory" }, { ".Chmod" }, { ".Chown" }, { ".Chgrp" }, { ".Use" }, { ".CheckExec" }, { ".MoveKill" }, { ".WaitDelay" }, { ".WaitNext" }, { ".Peek" } }; TRANS_SUBR_INFO *tsi = &subr_info[subr]; if (tsi->info == NULL) { tsi->info = TRANS_find_subr(SUBR_find(tsi->name)); if (!tsi->info) ERROR_panic("Unknown intern subroutine: %s", tsi->name); } if (subr == TS_SUBR_ARRAY && nparam > MAX_PARAM_OP) CODE_subr(tsi->info->opcode, MAX_PARAM_OP + 1, CODE_CALL_VARIANT + MAX_PARAM_OP, FALSE); else if (subr == TS_SUBR_COLLECTION && nparam > MAX_PARAM_OP) CODE_subr(tsi->info->opcode, MAX_PARAM_OP, CODE_CALL_VARIANT + MAX_PARAM_OP - 1, FALSE); else CODE_subr(tsi->info->opcode, nparam, tsi->info->optype, tsi->info->min_param == tsi->info->max_param); } static bool trans_stream_check(int default_stream, bool check) { if (TRANS_is(RS_SHARP) || default_stream == TS_NONE) { TRANS_expression(FALSE); if (check) { if (PATTERN_is(*JOB->current, RS_COMMA)) { JOB->current++; if (PATTERN_is_newline(*JOB->current)) THROW(E_SYNTAX); } else { if (!PATTERN_is_newline(*JOB->current)) THROW(E_SYNTAX); } } return FALSE; } else { //if (default_stream == TS_NONE) // THROW("Syntax error. &1 expected", "'#'"); CODE_push_number(default_stream); return TRUE; } } #define trans_stream(_default_stream) trans_stream_check(_default_stream, TRUE) #define trans_stream_no_check(_default_stream) trans_stream_check(_default_stream, FALSE) static void trans_print_debug() { int nparam = 1; bool semicolon_or_comma = FALSE; for(;;) { if (PATTERN_is_newline(*JOB->current)) break; TRANS_expression(FALSE); nparam++; semicolon_or_comma = FALSE; if (PATTERN_is_newline(*JOB->current)) break; if (TRANS_is(RS_SCOLON)) { if (TRANS_is(RS_SCOLON)) { CODE_push_char(' '); nparam++; } semicolon_or_comma = TRUE; } else if (TRANS_is(RS_COMMA)) { CODE_push_char('\t'); nparam++; semicolon_or_comma = TRUE; } else THROW(E_SYNTAX); } if (!semicolon_or_comma) { CODE_push_char('\n'); nparam++; } TRANS_subr(TS_SUBR_PRINT, nparam); CODE_drop(); } void TRANS_print(void) { trans_stream(TS_STDOUT); trans_print_debug(); } void TRANS_debug(void) { if (!JOB->debug) CODE_disable(); TRANS_subr(TS_SUBR_DEBUG, 0); trans_print_debug(); if (!JOB->debug) CODE_enable(); } void TRANS_error(void) { if (TRANS_is(RS_TO)) { if (TRANS_is(RS_DEFAULT)) CODE_push_null(); else trans_stream(TS_NONE); TRANS_subr(TS_SUBR_ERROR_TO, 1); if (!TRANS_in_assignment) CODE_drop(); } else { CODE_push_number(2); // stderr trans_print_debug(); } } static int trans_binary_type(void) { int index; int nparam = 0; if (PATTERN_is_class(*JOB->current)) { index = CLASS_add_class(JOB->class, PATTERN_index(*JOB->current)); if (PATTERN_is(JOB->current[1], RS_LSQR)) index = CLASS_get_array_class(JOB->class, T_OBJECT, index); CODE_push_class(index); } else if (PATTERN_is_type(*JOB->current)) { if (PATTERN_is(JOB->current[1], RS_LSQR)) { index = CLASS_get_array_class(JOB->class, RES_get_type(PATTERN_index(*JOB->current)), -1); CODE_push_class(index); } else { index = RES_get_type(PATTERN_index(*JOB->current)); CODE_push_number(index); } } else THROW(E_SYNTAX); JOB->current++; nparam++; #if 0 if (TRANS_is(RS_LSQR)) { TRANS_expression(FALSE); nparam++; TRANS_want(RS_RSQR, NULL); } else if (TRANS_is(RS_STAR)) { if (!string) THROW("Syntax error"); TRANS_expression(FALSE); nparam++; } #endif return nparam; } void TRANS_write(void) { trans_stream(TS_STDOUT); TRANS_expression(FALSE); if (TRANS_is(RS_AS)) { trans_binary_type(); TRANS_subr(TS_SUBR_WRITE, 3); } else { if (TRANS_is(RS_COMMA)) TRANS_expression(FALSE); else { if (JOB->no_old_read_syntax) THROW("Syntax error. &1 expected", "AS"); CODE_push_number(-1); } TRANS_subr(TS_SUBR_WRITE_BYTES, 3); } CODE_drop(); } void TRANS_output_to() { TRANS_want(RS_TO, NULL); if (TRANS_is(RS_DEFAULT)) CODE_push_null(); else trans_stream(TS_NONE); TRANS_subr(TS_SUBR_OUTPUT_TO, 1); if (!TRANS_in_assignment) CODE_drop(); } void TRANS_input_from() { if (TRANS_is(RS_DEFAULT)) CODE_push_null(); else trans_stream(TS_NONE); TRANS_subr(TS_SUBR_INPUT_FROM, 1); if (!TRANS_in_assignment) CODE_drop(); } void TRANS_input(void) { bool stream = TRUE; if (TRANS_is(RS_FROM)) { TRANS_input_from(); return; } trans_stream(TS_STDIN); for(;;) { TRANS_subr(TS_SUBR_INPUT, (stream ? 1 : 0)); stream = FALSE; TRANS_reference(); if (PATTERN_is_newline(*JOB->current)) break; if (!PATTERN_is(*JOB->current, RS_COMMA) && !PATTERN_is(*JOB->current, RS_SCOLON)) THROW(E_SYNTAX); JOB->current++; } } void TRANS_read_old(void) { PATTERN *save_var; PATTERN *save_current; TYPE type; if (JOB->no_old_read_syntax) THROW(E_UNEXPECTED, "READ"); trans_stream(TS_STDIN); save_var = JOB->current; type = TRANS_variable_get_type(); if (TRANS_is(RS_COMMA)) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_READ_BYTES, 2); } else { int id = TYPE_get_id(type); CODE_push_number(id); TRANS_subr(TS_SUBR_READ, 2); } save_current = JOB->current; JOB->current = save_var; TRANS_reference(); JOB->current = save_current; } void TRANS_read(void) { bool def = trans_stream_no_check(TS_STDIN); if (TRANS_is(RS_AS)) { trans_binary_type(); TRANS_subr(TS_SUBR_READ, 2); } else { if (!def) TRANS_want(RS_COMMA, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_READ_BYTES, 2); } } void TRANS_peek(void) { bool def = trans_stream_no_check(TS_STDIN); if (!def) TRANS_want(RS_COMMA, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_PEEK, 2); } static void TRANS_open_null(void) { int mode = TS_MODE_READ; // file name CODE_push_null(); // open mode if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_MODE_READ | TS_MODE_DIRECT; if (TRANS_is(RS_WRITE)) mode |= TS_MODE_WRITE | TS_MODE_DIRECT; } CODE_push_number(mode | TS_MODE_NULL); TRANS_subr(TS_SUBR_OPEN, 2); } static void TRANS_open_string(void) { int mode = TS_MODE_READ | TS_MODE_DIRECT; // file name if (!PATTERN_is(*JOB->current, RS_FOR) && !PATTERN_is_newline(*JOB->current)) TRANS_expression(FALSE); else CODE_push_null(); // open mode if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_MODE_READ | TS_MODE_DIRECT; if (TRANS_is(RS_WRITE)) mode |= TS_MODE_WRITE | TS_MODE_DIRECT; } CODE_push_number(mode | TS_MODE_STRING); TRANS_subr(TS_SUBR_OPEN, 2); } void TRANS_open(void) { int mode = TS_MODE_READ; if (TRANS_is(RS_PIPE)) { TRANS_pipe(); return; } else if (TRANS_is(RS_MEMORY)) { TRANS_memory(); return; } else if (TRANS_is(RS_STRING)) { TRANS_open_string(); return; } else if (TRANS_is(RS_NULL)) { TRANS_open_null(); return; } // file name TRANS_expression(FALSE); // open mode if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_MODE_READ | TS_MODE_DIRECT; else if (TRANS_is(RS_INPUT)) mode |= TS_MODE_READ; if (TRANS_is(RS_WRITE)) mode |= TS_MODE_WRITE | TS_MODE_DIRECT; else if (TRANS_is(RS_OUTPUT)) mode |= TS_MODE_WRITE; if (TRANS_is(RS_CREATE)) mode |= TS_MODE_CREATE; else if (TRANS_is(RS_APPEND)) mode |= TS_MODE_APPEND; if (TRANS_is(RS_WATCH)) mode |= TS_MODE_WATCH; /*if (TRANS_is(RS_BIG)) mode |= TS_MODE_BIG; else if (TRANS_is(RS_LITTLE)) mode |= TS_MODE_LITTLE;*/ /*JOB->current--; if (PATTERN_is(*JOB->current, RS_FOR)) THROW("Syntax error in file open mode"); JOB->current++;*/ } CODE_push_number(mode); TRANS_subr(TS_SUBR_OPEN, 2); } void TRANS_pipe(void) { int mode = TS_MODE_READ; // file name TRANS_expression(FALSE); // open mode if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_MODE_READ | TS_MODE_DIRECT; //else if (TRANS_is(RS_INPUT)) // mode |= TS_MODE_READ; if (TRANS_is(RS_WRITE)) mode |= TS_MODE_WRITE | TS_MODE_DIRECT; //else if (TRANS_is(RS_OUTPUT)) // mode |= TS_MODE_WRITE; if (TRANS_is(RS_WATCH)) mode |= TS_MODE_WATCH; } CODE_push_number(mode | TS_MODE_PIPE); TRANS_subr(TS_SUBR_OPEN, 2); } void TRANS_memory(void) { int mode = TS_MODE_READ; /* Memory address */ TRANS_expression(FALSE); /* Open mode */ if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_MODE_READ | TS_MODE_DIRECT; if (TRANS_is(RS_WRITE)) mode |= TS_MODE_WRITE | TS_MODE_DIRECT; } CODE_push_number(mode); TRANS_subr(TS_SUBR_OPEN_MEMORY, 2); } void TRANS_close(void) { if (PATTERN_is_newline(*JOB->current)) THROW(E_SYNTAX); TRANS_ignore(RS_SHARP); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_CLOSE, 1); if (!TRANS_in_assignment) CODE_drop(); } void TRANS_lock(void) { if (PATTERN_is_newline(*JOB->current)) THROW(E_SYNTAX); if (!TRANS_in_assignment) THROW("Useless LOCK"); TRANS_expression(FALSE); if (TRANS_is(RS_WAIT)) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_LOCK_WAIT, 2); } else TRANS_subr(TS_SUBR_LOCK, 1); } void TRANS_unlock(void) { if (PATTERN_is_newline(*JOB->current)) THROW(E_SYNTAX); TRANS_ignore(RS_SHARP); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_UNLOCK, 1); CODE_drop(); } void TRANS_seek(void) { int nparam; trans_stream(TS_NONE); TRANS_expression(FALSE); nparam = 2; /* if (TRANS_is(RS_COMMA)) { TRANS_expression(FALSE); nparam++; } */ TRANS_subr(TS_SUBR_SEEK, nparam); CODE_drop(); } void TRANS_line_input(void) { if (TRANS_is(RS_INPUT)) { trans_stream(TS_STDIN); TRANS_subr(TS_SUBR_LINE_INPUT, 1); TRANS_reference(); } else THROW(E_SYNTAX); } void TRANS_flush(void) { trans_stream(TS_STDOUT); TRANS_subr(TS_SUBR_FLUSH, 1); CODE_drop(); } void TRANS_quit(void) { if (PATTERN_is_newline(*JOB->current)) { CODE_quit(FALSE); } else { TRANS_expression(FALSE); CODE_quit(TRUE); } } void TRANS_randomize(void) { if (PATTERN_is_newline(*JOB->current)) { TRANS_subr(TS_SUBR_RANDOMIZE, 0); } else { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_RANDOMIZE, 1); } CODE_drop(); } static void trans_exec_shell(bool shell) { int mode = TS_EXEC_NONE; bool wait; bool as = TRUE; PATTERN *dest = NULL; PATTERN *save; TRANS_expression(FALSE); if (TRANS_is(RS_WITH)) TRANS_expression(FALSE); else CODE_push_null(); wait = TRANS_is(RS_WAIT); if (TRANS_is(RS_FOR)) { if (TRANS_is(RS_READ)) mode |= TS_EXEC_READ; if (TRANS_is(RS_WRITE)) mode |= TS_EXEC_WRITE; if (mode == 0) { mode |= TS_EXEC_TERM; if (TRANS_is(RS_INPUT)) mode |= TS_EXEC_READ; if (TRANS_is(RS_OUTPUT)) mode |= TS_EXEC_WRITE; } } else { if (TRANS_is(RS_TO)) { if (TRANS_in_assignment) THROW("Syntax error. Cannot use this syntax in assignment"); mode = TS_EXEC_STRING; wait = TRUE; as = FALSE; dest = JOB->current; TRANS_ignore_expression(); if (TRANS_is(RS_WITH) && TRANS_is(RS_ERROR)) mode += TS_EXEC_ERROR; } } if (wait) mode |= TS_EXEC_WAIT; CODE_push_number(mode); if (as && TRANS_is(RS_AS)) TRANS_expression(FALSE); else CODE_push_null(); TRANS_subr(shell ? TS_SUBR_SHELL : TS_SUBR_EXEC, 4); if (dest) { save = JOB->current; JOB->current = dest; TRANS_reference(); JOB->current = save; } else if (!TRANS_in_assignment) CODE_drop(); } void TRANS_exec(void) { trans_exec_shell(FALSE); } void TRANS_shell(void) { trans_exec_shell(TRUE); } void TRANS_wait(void) { if (TRANS_is(RS_NEXT)) TRANS_subr(TS_SUBR_WAIT_NEXT, 0); else if (!PATTERN_is_newline(*JOB->current)) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_WAIT_DELAY, 1); } else TRANS_subr(TS_SUBR_WAIT, 0); CODE_drop(); } void TRANS_sleep(void) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_SLEEP, 1); CODE_drop(); } void TRANS_kill(void) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_KILL, 1); CODE_drop(); } void TRANS_move(void) { int subr; TRANS_expression(FALSE); if (TRANS_is(RS_TO)) subr = TS_SUBR_MOVE; else if (TRANS_is(RS_DOWNTO) || TRANS_is(RS_KILL)) subr = TS_SUBR_MOVE_KILL; else THROW_UNEXPECTED(JOB->current); TRANS_expression(FALSE); TRANS_subr(subr, 2); CODE_drop(); } void TRANS_copy(void) { TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_COPY, 2); CODE_drop(); } void TRANS_link(void) { TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_LINK, 2); CODE_drop(); } void TRANS_chmod(void) { TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_CHMOD, 2); CODE_drop(); } void TRANS_chown(void) { TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_CHOWN, 2); CODE_drop(); } void TRANS_chgrp(void) { TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); TRANS_expression(FALSE); TRANS_subr(TS_SUBR_CHGRP, 2); CODE_drop(); } void TRANS_inc(void) { PATTERN *save = JOB->current; TRANS_expression(FALSE); CODE_push_number(1); CODE_op(C_ADD, 0, 2, TRUE); JOB->current = save; TRANS_reference(); } void TRANS_dec(void) { PATTERN *save = JOB->current; TRANS_expression(FALSE); CODE_push_number(1); CODE_op(C_SUB, 0, 2, TRUE); JOB->current = save; TRANS_reference(); } void TRANS_swap(void) { PATTERN *sa; PATTERN *sb; PATTERN *current; sa = JOB->current; TRANS_expression(FALSE); TRANS_want(RS_COMMA, "Comma"); sb = JOB->current; TRANS_expression(FALSE); current = JOB->current; JOB->current = sa; TRANS_reference(); JOB->current = sb; TRANS_reference(); JOB->current = current; } void TRANS_mkdir(void) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_MKDIR, 1); CODE_drop(); } void TRANS_rmdir(void) { TRANS_expression(FALSE); TRANS_subr(TS_SUBR_RMDIR, 1); CODE_drop(); } void TRANS_mid(void) { PATTERN *str; PATTERN *pos; PATTERN *len; PATTERN *save; TRANS_want(RS_LBRA, "Left bracket"); str = JOB->current; TRANS_expression(FALSE); TRANS_want(RS_COMMA, "Comma"); pos = JOB->current; TRANS_expression(FALSE); CODE_push_number(1); CODE_op(C_SUB, 0, 2, TRUE); TRANS_subr(TS_SUBR_LEFT, 2); if (TRANS_is(RS_COMMA)) { len = JOB->current; TRANS_ignore_expression(); } else { len = NULL; } TRANS_want(RS_RBRA, "Right bracket"); TRANS_want(RS_EQUAL, "Equal"); TRANS_expression(FALSE); save = JOB->current; if (len) { JOB->current = str; TRANS_expression(FALSE); JOB->current = pos; TRANS_expression(FALSE); JOB->current = len; TRANS_expression(FALSE); CODE_op(C_ADD, 0, 2, TRUE); TRANS_subr(TS_SUBR_MID, 2); } CODE_op(C_CAT, 0, len ? 3 : 2, FALSE); JOB->current = str; TRANS_reference(); JOB->current = save; } void TRANS_poke(void) { int index; index = PATTERN_index(*JOB->current); JOB->current++; TRANS_want(RS_LBRA, "Left bracket"); TRANS_expression(FALSE); TRANS_want(RS_RBRA, "Right bracket"); TRANS_want(RS_EQUAL, "Equal"); TRANS_expression(FALSE); CODE_op(C_POKE, COMP_subr_info[index].optype, 2, TRUE); CODE_drop(); } #if 0 void TRANS_scan(void) { PATTERN *save; int noutput = 0; TRANS_expression(FALSE); TRANS_want(RS_WITH, NULL); TRANS_expression(FALSE); TRANS_want(RS_TO, NULL); save = JOB->current; for(;;) { TRANS_expression(FALSE); noutput++; if (!TRANS_is(RS_COMMA)) break; } JOB->current = save; trans_subr_output(TSO_SUBR_SCAN, 2 + noutput, noutput); for(;;) { TRANS_reference(); if (!TRANS_is(RS_COMMA)) break; } } #endif