mirror of
https://git.code.sf.net/p/quake/quakeforge
synced 2024-11-30 16:10:53 +00:00
8c01497936
When linking a def into a defspace, ensure it can be removed. This fixes the missing __obj_exec_class def in qwaq.
238 lines
5.2 KiB
C
238 lines
5.2 KiB
C
/*
|
|
def.c
|
|
|
|
def management and symbol tables
|
|
|
|
Copyright (C) 2002 Bill Currie <bill@taniwha.org>
|
|
Copyright (C) 1996-1997 Id Software, Inc.
|
|
|
|
Author: Bill Currie <bill@taniwha.org>
|
|
Date: 2002/06/09
|
|
|
|
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
|
|
of the License, 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:
|
|
|
|
Free Software Foundation, Inc.
|
|
59 Temple Place - Suite 330
|
|
Boston, MA 02111-1307, USA
|
|
|
|
*/
|
|
#ifdef HAVE_CONFIG_H
|
|
# include "config.h"
|
|
#endif
|
|
|
|
static __attribute__ ((used)) const char rcsid[] =
|
|
"$Id$";
|
|
|
|
#ifdef HAVE_STRING_H
|
|
# include <string.h>
|
|
#endif
|
|
#ifdef HAVE_STRINGS_H
|
|
# include <strings.h>
|
|
#endif
|
|
#include <stdlib.h>
|
|
|
|
#include <QF/hash.h>
|
|
#include <QF/sys.h>
|
|
#include <QF/va.h>
|
|
|
|
#include "qfcc.h"
|
|
#include "def.h"
|
|
#include "defspace.h"
|
|
#include "diagnostic.h"
|
|
#include "expr.h"
|
|
#include "immediate.h"
|
|
#include "options.h"
|
|
#include "reloc.h"
|
|
#include "strpool.h"
|
|
#include "struct.h"
|
|
#include "symtab.h"
|
|
#include "type.h"
|
|
|
|
static def_t *free_defs;
|
|
|
|
static void
|
|
set_storage_bits (def_t *def, storage_class_t storage)
|
|
{
|
|
switch (storage) {
|
|
case st_system:
|
|
def->system = 1;
|
|
// fall through
|
|
case st_global:
|
|
def->global = 1;
|
|
def->external = 0;
|
|
def->local = 0;
|
|
break;
|
|
case st_extern:
|
|
def->global = 1;
|
|
def->external = 1;
|
|
def->local = 0;
|
|
break;
|
|
case st_static:
|
|
def->external = 0;
|
|
def->global = 0;
|
|
def->local = 0;
|
|
break;
|
|
case st_local:
|
|
def->external = 0;
|
|
def->global = 0;
|
|
def->local = 1;
|
|
break;
|
|
}
|
|
def->initialized = 0;
|
|
}
|
|
|
|
def_t *
|
|
new_def (const char *name, type_t *type, defspace_t *space,
|
|
storage_class_t storage)
|
|
{
|
|
def_t *def;
|
|
|
|
ALLOC (16384, def_t, defs, def);
|
|
|
|
if (!space && storage != st_extern)
|
|
internal_error (0, "non-external def with no storage space");
|
|
if (!type)
|
|
internal_error (0, "attempt to create def '%s' with a null type",
|
|
name);
|
|
|
|
def->return_addr = __builtin_return_address (0);
|
|
|
|
def->name = name ? save_string (name) : 0;
|
|
def->type = type;
|
|
|
|
if (storage != st_extern) {
|
|
def->offset = defspace_new_loc (space, type_size (type));
|
|
}
|
|
if (space) {
|
|
def->space = space;
|
|
*space->def_tail = def;
|
|
space->def_tail = &def->next;
|
|
}
|
|
|
|
def->file = pr.source_file;
|
|
def->line = pr.source_line;
|
|
|
|
set_storage_bits (def, storage);
|
|
|
|
return def;
|
|
}
|
|
|
|
def_t *
|
|
alias_def (def_t *def, type_t *type)
|
|
{
|
|
def_t *alias;
|
|
|
|
ALLOC (16384, def_t, defs, alias);
|
|
alias->offset = def->offset;
|
|
alias->type = type;
|
|
alias->alias = def;
|
|
return alias;
|
|
}
|
|
|
|
void
|
|
free_def (def_t *def)
|
|
{
|
|
if (!def->alias && def->space) {
|
|
def_t **d;
|
|
|
|
for (d = &def->space->defs; *d && *d != def; d = &(*d)->next)
|
|
;
|
|
*d = def->next;
|
|
if (!def->external)
|
|
defspace_free_loc (def->space, def->offset, type_size (def->type));
|
|
}
|
|
def->next = free_defs;
|
|
free_defs = def;
|
|
}
|
|
|
|
void
|
|
def_to_ddef (def_t *def, ddef_t *ddef, int aux)
|
|
{
|
|
type_t *type = def->type;
|
|
|
|
if (aux)
|
|
type = type->t.fldptr.type; // aux is true only for fields
|
|
ddef->type = type->type;
|
|
ddef->ofs = def->offset;
|
|
ddef->s_name = ReuseString (def->name);
|
|
}
|
|
|
|
void
|
|
initialize_def (symbol_t *sym, type_t *type, expr_t *init, defspace_t *space,
|
|
storage_class_t storage)
|
|
{
|
|
if (sym->table == current_symtab) {
|
|
if (sym->sy_type != sy_var || sym->type != type) {
|
|
error (0, "%s redefined", sym->name);
|
|
sym = new_symbol (sym->name);
|
|
} else {
|
|
// is var and same type
|
|
if (!sym->s.def)
|
|
internal_error (0, "half defined var");
|
|
if (storage == st_extern) {
|
|
if (init)
|
|
warning (0, "initializing external variable");
|
|
return;
|
|
}
|
|
if (init && sym->s.def->initialized) {
|
|
error (0, "%s redefined", sym->name);
|
|
return;
|
|
}
|
|
}
|
|
} else if (sym->table) {
|
|
sym = new_symbol (sym->name);
|
|
}
|
|
sym->type = type;
|
|
symtab_addsymbol (current_symtab, sym);
|
|
if (storage == st_global && init) {
|
|
sym->sy_type = sy_const;
|
|
memset (&sym->s.value, 0, sizeof (&sym->s.value));
|
|
if (init->type != ex_value) { //FIXME arrays/structs
|
|
error (0, "non-constant initializier");
|
|
} else {
|
|
sym->s.value = init->e.value;
|
|
}
|
|
return;
|
|
}
|
|
if (sym->s.def && sym->s.def->external) {
|
|
free_def (sym->s.def);
|
|
sym->s.def = 0;
|
|
}
|
|
if (!sym->s.def)
|
|
sym->s.def = new_def (sym->name, type, space, storage);
|
|
if (storage == st_extern) {
|
|
if (init)
|
|
warning (0, "initializing external variable");
|
|
return;
|
|
}
|
|
if (!init)
|
|
return;
|
|
if (init->type == ex_nil)
|
|
convert_nil (init, type);
|
|
if (!type_assignable (type, get_type (init))) {
|
|
error (init, "type mismatch in initializer");
|
|
return;
|
|
}
|
|
if (local_expr) {
|
|
append_expr (local_expr, assign_expr (new_symbol_expr (sym), init));
|
|
} else {
|
|
if (init->type != ex_value) { //FIXME arrays/structs
|
|
error (0, "non-constant initializier");
|
|
return;
|
|
}
|
|
memcpy (D_POINTER (void, sym->s.def), &init->e.value,
|
|
type_size (type) * sizeof (pr_type_t));
|
|
}
|
|
}
|