2007-12-30 17:41:49 +01:00
|
|
|
/***************************************************************************
|
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
gbc_trans_ctrl.c
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-09-12 15:13:13 +02:00
|
|
|
(c) 2000-2017 Benoît Minisini <benoit.minisini@gambas-basic.org>
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
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.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
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.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-12-31 03:39:20 +01:00
|
|
|
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.
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
***************************************************************************/
|
|
|
|
|
|
|
|
#define _TRANS_CTRL_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*/
|
|
|
|
/*#define DEBUG_GOTO*/
|
|
|
|
|
2009-04-08 12:11:16 +02:00
|
|
|
#define BEGIN_NO_BREAK \
|
|
|
|
{ \
|
|
|
|
bool nobreak = JOB->nobreak; \
|
|
|
|
JOB->nobreak = TRUE;
|
|
|
|
|
|
|
|
#define END_NO_BREAK \
|
|
|
|
JOB->nobreak = nobreak; \
|
|
|
|
}
|
|
|
|
|
2020-01-08 22:15:45 +01:00
|
|
|
#if defined(__GNUC__) && __GNUC__ == 8
|
2020-01-09 18:31:23 +01:00
|
|
|
static volatile int ctrl_level; // Workaround a gcc compiler optimization bug
|
2020-01-08 22:15:45 +01:00
|
|
|
#else
|
2007-12-30 17:41:49 +01:00
|
|
|
static int ctrl_level;
|
2020-01-08 22:15:45 +01:00
|
|
|
#endif
|
2007-12-30 17:41:49 +01:00
|
|
|
static int ctrl_id;
|
|
|
|
static int ctrl_local;
|
|
|
|
|
|
|
|
static TRANS_CTRL ctrl_data[MAX_CTRL_LEVEL];
|
|
|
|
|
|
|
|
static TRANS_CTRL *current_ctrl;
|
|
|
|
|
|
|
|
static TRANS_GOTO *goto_info;
|
|
|
|
static TRANS_LABEL *label_info;
|
|
|
|
static short *ctrl_parent;
|
|
|
|
|
2022-12-09 15:00:55 +01:00
|
|
|
static ushort *_relocation = NULL;
|
2018-12-12 01:40:44 +01:00
|
|
|
|
2022-12-09 15:00:55 +01:00
|
|
|
typedef
|
|
|
|
struct {
|
|
|
|
ushort src;
|
|
|
|
ushort dst;
|
|
|
|
}
|
|
|
|
JUMP;
|
|
|
|
|
|
|
|
static JUMP *_jumps = NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
static void jump_length(ushort src, ushort dst, bool can_optimize)
|
2018-05-24 00:31:58 +02:00
|
|
|
{
|
2022-12-09 15:00:55 +01:00
|
|
|
JUMP *jump;
|
|
|
|
|
2018-05-24 00:31:58 +02:00
|
|
|
CODE_jump_length(src, dst);
|
2022-12-09 15:00:55 +01:00
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
if (can_optimize)
|
|
|
|
{
|
|
|
|
jump = ARRAY_add(&_jumps);
|
|
|
|
jump->src = src;
|
|
|
|
jump->dst = dst;
|
|
|
|
}
|
2018-05-24 00:31:58 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
static void control_set_value(int value)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level <= 0)
|
|
|
|
return;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl->value = value;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
static int control_get_value(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level <= 0)
|
|
|
|
return 0;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
return current_ctrl->value;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
static void control_add_relocation()
|
|
|
|
{
|
|
|
|
if (!_relocation)
|
|
|
|
ARRAY_create(&_relocation);
|
|
|
|
|
|
|
|
*((ushort *)ARRAY_add(&_relocation)) = CODE_get_current_pos();
|
|
|
|
}
|
|
|
|
|
2013-04-29 23:27:11 +02:00
|
|
|
static void control_add_pos(ushort **tab_pos, ushort pos)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!(*tab_pos))
|
|
|
|
ARRAY_create(tab_pos);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2013-04-29 23:27:11 +02:00
|
|
|
*((ushort *)ARRAY_add(tab_pos)) = pos;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
static void control_add_current_pos(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_pos(¤t_ctrl->pos, CODE_get_current_pos());
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2018-05-25 21:01:38 +02:00
|
|
|
static void control_add_current_pos_break()
|
|
|
|
{
|
|
|
|
control_add_pos(¤t_ctrl->pos_break, CODE_get_current_pos());
|
|
|
|
}
|
|
|
|
|
|
|
|
static void control_add_current_pos_continue()
|
|
|
|
{
|
|
|
|
control_add_pos(¤t_ctrl->pos_continue, CODE_get_current_pos());
|
|
|
|
}
|
|
|
|
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
static void control_add_this_pos(ushort pos)
|
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_pos(¤t_ctrl->pos, pos);
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
}
|
|
|
|
|
2018-05-25 21:01:38 +02:00
|
|
|
static void control_add_this_pos_continue(ushort pos)
|
|
|
|
{
|
|
|
|
control_add_pos(¤t_ctrl->pos_continue, pos);
|
|
|
|
}
|
|
|
|
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2013-04-29 23:27:11 +02:00
|
|
|
static void control_jump_each_pos_with(ushort *tab_pos)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
int i;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!tab_pos)
|
|
|
|
return;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
for (i = 0; i < ARRAY_count(tab_pos); i++)
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(tab_pos[i], CODE_get_current_pos(), TRUE);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
static void control_jump_each_pos(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos_with(current_ctrl->pos);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static TRANS_CTRL *control_get_inner(void)
|
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
int level = ctrl_level;
|
|
|
|
TRANS_CTRL *ctrl_look;
|
|
|
|
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
if (!level)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
level--;
|
|
|
|
ctrl_look = &ctrl_data[level];
|
|
|
|
|
|
|
|
if ((ctrl_look->type == RS_DO)
|
|
|
|
|| (ctrl_look->type == RS_WHILE)
|
|
|
|
|| (ctrl_look->type == RS_REPEAT)
|
|
|
|
|| (ctrl_look->type == RS_FOR)
|
|
|
|
|| (ctrl_look->type == RS_EACH))
|
|
|
|
return ctrl_look;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static TRANS_CTRL *control_get_inner_with(void)
|
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
int level = ctrl_level;
|
|
|
|
TRANS_CTRL *ctrl_look;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
if (!level)
|
|
|
|
return NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
level--;
|
|
|
|
ctrl_look = &ctrl_data[level];
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_look->type == RS_WITH)
|
|
|
|
return ctrl_look;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
static void add_goto(int index, int mode)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_GOTO *info;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (goto_info == NULL)
|
|
|
|
ARRAY_create(&goto_info);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
info = ARRAY_add(&goto_info);
|
|
|
|
info->index = index;
|
|
|
|
info->pos = CODE_get_current_pos();
|
|
|
|
info->ctrl_id = (ctrl_level == 0) ? 0 : current_ctrl->id;
|
|
|
|
info->line = JOB->line;
|
2012-05-21 03:09:35 +02:00
|
|
|
info->gosub = mode == RS_GOSUB;
|
|
|
|
info->on_goto = mode == RS_NONE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
#ifdef DEBUG_GOTO
|
|
|
|
printf("add_goto: ctrl_id = %d (%ld)\n", info->ctrl_id, ARRAY_count(goto_info));
|
|
|
|
#endif
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
if (mode == RS_GOSUB)
|
2018-12-12 01:40:44 +01:00
|
|
|
{
|
2018-12-12 16:27:02 +01:00
|
|
|
//control_add_relocation();
|
2012-05-23 21:26:15 +02:00
|
|
|
CODE_gosub(ctrl_local);
|
2018-12-12 01:40:44 +01:00
|
|
|
}
|
2012-05-21 03:09:35 +02:00
|
|
|
else if (mode == RS_GOTO)
|
2012-03-04 00:57:36 +01:00
|
|
|
CODE_jump();
|
2012-05-21 03:09:35 +02:00
|
|
|
else
|
|
|
|
CODE_nop();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
static void control_enter(int type)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
short *parent;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level >= MAX_CTRL_LEVEL)
|
|
|
|
THROW("Too many nested control structures.");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ctrl_id++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl = &ctrl_data[ctrl_level];
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl->type = type;
|
|
|
|
current_ctrl->value = 0;
|
|
|
|
current_ctrl->pos = NULL;
|
|
|
|
current_ctrl->state = 0;
|
|
|
|
current_ctrl->local = ctrl_local;
|
|
|
|
current_ctrl->id = ctrl_id;
|
|
|
|
current_ctrl->loop_var = -1;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
parent = ARRAY_add(&ctrl_parent);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level == 0)
|
|
|
|
*parent = 0;
|
|
|
|
else
|
|
|
|
*parent = ctrl_data[ctrl_level - 1].id;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case RS_FOR:
|
|
|
|
case RS_EACH:
|
|
|
|
ctrl_local += 2;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case RS_SELECT:
|
|
|
|
case RS_WITH:
|
|
|
|
ctrl_local += 1;
|
|
|
|
break;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
JOB->func->nctrl = Max(JOB->func->nctrl, ctrl_local);
|
2019-03-13 14:25:32 +01:00
|
|
|
if ((JOB->func->nctrl + JOB->func->nlocal) > MAX_LOCAL_SYMBOL)
|
|
|
|
THROW("Too many local variables");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ctrl_level++;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
static void control_leave(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos_with(current_ctrl->pos_break);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ARRAY_delete(¤t_ctrl->pos);
|
|
|
|
ARRAY_delete(¤t_ctrl->pos_break);
|
|
|
|
ARRAY_delete(¤t_ctrl->pos_continue);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ctrl_local = current_ctrl->local;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ctrl_level--;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level > 0)
|
|
|
|
current_ctrl = &ctrl_data[ctrl_level - 1];
|
|
|
|
else
|
|
|
|
current_ctrl = NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
static void control_check(int type, const char *msg1, const char *msg2)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level <= 0)
|
|
|
|
THROW(msg1);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (current_ctrl->type != type)
|
|
|
|
THROW(E_UNEXPECTED, msg2);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
static void control_check_two(int type1, int type2, const char *msg1, const char *msg2)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_level <= 0)
|
|
|
|
THROW(msg1);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (current_ctrl->type != type1 && current_ctrl->type != type2)
|
|
|
|
THROW(E_UNEXPECTED, msg2);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void control_check_loop_var(short var)
|
|
|
|
{
|
|
|
|
int i;
|
|
|
|
|
|
|
|
for (i = 0; i < (ctrl_level - 1); i++)
|
|
|
|
{
|
|
|
|
if (ctrl_data[i].loop_var == var)
|
|
|
|
THROW("Loop variable already in use");
|
|
|
|
}
|
|
|
|
|
|
|
|
current_ctrl->loop_var = var;
|
|
|
|
}
|
|
|
|
|
2012-03-02 23:05:55 +01:00
|
|
|
static void check_try(const char *name)
|
|
|
|
{
|
|
|
|
if (TRANS_in_try)
|
|
|
|
{
|
2019-03-15 19:36:22 +01:00
|
|
|
const char *keyword = TRANS_in_try == RS_TRY ? "TRY" : "ASSERT";
|
|
|
|
TRANS_in_try = RS_NONE;
|
|
|
|
if (strcmp(name, keyword))
|
|
|
|
THROW("Cannot use &1 with &2", keyword, name);
|
2012-03-02 23:05:55 +01:00
|
|
|
else
|
2019-03-15 19:36:22 +01:00
|
|
|
THROW("Cannot use &1 twice", keyword);
|
2012-03-02 23:05:55 +01:00
|
|
|
}
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_control_init(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ctrl_level = 0;
|
|
|
|
ctrl_id = 0;
|
|
|
|
current_ctrl = NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
goto_info = NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
label_info = NULL;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
ctrl_local = 0; //JOB->func->nlocal;
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->func->nctrl = 0;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
ARRAY_create(&ctrl_parent);
|
2022-12-09 15:00:55 +01:00
|
|
|
ARRAY_create(&_jumps);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_control_exit(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2022-12-09 15:00:55 +01:00
|
|
|
int i, j;
|
2010-08-28 00:02:51 +02:00
|
|
|
CLASS_SYMBOL *sym;
|
|
|
|
int line;
|
|
|
|
TRANS_LABEL *label;
|
|
|
|
short id;
|
2018-12-12 16:27:02 +01:00
|
|
|
ushort *pcode;
|
2022-12-09 15:00:55 +01:00
|
|
|
JUMP *jump;
|
2010-08-28 00:02:51 +02:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
// Relocate locals
|
|
|
|
|
|
|
|
if (_relocation)
|
|
|
|
{
|
|
|
|
for (i = 0; i < ARRAY_count(_relocation); i++)
|
2018-12-12 16:27:02 +01:00
|
|
|
{
|
|
|
|
pcode = &JOB->func->code[_relocation[i]];
|
|
|
|
if (PCODE_is_breakpoint(*pcode))
|
|
|
|
pcode++;
|
|
|
|
|
|
|
|
*pcode += JOB->func->nlocal;
|
|
|
|
|
|
|
|
/*fprintf(stderr, "%04d : %04X --> ", _relocation[i], pcode);
|
|
|
|
pcode = (pcode & 0xFF00) | ((pcode & 0xFF) + JOB->func->nlocal);
|
|
|
|
fprintf(stderr, "%04X\n", pcode);
|
|
|
|
JOB->func->code[_relocation[i]] = pcode;*/
|
|
|
|
}
|
2018-12-12 01:40:44 +01:00
|
|
|
|
|
|
|
ARRAY_delete(&_relocation);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Resolve GOTOs
|
2010-08-28 00:02:51 +02:00
|
|
|
|
|
|
|
if (goto_info)
|
|
|
|
{
|
|
|
|
line = JOB->line;
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_count(goto_info); i++)
|
|
|
|
{
|
|
|
|
JOB->line = goto_info[i].line;
|
|
|
|
/*printf("%d\n", JOB->line);*/
|
|
|
|
|
|
|
|
sym = CLASS_get_symbol(JOB->class, goto_info[i].index);
|
|
|
|
|
|
|
|
if (TYPE_get_kind(sym->local.type) != TK_LABEL)
|
|
|
|
THROW("Label '&1' not declared", TABLE_get_symbol_name(JOB->class->table, goto_info[i].index));
|
|
|
|
|
|
|
|
label = &label_info[sym->local.value];
|
|
|
|
|
|
|
|
id = goto_info[i].ctrl_id;
|
2022-12-10 19:39:17 +01:00
|
|
|
|
2012-03-04 10:13:59 +01:00
|
|
|
if (goto_info[i].gosub)
|
2010-08-28 00:02:51 +02:00
|
|
|
{
|
2012-03-05 00:43:07 +01:00
|
|
|
if (label->ctrl_id)
|
2012-03-04 10:13:59 +01:00
|
|
|
THROW("Forbidden GOSUB");
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
if (label->ctrl_id == id)
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (id == 0)
|
|
|
|
THROW("Forbidden GOTO");
|
|
|
|
|
|
|
|
#ifdef DEBUG_GOTO
|
|
|
|
printf("id = %d ctrl_parent[id - 1] = %d (%ld)\n", id, ctrl_parent[id - 1], ARRAY_count(ctrl_parent));
|
|
|
|
#endif
|
|
|
|
id = ctrl_parent[id - 1];
|
|
|
|
}
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2022-12-10 19:39:17 +01:00
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(goto_info[i].pos, label->pos, !goto_info[i].on_goto);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
JOB->line = line;
|
|
|
|
}
|
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
// Optimize jumps
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_count(_jumps); i++)
|
|
|
|
{
|
|
|
|
jump = &_jumps[i];
|
|
|
|
|
|
|
|
for(j = 1; j <= 4; j++) // avoid infinite loop
|
|
|
|
{
|
|
|
|
if (jump->dst >= JOB->func->ncode)
|
|
|
|
break;
|
|
|
|
pcode = &JOB->func->code[jump->dst];
|
|
|
|
|
|
|
|
if (PCODE_is_breakpoint(*pcode))
|
|
|
|
{
|
|
|
|
pcode++;
|
|
|
|
jump->dst++;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!PCODE_is_jump(*pcode))
|
|
|
|
break;
|
|
|
|
|
|
|
|
jump->dst += ((short *)pcode)[1] + 2;
|
|
|
|
CODE_jump_length(jump->src, jump->dst);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-09 15:00:55 +01:00
|
|
|
// Remove previously declared labels
|
2010-08-28 00:02:51 +02:00
|
|
|
|
|
|
|
if (label_info)
|
|
|
|
{
|
|
|
|
for (i = 0; i < ARRAY_count(label_info); i++)
|
|
|
|
{
|
|
|
|
sym = CLASS_get_symbol(JOB->class, label_info[i].index);
|
|
|
|
TYPE_clear(&sym->local.type);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
ARRAY_delete(&goto_info);
|
|
|
|
ARRAY_delete(&ctrl_parent);
|
|
|
|
ARRAY_delete(&label_info);
|
2022-12-09 15:00:55 +01:00
|
|
|
ARRAY_delete(&_jumps);
|
2010-08-28 00:02:51 +02:00
|
|
|
|
2022-12-09 15:00:55 +01:00
|
|
|
// Detect structures still opened
|
2010-08-28 00:02:51 +02:00
|
|
|
|
|
|
|
if (ctrl_level == 0) return;
|
|
|
|
|
|
|
|
switch (ctrl_data[ctrl_level - 1].type)
|
|
|
|
{
|
|
|
|
case RS_IF:
|
|
|
|
THROW(E_MISSING, "ENDIF");
|
|
|
|
case RS_FOR:
|
|
|
|
case RS_EACH:
|
|
|
|
THROW(E_MISSING, "NEXT");
|
|
|
|
case RS_DO:
|
|
|
|
THROW(E_MISSING, "LOOP");
|
|
|
|
case RS_REPEAT:
|
|
|
|
THROW(E_MISSING, "UNTIL");
|
|
|
|
case RS_WHILE:
|
|
|
|
THROW(E_MISSING, "WEND");
|
|
|
|
case RS_SELECT:
|
|
|
|
THROW(E_MISSING, "END SELECT");
|
|
|
|
case RS_WITH:
|
|
|
|
THROW(E_MISSING, "END WITH");
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
static ushort trans_jump_if(bool if_true, bool fast)
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
{
|
|
|
|
ushort pos;
|
|
|
|
|
|
|
|
if (CODE_check_jump_not())
|
2022-12-10 12:54:42 +01:00
|
|
|
{
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
if_true = !if_true;
|
2022-12-10 12:54:42 +01:00
|
|
|
fast = FALSE;
|
|
|
|
}
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
|
|
|
|
pos = CODE_get_current_pos();
|
2022-12-10 12:54:42 +01:00
|
|
|
|
|
|
|
CODE_jump_if(if_true, fast);
|
|
|
|
|
[DEVELOPMENT ENVIRONMENT]
* BUG: Use TextEdit.RichText insted of TextEdit.Text.
* BUG: END SUB can be the end of a method. The class analyze now takes
that into account.
[HELP]
* BUG: Fixed the generated treeview.
[COMPILER]
* OPT: The NOT operator used just at the beginning of a conditional
expression is optimized. Consequently, an expression like 'IF NOT 2' is
now equivalent to 'IF 2 = 0' and not to 'IF (NOT 2) <> 0' as before. In
other words, the boolean conversion is now done before the NOT, and not
after. The following instructions are concerned: IF, WHILE, UNTIL.
* NEW: BYREF is new keyword that is a more readable synonymous of '@'.
[GB.DB.FORM]
* BUG: Correctly manage data controls inside TabStrip-like containers.
* BUG: Setting the focus on a non-initialized DataControl does not raise
an error anymore.
[GB.GTK]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Window arrangement is done before the Open event is raised, as in
gb.qt.
* BUG: Keyboard, focus and mouse events now work correctly on Window and
DrawingArea controls.
[GB.QT]
* BUG: HSplitter.Layout and VSplitter.Layout now work correctly. It is a
list of children widths, hidden children having a zero width.
* BUG: Many warning fixes.
* BUG: Now the Control.Visible property works like in gb.gtk, i.e. it
returns if the control was not explicitely hidden.
git-svn-id: svn://localhost/gambas/trunk@1060 867c0c6c-44f3-4631-809d-bfa615b0a4ec
2008-02-06 01:25:48 +01:00
|
|
|
return pos;
|
|
|
|
}
|
|
|
|
|
2018-05-25 21:01:38 +02:00
|
|
|
/*static void trans_endif(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (current_ctrl->state == 0)
|
2018-05-24 00:31:58 +02:00
|
|
|
jump_length(control_get_value(), CODE_get_current_pos());
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos();
|
2018-05-25 21:01:38 +02:00
|
|
|
}*/
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
static void trans_else(void)
|
|
|
|
{
|
2009-04-08 12:11:16 +02:00
|
|
|
BEGIN_NO_BREAK
|
|
|
|
{
|
2018-05-25 21:01:38 +02:00
|
|
|
control_add_current_pos_break();
|
2009-04-08 12:11:16 +02:00
|
|
|
CODE_jump();
|
|
|
|
}
|
|
|
|
END_NO_BREAK
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-05-25 21:01:38 +02:00
|
|
|
control_jump_each_pos_with(current_ctrl->pos_continue);
|
|
|
|
ARRAY_delete(¤t_ctrl->pos_continue);
|
|
|
|
//jump_length(control_get_value(), CODE_get_current_pos());
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl->state = 1;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
static bool trans_boolean_expr(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_expression(FALSE);
|
2022-12-10 12:54:42 +01:00
|
|
|
return TYPE_is_boolean(TRANS_get_last_type());
|
|
|
|
}
|
|
|
|
|
|
|
|
static void trans_if(void)
|
|
|
|
{
|
|
|
|
bool fast;
|
|
|
|
|
|
|
|
fast = trans_boolean_expr();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is(*JOB->current, RS_AND))
|
|
|
|
{
|
2018-05-25 21:01:38 +02:00
|
|
|
for(;;)
|
|
|
|
{
|
2022-12-10 12:54:42 +01:00
|
|
|
control_add_this_pos_continue(trans_jump_if(FALSE, fast));
|
2018-05-25 21:01:38 +02:00
|
|
|
|
|
|
|
if (!PATTERN_is(*JOB->current, RS_AND))
|
|
|
|
break;
|
|
|
|
|
|
|
|
JOB->current++;
|
|
|
|
|
|
|
|
TRANS_want(RS_IF, "AND IF");
|
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
fast = trans_boolean_expr();
|
2018-05-25 21:01:38 +02:00
|
|
|
}
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
|
|
|
else if (PATTERN_is(*JOB->current, RS_OR))
|
|
|
|
{
|
|
|
|
for(;;)
|
|
|
|
{
|
2022-12-10 12:54:42 +01:00
|
|
|
control_add_this_pos(trans_jump_if(TRUE, fast));
|
2018-05-25 21:01:38 +02:00
|
|
|
|
|
|
|
if (!PATTERN_is(*JOB->current, RS_OR))
|
2010-08-28 00:02:51 +02:00
|
|
|
break;
|
2018-05-25 21:01:38 +02:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->current++;
|
2018-05-25 21:01:38 +02:00
|
|
|
|
|
|
|
TRANS_want(RS_IF, "OR IF");
|
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
fast = trans_boolean_expr();
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2018-05-25 21:01:38 +02:00
|
|
|
|
|
|
|
control_add_current_pos_continue();
|
|
|
|
CODE_jump();
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2018-05-25 21:01:38 +02:00
|
|
|
else
|
|
|
|
{
|
2022-12-10 12:54:42 +01:00
|
|
|
control_add_this_pos_continue(trans_jump_if(FALSE, fast));
|
2018-05-25 21:01:38 +02:00
|
|
|
}
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is(*JOB->current, RS_THEN))
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
else if (!PATTERN_is_newline(*JOB->current))
|
2010-08-28 00:02:51 +02:00
|
|
|
THROW(E_EXPECTED, "THEN");
|
2018-05-25 21:01:38 +02:00
|
|
|
|
|
|
|
control_jump_each_pos();
|
|
|
|
ARRAY_delete(¤t_ctrl->pos);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
static void trans_else_if(void)
|
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
trans_else();
|
|
|
|
current_ctrl->state = 0;
|
|
|
|
trans_if();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_if(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2012-10-14 00:43:36 +02:00
|
|
|
PATTERN *look;
|
|
|
|
bool has_else;
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(RS_IF);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
trans_if();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is_newline(*JOB->current))
|
|
|
|
return;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-10-14 00:43:36 +02:00
|
|
|
has_else = FALSE;
|
|
|
|
look = JOB->current;
|
|
|
|
for(;;)
|
|
|
|
{
|
|
|
|
look++;
|
|
|
|
if (PATTERN_is_newline(*look))
|
|
|
|
break;
|
|
|
|
if (PATTERN_is(*look, RS_ELSE))
|
|
|
|
{
|
|
|
|
has_else = TRUE;
|
|
|
|
*look = PATTERN_make(RT_NEWLINE, 0);
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_statement();
|
2012-10-14 00:43:36 +02:00
|
|
|
|
|
|
|
if (has_else)
|
|
|
|
{
|
|
|
|
JOB->current++;
|
|
|
|
trans_else();
|
|
|
|
TRANS_statement();
|
|
|
|
}
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_endif();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_else(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_IF, "ELSE without IF", "ELSE");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (current_ctrl->state)
|
|
|
|
THROW(E_UNEXPECTED, "ELSE");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (TRANS_is(RS_IF))
|
|
|
|
trans_else_if();
|
|
|
|
else
|
|
|
|
trans_else();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_endif(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_IF, "ENDIF without IF", "ENDIF");
|
2018-05-25 21:01:38 +02:00
|
|
|
control_jump_each_pos_with(current_ctrl->pos_continue);
|
2010-08-28 00:02:51 +02:00
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_goto(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
int index;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-03-02 23:05:55 +01:00
|
|
|
check_try("GOTO");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!PATTERN_is_identifier(*JOB->current))
|
|
|
|
THROW(E_SYNTAX);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
index = PATTERN_index(*JOB->current);
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
add_goto(index, RS_GOTO);
|
2012-03-04 00:57:36 +01:00
|
|
|
}
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_gosub(void)
|
2012-03-04 00:57:36 +01:00
|
|
|
{
|
|
|
|
int index;
|
|
|
|
|
|
|
|
check_try("GOSUB");
|
|
|
|
|
|
|
|
if (!PATTERN_is_identifier(*JOB->current))
|
|
|
|
THROW(E_SYNTAX);
|
|
|
|
|
|
|
|
index = PATTERN_index(*JOB->current);
|
|
|
|
JOB->current++;
|
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
add_goto(index, RS_GOSUB);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2018-12-13 13:15:33 +01:00
|
|
|
|
2012-05-21 03:09:35 +02:00
|
|
|
void TRANS_on_goto_gosub(void)
|
|
|
|
{
|
|
|
|
bool gosub;
|
|
|
|
int pos, n;
|
|
|
|
int index;
|
|
|
|
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
|
|
|
|
if (TRANS_is(RS_GOTO))
|
|
|
|
gosub = FALSE;
|
|
|
|
else if (TRANS_is(RS_GOSUB))
|
|
|
|
gosub = TRUE;
|
|
|
|
else
|
|
|
|
THROW(E_SYNTAX);
|
|
|
|
|
|
|
|
check_try(gosub ? "ON GOSUB" : "ON GOTO");
|
|
|
|
|
|
|
|
pos = CODE_get_current_pos();
|
|
|
|
CODE_nop();
|
|
|
|
|
|
|
|
for (n = 1;; n++)
|
|
|
|
{
|
|
|
|
if (!PATTERN_is_identifier(*JOB->current))
|
|
|
|
THROW(E_SYNTAX);
|
|
|
|
|
|
|
|
index = PATTERN_index(*JOB->current);
|
|
|
|
JOB->current++;
|
|
|
|
|
|
|
|
add_goto(index, RS_NONE);
|
|
|
|
|
|
|
|
if (!TRANS_is(RS_COMMA))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (n == 127)
|
|
|
|
THROW("Too many labels");
|
2021-02-25 23:01:04 +01:00
|
|
|
|
|
|
|
TRANS_newline();
|
2012-05-21 03:09:35 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
pos = CODE_set_current_pos(pos);
|
|
|
|
CODE_on(n);
|
|
|
|
CODE_set_current_pos(pos);
|
|
|
|
|
|
|
|
if (gosub)
|
2018-12-12 01:40:44 +01:00
|
|
|
{
|
2018-12-12 16:27:02 +01:00
|
|
|
//control_add_relocation();
|
2012-05-23 21:26:15 +02:00
|
|
|
CODE_gosub(ctrl_local);
|
2018-12-12 01:40:44 +01:00
|
|
|
}
|
2012-05-21 03:09:35 +02:00
|
|
|
else
|
|
|
|
CODE_jump();
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-13 13:15:33 +01:00
|
|
|
|
2010-05-22 20:02:34 +02:00
|
|
|
void TRANS_do(int type)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
bool is_until;
|
2022-12-10 12:54:42 +01:00
|
|
|
bool fast;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(type);
|
|
|
|
control_set_value(CODE_get_current_pos());
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (type == RS_REPEAT)
|
|
|
|
return;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
is_until = PATTERN_is(*JOB->current, RS_UNTIL);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is(*JOB->current, RS_WHILE)
|
|
|
|
|| is_until)
|
|
|
|
{
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
fast = trans_boolean_expr();
|
|
|
|
control_add_this_pos(trans_jump_if(is_until, fast));
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2010-05-22 20:02:34 +02:00
|
|
|
void TRANS_loop(int type)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ushort pos;
|
|
|
|
bool is_until;
|
2022-12-10 12:54:42 +01:00
|
|
|
bool fast;
|
2010-08-28 00:02:51 +02:00
|
|
|
|
|
|
|
if (type == RS_LOOP)
|
|
|
|
control_check(RS_DO, "LOOP without DO", "LOOP");
|
|
|
|
else if (type == RS_UNTIL)
|
|
|
|
control_check(RS_REPEAT, "UNTIL without REPEAT", "UNTIL");
|
|
|
|
else if (type == RS_WEND)
|
|
|
|
control_check(RS_WHILE, "WEND without WHILE", "WEND");
|
|
|
|
|
|
|
|
control_jump_each_pos_with(current_ctrl->pos_continue);
|
|
|
|
|
|
|
|
is_until = PATTERN_is(*JOB->current, RS_UNTIL);
|
|
|
|
|
|
|
|
if ((type != RS_WEND) && (PATTERN_is(*JOB->current, RS_WHILE) || is_until))
|
|
|
|
{
|
|
|
|
JOB->current++;
|
|
|
|
|
2022-12-10 12:54:42 +01:00
|
|
|
fast = trans_boolean_expr();
|
|
|
|
pos = trans_jump_if(!is_until, fast);
|
2010-08-28 00:02:51 +02:00
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(pos, control_get_value(), TRUE);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
pos = CODE_get_current_pos();
|
|
|
|
CODE_jump();
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(pos, control_get_value(), TRUE);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
control_jump_each_pos();
|
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2009-07-08 21:57:50 +02:00
|
|
|
static void trans_select_break(bool do_not_add_pos)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2009-04-08 12:11:16 +02:00
|
|
|
BEGIN_NO_BREAK
|
|
|
|
{
|
|
|
|
if (current_ctrl->value)
|
|
|
|
{
|
|
|
|
if (!do_not_add_pos)
|
|
|
|
{
|
|
|
|
control_add_current_pos();
|
|
|
|
CODE_jump();
|
|
|
|
}
|
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(current_ctrl->value, CODE_get_current_pos(), TRUE);
|
2009-04-08 12:11:16 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
END_NO_BREAK
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_select(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(RS_SELECT);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is(*JOB->current, RS_CASE))
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_expression(FALSE);
|
2018-12-12 01:40:44 +01:00
|
|
|
|
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_pop_ctrl(current_ctrl->local);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_case(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
int i;
|
|
|
|
short pos;
|
|
|
|
short local;
|
2012-05-23 01:32:38 +02:00
|
|
|
bool like;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_SELECT, "CASE without SELECT", "CASE");
|
2011-11-10 11:09:25 +01:00
|
|
|
|
|
|
|
if (current_ctrl->state)
|
|
|
|
THROW("Default case must be the last one");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2009-04-08 12:11:16 +02:00
|
|
|
trans_select_break(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
local = current_ctrl->local;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(RS_CASE);
|
2012-05-23 01:32:38 +02:00
|
|
|
|
2012-07-06 23:50:15 +02:00
|
|
|
like = TRANS_is(RS_LIKE);
|
2012-05-23 01:32:38 +02:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
for(i = 0; ; i++)
|
|
|
|
{
|
|
|
|
if (i > MAX_CASE_EXPR)
|
|
|
|
THROW("Too many expressions in CASE");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-07-19 11:53:34 +02:00
|
|
|
if (PATTERN_is_newline_end(*JOB->current))
|
|
|
|
THROW("Unexpected end of line");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
/*CODE_dup();
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
CODE_op(C_EQ, 2);*/
|
2010-07-19 12:10:37 +02:00
|
|
|
|
2012-07-06 23:50:15 +02:00
|
|
|
if (like)
|
2010-07-19 12:10:37 +02:00
|
|
|
{
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_push_local(local);
|
2010-07-19 12:10:37 +02:00
|
|
|
TRANS_expression(FALSE);
|
2012-07-06 23:50:15 +02:00
|
|
|
CODE_op(C_LIKE, 0, 2, TRUE);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2012-07-06 23:50:15 +02:00
|
|
|
else if (TRANS_is(RS_TO))
|
2010-08-28 00:02:51 +02:00
|
|
|
{
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_push_local(local);
|
|
|
|
TRANS_expression(FALSE);
|
2012-07-06 23:50:15 +02:00
|
|
|
CODE_op(C_LE, 0, 2, TRUE);
|
2010-07-19 12:10:37 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
|
|
|
|
if (TRANS_is(RS_TO))
|
|
|
|
{
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-07-19 12:10:37 +02:00
|
|
|
CODE_push_local(local);
|
|
|
|
CODE_op(C_LE, 0, 2, TRUE);
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-07-19 12:10:37 +02:00
|
|
|
CODE_push_local(local);
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
CODE_op(C_LE, 0, 2, TRUE);
|
|
|
|
CODE_op(C_AND, 0, 2, TRUE);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-07-19 12:10:37 +02:00
|
|
|
CODE_push_local(local);
|
|
|
|
CODE_op(C_EQ, 0, 2, TRUE);
|
|
|
|
}
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!PATTERN_is(*JOB->current, RS_COMMA))
|
|
|
|
{
|
2021-05-07 19:20:30 +02:00
|
|
|
//TRANS_ignore(RS_THEN);
|
2010-08-28 00:02:51 +02:00
|
|
|
pos = CODE_get_current_pos();
|
2022-12-10 12:54:42 +01:00
|
|
|
CODE_jump_if(FALSE, FALSE);
|
2010-08-28 00:02:51 +02:00
|
|
|
break;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_current_pos();
|
2022-12-10 12:54:42 +01:00
|
|
|
CODE_jump_if(TRUE, FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->current++;
|
2014-01-14 23:43:35 +01:00
|
|
|
TRANS_newline();
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos();
|
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl->value = pos;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_default(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_SELECT, "DEFAULT without SELECT", "DEFAULT");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2011-11-10 11:09:25 +01:00
|
|
|
if (current_ctrl->state)
|
|
|
|
THROW("Default case already defined");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
trans_select_break(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
current_ctrl->value = 0; /*CODE_get_current_pos();*/
|
2011-11-10 11:09:25 +01:00
|
|
|
current_ctrl->state = 1;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_end_select(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_SELECT, "END SELECT without SELECT", "END SELECT");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
/*
|
|
|
|
if (current_ctrl->value)
|
2018-05-24 00:31:58 +02:00
|
|
|
jump_length(current_ctrl->value, CODE_get_current_pos());
|
2010-08-28 00:02:51 +02:00
|
|
|
*/
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
trans_select_break(TRUE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_break(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_CTRL *ctrl_inner = control_get_inner();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-03-02 23:05:55 +01:00
|
|
|
check_try("BREAK");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!ctrl_inner)
|
|
|
|
THROW(E_UNEXPECTED, "BREAK");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_pos(&ctrl_inner->pos_break, CODE_get_current_pos());
|
|
|
|
CODE_jump();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_continue(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_CTRL *ctrl_inner = control_get_inner();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-03-02 23:05:55 +01:00
|
|
|
check_try("CONTINUE");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!ctrl_inner)
|
|
|
|
THROW(E_UNEXPECTED, "CONTINUE");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_pos(&ctrl_inner->pos_continue, CODE_get_current_pos());
|
|
|
|
CODE_jump();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_return(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
if (FUNCTION_is_procedure(JOB->func))
|
|
|
|
{
|
|
|
|
if (!(PATTERN_is_newline(*JOB->current) || PATTERN_is_end(*JOB->current)))
|
|
|
|
THROW("Return value datatype not specified in function declaration");
|
|
|
|
}
|
|
|
|
|
|
|
|
if (PATTERN_is_newline(*JOB->current) || PATTERN_is_end(*JOB->current))
|
|
|
|
CODE_return(0);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
CODE_return(1);
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_for(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2018-12-13 13:15:33 +01:00
|
|
|
int local;
|
2010-12-28 03:24:20 +01:00
|
|
|
bool downto = FALSE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(RS_FOR);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-13 13:15:33 +01:00
|
|
|
local = TRANS_loop_local(FALSE);
|
|
|
|
|
|
|
|
TRANS_want(RS_EQUAL, "=");
|
|
|
|
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
|
2022-10-04 22:49:01 +02:00
|
|
|
CODE_pop_local_noref(local);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
|
|
|
control_check_loop_var(local);
|
|
|
|
|
2010-12-28 03:24:20 +01:00
|
|
|
if (TRANS_is(RS_DOWNTO))
|
|
|
|
downto = TRUE;
|
|
|
|
else
|
|
|
|
TRANS_want(RS_TO, "TO");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_expression(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (PATTERN_is(*JOB->current, RS_STEP))
|
|
|
|
{
|
|
|
|
JOB->current++;
|
|
|
|
TRANS_expression(FALSE);
|
2010-12-28 03:24:20 +01:00
|
|
|
if (downto)
|
|
|
|
CODE_op(C_NEG, 0, 1, TRUE);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2010-12-28 03:24:20 +01:00
|
|
|
CODE_push_number(downto ? -1 : 1);
|
2010-08-28 00:02:51 +02:00
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!PATTERN_is_newline(*JOB->current))
|
|
|
|
THROW(E_UNEXPECTED, READ_get_pattern(JOB->current));
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_jump_first(current_ctrl->local);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_set_value(CODE_get_current_pos());
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_add_current_pos();
|
|
|
|
CODE_jump_next();
|
2018-02-14 17:03:14 +01:00
|
|
|
|
2022-10-04 22:49:01 +02:00
|
|
|
CODE_pop_local_noref(local);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_for_each(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
PATTERN *iterator = JOB->current;
|
|
|
|
PATTERN *save;
|
2018-12-13 13:15:33 +01:00
|
|
|
int local;
|
2010-08-28 00:02:51 +02:00
|
|
|
bool var = TRUE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
while (!PATTERN_is(*JOB->current, RS_IN))
|
|
|
|
{
|
|
|
|
if (PATTERN_is_newline_end(*JOB->current))
|
|
|
|
{
|
|
|
|
JOB->current = iterator;
|
|
|
|
var = FALSE;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
JOB->current++;
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_enter(RS_EACH);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (var)
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_expression(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
/*CODE_pop_ctrl(current_ctrl->local);*/
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_first(current_ctrl->local);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_set_value(CODE_get_current_pos());
|
|
|
|
control_add_current_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (var)
|
|
|
|
{
|
|
|
|
CODE_next(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
save = JOB->current;
|
|
|
|
JOB->current = iterator;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-13 13:15:33 +01:00
|
|
|
local = TRANS_loop_local(TRUE);
|
|
|
|
CODE_pop_local(local);
|
|
|
|
|
|
|
|
//TRANS_reference();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2013-10-03 15:21:36 +02:00
|
|
|
TRANS_want(RS_IN, "IN");
|
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->current = save;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
CODE_next(TRUE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
return;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_next(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ushort pos;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check_two(RS_FOR, RS_EACH, "NEXT without FOR", "NEXT");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
/*
|
|
|
|
if (current_ctrl->type == RS_FOR)
|
|
|
|
{
|
|
|
|
control_jump_each_pos_with(current_ctrl->pos_continue);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
pos = CODE_get_current_pos();
|
|
|
|
CODE_jump();
|
2018-05-24 00:31:58 +02:00
|
|
|
jump_length(pos, control_get_value());
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos();
|
|
|
|
control_leave();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
*/
|
|
|
|
control_jump_each_pos_with(current_ctrl->pos_continue);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
pos = CODE_get_current_pos();
|
|
|
|
CODE_jump();
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(pos, control_get_value(), TRUE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_jump_each_pos();
|
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-03-15 19:36:22 +01:00
|
|
|
void TRANS_assert(void)
|
|
|
|
{
|
|
|
|
ushort pos;
|
|
|
|
|
|
|
|
check_try("ASSERT");
|
|
|
|
|
|
|
|
if (!JOB->debug || JOB->exec)
|
|
|
|
CODE_disable();
|
|
|
|
|
|
|
|
TRANS_in_try = RS_ASSERT;
|
|
|
|
|
|
|
|
TRANS_expression(FALSE);
|
|
|
|
|
|
|
|
if (PATTERN_is(*JOB->current, RS_PRINT) || PATTERN_is(*JOB->current, RS_ERROR))
|
|
|
|
{
|
|
|
|
CODE_dup();
|
|
|
|
pos = CODE_get_current_pos();
|
2022-12-10 12:54:42 +01:00
|
|
|
CODE_jump_if(TRUE, FALSE);
|
2019-03-15 19:36:22 +01:00
|
|
|
|
|
|
|
TRANS_statement();
|
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(pos, CODE_get_current_pos(), TRUE);
|
2019-03-15 19:36:22 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
TRANS_subr(TS_SUBR_DEBUG, 1);
|
|
|
|
CODE_drop();
|
|
|
|
|
|
|
|
TRANS_in_try = RS_NONE;
|
|
|
|
|
|
|
|
if (!JOB->debug || JOB->exec)
|
|
|
|
CODE_enable();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_try(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ushort pos;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2019-03-15 19:36:22 +01:00
|
|
|
check_try("TRY");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
pos = CODE_get_current_pos();
|
|
|
|
CODE_try();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2019-03-15 19:36:22 +01:00
|
|
|
TRANS_in_try = RS_TRY;
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_statement();
|
2019-03-15 19:36:22 +01:00
|
|
|
TRANS_in_try = RS_NONE;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2022-12-10 20:36:49 +01:00
|
|
|
jump_length(pos, CODE_get_current_pos(), TRUE);
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_end_try();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_finally(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ushort pos = CODE_get_current_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if ((JOB->func->finally != 0)
|
|
|
|
|| (JOB->func->catch != 0)
|
|
|
|
|| (pos == 0))
|
|
|
|
THROW(E_UNEXPECTED, "FINALLY");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->func->finally = pos;
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_catch(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
ushort pos = CODE_get_current_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if ((JOB->func->catch != 0)
|
|
|
|
|| (pos == 0))
|
|
|
|
THROW(E_UNEXPECTED, "CATCH");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_catch();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->func->catch = CODE_get_current_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_label(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
CLASS_SYMBOL *sym;
|
2012-03-05 00:43:07 +01:00
|
|
|
int sym_index;
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_LABEL *label;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
sym_index = PATTERN_index(*JOB->current);
|
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2015-01-16 01:31:59 +01:00
|
|
|
sym = CLASS_declare(JOB->class, sym_index, TK_LABEL, FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (label_info == NULL)
|
|
|
|
ARRAY_create(&label_info);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2016-03-20 02:27:31 +01:00
|
|
|
sym->local.type = TYPE_make(T_VOID, -1, TK_LABEL);
|
2010-08-28 00:02:51 +02:00
|
|
|
sym->local.value = ARRAY_count(label_info);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
label = ARRAY_add(&label_info);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
label->index = sym_index;
|
|
|
|
label->ctrl_id = (ctrl_level == 0) ? 0 : current_ctrl->id;
|
|
|
|
label->pos = CODE_get_current_pos();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
#ifdef DEBUG_GOTO
|
|
|
|
printf("TRANS_label: %.*s ctrl_id = %d\n", sym->symbol.len, sym->symbol.name, label->ctrl_id);
|
|
|
|
#endif
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
/* on saute le ':' */
|
|
|
|
JOB->current++;
|
2012-03-04 10:13:59 +01:00
|
|
|
|
2012-03-05 00:43:07 +01:00
|
|
|
/*
|
2012-03-04 10:13:59 +01:00
|
|
|
// A new set of control stack slots must be used for a GOSUB subroutine
|
|
|
|
|
|
|
|
for (i = 0; i < ARRAY_count(goto_info); i++)
|
|
|
|
{
|
|
|
|
if (goto_info[i].gosub && goto_info[i].index == sym_index)
|
|
|
|
{
|
|
|
|
ctrl_local = JOB->func->nctrl + JOB->func->nlocal;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
2012-03-05 00:43:07 +01:00
|
|
|
*/
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_with(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2018-06-02 17:59:47 +02:00
|
|
|
if (!TRANS_affectation(TRUE))
|
|
|
|
TRANS_expression(FALSE);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2017-08-01 00:49:57 +02:00
|
|
|
control_enter(RS_WITH);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 01:40:44 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_pop_ctrl(current_ctrl->local);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_use_with(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_CTRL *ctrl_inner = control_get_inner_with();
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (ctrl_inner == NULL)
|
|
|
|
THROW("Syntax error. Point syntax used outside of WITH / END WITH");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-12-12 16:27:02 +01:00
|
|
|
control_add_relocation();
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_push_local(ctrl_inner->local);
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_end_with(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
control_check(RS_WITH, "END WITH without WITH", "END WITH");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
control_leave();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2008-01-06 19:49:23 +01:00
|
|
|
void TRANS_raise(void)
|
2007-12-30 17:41:49 +01:00
|
|
|
{
|
2010-08-28 00:02:51 +02:00
|
|
|
CLASS_SYMBOL *sym;
|
2012-11-24 19:52:11 +01:00
|
|
|
int index;
|
2010-08-28 00:02:51 +02:00
|
|
|
int np;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (!PATTERN_is_identifier(*JOB->current))
|
|
|
|
THROW("Syntax error in event name");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-11-24 19:52:11 +01:00
|
|
|
index = PATTERN_index(*JOB->current);
|
|
|
|
sym = CLASS_get_symbol(JOB->class, index);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
JOB->current++;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-11-24 19:52:11 +01:00
|
|
|
if (TYPE_get_kind(sym->global.type) == TK_EVENT)
|
|
|
|
{
|
|
|
|
if (TYPE_is_static(JOB->func->type))
|
|
|
|
THROW("Cannot raise events in static function");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2012-11-24 19:52:11 +01:00
|
|
|
CODE_push_event(sym->global.value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
CODE_push_unknown_event(CLASS_add_unknown(JOB->class, index));
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
np = 0;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (TRANS_is(RS_LBRA))
|
|
|
|
{
|
|
|
|
for (;;)
|
|
|
|
{
|
|
|
|
if (TRANS_is(RS_RBRA))
|
|
|
|
break;
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (np > 0)
|
|
|
|
TRANS_want(RS_COMMA, "comma");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
if (np >= MAX_PARAM_FUNC)
|
|
|
|
THROW("Too many arguments");
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
TRANS_expression(FALSE);
|
|
|
|
np++;
|
|
|
|
}
|
|
|
|
}
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_call(np);
|
2007-12-30 17:41:49 +01:00
|
|
|
|
2018-02-14 17:03:14 +01:00
|
|
|
if (!TRANS_in_assignment)
|
2010-08-28 00:02:51 +02:00
|
|
|
CODE_drop();
|
2007-12-30 17:41:49 +01:00
|
|
|
}
|