From 66971055da035d25f4c946a3e897ac637c5de2f6 Mon Sep 17 00:00:00 2001 From: havoc Date: Tue, 25 Apr 2006 13:20:05 +0000 Subject: [PATCH] added prvm_printfunction command which prints an assembly dump of the specified qc function, with coloring and optionally statement profiling to show how many times each statement has been executed (prvm_statementprofiling must be on for this), also heavily modified statement printing to be more readable git-svn-id: svn://svn.icculus.org/twilight/trunk/darkplaces@6327 d7cf8633-e32d-0410-b094-e92efae38249 --- progsvm.h | 4 + prvm_edict.c | 34 ++++--- prvm_exec.c | 248 +++++++++++++++++++++++++++++++++------------ prvm_execprogram.h | 3 + 4 files changed, 209 insertions(+), 80 deletions(-) diff --git a/progsvm.h b/progsvm.h index 673caea3..8f77619d 100644 --- a/progsvm.h +++ b/progsvm.h @@ -266,6 +266,8 @@ typedef struct prvm_prog_s int *statement_linenums; // NULL if not available + int *statement_profile; // only incremented if prvm_statementprofiling is on + union { float *generic; globalvars_t *server; @@ -428,6 +430,7 @@ void _PRVM_Free (void *buffer, const char *filename, int fileline); void _PRVM_FreeAll (const char *filename, int fileline); void PRVM_Profile_f (void); +void PRVM_PrintFunction_f (void); void PRVM_PrintState(void); void PRVM_CrashAll (void); @@ -444,6 +447,7 @@ prvm_edict_t *PRVM_ED_Alloc (void); void PRVM_ED_Free (prvm_edict_t *ed); void PRVM_ED_ClearEdict (prvm_edict_t *e); +void PRVM_PrintFunctionStatements (const char *name); void PRVM_ED_Print(prvm_edict_t *ed); void PRVM_ED_Write (qfile_t *f, prvm_edict_t *ed); const char *PRVM_ED_ParseEdict (const char *data, prvm_edict_t *ent); diff --git a/prvm_edict.c b/prvm_edict.c index 0389b8fc..29e5998a 100644 --- a/prvm_edict.c +++ b/prvm_edict.c @@ -35,6 +35,8 @@ qboolean PRVM_ED_ParseEpair(prvm_edict_t *ent, ddef_t *key, const char *s); cvar_t prvm_boundscheck = {0, "prvm_boundscheck", "1", "enables detection of out of bounds memory access in the QuakeC code being run (in other words, prevents really exceedingly bad QuakeC code from doing nasty things to your computer)"}; // LordHavoc: prints every opcode as it executes - warning: this is significant spew cvar_t prvm_traceqc = {0, "prvm_traceqc", "0", "prints every QuakeC statement as it is executed (only for really thorough debugging!)"}; +// LordHavoc: counts usage of each QuakeC statement +cvar_t prvm_statementprofiling = {0, "prvm_statementprofiling", "0", "counts how many times each QuakeC statement has been executed, these counts are displayed in prvm_printfunction output (if enabled)"}; //============================================================================ // mempool handling @@ -509,7 +511,7 @@ padded to 20 field width char *PRVM_GlobalString (int ofs) { char *s; - size_t i; + //size_t i; ddef_t *def; void *val; static char line[128]; @@ -517,37 +519,37 @@ char *PRVM_GlobalString (int ofs) val = (void *)&prog->globals.generic[ofs]; def = PRVM_ED_GlobalAtOfs(ofs); if (!def) - sprintf (line,"%i(?)", ofs); + sprintf (line,"GLOBAL%i", ofs); else { s = PRVM_ValueString ((etype_t)def->type, (prvm_eval_t *)val); - sprintf (line,"%i(%s)%s", ofs, PRVM_GetString(def->s_name), s); + sprintf (line,"%s (=%s)", PRVM_GetString(def->s_name), s); } - i = strlen(line); - for ( ; i<20 ; i++) - strcat (line," "); - strcat (line," "); + //i = strlen(line); + //for ( ; i<20 ; i++) + // strcat (line," "); + //strcat (line," "); return line; } char *PRVM_GlobalStringNoContents (int ofs) { - size_t i; + //size_t i; ddef_t *def; static char line[128]; def = PRVM_ED_GlobalAtOfs(ofs); if (!def) - sprintf (line,"%i(?)", ofs); + sprintf (line,"GLOBAL%i", ofs); else - sprintf (line,"%i(%s)", ofs, PRVM_GetString(def->s_name)); + sprintf (line,"%s", PRVM_GetString(def->s_name)); - i = strlen(line); - for ( ; i<20 ; i++) - strcat (line," "); - strcat (line," "); + //i = strlen(line); + //for ( ; i<20 ; i++) + // strcat (line," "); + //strcat (line," "); return line; } @@ -1347,6 +1349,8 @@ void PRVM_LoadProgs (const char * filename, int numrequiredfunc, char **required prog->statements = (dstatement_t *)((unsigned char *)prog->progs + prog->progs->ofs_statements); + prog->statement_profile = (int *)Mem_Alloc(prog->progs_mempool, prog->progs->numstatements * sizeof(*prog->statement_profile)); + // moved edict_size calculation down below field adding code //pr_global_struct = (globalvars_t *)((unsigned char *)progs + progs->ofs_globals); @@ -1764,9 +1768,11 @@ void PRVM_Init (void) Cmd_AddCommand ("prvm_global", PRVM_Global_f, "prints value of a specified global variable in the selected VM (server, client, menu)"); Cmd_AddCommand ("prvm_globalset", PRVM_GlobalSet_f, "sets value of a specified global variable in the selected VM (server, client, menu)"); Cmd_AddCommand ("prvm_edictset", PRVM_ED_EdictSet_f, "changes value of a specified property of a specified entity in the selected VM (server, client, menu)"); + Cmd_AddCommand ("prvm_printfunction", PRVM_PrintFunction_f, "prints a disassembly (QuakeC instructions) of the specified function in the selected VM (server, client, menu)"); // LordHavoc: optional runtime bounds checking (speed drain, but worth it for security, on by default - breaks most QCCX features (used by CRMod and others)) Cvar_RegisterVariable (&prvm_boundscheck); Cvar_RegisterVariable (&prvm_traceqc); + Cvar_RegisterVariable (&prvm_statementprofiling); //VM_Cmd_Init(); } diff --git a/prvm_exec.c b/prvm_exec.c index e85c3eed..eb8aa349 100644 --- a/prvm_exec.c +++ b/prvm_exec.c @@ -23,7 +23,7 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. char *prvm_opnames[] = { -"DONE", +"^5DONE", "MUL_F", "MUL_V", @@ -38,31 +38,31 @@ char *prvm_opnames[] = "SUB_F", "SUB_V", -"EQ_F", -"EQ_V", -"EQ_S", -"EQ_E", -"EQ_FNC", +"^2EQ_F", +"^2EQ_V", +"^2EQ_S", +"^2EQ_E", +"^2EQ_FNC", -"NE_F", -"NE_V", -"NE_S", -"NE_E", -"NE_FNC", +"^2NE_F", +"^2NE_V", +"^2NE_S", +"^2NE_E", +"^2NE_FNC", -"LE", -"GE", -"LT", -"GT", +"^2LE", +"^2GE", +"^2LT", +"^2GT", -"INDIRECT", -"INDIRECT", -"INDIRECT", -"INDIRECT", -"INDIRECT", -"INDIRECT", +"^6FIELD_F", +"^6FIELD_V", +"^6FIELD_S", +"^6FIELD_ENT", +"^6FIELD_FLD", +"^6FIELD_FNC", -"ADDRESS", +"^1ADDRESS", "STORE_F", "STORE_V", @@ -71,40 +71,40 @@ char *prvm_opnames[] = "STORE_FLD", "STORE_FNC", -"STOREP_F", -"STOREP_V", -"STOREP_S", -"STOREP_ENT", -"STOREP_FLD", -"STOREP_FNC", +"^1STOREP_F", +"^1STOREP_V", +"^1STOREP_S", +"^1STOREP_ENT", +"^1STOREP_FLD", +"^1STOREP_FNC", -"RETURN", +"^5RETURN", -"NOT_F", -"NOT_V", -"NOT_S", -"NOT_ENT", -"NOT_FNC", +"^2NOT_F", +"^2NOT_V", +"^2NOT_S", +"^2NOT_ENT", +"^2NOT_FNC", -"IF", -"IFNOT", +"^5IF", +"^5IFNOT", -"CALL0", -"CALL1", -"CALL2", -"CALL3", -"CALL4", -"CALL5", -"CALL6", -"CALL7", -"CALL8", +"^3CALL0", +"^3CALL1", +"^3CALL2", +"^3CALL3", +"^3CALL4", +"^3CALL5", +"^3CALL6", +"^3CALL7", +"^3CALL8", -"STATE", +"^1STATE", -"GOTO", +"^5GOTO", -"AND", -"OR", +"^2AND", +"^2OR", "BITAND", "BITOR" @@ -121,48 +121,127 @@ char *PRVM_GlobalStringNoContents (int ofs); PRVM_PrintStatement ================= */ +extern cvar_t prvm_statementprofiling; void PRVM_PrintStatement (dstatement_t *s) { size_t i; + int opnum = (int)(s - prog->statements); - if( prog->statement_linenums ) { - int opnum; - - opnum = s - prog->statements; + Con_Printf("s%i: ", opnum); + if( prog->statement_linenums ) Con_Printf( "%s:%i: ", PRVM_GetString( prog->xfunction->s_file ), prog->statement_linenums[ opnum ] ); - } + + if (prvm_statementprofiling.integer) + Con_Printf("%7i ", prog->statement_profile[s - prog->statements]); if ( (unsigned)s->op < sizeof(prvm_opnames)/sizeof(prvm_opnames[0])) { Con_Printf("%s ", prvm_opnames[s->op]); i = strlen(prvm_opnames[s->op]); + // don't count a preceding color tag when padding the name + if (prvm_opnames[s->op][0] == STRING_COLOR_TAG) + i -= 2; for ( ; i<10 ; i++) Con_Print(" "); } - if (s->op == OP_IF || s->op == OP_IFNOT) - Con_Printf("%sbranch %i",PRVM_GlobalString((unsigned short) s->a),s->b); + Con_Printf("%s, s%i",PRVM_GlobalString((unsigned short) s->a),(signed short)s->b + opnum); else if (s->op == OP_GOTO) - { - Con_Printf("branch %i",s->a); - } + Con_Printf("s%i",(signed short)s->a + opnum); else if ( (unsigned)(s->op - OP_STORE_F) < 6) { Con_Print(PRVM_GlobalString((unsigned short) s->a)); + Con_Print(", "); Con_Print(PRVM_GlobalStringNoContents((unsigned short) s->b)); } + else if (s->op == OP_ADDRESS || (unsigned)(s->op - OP_LOAD_F) < 6) + { + if (s->a) + Con_Print(PRVM_GlobalString((unsigned short) s->a)); + if (s->b) + { + Con_Print(", "); + Con_Print(PRVM_GlobalStringNoContents((unsigned short) s->b)); + } + if (s->c) + { + Con_Print(", "); + Con_Print(PRVM_GlobalStringNoContents((unsigned short) s->c)); + } + } else { if (s->a) Con_Print(PRVM_GlobalString((unsigned short) s->a)); if (s->b) + { + Con_Print(", "); Con_Print(PRVM_GlobalString((unsigned short) s->b)); + } if (s->c) + { + Con_Print(", "); Con_Print(PRVM_GlobalStringNoContents((unsigned short) s->c)); + } } Con_Print("\n"); } +void PRVM_PrintFunctionStatements (const char *name) +{ + int i, firststatement, endstatement; + mfunction_t *func; + func = PRVM_ED_FindFunction (name); + if (!func) + { + Con_Printf("%s progs: no function named %s\n", PRVM_NAME, name); + return; + } + firststatement = func->first_statement; + if (firststatement < 0) + { + Con_Printf("%s progs: function %s is builtin #%i\n", PRVM_NAME, name, -firststatement); + return; + } + + // find the end statement + endstatement = prog->progs->numstatements; + for (i = 0;i < prog->progs->numfunctions;i++) + if (endstatement > prog->functions[i].first_statement && firststatement < prog->functions[i].first_statement) + endstatement = prog->functions[i].first_statement; + + // now print the range of statements + Con_Printf("%s progs: disassembly of function %s (statements %i-%i):\n", PRVM_NAME, name, firststatement, endstatement); + for (i = firststatement;i < endstatement;i++) + { + PRVM_PrintStatement(prog->statements + i); + prog->statement_profile[i] = 0; + } +} + +/* +============ +PRVM_PrintFunction_f + +============ +*/ +void PRVM_PrintFunction_f (void) +{ + if (Cmd_Argc() != 3) + { + Con_Printf("usage: prvm_printfunction \n"); + return; + } + + PRVM_Begin; + if(!PRVM_SetProgFromString(Cmd_Argv(1))) + return; + + PRVM_PrintFunctionStatements(Cmd_Argv(2)); + + PRVM_End; +} + /* ============ PRVM_StackTrace @@ -231,9 +310,9 @@ void PRVM_Profile_f (void) { //if (num < howmany) if (best->first_statement < 0) - Con_Printf("%7i -- builtin -- %s\n", best->callcount, PRVM_GetString(best->s_name)); + Con_Printf("%10i ----- builtin ----- %s\n", best->callcount, PRVM_GetString(best->s_name)); else - Con_Printf("%7i%7i%7i %s\n", best->callcount, best->profile, best->builtinsprofile, PRVM_GetString(best->s_name)); + Con_Printf("%10i%10i%10i %s\n", best->callcount, best->profile, best->builtinsprofile, PRVM_GetString(best->s_name)); num++; best->profile = 0; best->builtinsprofile = 0; @@ -394,6 +473,7 @@ PRVM_ExecuteProgram #define OPC ((prvm_eval_t *)&prog->globals.generic[(unsigned short) st->c]) extern cvar_t prvm_boundscheck; extern cvar_t prvm_traceqc; +extern cvar_t prvm_statementprofiling; extern int PRVM_ED_FindFieldOffset (const char *field); extern ddef_t* PRVM_ED_FindGlobal(const char *name); void PRVM_ExecuteProgram (func_t fnum, const char *errormessage) @@ -431,32 +511,68 @@ void PRVM_ExecuteProgram (func_t fnum, const char *errormessage) chooseexecprogram: cachedpr_trace = prog->trace; - if (prvm_boundscheck.integer) + if (prvm_statementprofiling.integer) { -#define PRVMBOUNDSCHECK 1 - if (prog->trace) +#define PRVMSTATEMENTPROFILING 1 + if (prvm_boundscheck.integer) { +#define PRVMBOUNDSCHECK 1 + if (prog->trace) + { #define PRVMTRACE 1 #include "prvm_execprogram.h" +#undef PRVMTRACE + } + else + { +#include "prvm_execprogram.h" + } +#undef PRVMBOUNDSCHECK } else { + if (prog->trace) + { +#define PRVMTRACE 1 +#include "prvm_execprogram.h" #undef PRVMTRACE + } + else + { #include "prvm_execprogram.h" + } } +#undef PRVMSTATEMENTPROFILING } else { -#undef PRVMBOUNDSCHECK - if (prog->trace) + if (prvm_boundscheck.integer) { +#define PRVMBOUNDSCHECK 1 + if (prog->trace) + { #define PRVMTRACE 1 #include "prvm_execprogram.h" +#undef PRVMTRACE + } + else + { +#include "prvm_execprogram.h" + } +#undef PRVMBOUNDSCHECK } else { + if (prog->trace) + { +#define PRVMTRACE 1 +#include "prvm_execprogram.h" #undef PRVMTRACE + } + else + { #include "prvm_execprogram.h" + } } } } diff --git a/prvm_execprogram.h b/prvm_execprogram.h index 97f50b94..aa79fdb3 100644 --- a/prvm_execprogram.h +++ b/prvm_execprogram.h @@ -8,6 +8,9 @@ #if PRVMTRACE PRVM_PrintStatement(st); #endif +#if PRVMSTATEMENTPROFILING + prog->statement_profile[st - prog->statements]++; +#endif switch (st->op) { -- 2.39.5