gambas-source-code/main/gbc/gbc_trans_code.c
Benoît Minisini af950247c4 Implement overflow detection in JIT compiler. Some jump optimizations in the compiler and the interpreter.
[COMPILER]
* OPT: Optimize jumps pointing at a JUMP instruction.

[INTERPRETER]
* OPT: Avoid conversion to boolean when possible in conditional jumps.

[GB.JIT]
* NEW: Implement overflow detection in integer arithmetic operations. The UNSAFE keyword disable them to get full speed.
2022-12-09 15:00:55 +01:00

781 lines
15 KiB
C

/***************************************************************************
gbc_trans_code.c
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
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_CODE_C
#include <stdlib.h>
#include <string.h>
#include <stdio.h>
#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
//ushort *TRANS_labels = NULL;
static FUNCTION *_func;
static CLASS_SYMBOL *add_local(int sym_index, TYPE type, int value, bool used)
{
CLASS_SYMBOL *sym;
PARAM *loc;
bool warnings = JOB->warnings;
if (used)
JOB->warnings = FALSE;
sym = CLASS_declare(JOB->class, sym_index, TK_VARIABLE, FALSE);
if (used)
JOB->warnings = warnings;
if (TYPE_is_static(type))
{
CLASS_add_static_declaration(JOB->class, sym_index, type, sym, TRUE);
loc = ARRAY_add(&_func->stat);
}
else
{
if (value > MAX_LOCAL_SYMBOL)
THROW("Too many local variables");
sym->local.value = value;
sym->local_used = used;
loc = ARRAY_add(&_func->local);
}
loc->index = sym_index;
loc->type = type;
loc->value = sym->local.value;
sym->local.type = type;
CLASS_check_variable_prefix(sym, TRUE);
return sym;
}
static void create_local_from_param()
{
int i;
JOB->line--; // For line number of declaration
for (i = 0; i < _func->nparam; i++)
{
if (TYPE_get_id(_func->param[i].type) != T_VOID)
add_local(_func->param[i].index, _func->param[i].type, (i - _func->nparam), _func->param[i].ignore);
}
JOB->line++;
}
static void check_local(CLASS_SYMBOL *sym)
{
if (!sym->local_used)
{
if (sym->local.value < 0)
COMPILE_print(MSG_WARNING, sym->local.line, "unused argument: &1", SYMBOL_get_name(&sym->symbol));
else
COMPILE_print(MSG_WARNING, sym->local.line, "unused variable: &1", SYMBOL_get_name(&sym->symbol));
}
else if (!sym->local_assigned)
{
if (sym->local.value >= 0)
COMPILE_print(MSG_WARNING, sym->local.line, "uninitialized variable: &1", SYMBOL_get_name(&sym->symbol));
}
TYPE_clear(&sym->local.type);
}
static void remove_local()
{
int i;
for (i = 0; i < ARRAY_count(_func->local); i++)
check_local(CLASS_get_symbol(JOB->class, _func->local[i].index));
for (i = 0; i < ARRAY_count(_func->stat); i++)
check_local(CLASS_get_symbol(JOB->class, _func->stat[i].index));
}
static bool TRANS_local(void)
{
int sym_index;
TRANS_DECL decl;
PATTERN *pattern;
CLASS_SYMBOL *sym;
int f;
bool no_warning;
bool save_warnings;
bool is_static;
if (TRANS_is(RS_DIM))
is_static = FALSE;
else if (TRANS_is(RS_STATIC))
is_static = TRUE;
else
return FALSE;
for(;;)
{
//many = FALSE;
pattern = JOB->current;
for(;;)
{
no_warning = TRANS_is(RS_LBRA);
if (!PATTERN_is_identifier(*JOB->current))
THROW(E_SYNTAX);
JOB->current++;
if (no_warning)
{
if (!PATTERN_is(*JOB->current, RS_RBRA))
THROW("Missing right brace");
JOB->current++;
}
if (!TRANS_is(RS_COMMA))
break;
//many = TRUE;
}
f = TT_DO_NOT_CHECK_AS | TT_CAN_ARRAY | TT_CAN_NEW;
if (is_static)
f |= TT_CAN_EMBED;
if (!TRANS_type(f, &decl))
THROW(E_SYNTAX);
if (is_static)
TYPE_set_flag(&decl.type, TF_STATIC);
for(;;)
{
if (PATTERN_is(*pattern, RS_LBRA))
{
pattern++;
no_warning = TRUE;
save_warnings = JOB->warnings;
JOB->warnings = FALSE;
}
else
no_warning = FALSE;
sym_index = PATTERN_index(*pattern);
sym = add_local(sym_index, decl.type, _func->nlocal, FALSE);
pattern++;
if (no_warning)
{
pattern++;
JOB->warnings = save_warnings;
}
if (is_static)
{
CLASS_init_global_declaration(JOB->class, &decl, sym, TRUE);
if (COMP_verbose)
printf("STATIC %s AS %s\n", TABLE_get_symbol_name(JOB->class->table, sym_index), TYPE_get_desc(decl.type));
}
else
{
_func->nlocal++;
if (TRANS_init_var(&decl))
{
CODE_pop_local(sym->local.value);
sym->local_assigned = TRUE;
}
if (COMP_verbose)
printf("DIM %s AS %s\n", TABLE_get_symbol_name(JOB->class->table, sym_index), TYPE_get_desc(decl.type));
}
if (!PATTERN_is(*pattern, RS_COMMA))
break;
pattern++;
}
/*if (!many)
{
if (TRANS_init_var(&decl))
CODE_pop_local(_func->nlocal - 1);
}*/
if (!TRANS_is(RS_COMMA))
break;
}
return TRUE;
}
int TRANS_loop_local(bool allow_arg)
{
int sym_index;
TRANS_DECL decl;
CLASS_SYMBOL *sym;
if (!PATTERN_is_identifier(*JOB->current))
THROW("Syntax error. Identifier expected");
sym_index = PATTERN_index(*JOB->current);
JOB->current++;
sym = CLASS_get_symbol(JOB->class, sym_index);
if (TRANS_type(TT_DO_NOT_CHECK_AS | TT_CAN_ARRAY, &decl))
{
if (!TYPE_compare(&sym->local.type, &decl.type))
{
add_local(sym_index, decl.type, _func->nlocal, FALSE);
_func->nlocal++;
}
}
if (TYPE_is_null(sym->local.type) || TYPE_is_static(sym->local.type))
{
if (TYPE_is_null(sym->global.type))
THROW("Unknown identifier: &1", TABLE_get_symbol_name(JOB->class->table, sym_index));
else
THROW("Loop variable cannot be global");
}
if (!allow_arg && sym->local.value < 0)
THROW("Loop variable cannot be an argument");
sym->local_assigned = TRUE;
sym->local_used = TRUE;
return sym->local.value;
}
void TRANS_stop(void)
{
if (TRANS_is(RS_EVENT))
CODE_stop_event();
else
CODE_stop();
}
void TRANS_statement(void)
{
static TRANS_STATEMENT statement[] = {
{ RS_EXIT, TRANS_break },
{ RS_BREAK, TRANS_break },
{ RS_CONTINUE, TRANS_continue },
{ RS_GOTO, TRANS_goto },
{ RS_RETURN, TRANS_return },
{ RS_PRINT, TRANS_print },
{ RS_INPUT, TRANS_input },
{ RS_WRITE, TRANS_write },
{ RS_READ, TRANS_read_old },
//{ RS_OPEN, TRANS_open },
{ RS_CLOSE, TRANS_close },
{ RS_SEEK, TRANS_seek },
{ RS_FLUSH, TRANS_flush },
{ RS_STOP, TRANS_stop },
{ RS_QUIT, TRANS_quit },
{ RS_EXEC, TRANS_exec },
{ RS_SHELL, TRANS_shell },
{ RS_WAIT, TRANS_wait },
{ RS_SLEEP, TRANS_sleep },
{ RS_KILL, TRANS_kill },
{ RS_MOVE, TRANS_move },
{ RS_INC, TRANS_inc },
{ RS_DEC, TRANS_dec },
{ RS_SWAP, TRANS_swap },
{ RS_MKDIR, TRANS_mkdir },
{ RS_RMDIR, TRANS_rmdir },
{ RS_COPY, TRANS_copy },
{ RS_RAISE, TRANS_raise },
{ RS_LINK, TRANS_link },
{ RS_LOCK, TRANS_lock },
{ RS_UNLOCK, TRANS_unlock },
{ RS_TRY, TRANS_try },
{ RS_LINE, TRANS_line_input },
{ RS_OUTPUT, TRANS_output_to },
{ RS_DEBUG, TRANS_debug },
{ RS_ASSERT, TRANS_assert },
{ RS_ERROR, TRANS_error },
{ RS_RANDOMIZE, TRANS_randomize },
{ RS_CHMOD, TRANS_chmod },
{ RS_CHOWN, TRANS_chown },
{ RS_CHGRP, TRANS_chgrp },
{ RS_GOSUB, TRANS_gosub },
{ RS_ON, TRANS_on_goto_gosub },
{ RS_NONE, NULL }
};
PATTERN *look = JOB->current;
TRANS_STATEMENT *st;
COMP_INFO *info;
int index;
if (PATTERN_is_reserved(look[0]))
{
info = &COMP_res_info[PATTERN_index(*look)];
if (!info->func)
{
for (st = statement; st->id; st++)
{
if (PATTERN_is(look[0], st->id))
{
info->func = st->func;
break;
}
}
if (!info->func)
info->func = TS_NO_SUBR;
}
if (info->func && info->func != TS_NO_SUBR)
{
JOB->current++;
(*info->func)();
return;
}
}
else if (PATTERN_is_subr(look[0]))
{
index = PATTERN_index(look[0]);
if (index == SUBR_Mid || index == SUBR_MidS)
{
JOB->current++;
TRANS_mid();
return;
}
else if (COMP_version >= 0x03180000 && COMP_subr_info[index].opcode == (CODE_POKE - CODE_FIRST_SUBR))
{
TRANS_poke();
return;
}
}
if (!TRANS_affectation(FALSE))
TRANS_expression(TRUE);
}
static void translate_body()
{
PATTERN *look;
bool is_proc = (TYPE_get_id(_func->type) == T_VOID);
bool test_newline;
//int line = JOB->line - 1;
bool just_got_select = FALSE;
for(;;)
{
test_newline = TRUE;
CODE_allow_break();
FUNCTION_add_all_pos_line();
look = JOB->current;
if (PATTERN_is(look[0], RS_END))
if (TRANS_is_end_function(is_proc, &look[1]))
break;
if (TRANS_newline())
test_newline = FALSE;
else if (!TRANS_local())
break;
if (test_newline)
if (!PATTERN_is_newline(*JOB->current))
THROW_UNEXPECTED(JOB->current);
}
TRANS_control_init();
for(;;)
{
test_newline = TRUE;
CODE_allow_break();
FUNCTION_add_all_pos_line();
look = JOB->current;
if (PATTERN_is(look[0], RS_END))
if (TRANS_is_end_function(is_proc, &look[1]))
break;
if (TRANS_newline())
continue;
if (just_got_select)
{
if (!PATTERN_is(look[0], RS_CASE) && !PATTERN_is(look[0], RS_DEFAULT))
THROW("Syntax error. CASE or DEFAULT expected after SELECT");
just_got_select = FALSE;
}
if (PATTERN_is(look[0], RS_DIM) || PATTERN_is(look[0], RS_STATIC))
{
TRANS_local();
}
else if (PATTERN_is_identifier(look[0]) && PATTERN_is(look[1], RS_COLON))
{
TRANS_label();
}
else if (PATTERN_is(look[0], RS_IF))
{
JOB->current++;
TRANS_if();
}
else if (PATTERN_is(look[0], RS_ELSE))
{
JOB->current++;
TRANS_else();
}
else if ((PATTERN_is(look[0], RS_END)
&& PATTERN_is(look[1], RS_IF))
|| PATTERN_is(look[0], RS_ENDIF))
{
if (PATTERN_is(look[0], RS_END))
JOB->current += 2;
else
JOB->current++;
TRANS_endif();
}
else if (PATTERN_is(look[0], RS_DO))
{
JOB->current++;
TRANS_do(RS_DO);
}
else if (PATTERN_is(look[0], RS_WHILE))
{
TRANS_do(RS_WHILE);
}
else if (PATTERN_is(*look, RS_REPEAT))
{
JOB->current++;
TRANS_do(RS_REPEAT);
}
else if (PATTERN_is(look[0], RS_LOOP))
{
JOB->current++;
TRANS_loop(RS_LOOP);
}
else if (PATTERN_is(look[0], RS_UNTIL))
{
TRANS_loop(RS_UNTIL);
}
else if (PATTERN_is(look[0], RS_WEND))
{
JOB->current++;
TRANS_loop(RS_WEND);
}
else if (PATTERN_is(look[0], RS_FOR))
{
if (PATTERN_is(look[1], RS_EACH))
{
JOB->current += 2;
TRANS_for_each();
}
else
{
JOB->current++;
TRANS_for();
}
}
else if (PATTERN_is(look[0], RS_NEXT))
{
JOB->current++;
TRANS_next();
}
else if (PATTERN_is(look[0], RS_SELECT))
{
JOB->current++;
TRANS_select();
just_got_select = TRUE;
}
else if (PATTERN_is(look[0], RS_CASE))
{
JOB->current++;
if (PATTERN_is(look[1], RS_ELSE))
{
JOB->current++;
TRANS_default();
}
else
TRANS_case();
}
else if (PATTERN_is(look[0], RS_DEFAULT))
{
JOB->current++;
TRANS_default();
}
else if (PATTERN_is(look[0], RS_END)
&& PATTERN_is(look[1], RS_SELECT))
{
JOB->current += 2;
TRANS_end_select();
}
else if (PATTERN_is(look[0], RS_FINALLY))
{
JOB->current++;
TRANS_finally();
}
else if (PATTERN_is(look[0], RS_CATCH))
{
JOB->current++;
TRANS_catch();
}
else if (PATTERN_is(*look, RS_WITH))
{
JOB->current++;
TRANS_with();
}
else if (PATTERN_is(look[0], RS_END)
&& PATTERN_is(look[1], RS_WITH))
{
JOB->current += 2;
TRANS_end_with();
}
else if (PATTERN_is(look[0], RS_LET))
{
JOB->current++;
if (!TRANS_affectation(FALSE))
THROW(E_SYNTAX);
}
else
TRANS_statement();
/*
if (next_newline)
{
for(;;)
{
if (PATTERN_is_NEWLINE(*JOB->current)
|| PATTERN_is_END(*JOB->current))
break;
JOB->current++;
}
}
*/
if (test_newline)
if (!PATTERN_is_newline(*JOB->current))
THROW_UNEXPECTED(JOB->current);
}
TRANS_control_exit();
}
static void trans_call(const char *name, int nparam)
{
CLASS_SYMBOL *sym;
int index;
if (!TABLE_find_symbol(JOB->class->table, name, strlen(name), &index))
return;
sym = (CLASS_SYMBOL *)TABLE_get_symbol(JOB->class->table, index);
if (TYPE_get_kind(sym->global.type) != TK_FUNCTION)
return;
sym->global_used = TRUE;
CODE_push_global(sym->global.value, FALSE, TRUE);
CODE_call(nparam);
CODE_drop();
}
void TRANS_code(void)
{
int i, j, n;
// We must compile initialization functions at the end, because static local variables can modify them.
n = ARRAY_count(JOB->class->function);
for (j = 0; j < n; j++)
{
i = (j + FUNC_INIT_MAX + 1) % n;
_func = &JOB->class->function[i];
CODE_begin_function(_func);
if (COMP_verbose)
{
printf("Compiling %s()...\n", TABLE_get_symbol_name(JOB->class->table, _func->name));
if (_func->fast)
printf("Fast\n");
}
/* Do not debug implicit or generated functions */
if (!_func->start || _func->name == NO_SYMBOL || TABLE_get_symbol_name(JOB->class->table, _func->name)[0] == '@')
JOB->nobreak = TRUE;
else
JOB->nobreak = FALSE;
JOB->line = _func->line;
JOB->current = _func->start;
JOB->func = _func;
/* fonction implicite ? */
if (!_func->start)
{
if ((i == FUNC_INIT_DYNAMIC) && (JOB->form != NULL))
{
//CODE_event(FALSE);
trans_call("@load", 0);
//CODE_event(TRUE);
}
FUNCTION_add_last_pos_line();
CODE_op(C_RETURN, 0, 0, TRUE);
if (COMP_verbose)
CODE_dump(_func->code, _func->ncode);
continue;
}
create_local_from_param();
translate_body();
CODE_return(2); // Return from function, ignore Gosub stack
CODE_end_function(_func);
FUNCTION_add_last_pos_line();
_func->stack = _func->nlocal + _func->nctrl + CODE_stack_usage;
if (COMP_verbose)
{
CODE_dump(_func->code, _func->ncode);
printf("%d local(s) %d control(s) ", _func->nlocal, _func->nctrl);
printf("%d stack\n", _func->stack);
printf("\n");
}
remove_local();
}
CLASS_check_properties(JOB->class);
CLASS_check_unused_global(JOB->class);
JOB->func = NULL;
}
bool TRANS_init_var(TRANS_DECL *decl)
{
int i;
TRANS_ARRAY *array;
//PATTERN *save;
if (decl->is_new)
{
if (TYPE_is_array(decl->type) && decl->array.ndim > 0)
{
array = &decl->array;
if (TYPE_is_object(decl->type))
CODE_push_class(TYPE_get_value(decl->type));
else
CODE_push_number(TYPE_get_id(decl->type));
for (i = 0; i < array->ndim; i++)
CODE_push_number(array->dim[i]);
CODE_new(array->ndim + 1, TRUE, FALSE);
}
else
{
JOB->current = decl->init;
TRANS_new();
}
return TRUE;
}
else if (decl->init)
{
JOB->current = decl->init;
TRANS_expression(FALSE);
return TRUE;
}
else
return FALSE;
}
/*
void TRANS_init_object()
{
}
*/
void TRANS_init_optional(TRANS_PARAM *param)
{
PATTERN *look = param->optional;
PATTERN *save;
if (look == NULL)
return;
save = JOB->current;
if (PATTERN_is(*look, RS_COMMA) || PATTERN_is(*look, RS_RBRA))
{
CODE_push_void();
}
else
{
if (!PATTERN_is(*look, RS_EQUAL))
THROW("Syntax error. Invalid optional parameter");
look++;
JOB->current = look;
TRANS_expression(FALSE);
if (!PATTERN_is(*JOB->current, RS_COMMA) && !PATTERN_is(*JOB->current, RS_RBRA))
THROW("Syntax error. Invalid optional parameter");
}
JOB->current = save;
}