/*************************************************************************** gbc_class.c (c) 2000-2017 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., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. ***************************************************************************/ #define __GBC_CLASS_C #include #include #include #include "gb_common.h" #include "gb_error.h" #include "gb_str.h" #include "gb_file.h" #include "gb_table.h" #include "gbc_compile.h" #include "gb_code.h" static int _array_class[17]; void CLASS_create(CLASS **result) { CLASS *class; TRANS_FUNC func; ALLOC_ZERO(&class, sizeof(CLASS)); ARRAY_create_inc(&class->function, 256); ARRAY_create(&class->event); ARRAY_create(&class->prop); ARRAY_create(&class->ext_func); ARRAY_create_inc(&class->constant, 256); ARRAY_create(&class->class); ARRAY_create_inc(&class->unknown, 256); ARRAY_create_inc(&class->stat, 256); ARRAY_create_inc(&class->dyn, 256); ARRAY_create(&class->array); ARRAY_create(&class->structure); ARRAY_create(&class->names); TABLE_create(&class->table, sizeof(CLASS_SYMBOL), TF_IGNORE_CASE); TABLE_create(&class->string, sizeof(SYMBOL), TF_NORMAL); CLEAR(&func); TYPE_clear(&func.type); func.index = CLASS_add_symbol(class, "@init"); CLASS_add_function(class, &func); func.index = CLASS_add_symbol(class, "@new"); CLASS_add_function(class, &func); class->name = STR_copy(FILE_set_ext(FILE_get_name(JOB->name), NULL)); class->parent = NO_SYMBOL; memset(_array_class, 0, sizeof(_array_class)); *result = class; } static void delete_function(FUNCTION *func) { ARRAY_delete(&func->local); ARRAY_delete(&func->stat); if (func->code) FREE(&func->code); if (JOB->debug) ARRAY_delete(&func->pos_line); if (func->param) FREE(&func->param); } static void delete_event(EVENT *event) { if (event->param) FREE(&event->param); } static void delete_extfunc(EXTFUNC *extfunc) { if (extfunc->param) FREE(&extfunc->param); } static void delete_structure(CLASS_STRUCT *structure) { if (structure->field) ARRAY_delete(&structure->field); } void CLASS_delete(CLASS **class) { int i; if (!*class) return; CLASS_set_current_function(NULL); TABLE_delete(&((*class)->table)); TABLE_delete(&((*class)->string)); for (i = 0; i < ARRAY_count((*class)->function); i++) delete_function(&((*class)->function[i])); for (i = 0; i < ARRAY_count((*class)->event); i++) delete_event(&((*class)->event[i])); for (i = 0; i < ARRAY_count((*class)->ext_func); i++) delete_extfunc(&((*class)->ext_func[i])); for (i = 0; i < ARRAY_count((*class)->structure); i++) delete_structure(&((*class)->structure[i])); for (i = 0; i < ARRAY_count((*class)->names); i++) FREE(&((*class)->names[i])); ARRAY_delete(&((*class)->function)); ARRAY_delete(&((*class)->event)); ARRAY_delete(&((*class)->prop)); ARRAY_delete(&((*class)->ext_func)); ARRAY_delete(&((*class)->constant)); ARRAY_delete(&((*class)->class)); ARRAY_delete(&((*class)->unknown)); ARRAY_delete(&((*class)->stat)); ARRAY_delete(&((*class)->dyn)); ARRAY_delete(&((*class)->array)); ARRAY_delete(&((*class)->structure)); ARRAY_delete(&((*class)->names)); if ((*class)->name) STR_free((*class)->name); if ((*class)->export_name) STR_free((*class)->export_name); FREE(class); } CLASS_SYMBOL *CLASS_declare(CLASS *class, int index, int type, bool global) { const char *name; CLASS_SYMBOL *sym; sym = CLASS_get_symbol(class, index); if ((global && !TYPE_is_null(sym->global.type)) || (!global && !TYPE_is_null(sym->local.type))) // || !TYPE_is_null(sym->global.type)))) { char name[sym->symbol.len + 1]; memcpy(name, sym->symbol.name, sym->symbol.len); name[sym->symbol.len] = 0; THROW("'&1' already declared", name); } if (type == TK_VARIABLE && sym->class && JOB->class->class[sym->class - 1].has_static) { name = SYMBOL_get_name(&sym->symbol); if (global) COMPILE_print(MSG_WARNING, -1, "class name hidden by global declaration: &1", name); else COMPILE_print(MSG_WARNING, -1, "class name hidden by local declaration: &1", name); } if (!global && !TYPE_is_null(sym->global.type)) { name = SYMBOL_get_name(&sym->symbol); switch (TYPE_get_kind(sym->global.type)) { case TK_VARIABLE: COMPILE_print(MSG_WARNING, -1, "global variable hidden by local declaration: &1", name); break; case TK_FUNCTION: COMPILE_print(MSG_WARNING, -1, "function hidden by local declaration: &1", name); break; case TK_EXTERN: COMPILE_print(MSG_WARNING, -1, "extern function hidden by local declaration: &1", name); break; case TK_CONST: COMPILE_print(MSG_WARNING, -1, "constant hidden by local declaration: &1", name); break; } } if (global) sym->global.line = JOB->line; else sym->local.line = JOB->line; return sym; } void CLASS_check_unused_global(CLASS *class) { CLASS_SYMBOL *sym; int i; TYPE type; for (i = 0; i < TABLE_count(class->table); i++) { sym = CLASS_get_symbol(class, i); type = sym->global.type; if (sym->global_no_warning) continue; if (sym->global_used && sym->global_assigned) continue; if (TYPE_is_null(type) || TYPE_is_public(type)) continue; if (!sym->global_used) { if (TYPE_get_kind(type) == TK_VARIABLE) COMPILE_print(MSG_WARNING, sym->global.line, "unused global variable: &1", SYMBOL_get_name(&sym->symbol)); else if (TYPE_get_kind(type) == TK_FUNCTION) COMPILE_print(MSG_WARNING, class->function[sym->global.value].line - 1, "unused function: &1", SYMBOL_get_name(&sym->symbol)); else if (TYPE_get_kind(type) == TK_EXTERN) COMPILE_print(MSG_WARNING, sym->global.line, "unused extern function: &1", SYMBOL_get_name(&sym->symbol)); } else { if (TYPE_get_kind(type) == TK_VARIABLE) COMPILE_print(MSG_WARNING, sym->global.line, "uninitialized global variable: &1", SYMBOL_get_name(&sym->symbol)); } } } int CLASS_add_function(CLASS *class, TRANS_FUNC *decl) { FUNCTION *func; int i; CLASS_SYMBOL *sym; PARAM *param; int count; count = ARRAY_count(class->function); if (count >= MAX_CLASS_FUNCTION) THROW("Too many functions"); func = ARRAY_add_void(&class->function); TYPE_clear(&func->type); func->nparam = 0; func->name = NO_SYMBOL; ARRAY_create(&func->local); ARRAY_create(&func->stat); //ARRAY_create_inc(&func->code, 512); func->code = NULL; func->ncode = 0; func->ncode_max = 0; if (JOB->debug) ARRAY_create(&func->pos_line); if (decl->index != NO_SYMBOL) { sym = CLASS_declare(class, decl->index, TK_FUNCTION, TRUE); sym->global.type = decl->type; sym->global.value = count; sym->global_no_warning = decl->no_warning; } if (TYPE_is_static(decl->type)) class->has_static = TRUE; func->nparam = decl->nparam; if (decl->nparam) { ALLOC(&func->param, decl->nparam * sizeof(PARAM)); for (i = 0; i < decl->nparam; i++) func->param[i] = decl->param[i]; } func->type = decl->type; func->start = decl->start; func->line = decl->line; func->name = decl->index; func->last_code = CODE_NO_POS; func->last_code2 = CODE_NO_POS; func->stack = 8; // Some stack may be needed for initialization functions func->finally = 0; func->catch = 0; func->npmin = -1; func->vararg = decl->vararg; func->fast = decl->fast; func->unsafe = decl->unsafe; // Function startup CODE_begin_function(func); JOB->func = func; // Byref check at function startup if (decl->byref) CODE_byref(decl->byref); // Optional parameters for (i = 0; i < func->nparam; i++) { param = &func->param[i]; if (param->optional == NULL) continue; if (func->npmin < 0) func->npmin = i; TRANS_init_optional(param); CODE_pop_optional(i - func->nparam); } if (func->npmin < 0) func->npmin = func->nparam; return count; } void CLASS_add_event(CLASS *class, TRANS_EVENT *decl) { EVENT *event; int i; CLASS_SYMBOL *sym; int count; count = ARRAY_count(class->event); if (count >= MAX_CLASS_EVENT) THROW("Too many events"); event = ARRAY_add_void(&class->event); TYPE_clear(&event->type); event->nparam = 0; event->name = NO_SYMBOL; if (!decl) return; sym = CLASS_declare(class, decl->index, TK_EVENT, TRUE); /*CLASS_add_symbol(class, JOB->table, decl->index, &sym, NULL);*/ sym->global.type = decl->type; sym->global.value = ARRAY_count(class->event) - 1; event->nparam = decl->nparam; if (event->nparam) { ALLOC_ZERO(&event->param, decl->nparam * sizeof(PARAM)); for (i = 0; i < decl->nparam; i++) { event->param[i].type = decl->param[i].type; event->param[i].index = decl->param[i].index; } } event->type = decl->type; event->name = decl->index; } void CLASS_add_property(CLASS *class, TRANS_PROPERTY *decl) { PROPERTY *prop; CLASS_SYMBOL *sym; int index; int i; index = ARRAY_count(class->prop); prop = ARRAY_add_void(&class->prop); TYPE_clear(&prop->type); prop->name = NO_SYMBOL; prop->read = TRUE; prop->write = decl->read == 0; prop->synonymous = -1; sym = CLASS_declare(class, decl->index, TK_PROPERTY, TRUE); /*CLASS_add_symbol(class, JOB->table, decl->index, &sym, NULL);*/ sym->global.type = decl->type; sym->global.value = ARRAY_count(class->prop) - 1; if (TYPE_is_static(decl->type)) class->has_static = TRUE; prop->type = decl->type; prop->name = decl->index; prop->line = decl->line; prop->comment = decl->comment; prop->use = decl->use; for (i = 0; i < decl->nsynonymous; i++) { prop = ARRAY_add_void(&class->prop); TYPE_clear(&prop->type); prop->name = NO_SYMBOL; prop->read = TRUE; prop->write = decl->read == 0; prop->synonymous = index; sym = CLASS_declare(class, decl->synonymous[i], TK_PROPERTY, TRUE); /*CLASS_add_symbol(class, JOB->table, decl->index, &sym, NULL);*/ sym->global.type = decl->type; sym->global.value = ARRAY_count(class->prop) - 1; prop->type = decl->type; prop->name = decl->synonymous[i]; prop->line = decl->line; prop->comment = decl->comment; } } void CLASS_add_extern(CLASS *class, TRANS_EXTERN *decl) { EXTFUNC *extfunc; int i; CLASS_SYMBOL *sym; int count; count = ARRAY_count(class->ext_func); if (count >= MAX_CLASS_EXTERN) THROW("Too many external functions"); extfunc = ARRAY_add_void(&class->ext_func); TYPE_clear(&extfunc->type); extfunc->nparam = 0; extfunc->vararg = FALSE; extfunc->name = NO_SYMBOL; if (!decl) return; sym = CLASS_declare(class, decl->index, TK_EXTERN, TRUE); /*CLASS_add_symbol(class, JOB->table, decl->index, &sym, NULL);*/ sym->global.type = decl->type; sym->global.value = ARRAY_count(class->ext_func) - 1; extfunc->nparam = decl->nparam; extfunc->vararg = decl->vararg; if (extfunc->nparam) { ALLOC(&extfunc->param, decl->nparam * sizeof(PARAM)); for (i = 0; i < decl->nparam; i++) { extfunc->param[i].type = decl->param[i].type; extfunc->param[i].index = decl->param[i].index; } } extfunc->type = decl->type; extfunc->name = decl->index; extfunc->library = decl->library; extfunc->alias = decl->alias; } int CLASS_add_constant(CLASS *class, TRANS_DECL *decl) { CONSTANT *desc; int num; num = ARRAY_count(class->constant); if (num >= MAX_CLASS_CONST) THROW("Too many constants"); desc = ARRAY_add(&class->constant); desc->type = decl->type; desc->index = decl->index; desc->value = decl->value; if (TYPE_get_id(decl->type) == T_LONG) desc->lvalue = decl->lvalue; desc->is_integer = decl->is_integer; desc->line = JOB->line; return num; } static int add_class(CLASS *class, int index, bool used, bool exported) { int num; CLASS_REF *desc; CLASS_SYMBOL *sym = CLASS_get_symbol(class, index); num = sym->class - 1; if (num < 0) { num = ARRAY_count(class->class); if (num >= MAX_CLASS_CLASS) THROW("Too many different classes used"); desc = ARRAY_add_void(&class->class); desc->index = index; sym->class = num + 1; if (COMP_verbose) fprintf(stderr, "Adding class %.*s %s%s\n", sym->symbol.len, sym->symbol.name, used ? "" : "Unused ", exported ? "Exported" : ""); JOB->class->class[num].exported = exported; } if (used != JOB->class->class[num].used) { if (COMP_verbose) fprintf(stderr, "Switching class %.*s to %s\n", sym->symbol.len, sym->symbol.name, used ? "Used" : "Unused"); JOB->class->class[num].used = used; } return num; } int CLASS_add_class(CLASS *class, int index) { return add_class(class, index, TRUE, FALSE); } int CLASS_add_class_unused(CLASS *class, int index) { return add_class(class, index, FALSE, FALSE); } int CLASS_add_class_exported(CLASS *class, int index) { return add_class(class, index, TRUE, TRUE); } int CLASS_add_class_exported_unused(CLASS *class, int index) { return add_class(class, index, FALSE, TRUE); } bool CLASS_exist_class(CLASS *class, int index) { return CLASS_get_symbol(class, index)->class > 0; } static char *CLASS_add_name(CLASS *class, const char *name, int len) { char *ret; ALLOC(&ret, len + 1); memcpy(ret, name, len); ret[len] = 0; *((char **)ARRAY_add(&class->names)) = ret; return ret; } int CLASS_get_array_class(CLASS *class, int type, int value) { static char *names[] = { NULL, "Boolean[]", "Byte[]", "Short[]", "Integer[]", "Long[]", "Single[]", "Float[]", "Date[]", "String[]", "String[]", "Pointer[]", "Variant[]", NULL, NULL, NULL, "Object[]" }; int index; //CLASS_REF *cref; if (value < 0) { if (type <= T_VOID || type > T_OBJECT) ERROR_panic("Bad native array class"); index = _array_class[type]; if (index == 0) { if (!TABLE_find_symbol(class->table, names[type], strlen(names[type]), &index)) index = CLASS_add_symbol(class, names[type]); index = CLASS_add_class_exported(class, index); _array_class[type] = index; } } else { CLASS_SYMBOL *sym = CLASS_get_symbol(class, class->class[value].index); int len = sym->symbol.len; char name[len + 2]; memcpy(name, sym->symbol.name, len); memcpy(&name[len], "[]", 2); if (!TABLE_find_symbol(class->table, name, len + 2, &index)) { char *name_alloc = CLASS_add_name(class, name, len + 2); index = TABLE_add_symbol(class->table, name_alloc, len + 2); } if (class->class[value].exported) index = CLASS_add_class_exported(JOB->class, index); else index = CLASS_add_class(JOB->class, index); } JOB->class->class[index].type = TYPE_make(type, value, 0); /*cref = &class->class[index]; if (TYPE_is_null(cref->array)) { cref->array.t.id = type; cref->array.t.value = value; }*/ return index; } int CLASS_add_unknown(CLASS *class, int index) { int num; int *desc; CLASS_SYMBOL *sym = CLASS_get_symbol(class, index); num = sym->unknown - 1; if (num < 0) { num = ARRAY_count(class->unknown); if (num >= MAX_CLASS_UNKNOWN) THROW("Too many unknown symbols"); desc = ARRAY_add(&class->unknown); *desc = index; sym->unknown = num + 1; } return num; } int CLASS_add_array(CLASS *class, TRANS_ARRAY *array) { CLASS_ARRAY *desc; int num; int i; num = ARRAY_count(class->array); if (num >= MAX_CLASS_ARRAY) THROW("Too many array declarations"); desc = ARRAY_add(&class->array); desc->type = array->type; desc->ndim = array->ndim; for (i = 0; i < desc->ndim; i++) desc->dim[i] = array->dim[i]; return num; } FUNCTION *CLASS_set_current_function(FUNCTION *func) { JOB->func = func; return CODE_set_function(func); } void CLASS_add_static_declaration(CLASS *class, int index, TYPE type, CLASS_SYMBOL *sym, bool local) { VARIABLE *var; int count = ARRAY_count(class->stat); if (count >= MAX_CLASS_SYMBOL) THROW("Too many static variables"); if (local) sym->local.value = count; else sym->global.value = count; var = ARRAY_add(&class->stat); var->type = type; var->index = index; class->has_static = TRUE; } void CLASS_init_global_declaration(CLASS *class, TRANS_DECL *decl, CLASS_SYMBOL *sym, bool local) { FUNCTION *prev_func; bool is_static; if (!TRANS_has_init_var(decl)) return; is_static = TYPE_is_static(decl->type); prev_func = CLASS_set_current_init_function(class, is_static ? FUNC_INIT_STATIC : FUNC_INIT_DYNAMIC); FUNCTION_add_all_pos_line(); TRANS_init_var(decl); if (local) { CODE_pop_global(sym->local.value, is_static); sym->local_assigned = TRUE; } else { CODE_pop_global(sym->global.value, is_static); sym->global_assigned = TRUE; } CLASS_set_current_function(prev_func); } void CLASS_add_declaration(CLASS *class, TRANS_DECL *decl) { CLASS_SYMBOL *sym; VARIABLE *var; int count; bool save_warnings = FALSE; if (decl->no_warning) { save_warnings = JOB->warnings; JOB->warnings = FALSE; } sym = CLASS_declare(class, decl->index, TYPE_get_kind(decl->type), TRUE); if (decl->no_warning) { JOB->warnings = save_warnings; } sym->global.type = decl->type; if (TYPE_get_kind(decl->type) == TK_CONST) { sym->global.value = CLASS_add_constant(class, decl); class->has_static = TRUE; } else if (TYPE_is_static(decl->type)) { CLASS_add_static_declaration(class, decl->index, decl->type, sym, FALSE); CLASS_init_global_declaration(class, decl, sym, FALSE); } else { count = ARRAY_count(class->dyn); if (count >= MAX_CLASS_SYMBOL) THROW("Too many dynamic variables"); sym->global.value = count; var = ARRAY_add(&class->dyn); var->type = decl->type; var->index = decl->index; CLASS_init_global_declaration(class, decl, sym, FALSE); /*CLASS_begin_init_function(class, FUNC_INIT_DYNAMIC); if (TRANS_has_init_var(decl)) { FUNCTION_add_all_pos_line(); TRANS_init_var(decl); CODE_pop_global(sym->global.value, FALSE); sym->global_assigned = TRUE; }*/ } CLASS_check_variable_prefix(sym, FALSE); } // Don't do that! The order of variables must be kept. void CLASS_sort_declaration(CLASS *class) { //reorder_decl(class, class->stat, "static"); //reorder_decl(class, class->dyn, "dynamic"); } int CLASS_add_symbol(CLASS *class, const char *name) { int index; index = TABLE_add_symbol(class->table, name, strlen(name)); return index; } void FUNCTION_add_last_pos_line(void) { int current_pos; if (!JOB->debug || JOB->nobreak) return; current_pos = CODE_get_current_pos(); *ARRAY_add(&JOB->func->pos_line) = current_pos; } void FUNCTION_add_all_pos_line(void) { int line; int current_pos; if (!JOB->debug || JOB->nobreak) return; line = JOB->func->line + ARRAY_count(JOB->func->pos_line) - 1; current_pos = CODE_get_current_pos(); //fprintf(stderr, "FUNCTION_add_all_pos_line: line = %d JOB->line = %d\n", line, JOB->line); while (line < JOB->line) { *ARRAY_add(&JOB->func->pos_line) = current_pos; line++; } } char *FUNCTION_get_fake_name(int func) { static char buf[6]; snprintf(buf, sizeof(buf), "$%d", func); return buf; } static int check_one_property_func(CLASS *class, PROPERTY *prop, bool write) { CLASS_SYMBOL *sym; char *name; //char *name_alloc; bool is_static; FUNCTION *func; int index; int value = 0; JOB->line = prop->line; JOB->current = NULL; is_static = TYPE_is_static(prop->type); name = STR_copy(TABLE_get_symbol_name_suffix(class->table, prop->name, write ? "_Write" : "_Read")); if (!TABLE_find_symbol(class->table, name, strlen(name), &index)) { TRANS_FUNC decl; if (prop->use == 0) THROW("&1 is not declared", name); CLEAR(&decl); //name_alloc = CLASS_add_name(class, name, strlen(name)); decl.index = index = NO_SYMBOL; //CLASS_add_symbol(class, name_alloc); if (write) { decl.nparam = 1; decl.param[0].type = prop->type; decl.param[0].index = NO_SYMBOL; } else { decl.type = prop->type; TYPE_clear_flag(&decl.type, TF_PUBLIC); } decl.line = prop->line; TYPE_set_kind(&decl.type, TK_FUNCTION); if (is_static) TYPE_set_flag(&decl.type, TF_STATIC); value = CLASS_add_function(JOB->class, &decl); JOB->nobreak = TRUE; sym = CLASS_get_symbol(JOB->class, prop->use); if (write) { sym->global_assigned = TRUE; CODE_push_local(-1); CODE_pop_global(sym->global.value, is_static); CODE_return(2); } else { sym->global_used = TRUE; CODE_push_global(sym->global.value, is_static, FALSE); CODE_return(1); } FUNCTION_add_last_pos_line(); JOB->func->stack = CODE_stack_usage; JOB->nobreak = FALSE; } if (index != NO_SYMBOL) { sym = (CLASS_SYMBOL *)TABLE_get_symbol(class->table, index); if (TYPE_get_kind(sym->global.type) != TK_FUNCTION) THROW("&1 is declared but is not a function", name); func = &class->function[sym->global.value]; JOB->line = func->line; if (TYPE_is_public(sym->global.type)) THROW("A property implementation cannot be public"); if (is_static != TYPE_is_static(sym->global.type)) { if (is_static) THROW("&1 must be static", name); else THROW("&1 cannot be static", name); } if (write) { if (TYPE_get_id(func->type) != T_VOID) goto _BAD_SIGNATURE; if (func->nparam != 1 || func->npmin != 1) goto _BAD_SIGNATURE; if (!TYPE_compare(&func->param[0].type, &prop->type)) goto _BAD_SIGNATURE; } else { if (!TYPE_compare(&func->type, &prop->type)) goto _BAD_SIGNATURE; if (func->nparam != 0 || func->npmin != 0) goto _BAD_SIGNATURE; } sym->global_used = TRUE; value = sym->global.value; } STR_free(name); return value; _BAD_SIGNATURE: THROW("&1 declaration does not match", name); } void CLASS_check_properties(CLASS *class) { int i; PROPERTY *prop; for (i = 0; i < ARRAY_count(class->prop); i++) { prop = &class->prop[i]; if (prop->synonymous >= 0) { prop->read = class->prop[prop->synonymous].read; prop->write = class->prop[prop->synonymous].write; } else { prop->read = check_one_property_func(class, prop, FALSE); if (prop->write) prop->write = check_one_property_func(class, prop, TRUE); else prop->write = NO_SYMBOL; } } } void CLASS_check_variable_prefix(CLASS_SYMBOL *sym, bool local) { char *name = sym->symbol.name; int len = sym->symbol.len; int len_prefix; char *p; TYPE type; if (!JOB->check_prefix) return; type = local ? sym->local.type : sym->global.type; if (TYPE_is_null(type)) return; if (len >= 1 && *name == '$') { name++; len--; } if (len <= 1) return; len_prefix = 0; p = name; while (len > 0 && islower(*p)) { p++; len--; len_prefix++; } if (TYPE_check_prefix(type, name, len_prefix)) COMPILE_print(MSG_WARNING, -1, "variable prefix does not match its datatype: &1", SYMBOL_get_name(&sym->symbol)); } char *CLASS_get_export_name(CLASS *class) { if (!class->exported) return NULL; if (class->export_name) return class->export_name; return class->name; }