Fixed building on GCC and some valgrind warnings. Read .dbg.natives table now.

This commit is contained in:
David Anderson 2008-09-27 15:34:59 -05:00
parent a935e6f90d
commit 32f7bec6c8
10 changed files with 259 additions and 42 deletions

View File

@ -0,0 +1,98 @@
# (C)2004-2008 SourceMod Development Team
# Makefile written by David "BAILOPAN" Anderson
SMSDK = ../..
SOURCEHOOK = ../../../sourcemm-1.6/sourcehook
#####################################
### EDIT BELOW FOR OTHER PROJECTS ###
#####################################
PROJECT = spdecomp
OBJECTS = main.cpp \
decompiler.cpp \
code_analyzer.cpp \
file_format.cpp \
platform_util.cpp \
zlib/adler32.c \
zlib/compress.c \
zlib/crc32.c \
zlib/deflate.c \
zlib/gzio.c \
zlib/infback.c \
zlib/inffast.c \
zlib/inflate.c \
zlib/inftrees.c \
zlib/trees.c \
zlib/uncompr.c \
zlib/zutil.c
##############################################
### CONFIGURE ANY OTHER FLAGS/OPTIONS HERE ###
##############################################
C_OPT_FLAGS = -DNDEBUG -O3 -funroll-loops -pipe -fno-strict-aliasing
C_DEBUG_FLAGS = -D_DEBUG -DDEBUG -g -ggdb3
C_GCC4_FLAGS = -fvisibility=hidden
CPP_GCC4_FLAGS = -fvisibility-inlines-hidden
CPP = gcc-4.1
LINK = -static-libgcc
INCLUDE = -I. -I.. -I$(SMSDK)/public -I$(SMSDK)/public/jit -I$(SMSDK)/public/jit/x86 \
-I$(SMSDK)/public/sourcepawn -I$(SOURCEHOOK) -I$(SMSDK)/knight/shared -Ix86
CFLAGS += -D_LINUX -Dstricmp=strcasecmp -D_stricmp=strcasecmp -D_strnicmp=strncasecmp -Dstrnicmp=strncasecmp \
-D_snprintf=snprintf -D_vsnprintf=vsnprintf -D_alloca=alloca -Dstrcmpi=strcasecmp -Wall -DHAVE_STDINT_H \
-m32 -Wno-uninitialized -Werror
CPPFLAGS += -Wno-non-virtual-dtor -fno-exceptions -fno-rtti
################################################
### DO NOT EDIT BELOW HERE FOR MOST PROJECTS ###
################################################
ifeq "$(DEBUG)" "true"
BIN_DIR = Debug
CFLAGS += $(C_DEBUG_FLAGS)
else
BIN_DIR = Release
CFLAGS += $(C_OPT_FLAGS)
endif
GCC_VERSION := $(shell $(CPP) -dumpversion >&1 | cut -b1)
ifeq "$(GCC_VERSION)" "4"
CFLAGS += $(C_GCC4_FLAGS)
CPPFLAGS += $(CPP_GCC4_FLAGS)
endif
BINARY = $(PROJECT)
OBJ_LINUX := $(OBJECTS:%.cpp=$(BIN_DIR)/%.o)
OBJ_LINUX := $(OBJ_LINUX:%.c=$(BIN_DIR)/%.o)
default: all
$(BIN_DIR)/%.o: %.c
$(CPP) $(INCLUDE) $(CFLAGS) -o $@ -c $<
$(BIN_DIR)/%.o: %.cpp
$(CPP) $(INCLUDE) $(CFLAGS) $(CPPFLAGS) -o $@ -c $<
all:
mkdir -p $(BIN_DIR)/x86
mkdir -p $(BIN_DIR)/zlib
mkdir -p $(BIN_DIR)/knight
$(MAKE) -f Makefile jit
jit: $(OBJ_LINUX)
$(CPP) $(INCLUDE) $(OBJ_LINUX) $(LINK) -m32 -ldl -lm -o$(BIN_DIR)/$(BINARY)
debug:
$(MAKE) -f Makefile all DEBUG=true
clean:
rm -rf $(BIN_DIR)/zlib/*.o
rm -rf $(BIN_DIR)/*.o
rm -rf $(BIN_DIR)/$(BINARY)

View File

@ -146,6 +146,7 @@ static FunctionInfo *sp_InferFuncPrototype(sp_decomp_t *dc, uint32_t code_addr,
}
info->known_args[i].name = plugin->debug.stringbase + sym->name;
info->known_args[i].dimcount = sym->dimcount;
if (sym->dimcount > 0)
{
info->known_args[i].dims =
@ -260,6 +261,7 @@ static void sp_CacheLocals(sp_decomp_t *dc, FunctionInfo *info)
info->known_vars[num].name = plugin->debug.stringbase + sym->name;
info->known_vars[num].tag =
(sym->tagid == 0) ? NULL : Sp_FindTag(plugin, sym->tagid);
info->known_vars[num].dimcount = sym->dimcount;
info->known_vars[num].dims =
(sym->dimcount == 0)
? NULL
@ -633,35 +635,60 @@ int sp_AnalyzeGraph(sp_decomp_t *dc, ControlFlowGraph *graph)
break;
}
case OP_CALL:
case OP_SYSREQ_N:
{
p1 = PCODE_PARAM(pcode, cip, 1);
tfunc = sp_GetFuncPrototype(dc, p1, false);
if (op == OP_CALL)
{
tfunc = sp_GetFuncPrototype(dc, p1, false);
}
else if (op == OP_SYSREQ_N)
{
assert(dc->natives);
assert(uint32_t(p1) < dc->plugin->num_natives);
tfunc = &dc->natives[p1];
}
CHECK_VALUE(tfunc);
/* Sanity checks, :TODO: runtime checks later. */
assert(stack_entries >= 1);
ConstNode *cnode;
cnode = (ConstNode *)eval_stack[stack_entries-1].expr;
stack_entries--;
if (op == OP_SYSREQ_N)
{
p2 = PCODE_PARAM(pcode, cip, 2);
}
else
{
/* Try to read the value off the stack. */
ConstNode *cnode;
cnode = (ConstNode *)eval_stack[stack_entries-1].expr;
/* Sanity checks, :TODO: runtime checks later. */
assert(cnode->op == _OP_CONST);
assert(cnode->val >= 0);
assert(uint32_t(cnode->val) == tfunc->num_known_args);
p2 = cnode->val;
stack_entries--;
}
/* Sanity checks, :TODO: runtime checks later. */
assert(cnode->op == _OP_CONST);
assert(cnode->val >= 0);
assert(cnode->val == tfunc->num_known_args);
assert(unsigned int(cnode->val) <= stack_entries);
assert(uint32_t(p2) <= stack_entries);
cell_t argc = cnode->val;
BaseNode **nodes = new (graph) BaseNode *[cnode->val];
for (cell_t argno = 0; argno < cnode->val; argno++, stack_entries--)
/* Build arg SSA */
BaseNode **nodes = new (graph) BaseNode *[p2];
for (cell_t argno = 0; argno < p2; argno++, stack_entries--)
{
nodes[argno] = eval_stack[stack_entries-1].expr;
}
sp += (cnode->val + 1) * sizeof(cell_t);
/* Get rid of any args we pushed. */
sp += p2 * sizeof(cell_t);
if (op == OP_CALL)
{
sp += sizeof(cell_t);
}
assert(sp <= 0);
pri = new (graph) CallNode(tfunc, nodes, cnode->val);
/* Build function call SSA */
pri = new (graph) CallNode(tfunc, nodes, p2);
pri = new (graph) DefineNode(pri, "call%03x", ++callnumber);
rtemp = new (graph) StmtNode(_OP_STMT, pri, NULL);
root_tail->next = rtemp;
@ -677,32 +704,38 @@ int sp_AnalyzeGraph(sp_decomp_t *dc, ControlFlowGraph *graph)
break;
}
case OP_PUSH_C:
case OP_PUSH2_C:
case OP_PUSH3_C:
case OP_PUSH5_C:
{
p1 = PCODE_PARAM(pcode, cip, 1);
/* Adjust stack pointer. */
sp -= sizeof(cell_t);
if ((v1 = sp_FindGraphVar(func, cip, sp, sp)) == NULL)
for (int N = 1; N <= dc->opdef[op].params; N++)
{
/* Add a new stack entry. */
assert(stack_entries < MAX_STACK_ENTRIES);
eval_stack[stack_entries++] = StackEntry(sp, new ConstNode(p1));
}
else
{
assert(v1->new_stmt == NULL);
if (v1->new_stmt == NULL)
p1 = PCODE_PARAM(pcode, cip, N);
/* Adjust stack pointer. */
sp -= sizeof(cell_t);
if ((v1 = sp_FindGraphVar(func, cip, sp, sp)) == NULL)
{
v1->new_stmt = new (graph) DeclNode(
new (graph) VarNode(v1),
new (graph) ConstNode(p1));
rtemp = new (graph) StmtNode(
_OP_STMT,
v1->new_stmt,
NULL);
root_tail->next = rtemp;
root_tail = rtemp;
/* Add a new stack entry. */
assert(stack_entries < MAX_STACK_ENTRIES);
eval_stack[stack_entries++] = StackEntry(sp, new ConstNode(p1));
}
else
{
assert(v1->new_stmt == NULL);
if (v1->new_stmt == NULL)
{
v1->new_stmt = new (graph) DeclNode(
new (graph) VarNode(v1),
new (graph) ConstNode(p1));
rtemp = new (graph) StmtNode(
_OP_STMT,
v1->new_stmt,
NULL);
root_tail->next = rtemp;
root_tail = rtemp;
}
}
}
break;
@ -1160,7 +1193,7 @@ int Sp_DecompFunction(sp_decomp_t *dc, uint32_t code_addr, bool is_public)
goto return_error;
}
if ((err = sp_AnalyzeGraph(dc, &graph)) == NULL)
if ((err = sp_AnalyzeGraph(dc, &graph)) != SP_ERROR_NONE)
{
goto return_error;
}

