/*************************************************************************** gbx_exec_push.c (c) 2000-2009 Benoît Minisini This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 675 Mass Ave, Cambridge, MA 02139, USA. ***************************************************************************/ #include "gb_common.h" #include "gb_limit.h" #include "gbx_exec.h" #include "gb_pcode.h" #include "gbx_api.h" #include "gbx_string.h" #include "gbx_c_array.h" #include "gbx_c_collection.h" #include "gbx_api.h" #include "gbx_struct.h" void EXEC_push_unknown(ushort code) { static void *jump[] = { /* 0 */ &&_PUSH_GENERIC, /* 1 */ &&_PUSH_CONSTANT, /* 2 */ &&_PUSH_VARIABLE, /* 3 */ &&_PUSH_STATIC_VARIABLE, /* 4 */ &&_PUSH_PROPERTY, /* 5 */ &&_PUSH_METHOD, /* 6 */ &&_PUSH_STATIC_METHOD, /* 7 */ &&_PUSH_VARIABLE_AUTO, /* 8 */ &&_PUSH_PROPERTY_AUTO, /* 9 */ &&_PUSH_METHOD_AUTO, /* 10 */ &&_PUSH_EXTERN, /* 11 */ &&_PUSH_STRUCT_FIELD }; const char *name; int index; CLASS_DESC *desc; CLASS *class; OBJECT *object; void *ref; char *addr; bool defined; VALUE *val; defined = EXEC_object(&SP[-1], &class, &object); goto *jump[code & 0xF]; _PUSH_GENERIC: name = CP->load->unknown[PC[1]]; // The first time we access a symbol, we must not be virtual to find it val = &SP[-1]; if (LIKELY(defined && object && !VALUE_is_super(val))) index = CLASS_find_symbol(val->_object.class, name); else index = CLASS_find_symbol(class, name); if (UNLIKELY(index == NO_SYMBOL)) { //index = CLASS_find_symbol(class, name); if (class->special[SPEC_UNKNOWN] == NO_SYMBOL) { if (defined && object && !VALUE_is_super(val)) class = val->_object.class; THROW(E_NSYMBOL, name, class->name); } goto _PUSH_UNKNOWN_METHOD; } desc = class->table[index].desc; switch (CLASS_DESC_get_type(desc)) { case CD_CONSTANT: if (LIKELY(defined)) { if ((PC[-1] & 0xF800) == C_PUSH_CLASS) { if (desc->constant.type == T_BOOLEAN) { PC[-1] = C_PUSH_MISC | (desc->constant.value._integer ? CPM_TRUE : CPM_FALSE); PC[0] = C_NOP; PC[1] = C_NOP; goto _PUSH_CONSTANT_2; } else if (desc->constant.type == T_BYTE || desc->constant.type == T_SHORT) { PC[-1] = C_PUSH_INTEGER; PC[0] = desc->constant.value._integer; PC[1] = (CODE_CONV << 8) | desc->constant.type; goto _PUSH_CONSTANT_2; } else if (desc->constant.type == T_INTEGER) { PC[-1] = C_PUSH_LONG; PC[0] = desc->constant.value._integer & 0xFFFF; PC[1] = desc->constant.value._integer >> 16; goto _PUSH_CONSTANT_2; } } *PC |= 1; PC[1] = index; } goto _PUSH_CONSTANT_2; case CD_VARIABLE: if (object == NULL) { if (UNLIKELY(!class->auto_create)) THROW(E_DYNAMIC, class->name, name); object = EXEC_auto_create(class, TRUE); *PC |= 7; } else { if (defined) *PC |= 2; } if (defined) PC[1] = index; goto _PUSH_VARIABLE_2; case CD_STRUCT_FIELD: if (object == NULL) THROW(E_DYNAMIC, class->name, name); if (defined) { *PC |= 11; PC[1] = index; } goto _PUSH_STRUCT_FIELD_2; case CD_STATIC_VARIABLE: if (UNLIKELY(object != NULL)) THROW(E_STATIC, class->name, name); if (defined) *PC |= 3; if (defined) PC[1] = index; goto _PUSH_STATIC_VARIABLE_2; case CD_PROPERTY: case CD_PROPERTY_READ: if (object == NULL) { if (UNLIKELY(!class->auto_create)) THROW(E_DYNAMIC, class->name, name); object = EXEC_auto_create(class, TRUE); *PC |= 8; } else { if (defined) *PC |= 4; } if (defined) PC[1] = index; goto _PUSH_PROPERTY_2; case CD_STATIC_PROPERTY: case CD_STATIC_PROPERTY_READ: if (UNLIKELY(object != NULL)) THROW(E_STATIC, class->name, name); if (defined) *PC |= 4; if (defined) PC[1] = index; goto _PUSH_PROPERTY_2; case CD_METHOD: if (object == NULL) { if (UNLIKELY(!class->auto_create)) THROW(E_DYNAMIC, class->name, name); object = EXEC_auto_create(class, TRUE); *PC |= 9; } else { if (defined) *PC |= 5; } if (defined) PC[1] = index; goto _PUSH_METHOD_2; case CD_STATIC_METHOD: if (object) { OBJECT_UNREF(object, "EXEC_push_unknown"); object = NULL; } if (defined) *PC |= 6; if (defined) PC[1] = index; goto _PUSH_METHOD_2; case CD_EXTERN: if (object) { OBJECT_UNREF(object, "EXEC_push_unknown"); object = NULL; } if (defined) *PC |= 10; if (defined) PC[1] = index; goto _PUSH_EXTERN_2; default: THROW(E_NSYMBOL, name, class->name); } _PUSH_CONSTANT: desc = class->table[PC[1]].desc; _PUSH_CONSTANT_2: VALUE_read(&SP[-1], (void *)&desc->constant.value, desc->constant.type); goto _FIN_DEFINED; _PUSH_VARIABLE_AUTO: object = EXEC_auto_create(class, TRUE); _PUSH_VARIABLE: desc = class->table[PC[1]].desc; _PUSH_VARIABLE_2: addr = (char *)object + desc->variable.offset; ref = object; goto _READ_VARIABLE; _PUSH_STATIC_VARIABLE: desc = class->table[PC[1]].desc; _PUSH_STATIC_VARIABLE_2: addr = (char *)desc->variable.class->stat + desc->variable.offset; ref = desc->variable.class; goto _READ_VARIABLE; _PUSH_STRUCT_FIELD: desc = class->table[PC[1]].desc; _PUSH_STRUCT_FIELD_2: if (((CSTRUCT *)object)->ref) addr = (char *)((CSTATICSTRUCT *)object)->addr + desc->variable.offset; else addr = (char *)object + sizeof(CSTRUCT) + desc->variable.offset; ref = object; goto _READ_VARIABLE; _READ_VARIABLE: VALUE_class_read(desc->variable.class, &SP[-1], (void *)addr, desc->variable.ctype, ref); goto _FIN_DEFINED; _PUSH_PROPERTY_AUTO: object = EXEC_auto_create(class, TRUE); _PUSH_PROPERTY: desc = class->table[PC[1]].desc; _PUSH_PROPERTY_2: if (desc->property.native) { if (EXEC_call_native(desc->property.read, object, desc->property.type, NULL)) PROPAGATE(); //SP[-1] = TEMP; VALUE_copy(&SP[-1], &TEMP); goto _FIN_DEFINED; } else { EXEC.class = desc->property.class; EXEC.object = object; EXEC.nparam = 0; EXEC.native = FALSE; EXEC.index = (int)(intptr_t)desc->property.read; EXEC_function_keep(); //SP[-1] = *RP; VALUE_copy(&SP[-1], RP); RP->type = T_VOID; goto _FIN_DEFINED_NO_BORROW; } _PUSH_STATIC_METHOD: if (object) { OBJECT_UNREF(object, "EXEC_push_unknown"); object = NULL; } goto _PUSH_METHOD; _PUSH_METHOD_AUTO: object = EXEC_auto_create(class, TRUE); _PUSH_METHOD: index = PC[1]; desc = class->table[index].desc; _PUSH_METHOD_2: //printf("PUSH_METHOD: %d %s\n", index, desc->method.name); SP--; SP->type = T_FUNCTION; SP->_function.class = class; SP->_function.object = object; /*SP->_function.function = (int)&desc->method;*/ if (FUNCTION_is_native(&desc->method)) SP->_function.kind = FUNCTION_NATIVE; else SP->_function.kind = FUNCTION_PUBLIC; SP->_function.index = index; SP->_function.defined = defined; SP++; goto _FIN; _PUSH_EXTERN: if (object) { OBJECT_UNREF(object, "EXEC_push_unknown"); object = NULL; } index = PC[1]; desc = class->table[index].desc; _PUSH_EXTERN_2: //printf("PUSH_METHOD: %d %s\n", index, desc->method.name); SP--; SP->type = T_FUNCTION; SP->_function.class = class; SP->_function.object = object; SP->_function.kind = FUNCTION_EXTERN; SP->_function.index = (int)desc->ext.exec; SP->_function.defined = defined; SP++; goto _FIN; _PUSH_UNKNOWN_METHOD: SP--; SP->type = T_FUNCTION; SP->_function.class = class; SP->_function.object = object; SP->_function.kind = FUNCTION_UNKNOWN; SP->_function.index = PC[1]; SP->_function.defined = FALSE; SP++; //OBJECT_REF(&object, "EXEC_push_unknown: UNKNOWN"); goto _FIN; _FIN_DEFINED: BORROW(&SP[-1]); _FIN_DEFINED_NO_BORROW: if (UNLIKELY(!defined)) VALUE_conv_variant(&SP[-1]); /* sp[-1] contenait l'objet et a ���ras� Il faut donc le d���encer nous-m�e. Sauf si c'est un appel de m�hode statique (cf. plus haut) */ OBJECT_UNREF(object, "EXEC_push_unknown"); _FIN: PC++; } void EXEC_push_array(ushort code) { static const void *jump[] = { &&__PUSH_GENERIC, &&__PUSH_QUICK_ARRAY, &&__PUSH_QUICK_COLLECTION, &&__PUSH_ARRAY }; CLASS *class; OBJECT *object; GET_NPARAM(np); //int dim[MAX_ARRAY_DIM]; int i; void *data; bool defined; VALUE *val; int fast; //ARRAY_DESC *desc; val = &SP[-np]; np--; goto *jump[((unsigned char)code) >> 6]; __PUSH_GENERIC: defined = EXEC_object(val, &class, &object); // The first time we access a symbol, we must not be virtual to find it if (LIKELY(defined && object && !VALUE_is_super(val))) class = val->_object.class; fast = 3; if (LIKELY(defined)) { if (class->quick_array == CQA_ARRAY) fast = 1; else if (class->quick_array == CQA_COLLECTION) fast = 2; } *PC |= fast << 6; goto __PUSH_ARRAY_2; /*__PUSH_STATIC_ARRAY: for (i = 1; i <= np; i++) { VALUE_conv_integer(&val[i]; dim[i - 1] = val[i]._integer.value; } SP = val; desc = (ARRAY_DESC *)SP->_array.class->load->array[SP->_array.index]; data = ARRAY_get_address(desc, SP->_array.addr, np, dim); VALUE_read(SP, data, CLASS_ctype_to_type(SP->_array.class, desc->type)); PUSH(); return;*/ __PUSH_QUICK_ARRAY: EXEC_object_fast(val, &class, &object); VALUE_conv_integer(&val[1]); if (LIKELY(np == 1)) { data = CARRAY_get_data((CARRAY *)object, val[1]._integer.value); } else { for (i = 2; i <= np; i++) VALUE_conv_integer(&val[i]); data = CARRAY_get_data_multi((CARRAY *)object, (GB_INTEGER *)&val[1], np); } if (UNLIKELY(data == NULL)) PROPAGATE(); VALUE_read(val, data, ((CARRAY *)object)->type); goto __PUSH_QUICK_END; __PUSH_QUICK_COLLECTION: EXEC_object_fast(val, &class, &object); VALUE_conv_string(&val[1]); //fprintf(stderr, "GB_CollectionGet: %p '%.*s'\n", val[1]._string.addr, val[1]._string.len, val[1]._string.addr + val[1]._string.start); GB_CollectionGet((GB_COLLECTION)object, val[1]._string.addr + val[1]._string.start, val[1]._string.len, (GB_VARIANT *)val); RELEASE_STRING(&val[1]); __PUSH_QUICK_END: SP = val; PUSH(); OBJECT_UNREF(object, "EXEC_push_array"); return; __PUSH_ARRAY: defined = EXEC_object(val, &class, &object); __PUSH_ARRAY_2: if (UNLIKELY(EXEC_special(SPEC_GET, class, object, np, FALSE))) THROW(E_NARRAY, class->name); OBJECT_UNREF(object, "EXEC_push_array"); SP--; //SP[-1] = SP[0]; VALUE_copy(&SP[-1], &SP[0]); if (UNLIKELY(!defined)) VALUE_conv_variant(&SP[-1]); }