gambas 706a605cc3 Correctly restore the stack after calling a method through the interpreter, and handle null values as object values having a null address automatically.
[INTERPRETER]
* NEW: Creating a NULL value now clears the object address field so that it can be freely interpreted as a Object value with a null address by the JIT compiled code.

[GB.JIT]
* BUG: Correctly restore stack after calling a method through the interpreter.
2018-08-22 09:30:27 +02:00

657 lines
14 KiB
C

/***************************************************************************
gbx_value.h
(c) 2000-2017 Benoît Minisini <g4mba5@gmail.com>
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_VALUE_H
#define __GBX_VALUE_H
#include <time.h>
#include "gbx_type.h"
#include "gbx_class.h"
#ifndef __DATE_DECLARED
#define __DATE_DECLARED
typedef
struct {
int date;
int time;
}
DATE;
#endif
typedef
struct {
TYPE type;
int value;
}
VALUE_BOOLEAN;
typedef
struct {
TYPE type;
int value;
}
VALUE_BYTE;
typedef
struct {
TYPE type;
int value;
}
VALUE_SHORT;
typedef
struct {
TYPE type;
int value;
}
VALUE_INTEGER;
typedef
struct {
TYPE type;
#ifndef OS_64BITS
int _padding;
#endif
int64_t value;
}
VALUE_LONG;
typedef
struct {
TYPE type;
char *value;
}
VALUE_POINTER;
typedef
struct {
TYPE type;
float value;
}
VALUE_SINGLE;
typedef
struct {
TYPE type;
#ifndef OS_64BITS
int _padding;
#endif
double value;
}
VALUE_FLOAT;
typedef
struct {
TYPE type;
int date; /* number of days */
int time; /* number of milliseconds */
}
VALUE_DATE;
typedef
struct {
TYPE type;
char *addr;
int start;
int len;
}
VALUE_STRING;
typedef
struct {
TYPE type;
CLASS *class;
void *object;
char kind;
char defined;
short index;
}
VALUE_FUNCTION;
enum
{
FUNCTION_NULL,
FUNCTION_NATIVE,
FUNCTION_PRIVATE,
FUNCTION_PUBLIC,
FUNCTION_EVENT,
FUNCTION_EXTERN,
FUNCTION_UNKNOWN,
FUNCTION_CALL,
FUNCTION_SUBR
};
typedef
struct {
TYPE type;
TYPE ptype;
intptr_t value[2];
}
VALUE_VOID;
typedef
struct {
TYPE type;
TYPE vtype;
union {
char _boolean;
unsigned char _byte;
short _short;
int _integer;
int64_t _long;
float _single;
double _float;
DATE _date;
char *_string;
void *_pointer;
void *_object;
int64_t data;
}
value;
}
VALUE_VARIANT;
typedef
struct {
CLASS *class;
void *object;
void *super;
}
VALUE_OBJECT;
typedef
struct {
TYPE type;
CLASS *class;
void *super;
}
VALUE_CLASS;
typedef
union value {
TYPE type;
VALUE_BOOLEAN _boolean;
VALUE_BYTE _byte;
VALUE_SHORT _short;
VALUE_INTEGER _integer;
VALUE_LONG _long;
VALUE_SINGLE _single;
VALUE_FLOAT _float;
VALUE_DATE _date;
VALUE_STRING _string;
VALUE_POINTER _pointer;
VALUE_FUNCTION _function;
VALUE_VARIANT _variant;
VALUE_CLASS _class;
VALUE_OBJECT _object;
VALUE_VOID _void;
}
VALUE;
typedef
void (*VALUE_CONVERT_FUNC)(VALUE *);
#define VALUE_copy(_dst, _src) \
((_dst)->_void.type = (_src)->_void.type, \
(_dst)->_void.ptype = (_src)->_void.ptype, \
(_dst)->_void.value[0] = (_src)->_void.value[0], \
(_dst)->_void.value[1] = (_src)->_void.value[1])
#define VALUE_is_equal(_v1, _v2) (*_v1 == *v2)
#define VALUE_is_object(val) (TYPE_is_object((val)->type))
#define VALUE_is_string(val) ((val)->type == T_STRING || (val)->type == T_CSTRING)
#define VALUE_is_number(val) ((val)->type >= T_BYTE && (val)->type <= T_FLOAT)
void VALUE_default(VALUE *value, TYPE type);
void VALUE_convert(VALUE *value, TYPE type);
void VALUE_convert_boolean(VALUE *value);
void VALUE_convert_integer(VALUE *value);
void VALUE_convert_float(VALUE *value);
void VALUE_convert_string(VALUE *value);
void VALUE_convert_variant(VALUE *value);
void VALUE_read(VALUE *value, void *addr, TYPE type);
void VALUE_write(VALUE *value, void *addr, TYPE type);
void VALUE_undo_variant(VALUE *value);
void VALUE_write_variant(VALUE *value, void *addr);
//void VALUE_put(VALUE *value, void *addr, TYPE type);
void VALUE_free(void *addr, TYPE type);
void VALUE_to_string(VALUE *value, char **addr, int *len);
void VALUE_from_string(VALUE *value, const char *addr, int len);
void VALUE_class_read(CLASS *class, VALUE *value, char *addr, CTYPE ctype, void *ref);
void VALUE_class_write(CLASS *class, VALUE *value, char *addr, CTYPE ctype);
void VALUE_class_constant(CLASS *class, VALUE *value, int ind);
#define VALUE_null(_val) ({ (_val)->type = T_NULL; (_val)->_object.object = NULL; })
bool VALUE_is_null(VALUE *val);
//void VALUE_get_string(VALUE *val, char **text, int *length);
#define VALUE_get_string(_value, _ptext, _plen) \
({ \
*(_plen) = (_value)->_string.len; \
if (*(_plen)) \
*(_ptext) = (_value)->_string.start + (_value)->_string.addr; \
else \
*(_ptext) = NULL; \
})
void THROW_TYPE(TYPE wanted, TYPE got) NORETURN;
#define VALUE_conv(_value, _type) \
({ \
if ((_value)->type != (_type)) \
VALUE_convert(_value, _type); \
})
#if 0
#define VALUE_conv_boolean(_value) \
({ \
VALUE *v = _value; \
if (UNLIKELY(v->type != T_BOOLEAN)) \
{ \
VALUE_convert_boolean(v); \
} \
})
#define VALUE_conv_integer(_value) \
({ \
VALUE *v = _value; \
if (UNLIKELY(v->type != T_INTEGER)) \
{ \
if (TYPE_is_object(v->type)) \
THROW_TYPE_INTEGER(v->type); \
VALUE_convert_integer(v); \
} \
})
#define VALUE_conv_float(_value) \
({ \
VALUE *v = _value; \
if (UNLIKELY(v->type != T_FLOAT)) \
{ \
if (TYPE_is_object(v->type)) \
THROW_TYPE_FLOAT(v->type); \
VALUE_convert_float(v); \
} \
})
#define VALUE_conv_string(_value) \
({ \
VALUE *v = _value; \
if (UNLIKELY(v->type != T_STRING && v->type != T_CSTRING)) \
{ \
if (TYPE_is_object(v->type)) \
THROW_TYPE_STRING(v->type); \
VALUE_convert_string(v); \
} \
})
#define VALUE_conv_variant(_value) \
({ \
if (UNLIKELY((_value)->type != T_VARIANT)) \
VALUE_convert_variant(_value); \
})
#define VALUE_conv_object(_value, _type) \
({ \
if (UNLIKELY((_value)->type != (_type))) \
VALUE_convert_object(_value, _type); \
})
#else
#define VALUE_conv_boolean(_value) \
({ \
if (UNLIKELY((_value)->type != T_BOOLEAN)) \
VALUE_convert_boolean(_value); \
})
#define VALUE_conv_float(_value) \
({ \
if (UNLIKELY((_value)->type != T_FLOAT)) \
VALUE_convert_float(_value); \
})
#define VALUE_conv_variant(_value) \
({ \
if (UNLIKELY((_value)->type != T_VARIANT)) \
VALUE_convert_variant(_value); \
})
//#define VALUE_conv_boolean(_value) VALUE_conv(_value, T_BOOLEAN)
#define VALUE_conv_integer(_value) VALUE_conv(_value, T_INTEGER)
//#define VALUE_conv_float(_value) VALUE_conv(_value, T_FLOAT)
//#define VALUE_conv_variant(_value) VALUE_conv(_value, T_VARIANT)
#define VALUE_conv_object(_value, _type) VALUE_conv(_value, _type)
#define VALUE_conv_string(_value) \
({ \
if (UNLIKELY((_value)->type != T_STRING && (_value)->type != T_CSTRING)) \
VALUE_conv(_value, T_STRING); \
})
#endif
#define VALUE_is_super(_value) (EXEC_super && EXEC_super == (_value)->_object.super)
#define VALUE_class_read_inline(_class, _value, _addr, _ctype, _ref) \
({ \
static void *jump[17] = { \
&&__VOID, &&__BOOLEAN, &&__BYTE, &&__SHORT, &&__INTEGER, &&__LONG, &&__SINGLE, &&__FLOAT, &&__DATE, \
&&__STRING, &&__CSTRING, &&__POINTER, &&__VARIANT, &&__ARRAY, &&__STRUCT, &&__NULL, &&__OBJECT \
}; \
\
for(;;) \
{ \
(_value)->type = (_ctype).id; \
goto *jump[(_ctype).id]; \
\
__BOOLEAN: \
(_value)->_boolean.value = -(*((unsigned char *)(_addr)) != 0); \
break; \
\
__BYTE: \
(_value)->_byte.value = *((unsigned char *)(_addr)); \
break; \
\
__SHORT: \
(_value)->_short.value = *((short *)(_addr)); \
break; \
\
__INTEGER: \
(_value)->_integer.value = *((int *)(_addr)); \
break; \
\
__LONG: \
(_value)->_long.value = *((int64_t *)(_addr)); \
break; \
\
__SINGLE: \
(_value)->_single.value = *((float *)(_addr)); \
break; \
\
__FLOAT: \
(_value)->_float.value = *((double *)(_addr)); \
break; \
\
__DATE: \
(_value)->_date.date = ((int *)(_addr))[0]; \
(_value)->_date.time = ((int *)(_addr))[1]; \
break; \
\
__STRING: \
{ \
char *str = *((char **)(_addr)); \
\
(_value)->type = T_STRING; \
(_value)->_string.addr = str; \
(_value)->_string.start = 0; \
(_value)->_string.len = STRING_length(str); \
\
break; \
} \
\
__CSTRING: \
{ \
char *str = *((char **)(_addr)); \
\
(_value)->type = T_CSTRING; \
(_value)->_string.addr = str; \
(_value)->_string.start = 0; \
(_value)->_string.len = (str == NULL) ? 0 : strlen(str); \
\
break; \
} \
\
__OBJECT: \
(_value)->_object.object = *((void **)(_addr)); \
(_value)->type = ((_ctype).value >= 0) ? (TYPE)(_class)->load->class_ref[(_ctype).value] : T_OBJECT; \
break; \
\
__POINTER: \
(_value)->_pointer.value = *((void **)(_addr)); \
break; \
\
__VARIANT: \
(_value)->_variant.type = T_VARIANT; \
(_value)->_variant.vtype = ((VARIANT *)(_addr))->type; \
if ((_value)->_variant.vtype == T_VOID) \
(_value)->_variant.vtype = T_NULL; \
VARIANT_copy_value(&(_value)->_variant, ((VARIANT *)(_addr))); \
break; \
\
__ARRAY: \
{ \
void *object = CARRAY_create_static((_class), (_ref), (_class)->load->array[(_ctype).value], (_addr)); \
(_value)->_object.class = OBJECT_class(object); \
(_value)->_object.object = object; \
break; \
} \
\
__STRUCT: \
{ \
void *object = CSTRUCT_create_static((_ref), (_class)->load->class_ref[(_ctype).value], (_addr)); \
(_value)->_object.class = OBJECT_class(object); \
(_value)->_object.object = object; \
break; \
} \
\
__VOID: \
__NULL: \
THROW_ILLEGAL(); \
} \
})
#define VALUE_class_constant_inline(_class, _value, _ind) \
({ \
static void *jump[] = \
{ \
&&__ILLEGAL, &&__INTEGER, &&__INTEGER, &&__INTEGER, &&__INTEGER, &&__LONG, &&__SINGLE, &&__FLOAT, \
&&__ILLEGAL, &&__STRING, &&__CSTRING, &&__POINTER, &&__ILLEGAL, &&__ILLEGAL, &&__ILLEGAL, &&__ILLEGAL \
}; \
\
CLASS_CONST *cc; \
\
for(;;) \
{ \
cc = &(_class)->load->cst[_ind]; \
goto *jump[cc->type]; \
\
__INTEGER: \
\
(_value)->type = T_INTEGER; \
(_value)->_integer.value = cc->_integer.value; \
break; \
\
__LONG: \
\
(_value)->type = T_LONG; \
(_value)->_long.value = cc->_long.value; \
break; \
\
__SINGLE: \
\
(_value)->type = T_SINGLE; \
(_value)->_single.value = cc->_single.value; \
break; \
\
__FLOAT: \
\
(_value)->type = T_FLOAT; \
(_value)->_float.value = cc->_float.value; \
break; \
\
__STRING: \
\
(_value)->type = T_CSTRING; \
(_value)->_string.addr = (char *)cc->_string.addr; \
(_value)->_string.start = 0; \
(_value)->_string.len = cc->_string.len; \
break; \
\
__CSTRING: \
\
(_value)->type = T_CSTRING; \
(_value)->_string.addr = (char *)LOCAL_gettext(cc->_string.addr); \
(_value)->_string.start = 0; \
(_value)->_string.len = strlen((_value)->_string.addr); \
break; \
\
__POINTER: \
\
(_value)->type = T_POINTER; \
(_value)->_pointer.value = NULL; \
break; \
\
__ILLEGAL: \
\
THROW_ILLEGAL(); \
} \
})
#define VALUE_read_inline_type(_value, _addr, _ctype, _type, _label_noref, _label_ref) \
({ \
static void *jump[17] = { \
&&__VOID, &&__BOOLEAN, &&__BYTE, &&__SHORT, &&__INTEGER, &&__LONG, &&__SINGLE, &&__FLOAT, &&__DATE, \
&&__STRING, &&__CSTRING, &&__POINTER, &&__VARIANT, &&__FUNCTION, &&__CLASS, &&__NULL, &&__OBJECT \
}; \
\
for(;;) \
{ \
(_value)->type = (_type); \
goto *jump[_ctype]; \
\
__BOOLEAN: \
\
(_value)->_boolean.value = (*((unsigned char *)(_addr)) != 0) ? (-1) : 0; \
goto _label_noref; \
\
__BYTE: \
\
(_value)->_byte.value = *((unsigned char *)(_addr)); \
goto _label_noref; \
\
__SHORT: \
\
(_value)->_short.value = *((short *)(_addr)); \
goto _label_noref; \
\
__INTEGER: \
\
(_value)->_integer.value = *((int *)(_addr)); \
goto _label_noref; \
\
__LONG: \
\
(_value)->_long.value = *((int64_t *)(_addr)); \
goto _label_noref; \
\
__SINGLE: \
\
(_value)->_single.value = *((float *)(_addr)); \
goto _label_noref; \
\
__FLOAT: \
\
(_value)->_float.value = *((double *)(_addr)); \
goto _label_noref; \
\
__DATE: \
\
(_value)->_date.date = ((int *)(_addr))[0]; \
(_value)->_date.time = ((int *)(_addr))[1]; \
goto _label_noref; \
\
__STRING: \
\
{ \
char *str = *((char **)(_addr)); \
\
(_value)->type = T_STRING; \
(_value)->_string.addr = str; \
(_value)->_string.start = 0; \
(_value)->_string.len = STRING_length(str); \
\
goto _label_ref; \
} \
\
__CSTRING: \
\
{ \
char *str = *((char **)(_addr)); \
\
(_value)->type = T_CSTRING; \
(_value)->_string.addr = str; \
(_value)->_string.start = 0; \
(_value)->_string.len = (str == NULL) ? 0 : strlen(str); \
\
goto _label_noref; \
} \
\
__OBJECT: \
\
(_value)->_object.object = *((void **)(_addr)); \
goto _label_ref; \
\
__POINTER: \
\
(_value)->_pointer.value = *((void **)(_addr)); \
goto _label_noref; \
break; \
\
__VARIANT: \
\
(_value)->_variant.type = T_VARIANT; \
(_value)->_variant.vtype = ((VARIANT *)(_addr))->type; \
\
if ((_value)->_variant.vtype == T_VOID) \
(_value)->_variant.vtype = T_NULL; \
\
VARIANT_copy_value(&(_value)->_variant, ((VARIANT *)(_addr))); \
\
goto _label_ref; \
\
__CLASS: \
\
(_value)->_class.class = *((void **)(_addr)); \
(_value)->_class.super = NULL; \
goto _label_noref; \
\
__VOID: \
__FUNCTION: \
__NULL: \
THROW_ILLEGAL(); \
} \
})
#endif