View File

@ -126,6 +126,7 @@ struct FuncVar
sp_fdbg_arraydim_t *dims;
sp_tag_t *tag;
const char *name;
unsigned int dimcount;
DeclNode *new_stmt;
};

View File

@ -2,6 +2,7 @@
#include <stdlib.h>
#include <string.h>
#include <stdarg.h>
#include <assert.h>
#include "decompiler.h"
#include "platform_util.h"
@ -27,6 +28,52 @@ sp_decomp_t *Sp_InitDecomp(const char *file, int *err)
#include "opcodes.tbl"
#undef OPDEF
if (plugin->debug.ntv != NULL)
{
assert(plugin->debug.ntv->num_entries == plugin->num_natives);
dc->natives = new FunctionInfo[plugin->num_natives];
memset(dc->natives, 0, sizeof(FunctionInfo) * plugin->num_natives);
/* Unfurl this mess. */
uint8_t *cursor = (uint8_t *)plugin->debug.ntv;
cursor += sizeof(sp_fdbg_ntvtab_t);
for (uint32_t i = 0; i < plugin->num_natives; i++)
{
sp_fdbg_native_t *native = (sp_fdbg_native_t *)cursor;
uint32_t idx = native->index;
assert(dc->natives[idx].name[0] == '\0');
Sp_Format(dc->natives[idx].name,
sizeof(dc->natives[idx].name),
"%s",
plugin->debug.stringbase + native->name);
dc->natives[idx].tag = Sp_FindTag(plugin, native->tagid);
dc->natives[idx].num_known_args = native->nargs;
cursor += sizeof(sp_fdbg_native_t);
for (uint32_t j = 0; j < dc->natives[idx].num_known_args; j++)
{
sp_fdbg_ntvarg_t *arg = (sp_fdbg_ntvarg_t *)cursor;
dc->natives[idx].known_args[j].name =
plugin->debug.stringbase + arg->name;
dc->natives[idx].known_args[j].tag =
Sp_FindTag(plugin, arg->tagid);
dc->natives[idx].known_args[j].dimcount = arg->dimcount;
cursor += sizeof(sp_fdbg_ntvarg_t);
if (arg->dimcount != 0)
{
dc->natives[idx].known_args[j].dims = (sp_fdbg_arraydim_t *)cursor;
cursor += sizeof(sp_fdbg_arraydim_t) * arg->dimcount;
}
}
}
}
return dc;
}
@ -44,6 +91,7 @@ void Sp_FreeDecomp(sp_decomp_t *decomp)
Sp_FreePlugin(decomp->plugin);
free(decomp->pcode_map);
delete [] decomp->natives;
delete decomp;
}
@ -84,9 +132,8 @@ void PrintBuffer::Append(const char *fmt, ...)
len = Sp_FormatArgs(buffer, sizeof(buffer), fmt, ap);
va_end(ap);
Grow(len);
Grow(len + 1);
printbuf = (char *)realloc(printbuf, printbuf_alloc);
strcpy(&printbuf[printbuf_pos], buffer);
printbuf_pos += len;
}
@ -102,7 +149,7 @@ void PrintBuffer::Set(const char *fmt, ...)
va_end(ap);
Reset();
Grow(len);
Grow(len + 1);
printbuf = (char *)realloc(printbuf, printbuf_alloc);
strcpy(&printbuf[printbuf_pos], buffer);

