bdea5c8a7c
[INTERPRETER] * BUG: Fix compilation on Ubuntu Trusty.
465 lines
11 KiB
C
465 lines
11 KiB
C
/***************************************************************************
|
|
|
|
gbx_exec.h
|
|
|
|
(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.
|
|
|
|
***************************************************************************/
|
|
|
|
#ifndef __GBX_EXEC_H
|
|
#define __GBX_EXEC_H
|
|
|
|
#include "gb_alloc.h"
|
|
#include "gb_error.h"
|
|
#include "gb_overflow.h"
|
|
#include "gbx_class.h"
|
|
#include "gbx_value.h"
|
|
#include "gb_pcode.h"
|
|
#include "gbx_stack.h"
|
|
|
|
#include "gbx_string.h"
|
|
#include "gbx_object.h"
|
|
#include "gbx_variant.h"
|
|
|
|
#include "gbx_c_enum.h"
|
|
|
|
enum { GB_LITTLE_ENDIAN, GB_BIG_ENDIAN };
|
|
|
|
typedef
|
|
void (*EXEC_FUNC)();
|
|
|
|
typedef
|
|
void (*EXEC_FUNC_CODE)(ushort);
|
|
|
|
typedef
|
|
void (*EXEC_FUNC_CODE_SP)(ushort, VALUE *);
|
|
|
|
typedef
|
|
struct {
|
|
unsigned debug : 1; // running in debugging mode
|
|
unsigned got_error : 1; // if a native function has returned an error
|
|
unsigned debug_inside : 1; // debug inside components
|
|
unsigned debug_hold : 1; // hold execution at program end
|
|
unsigned task : 1; // I am a background task
|
|
unsigned profile : 1; // profiling mode
|
|
unsigned profile_instr : 1; // profiling mode at instruction level
|
|
unsigned trace : 1; // tracing mode
|
|
unsigned arch : 1; // executing an archive
|
|
unsigned fifo : 1; // debugging through a fifo
|
|
unsigned keep_library : 1; // do not unload libraries
|
|
unsigned main_hook_done : 1; // the main hook has been run
|
|
unsigned break_on_error : 1; // if we must break into the debugger as soon as there is an error.
|
|
unsigned in_event_loop : 1; // if we are in the event loop
|
|
unsigned check_overflow : 1; // if we should check for overflow
|
|
unsigned big_endian : 1; // if the CPU is big endian
|
|
unsigned has_forked : 1; // if the process has forked
|
|
}
|
|
EXEC_FLAG;
|
|
|
|
typedef
|
|
struct {
|
|
CLASS *class;
|
|
OBJECT *object;
|
|
CLASS_DESC_METHOD *desc;
|
|
FUNCTION *func;
|
|
int index;
|
|
char nparam;
|
|
bool native;
|
|
bool use_stack;
|
|
}
|
|
EXEC_GLOBAL;
|
|
|
|
typedef
|
|
struct {
|
|
void (*main)();
|
|
void (*loop)();
|
|
void (*wait)();
|
|
void (*timer)();
|
|
void (*lang)();
|
|
void (*watch)();
|
|
void (*post)();
|
|
void (*quit)();
|
|
bool (*error)();
|
|
double (*timeout)();
|
|
}
|
|
EXEC_HOOK;
|
|
|
|
enum {
|
|
OP_NOTHING = 0,
|
|
OP_OBJECT_FLOAT,
|
|
OP_FLOAT_OBJECT,
|
|
OP_OBJECT_OTHER,
|
|
OP_OTHER_OBJECT,
|
|
OP_OBJECT_OBJECT
|
|
};
|
|
|
|
|
|
#ifndef __GBX_EXEC_LOOP_C
|
|
|
|
extern STACK_CONTEXT EXEC_current;
|
|
extern VALUE *SP;
|
|
extern bool EXEC_bytecode_less_than_3_18;
|
|
|
|
#endif
|
|
|
|
#ifndef __GBX_EXEC_C
|
|
|
|
extern VALUE TEMP;
|
|
extern VALUE RET;
|
|
|
|
extern VALUE *EXEC_super;
|
|
|
|
/*
|
|
extern bool EXEC_debug;
|
|
extern bool EXEC_got_error;
|
|
extern bool EXEC_debug_inside;
|
|
extern bool EXEC_debug_hold;
|
|
extern bool EXEC_task;
|
|
extern bool EXEC_profile;
|
|
extern bool EXEC_trace;
|
|
extern bool EXEC_profile_instr;
|
|
extern bool EXEC_arch;
|
|
extern bool EXEC_fifo;
|
|
extern bool EXEC_keep_library;
|
|
extern bool EXEC_break_on_error;
|
|
extern bool EXEC_in_event_loop;
|
|
#if DO_NOT_CHECK_OVERFLOW
|
|
#else
|
|
extern bool EXEC_check_overflow;
|
|
#endif
|
|
extern bool EXEC_big_endian;
|
|
extern bool EXEC_main_hook_done;
|
|
*/
|
|
|
|
extern const char *EXEC_fifo_name;
|
|
extern const char *EXEC_profile_path;
|
|
|
|
extern EXEC_HOOK EXEC_Hook;
|
|
|
|
extern CENUM *EXEC_enum;
|
|
|
|
extern const char EXEC_should_borrow[];
|
|
|
|
extern const char *EXEC_unknown_name;
|
|
extern char EXEC_unknown_nparam;
|
|
extern uchar EXEC_quit_value;
|
|
|
|
extern EXEC_GLOBAL EXEC;
|
|
extern EXEC_FLAG FLAG;
|
|
|
|
extern const void *EXEC_subr_table[];
|
|
|
|
#endif
|
|
|
|
#define EXEC_debug FLAG.debug
|
|
#define EXEC_got_error FLAG.got_error
|
|
#define EXEC_debug_inside FLAG.debug_inside
|
|
#define EXEC_profile FLAG.profile
|
|
#define EXEC_fifo FLAG.fifo
|
|
#define EXEC_arch FLAG.arch
|
|
#define EXEC_profile_instr FLAG.profile_instr
|
|
#define EXEC_trace FLAG.trace
|
|
#define EXEC_big_endian FLAG.big_endian
|
|
#define EXEC_break_on_error FLAG.break_on_error
|
|
#define EXEC_debug_hold FLAG.debug_hold
|
|
#define EXEC_task FLAG.task
|
|
#define EXEC_keep_library FLAG.keep_library
|
|
#define EXEC_main_hook_done FLAG.main_hook_done
|
|
#define EXEC_check_overflow FLAG.check_overflow
|
|
|
|
|
|
// Local variables base pointer
|
|
#define BP EXEC_current.bp
|
|
// Arguments base pointer
|
|
#define PP EXEC_current.pp
|
|
// Current class
|
|
#define CP EXEC_current.cp
|
|
// Current object
|
|
#define OP EXEC_current.op
|
|
// Save stack pointer for a TRY
|
|
#define EP EXEC_current.ep
|
|
// Current function
|
|
#define FP EXEC_current.fp
|
|
// Program counter
|
|
#define PC EXEC_current.pc
|
|
// Where to go if there is an error
|
|
#define EC EXEC_current.ec
|
|
// Save register for TRY
|
|
#define ET EXEC_current.et
|
|
// GoSub stack
|
|
#define GP EXEC_current.gp
|
|
|
|
// Function return value pointer
|
|
#define RP (&RET)
|
|
|
|
#define HOOK(func) (!EXEC_Hook.func) ? 0 : (*EXEC_Hook.func)
|
|
#define HOOK_DEFAULT(func, def) (*((!EXEC_Hook.func) ? def : EXEC_Hook.func))
|
|
|
|
#define GET_NPARAM(var) short var = code & 0x3F
|
|
#define GET_PARAM(var, nparam) VALUE *var = &SP[-nparam]
|
|
|
|
void EXEC_init(void);
|
|
|
|
void EXEC_enter_check(bool defined);
|
|
void EXEC_enter(void);
|
|
void EXEC_enter_quick(void);
|
|
void EXEC_leave_keep();
|
|
void EXEC_leave_drop();
|
|
void EXEC_loop(void);
|
|
void EXEC_init_bytecode_check(void);
|
|
void EXEC_switch_bytecode(void);
|
|
|
|
#define EXEC_check_bytecode() \
|
|
({ \
|
|
if (CP && CP->less_than_3_18 != EXEC_bytecode_less_than_3_18) \
|
|
EXEC_switch_bytecode(); \
|
|
})
|
|
|
|
#define EXEC_object_2(_val, _pclass, _pobject) \
|
|
({ \
|
|
VALUE *v = (_val); \
|
|
\
|
|
*(_pobject) = v->_object.object; \
|
|
\
|
|
if (v->_object.class->is_simple) \
|
|
{ \
|
|
if (!*(_pobject)) \
|
|
THROW_NULL(); \
|
|
\
|
|
*(_pclass) = OBJECT_class(v->_object.object); \
|
|
} \
|
|
else \
|
|
*(_pclass) = EXEC_object_real(_val); \
|
|
})
|
|
|
|
#define EXEC_object(_val, _pclass, _pobject) \
|
|
((TYPE_is_pure_object((_val)->type)) ? EXEC_object_2(_val, _pclass, _pobject), TRUE : \
|
|
TYPE_is_variant((_val)->type) ? (*(_pclass) = EXEC_object_variant(_val, _pobject)), FALSE : \
|
|
EXEC_object_other(_val, _pclass, _pobject))
|
|
|
|
#define EXEC_object_fast(_val, _pclass, _pobject) \
|
|
({ \
|
|
if ((_val)->type != T_CLASS) \
|
|
EXEC_object_real(_val, _pclass, _pobject); \
|
|
else \
|
|
EXEC_object_other(_val, _pclass, _pobject); \
|
|
})
|
|
|
|
#define EXEC_object_array(_val, _vclass, _vobject) \
|
|
_vobject = (_val)->_object.object; \
|
|
if (!(_vobject)) \
|
|
THROW_NULL(); \
|
|
_vclass = (_vobject)->class; \
|
|
if ((_val) == EXEC_super) \
|
|
EXEC_super = (_val)->_object.super;
|
|
|
|
CLASS *EXEC_object_real(VALUE *val);
|
|
CLASS *EXEC_object_variant(VALUE *val, OBJECT **pobject);
|
|
bool EXEC_object_other(VALUE *val, CLASS **pclass, OBJECT **pobject);
|
|
|
|
void *EXEC_auto_create(CLASS *class, bool ref);
|
|
|
|
bool EXEC_call_native(void (*exec)(), void *object, TYPE type, VALUE *param);
|
|
void EXEC_native_check(bool defined);
|
|
void EXEC_native_quick(void);
|
|
void EXEC_native(void);
|
|
void EXEC_function_real(void);
|
|
void EXEC_function_loop(void);
|
|
|
|
#define EXEC_function() EXEC_function_real(), EXEC_release_return_value()
|
|
#define EXEC_function_keep() EXEC_function_real()
|
|
|
|
void EXEC_public(CLASS *class, void *object, const char *name, int nparam);
|
|
void EXEC_public_desc(CLASS *class, void *object, CLASS_DESC_METHOD *desc, int nparam);
|
|
|
|
bool EXEC_special_do(CLASS *class, int index, void *object, int nparam, bool drop);
|
|
|
|
#define EXEC_special(_special, _class, _object, _nparam, _drop) \
|
|
({ \
|
|
int __index = _class->special[_special]; \
|
|
(__index == NO_SYMBOL) ? TRUE : EXEC_special_do(_class, __index, _object, _nparam, _drop); \
|
|
})
|
|
|
|
void EXEC_special_inheritance(int special, CLASS *class, OBJECT *object, int nparam, bool drop);
|
|
|
|
void EXEC_nop(void);
|
|
|
|
void EXEC_push_unknown(void);
|
|
int EXEC_push_unknown_event(bool unknown);
|
|
|
|
void EXEC_pop_unknown(void);
|
|
|
|
void EXEC_enum_first(PCODE code, VALUE *local, VALUE *penum);
|
|
bool EXEC_enum_next(PCODE code, VALUE *local, VALUE *penum);
|
|
|
|
void *EXEC_create_object(CLASS *class, int np, char *event);
|
|
void EXEC_new(ushort code);
|
|
|
|
void EXEC_release_return_value(void);
|
|
void EXEC_do_quit(void);
|
|
|
|
void EXEC_dup(int n);
|
|
|
|
void EXEC_borrow(TYPE type, VALUE *val);
|
|
void EXEC_unborrow(VALUE *val);
|
|
void EXEC_release(TYPE type, VALUE *val);
|
|
void RELEASE_many(VALUE *val, int n);
|
|
|
|
void EXEC_push_complex(void);
|
|
|
|
void EXEC_push_vargs(void);
|
|
void EXEC_end_vargs(void);
|
|
void EXEC_drop_vargs(void);
|
|
|
|
#define BORROW(_value) \
|
|
do { \
|
|
VALUE *_v = (_value); \
|
|
TYPE type = _v->type; \
|
|
if (TYPE_is_object(type)) \
|
|
{ \
|
|
OBJECT_REF_CHECK(_v->_object.object); \
|
|
} \
|
|
else if (EXEC_should_borrow[type]) \
|
|
{ \
|
|
if (type == T_STRING) \
|
|
STRING_ref(_v->_string.addr); \
|
|
else \
|
|
EXEC_borrow(type, _v); \
|
|
} \
|
|
} while (0)
|
|
|
|
#define RELEASE(_value) \
|
|
do { \
|
|
VALUE *_v = (_value); \
|
|
TYPE type = _v->type; \
|
|
if (TYPE_is_object(type)) \
|
|
{ \
|
|
OBJECT_UNREF(_v->_object.object); \
|
|
} \
|
|
else if (EXEC_should_borrow[type]) \
|
|
{ \
|
|
if (type == T_STRING) \
|
|
STRING_unref(&_v->_string.addr); \
|
|
else \
|
|
EXEC_release(type, _v); \
|
|
} \
|
|
} while (0)
|
|
|
|
#if DEBUG_REF
|
|
|
|
#define UNBORROW(_value) \
|
|
do { \
|
|
fprintf(stderr, "@%s\n", OBJECT_where_am_i(__FILE__, __LINE__, __func__)); \
|
|
EXEC_unborrow(_value); \
|
|
} while (0)
|
|
|
|
#else
|
|
|
|
#define UNBORROW EXEC_unborrow
|
|
|
|
#endif
|
|
|
|
#define RELEASE_STRING(_value) \
|
|
do { \
|
|
if ((_value)->type == T_STRING) STRING_unref(&(_value)->_string.addr); \
|
|
} while(0)
|
|
|
|
#define RELEASE_OBJECT(_value) \
|
|
do { \
|
|
VALUE *_v = (_value); \
|
|
OBJECT_UNREF(_v->_object.object); \
|
|
} while (0)
|
|
|
|
#define RELEASE_VARIANT(_value) \
|
|
do { \
|
|
VALUE *_v = (_value); \
|
|
if (_v->_variant.vtype == T_STRING) \
|
|
STRING_unref(&_v->_variant.value._string); \
|
|
else if (TYPE_is_object(_v->_variant.vtype)) \
|
|
OBJECT_UNREF(_v->_variant.value._object); \
|
|
} while (0)
|
|
|
|
#define RELEASE_MANY(_val, _n) \
|
|
do { \
|
|
if (_n) \
|
|
{ \
|
|
if ((_n) == 1) \
|
|
{ \
|
|
_val--; \
|
|
RELEASE((_val)); \
|
|
} \
|
|
else \
|
|
{ \
|
|
RELEASE_many((_val), (_n)); \
|
|
_val -= (_n); \
|
|
} \
|
|
} \
|
|
} while (0)
|
|
|
|
#define PUSH() \
|
|
do { \
|
|
BORROW(SP); \
|
|
SP++; \
|
|
} while (0)
|
|
|
|
#define POP() \
|
|
do { \
|
|
SP--; \
|
|
RELEASE(SP); \
|
|
} while (0)
|
|
|
|
#define PUSH_OBJECT(__class, __object) \
|
|
({ \
|
|
SP->_object.class = (__class); \
|
|
SP->_object.object = OBJECT_REF(__object); \
|
|
SP++; \
|
|
})
|
|
|
|
#define COPY_VALUE(_dst, _src) VALUE_copy(_dst, _src)
|
|
|
|
#define EXEC_set_native_error(_err) (ERROR_current->info.native = (_err))
|
|
#define EXEC_has_native_error() (ERROR_current->info.native)
|
|
|
|
#define EXEC_move_ret_to_temp() \
|
|
({ \
|
|
TEMP = *RP; \
|
|
UNBORROW(RP); \
|
|
RP->type = T_VOID; \
|
|
})
|
|
|
|
bool EXEC_check_operator_single(VALUE *P1, uchar op);
|
|
int EXEC_check_operator(VALUE *P1, VALUE *P2, uchar op);
|
|
void EXEC_operator(uchar what, uchar op, VALUE *P1, VALUE *P2);
|
|
void EXEC_operator_object_add_quick(VALUE *P1, double val);
|
|
int EXEC_comparator(uchar what, uchar op, VALUE *P1, VALUE *P2);
|
|
void EXEC_operator_object_sgn(VALUE *P1);
|
|
void EXEC_operator_object_fabs(VALUE *P1);
|
|
void EXEC_operator_object_single(uchar op, VALUE *P1);
|
|
|
|
void EXEC_quit(ushort code);
|
|
|
|
void EXEC_push_array(ushort code);
|
|
void EXEC_pop_array(ushort code);
|
|
|
|
void EXEC_set_got_error(bool err);
|
|
|
|
PCODE *EXEC_on_goto_gosub(PCODE *pc);
|
|
void EXEC_gosub(PCODE *pc);
|
|
|
|
|
|
#endif /* */
|