quakeforge/tools/qfcc/source/dump_globals.c
Bill Currie 7970525ef4 [util] Make va thread-safe
It now takes a context pointer (opaque data) that holds the buffers it
uses for the temporary strings. If the context pointer is null, a static
context is used (making those uses of va NOT thread-safe). Most calls to
va use the static context, but all such calls have been formatted
consistently so they are easy to find when it comes time to do a full
audit.
2021-01-31 16:05:48 +09:00

639 lines
17 KiB
C

/*
dump_globals.c
Dump progs globals information.
Copyright (C) 2002 Bill Currie <bill@taniwha.org>
Author: Bill Currie <bill@taniwha.org>
Date: 2002/05/16
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
#include <stdlib.h>
#ifdef HAVE_STRING_H
# include <string.h>
#endif
#ifdef HAVE_STRINGS_H
# include <strings.h>
#endif
#include "QF/dstring.h"
#include "QF/progs.h"
#include "QF/va.h"
#include "tools/qfcc/include/obj_file.h"
#include "tools/qfcc/include/obj_type.h"
#include "tools/qfcc/include/qfprogs.h"
#include "tools/qfcc/include/reloc.h"
#include "tools/qfcc/include/strpool.h"
static int
cmp (const void *_a, const void *_b)
{
const pr_def_t *a = (const pr_def_t *)_a;
const pr_def_t *b = (const pr_def_t *)_b;
return a->ofs - b->ofs;
}
static void
dump_def (progs_t *pr, pr_def_t *def, int indent)
{
const char *name;
const char *type;
pr_uint_t offset;
const char *comment;
string_t string;
const char *str;
int saveglobal;
if (!def->type && !def->ofs && !def->name)
return;
name = PR_GetString (pr, def->name);
type = pr_type_name[def->type & ~DEF_SAVEGLOBAL];
saveglobal = (def->type & DEF_SAVEGLOBAL) != 0;
offset = def->ofs;
comment = " invalid offset";
if (offset < pr->progs->numglobals) {
comment = "";
switch (def->type & ~DEF_SAVEGLOBAL) {
case ev_void:
break;
case ev_string:
string = G_INT (pr, offset);
// at runtime, strings can be negative (thus string_t is
// signed), but negative strings means they have been
// dynamically allocated, thus a negative string index should
// never appear in compiled code
if (string < 0
|| (pr_uint_t) string >= pr->progs->numstrings) {
str = "invalid string offset";
comment = va (0, " %d %s", string, str);
} else {
str = quote_string (pr->pr_strings + G_INT (pr, offset));
comment = va (0, " %d \"%s\"", string, str);
}
break;
case ev_float:
comment = va (0, " %g", G_FLOAT (pr, offset));
break;
case ev_double:
comment = va (0, " %.17g", G_DOUBLE (pr, offset));
break;
case ev_vector:
comment = va (0, " '%g %g %g'",
G_VECTOR (pr, offset)[0],
G_VECTOR (pr, offset)[1],
G_VECTOR (pr, offset)[2]);
break;
case ev_entity:
break;
case ev_field:
comment = va (0, " %x", G_INT (pr, offset));
break;
case ev_func:
{
func_t func = G_FUNCTION (pr, offset);
int start;
if (func < pr->progs->numfunctions) {
start = pr->pr_functions[func].first_statement;
if (start > 0)
comment = va (0, " %d @ %x", func, start);
else
comment = va (0, " %d = #%d", func, -start);
} else {
comment = va (0, " %d = illegal function", func);
}
}
break;
case ev_pointer:
comment = va (0, " %x", G_INT (pr, offset));
break;
case ev_quat:
comment = va (0, " '%g %g %g %g'",
G_QUAT (pr, offset)[0],
G_QUAT (pr, offset)[1],
G_QUAT (pr, offset)[2],
G_QUAT (pr, offset)[3]);
break;
case ev_integer:
comment = va (0, " %d", G_INT (pr, offset));
break;
case ev_short:
break;
case ev_invalid:
comment = " struct?";
break;
case ev_type_count:
break;
}
}
printf ("%*s %x:%d %d %s %s:%x %s\n", indent * 12, "",
offset, def->size, saveglobal, name, type, def->type_encoding,
comment);
}
void
dump_globals (progs_t *pr)
{
unsigned int i;
pr_def_t *global_defs = pr->pr_globaldefs;
if (sorted) {
global_defs = malloc (pr->progs->numglobaldefs * sizeof (ddef_t));
memcpy (global_defs, pr->pr_globaldefs,
pr->progs->numglobaldefs * sizeof (ddef_t));
qsort (global_defs, pr->progs->numglobaldefs, sizeof (ddef_t), cmp);
}
for (i = 0; i < pr->progs->numglobaldefs; i++) {
pr_def_t *def = &global_defs[i];
dump_def (pr, def, 0);
}
}
void
dump_fields (progs_t *pr)
{
unsigned int i;
const char *name;
const char *type;
int offset;
const char *comment;
for (i = 0; i < pr->progs->numfielddefs; i++) {
pr_def_t *def = &pr->pr_fielddefs[i];
name = PR_GetString (pr, def->name);
type = pr_type_name[def->type & ~DEF_SAVEGLOBAL];
offset = def->ofs;
comment = "";
printf ("%d %s %s%s\n", offset, name, type, comment);
}
}
void
qfo_fields (qfo_t *qfo)
{
unsigned int i;
const char *name;
const char *typestr;
qfot_type_t *type;
int offset;
const char *comment;
qfo_mspace_t *space = &qfo->spaces[qfo_entity_space];
if (qfo_entity_space >= qfo->num_spaces) {
printf ("no entity space\n");
return;
}
if (!space->num_defs) {
printf ("no fields\n");
return;
}
for (i = 0; i < space->num_defs; i++) {
qfo_def_t *def = space->defs + i;
name = QFO_GETSTR (qfo, def->name);
//FIXME check type
type = QFO_POINTER (qfo, qfo_type_space, qfot_type_t, def->type);
typestr = QFO_GETSTR (qfo, type->encoding);
offset = def->offset;
comment = "";
printf ("%d %s %s%s\n", offset, name, typestr, comment);
}
}
void
dump_functions (progs_t *pr)
{
pr_uint_t i, j, count;
const char *name;
int start;
const char *comment;
pr_def_t *encodings_def;
pointer_t type_encodings = 0;
encodings_def = PR_FindGlobal (pr, ".type_encodings");
if (encodings_def) {
type_encodings = encodings_def->ofs;
}
for (i = 0; i < pr->progs->numfunctions; i++) {
dfunction_t *func = &pr->pr_functions[i];
name = PR_GetString (pr, func->s_name);
start = func->first_statement;
if (start > 0)
comment = va (0, " @ %x", start);
else
comment = va (0, " = #%d", -start);
printf ("%-5d %s%s: %d (", i, name, comment, func->numparms);
if (func->numparms < 0)
count = -func->numparms - 1;
else
count = func->numparms;
for (j = 0; j < count; j++)
printf (" %d:%d", func->parm_size[j].alignment,
func->parm_size[j].size);
printf (") %d @ %x", func->locals, func->parm_start);
puts ("");
if (type_encodings) {
pr_auxfunction_t *aux = PR_Debug_MappedAuxFunction (pr, i);
if (!aux) {
continue;
}
printf (" %d %s:%d %d %d %d %x\n", aux->function,
PR_GetString (pr, func->s_file), aux->source_line,
aux->line_info,
aux->local_defs, aux->num_locals,
aux->return_type);
pr_def_t *local_defs = PR_Debug_LocalDefs (pr, aux);
if (!local_defs) {
continue;
}
for (j = 0; j < aux->num_locals; j++) {
dump_def (pr, local_defs + j, 1);
}
}
}
}
static const char *
flags_string (pr_uint_t flags)
{
static dstring_t *str;
if (!str)
str = dstring_newstr ();
dstring_clearstr (str);
dstring_appendstr (str, (flags & QFOD_INITIALIZED) ? "I" : "-");
dstring_appendstr (str, (flags & QFOD_CONSTANT) ? "C" : "-");
dstring_appendstr (str, (flags & QFOD_ABSOLUTE) ? "A" : "-");
dstring_appendstr (str, (flags & QFOD_GLOBAL) ? "G" : "-");
dstring_appendstr (str, (flags & QFOD_EXTERNAL) ? "E" : "-");
dstring_appendstr (str, (flags & QFOD_LOCAL) ? "L" : "-");
dstring_appendstr (str, (flags & QFOD_SYSTEM) ? "S" : "-");
dstring_appendstr (str, (flags & QFOD_NOSAVE) ? "N" : "-");
dstring_appendstr (str, (flags & QFOD_PARAM) ? "P" : "-");
return str->str;
}
void
qfo_globals (qfo_t *qfo)
{
qfo_def_t *def;
unsigned i;
unsigned space;
int count = 0;
for (space = 0; space < qfo->num_spaces; space++) {
for (i = 0; i < qfo->spaces[space].num_defs; i++, count++) {
def = &qfo->spaces[space].defs[i];
printf ("%-5d %2d:%-5x %s %s %x %s", count, space, def->offset,
flags_string (def->flags),
QFO_GETSTR (qfo, def->name),
def->type,
QFO_TYPESTR (qfo, def->type));
if (!(def->flags & QFOD_EXTERNAL) && qfo->spaces[space].d.data)
printf (" %d",
qfo->spaces[space].d.data[def->offset].integer_var);
puts ("");
}
}
}
void
qfo_relocs (qfo_t *qfo)
{
qfo_reloc_t *reloc;
qfo_def_t *def;
qfo_func_t *func;
int opind;
dstatement_t *statement;
unsigned i;
for (i = 0; i < qfo->num_relocs; i++) {
if (i == qfo->num_relocs - qfo->num_loose_relocs) {
printf ("---- unbound relocs ----\n");
}
reloc = qfo->relocs + i;
if ((unsigned) reloc->type > rel_def_field_ofs) {
printf ("%d unknown reloc: %d\n", i, reloc->type);
continue;
}
printf ("%s", reloc_names[reloc->type]);
def = 0;
func = 0;
switch ((reloc_type) reloc->type) {
case rel_none:
break;
case rel_op_a_def:
case rel_op_b_def:
case rel_op_c_def:
def = qfo->defs + reloc->target;
printf (" op.%c@%x def#%d %s",
reloc->type - rel_op_a_def + 'a',
reloc->offset, reloc->target,
QFO_GETSTR (qfo, def->name));
break;
case rel_op_a_op:
case rel_op_b_op:
case rel_op_c_op:
printf (" op.%c op@%x", reloc->type - rel_op_a_def + 'a',
reloc->offset);
break;
case rel_def_op:
printf (" def@%d:%x op@%d", reloc->space, reloc->offset,
reloc->target);
break;
case rel_def_def:
def = qfo->defs + reloc->target;
printf (" def@%d:%x def#%d %s", reloc->space, reloc->offset,
reloc->target, QFO_GETSTR (qfo, def->name));
break;
case rel_def_func:
func = qfo->funcs + reloc->target;
//func = qfo->funcs + QFO_FUNCTION (qfo, reloc->space,
// reloc->offset);
printf (" def@%d:%x func#%d %s %x",
reloc->space, reloc->offset,
QFO_FUNCTION (qfo, reloc->space, reloc->offset),
QFO_GETSTR (qfo, func->name),
reloc->target);
break;
case rel_def_string:
printf (" def@%d:%x string:%x `%s'",
reloc->space, reloc->offset, reloc->target,
quote_string (QFO_GETSTR (qfo, reloc->target)));
break;
case rel_def_field:
def = qfo->defs + reloc->target;
printf (" def@%d:%x def#%d %s", reloc->space, reloc->offset,
reloc->target, QFO_GETSTR (qfo, def->name));
break;
case rel_op_a_def_ofs:
case rel_op_b_def_ofs:
case rel_op_c_def_ofs:
def = qfo->defs + reloc->target;
opind = reloc->type - rel_op_a_def_ofs;
statement = QFO_STATEMENT (qfo, reloc->offset);
printf (" op.%c@%x def#%d %s+%d",
opind + 'a',
reloc->offset, reloc->target,
QFO_GETSTR (qfo, def->name),
((pr_ushort_t *)statement)[opind + 1]);
break;
case rel_def_def_ofs:
def = qfo->defs + reloc->target;
printf (" def@%d:%x def#%d+%d %s+%d",
reloc->space, reloc->offset, reloc->target,
QFO_INT (qfo, reloc->space, reloc->offset),
QFO_GETSTR (qfo, def->name),
QFO_INT (qfo, reloc->space, reloc->offset));
break;
case rel_def_field_ofs:
def = qfo->defs + reloc->target;
printf (" def@%d:%x def#%d+%d %s(%d)+%d",
reloc->space, reloc->offset, reloc->target,
QFO_INT (qfo, reloc->space, reloc->offset),
QFO_GETSTR (qfo, def->name), def->offset,
QFO_INT (qfo, reloc->space, reloc->offset));
break;
}
if (def && def->flags & QFOD_EXTERNAL)
printf (" external");
if (func && qfo->defs[func->def].flags & QFOD_EXTERNAL)
printf (" external");
if (def && (i < def->relocs || i >= def->relocs + def->num_relocs))
printf (" BOGUS def reloc! %d %d %d",
i, def->relocs, def->num_relocs);
if (func && (i < func->relocs || i >= func->relocs + func->num_relocs))
printf (" BOGUS func reloc! %d %d %d",
i, func->relocs, func->num_relocs);
puts ("");
}
}
void
qfo_functions (qfo_t *qfo)
{
qfo_def_t *def;
qfo_func_t *func;
unsigned i, j, d;
unsigned space;
qfo_mspace_t *locals;
for (i = 0; i < qfo->num_funcs; i++) {
func = &qfo->funcs[i];
def = &qfo->defs[func->def];
for (space = 0; space < qfo->num_spaces; space++) {
if (!qfo->spaces[space].num_defs)
continue;
d = qfo->spaces[space].defs - qfo->defs;
if (func->def >= d && func->def < d + qfo->spaces[space].num_defs)
break;
}
if (space == qfo->num_spaces)
space = qfo_near_data_space;
printf ("%-5d %-5d %s %s %d %s", i, def->offset,
flags_string (def->flags),
QFO_GETSTR (qfo, func->name), func->def,
QFO_GETSTR (qfo, def->name));
if (!(def->flags & QFOD_EXTERNAL))
printf (" %d", QFO_FUNCTION (qfo, space, def->offset));
if (func->code > 0)
printf (" @ %x", func->code);
else
printf (" = #%d", -func->code);
printf (" loc: %d\n", func->locals_space);
if (func->locals_space) {
locals = &qfo->spaces[func->locals_space];
printf ("%*s%d %p %d %p %d %d\n", 16, "", locals->type,
locals->defs, locals->num_defs,
locals->d.data, locals->data_size, locals->id);
for (j = 0; j < locals->num_defs; j++) {
qfo_def_t *def = locals->defs + j;
int offset;
const char *typestr;
const char *name;
qfot_type_t *type;
name = QFO_GETSTR (qfo, def->name);
//FIXME check type
type = QFO_POINTER (qfo, qfo_type_space, qfot_type_t,
def->type);
typestr = QFO_GETSTR (qfo, type->encoding);
offset = def->offset;
printf ("%*s%d %s %s\n", 20, "", offset, name, typestr);
}
}
}
}
static const char *ty_meta_names[] = {
"ty_basic",
"ty_struct",
"ty_union",
"ty_enum",
"ty_array",
"ty_class",
"ty_alias",
};
#define NUM_META ((int)(sizeof (ty_meta_names) / sizeof (ty_meta_names[0])))
static void
dump_qfo_types (qfo_t *qfo, int base_address)
{
pointer_t type_ptr;
qfot_type_t *type;
const char *meta;
int i, count;
for (type_ptr = 4; type_ptr < qfo->spaces[qfo_type_space].data_size;
type_ptr += type->size) {
type = &QFO_STRUCT (qfo, qfo_type_space, qfot_type_t, type_ptr);
if (qfo->spaces[qfo_type_space].data_size - type_ptr < 2) {
printf ("%-5x overflow, can't check size. %x\n",
type_ptr + base_address,
qfo->spaces[qfo_type_space].data_size);
}
if (type_ptr + type->size > qfo->spaces[qfo_type_space].data_size) {
printf ("%-5x overflow by %d words. %x\n", type_ptr + base_address,
(type_ptr + type->size
- qfo->spaces[qfo_type_space].data_size),
qfo->spaces[qfo_type_space].data_size);
continue;
}
if (type->meta < 0 || type->meta >= NUM_META)
meta = va (0, "invalid meta: %d", type->meta);
else
meta = ty_meta_names[type->meta];
printf ("%-5x %-9s %-20s", type_ptr + base_address, meta,
QFO_TYPESTR (qfo, type_ptr));
if (type->meta < 0 || type->meta >= NUM_META) {
puts ("");
break;
}
switch ((ty_meta_e) type->meta) {
case ty_basic:
printf (" %-10s", (type->type < 0
|| type->type >= ev_type_count)
? "invalid type"
: pr_type_name[type->type]);
if (type->type == ev_func) {
printf (" %4x %d", type->func.return_type,
count = type->func.num_params);
if (count < 0)
count = ~count; //ones complement
for (i = 0; i < count; i++)
printf (" %x", type->func.param_types[i]);
} else if (type->type == ev_pointer
|| type->type == ev_field) {
printf (" %4x", type->fldptr.aux_type);
}
printf ("\n");
break;
case ty_struct:
case ty_union:
case ty_enum:
printf (" %s\n", QFO_GETSTR (qfo, type->strct.tag));
for (i = 0; i < type->strct.num_fields; i++) {
printf (" %-5x %4x %s\n",
type->strct.fields[i].type,
type->strct.fields[i].offset,
QFO_GETSTR (qfo, type->strct.fields[i].name));
}
break;
case ty_array:
printf (" %-5x %d %d\n", type->array.type,
type->array.base, type->array.size);
break;
case ty_class:
printf (" %-5x\n", type->class);
break;
case ty_alias:
printf (" %s %d %5x %5x\n",
QFO_GETSTR (qfo, type->alias.name),
type->alias.type, type->alias.aux_type,
type->alias.full_type);
break;
}
}
}
void
qfo_types (qfo_t *qfo)
{
dump_qfo_types (qfo, 0);
}
void
dump_types (progs_t *pr)
{
qfo_mspace_t spaces[qfo_num_spaces];
qfo_t qfo;
pr_def_t *encodings_def;
qfot_type_encodings_t *encodings;
encodings_def = PR_FindGlobal (pr, ".type_encodings");
if (!encodings_def) {
printf ("no type encodings found");
return;
}
if (encodings_def->ofs > pr->globals_size - 2) {
printf ("invalid .type_encodings address");
return;
}
encodings = &G_STRUCT (pr, qfot_type_encodings_t, encodings_def->ofs);
if (encodings->types > pr->globals_size) {
printf ("invalid encodings block start");
return;
}
if (encodings->types + encodings->size > pr->globals_size) {
printf ("truncating encodings block size");
encodings->size = pr->globals_size - encodings->types;
}
memset (spaces, 0, sizeof (spaces));
spaces[qfo_strings_space].type = qfos_string;
spaces[qfo_strings_space].d.strings = pr->pr_strings;
spaces[qfo_strings_space].data_size = pr->pr_stringsize;
spaces[qfo_type_space].type = qfos_type;
spaces[qfo_type_space].d.data = pr->pr_globals + encodings->types;
spaces[qfo_type_space].data_size = encodings->size;
memset (&qfo, 0, sizeof (qfo));
qfo.spaces = spaces;
qfo.num_spaces = qfo_num_spaces;
dump_qfo_types (&qfo, encodings->types);
}