View File

@ -30,6 +30,7 @@ struct sp_decomp_t
sp_opdef_t opdef[OP_TOTAL+1];
FunctionInfo *funcs;
cell_t *pcode_map;
FunctionInfo *natives;
};
sp_decomp_t *Sp_InitDecomp(const char *file, int *err);

View File

@ -246,6 +246,10 @@ sp_file_t *Sp_ReadPlugin(const char *file, int *err)
plugin->debug.lines_num = inf->num_lines;
plugin->debug.syms_num = inf->num_syms;
}
else if (plugin->debug.ntv == NULL && !strcmp(nameptr, ".dbg.natives"))
{
plugin->debug.ntv = (sp_fdbg_ntvtab_s *)(base + secptr->dataoffs);
}
else if (!(plugin->debug.stringbase) && !strcmp(nameptr, ".dbg.strings"))
{
plugin->debug.stringbase = (const char *)(base + secptr->dataoffs);

View File

@ -12,6 +12,7 @@ struct sp_debug_t
sp_fdbg_line_t *lines; /**< lines table */
uint32_t syms_num; /**< number of symbols */
sp_fdbg_symbol_t *symbols; /**< symbol table */
sp_fdbg_ntvtab_s *ntv; /**< native table */
};
struct sp_tag_t

View File

@ -7,7 +7,7 @@ int main()
int err;
sp_decomp_t *dc;
const char *file = "R:\\sourcemod\\sourcemod-central\\plugins\\test.smx";
const char *file = "../../plugins/test.smx";
if ((dc = Sp_InitDecomp(file, &err)) == NULL)
{

View File

@ -23,6 +23,10 @@ OPDEF(80, SUB_ALT, "sub.alt", 0)
OPDEF(87, ADD_C, "add.c", 1)
OPDEF(88, SMUL_C, "smul.c", 1)
OPDEF(89, ZERO_PRI, "zero.pri", 0)
OPDEF(135, SYSREQ_N, "sysreq.n", 2)
OPDEF(137, BREAK, "break", 0)
OPDEF(138, PUSH2_C, "push2.c", 2)
OPDEF(142, PUSH3_C, "push3.c", 3)
OPDEF(150, PUSH5_C, "push5.c", 5)
OPDEF(155, LOAD_S_BOTH, "load.s.both", 2)
OPDEF(166, TOTAL, NULL, 0)

View File

@ -1,4 +1,6 @@
#include <new>
#include <stdio.h>
#include <stdlib.h>
#include "platform_util.h"
size_t Sp_Format(char *buffer, size_t maxlength, const char *fmt, ...)
@ -30,3 +32,29 @@ size_t Sp_FormatArgs(char *buffer, size_t maxlength, const char *fmt, va_list ap
}
}
/* Overload a few things to prevent libstdc++ linking */
#if defined __linux__
extern "C" void __cxa_pure_virtual(void)
{
}
void *operator new(size_t size)
{
return malloc(size);
}
void *operator new[](size_t size)
{
return malloc(size);
}
void operator delete(void *ptr)
{
free(ptr);
}
void operator delete[](void * ptr)
{
free(ptr);
}
#endif