From 8770df39e5d78f5a3425d98bb3c6cd581682e671 Mon Sep 17 00:00:00 2001 From: Bill Currie Date: Mon, 8 Nov 2004 23:27:00 +0000 Subject: [PATCH] start documenting things (progs building still borked) This is an imperfect revision of history. --- doc/Makefile.am | 17 +- doc/progs/vm-exec.c | 29 + doc/quakeforge.dox.in | 27 +- include/QF/cbuf.h | 10 + include/QF/checksum.h | 10 + include/QF/cmd.h | 10 + include/QF/crc.h | 10 + include/QF/cvar.h | 10 + include/QF/dstring.h | 10 + include/QF/hash.h | 10 + include/QF/idparse.h | 10 + include/QF/info.h | 10 + include/QF/mathlib.h | 10 + include/QF/mdfour.h | 10 + include/QF/msg.h | 10 + include/QF/pak.h | 10 + include/QF/pakfile.h | 10 + include/QF/plugin.h | 10 + include/QF/progs.h | 1117 +++++++++++++++++++++++++++---- include/QF/qargs.h | 10 + include/QF/qendian.h | 10 + include/QF/quakefs.h | 10 + include/QF/quakeio.h | 10 + include/QF/sizebuf.h | 11 + include/QF/sys.h | 11 + include/QF/va.h | 11 + include/QF/ver_check.h | 10 + include/QF/wad.h | 11 + include/QF/wadfile.h | 12 +- include/QF/zone.h | 88 +-- libs/gamecode/builtins/bi_gib.c | 2 +- qw/include/sv_pr_cmds.h | 5 - qw/source/sv_pr_cmds.c | 1 + tools/qfcc/source/modules.c | 20 +- tools/qwaq/main.c | 1 + 35 files changed, 1367 insertions(+), 196 deletions(-) create mode 100644 doc/progs/vm-exec.c diff --git a/doc/Makefile.am b/doc/Makefile.am index 4e67a6d8b..dd7caafa7 100644 --- a/doc/Makefile.am +++ b/doc/Makefile.am @@ -24,7 +24,20 @@ EXTRA_DIST= 3dfx.txt CodingStyle glqnotes.txt \ data/docs/install.quake data/docs/install.quake2 data/docs/readme \ data/docs/readme.glquake data/docs/readme.squake data/docs/readme.x11 \ \ - ideas/quakedownload.txt ideas/rhamph.txt ideas/serverlist.txt + ideas/quakedownload.txt ideas/rhamph.txt ideas/serverlist.txt \ + \ + progs/vm-mem.fig -doc: quakeforge.dox +%.png: %.fig + @mkdir -p `dirname $@` + fig2dev -L png $< $@ + +%.eps: %.fig + @mkdir -p `dirname $@` + fig2dev -L ps $< $@ + +progs/vm-mem.png: progs/vm-mem.fig +progs/vm-mem.eps: progs/vm-mem.fig + +doc: quakeforge.dox progs/vm-mem.png progs/vm-mem.eps doxygen quakeforge.dox diff --git a/doc/progs/vm-exec.c b/doc/progs/vm-exec.c new file mode 100644 index 000000000..91c0d4f6a --- /dev/null +++ b/doc/progs/vm-exec.c @@ -0,0 +1,29 @@ +#include "QF/progs.h" + +int +call_progs_main (progs_t *pr, int argc, const char **argv) +{ + int i; + dfunction_t *dfunc; + func_t progs_main = 0; + string_t *pr_argv; + + if ((dfunc = PR_FindFunction (pr, "main"))) { + progs_main = dfunc - pr->pr_functions; + } else { + PR_Undefined (pr, "function", "main"); + return -1; + } + + PR_PushFrame (pr); + pr_argv = PR_Zone_Malloc (pr, (argc + 1) * 4); + for (i = 0; i < argc; i++) + pr_argv[i] = PR_SetTempString (pr, argv[1 + i]); + pr_argv[i] = 0; + P_INT (pr, 0) = argc; + P_POINTER (pr, 1) = PR_SetPointer (pr, pr_argv); + PR_ExecuteProgram (pr, progs_main); + PR_PopFrame (pr); + PR_Zone_Free (pr, pr_argv); + return R_INT (pr); +} diff --git a/doc/quakeforge.dox.in b/doc/quakeforge.dox.in index 2143ff11b..64a9bd894 100644 --- a/doc/quakeforge.dox.in +++ b/doc/quakeforge.dox.in @@ -345,8 +345,12 @@ WARN_LOGFILE = # directories like "/usr/src/myproject". Separate the files or directories # with spaces. -INPUT = @top_srcdir@/include @top_srcdir@/libs @top_srcdir@/nq -INPUT += @top_srcdir@/qw @top_srcdir@/tools +INPUT = @top_srcdir@/include +INPUT += @top_srcdir@/libs +INPUT += @top_srcdir@/nq +INPUT += @top_srcdir@/qtv +INPUT += @top_srcdir@/qw +INPUT += @top_srcdir@/tools # If the value of the INPUT tag contains directories, you can use the # FILE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp @@ -367,9 +371,16 @@ RECURSIVE = YES # excluded from the INPUT source files. This way you can easily exclude a # subdirectory from a directory tree whose root is specified with the INPUT tag. -EXCLUDE = qc-lex.c qc-parse.c qc-parse.h -EXCLUDE += fbset_modes_l.c fbset_modes_y.c fbset_modes_y.h +EXCLUDE = @top_srcdir@/tools/qfcc/source/qc-lex.c +EXCLUDE += @top_srcdir@/tools/qfcc/source/qc-parse.c +EXCLUDE += @top_srcdir@/tools/qfcc/source/qc-parse.h +EXCLUDE += @top_srcdir@/tools/qfcc/test +EXCLUDE += @top_srcdir@/tools/texpaint +EXCLUDE += @top_srcdir@/libs/video/targets/fbset_modes_l.c +EXCLUDE += @top_srcdir@/libs/video/targets/fbset_modes_y.c +EXCLUDE += @top_srcdir@/libs/video/targets/fbset_modes_y.h EXCLUDE += @top_srcdir@/tools/Forge +EXCLUDE += @top_srcdir@/include/QF/GL # The EXCLUDE_SYMLINKS tag can be used select whether or not files or directories # that are symbolic links (a Unix filesystem feature) are excluded from the input. @@ -386,7 +397,8 @@ EXCLUDE_PATTERNS = # directories that contain example code fragments that are included (see # the \include command). -EXAMPLE_PATH = +EXAMPLE_PATH = @top_srcdir@/doc +EXAMPLE_PATH += @top_srcdir@/doc/progs # If the value of the EXAMPLE_PATH tag contains directories, you can use the # EXAMPLE_PATTERNS tag to specify one or more wildcard pattern (like *.cpp @@ -406,7 +418,8 @@ EXAMPLE_RECURSIVE = NO # directories that contain image that are included in the documentation (see # the \image command). -IMAGE_PATH = +IMAGE_PATH = @top_builddir@/doc +IMAGE_PATH += @top_builddir@/doc/progs # The INPUT_FILTER tag can be used to specify a program that doxygen should # invoke to filter for each input file. Doxygen will invoke the filter program @@ -430,7 +443,7 @@ FILTER_SOURCE_FILES = NO # If the SOURCE_BROWSER tag is set to YES then a list of source files will # be generated. Documented entities will be cross-referenced with these sources. -SOURCE_BROWSER = YES +SOURCE_BROWSER = NO # Setting the INLINE_SOURCES tag to YES will include the body # of functions and classes directly in the documentation. diff --git a/include/QF/cbuf.h b/include/QF/cbuf.h index 3411b94a0..544a9479c 100644 --- a/include/QF/cbuf.h +++ b/include/QF/cbuf.h @@ -32,6 +32,13 @@ #ifndef __QF_cbuf_h #define __QF_cbuf_h +/** \addtogroup utils */ +//@{ + +/** \defgroup cbuf Command buffer management. +*/ +//@{ + #include #include "QF/qtypes.h" @@ -95,4 +102,7 @@ void Cbuf_Execute (cbuf_t *cbuf); void Cbuf_Execute_Stack (cbuf_t *cbuf); void Cbuf_Execute_Sets (cbuf_t *cbuf); +//@} +//@} + #endif//__QF_cbuf_h diff --git a/include/QF/checksum.h b/include/QF/checksum.h index 729426b76..9c58bcbf4 100644 --- a/include/QF/checksum.h +++ b/include/QF/checksum.h @@ -29,10 +29,20 @@ #ifndef __checksum_h #define __checksum_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup crc +*/ +//@{ + #include "QF/qtypes.h" unsigned int Com_BlockChecksum (const void *buffer, int length); void Com_BlockFullChecksum (const void *buffer, int len, unsigned char *outbuf); byte COM_BlockSequenceCRCByte (const byte *base, int length, int sequence); +//@} +//@} + #endif // __checksum_h diff --git a/include/QF/cmd.h b/include/QF/cmd.h index 1c8f02dee..02a6564d6 100644 --- a/include/QF/cmd.h +++ b/include/QF/cmd.h @@ -29,6 +29,13 @@ #ifndef __QF_cmd_h #define __QF_cmd_h +/** \addtogroup utils */ +//@{ + +/** \defgroup cmd Command management. +*/ +//@{ + #include "QF/qtypes.h" typedef void (*xcommand_t) (void); @@ -66,4 +73,7 @@ void Cmd_Exec_File (struct cbuf_s *cbuf, const char *path, int qfs); extern struct cbuf_args_s *cmd_args; extern struct cvar_s *cmd_warncmd; +//@} +//@} + #endif//__QF_cmd_h diff --git a/include/QF/crc.h b/include/QF/crc.h index 79be2c028..1954634be 100644 --- a/include/QF/crc.h +++ b/include/QF/crc.h @@ -29,6 +29,13 @@ #ifndef __crc_h #define __crc_h +/** \addtogroup utils */ +//@{ + +/** \defgroup crc Checksum generation. +*/ +//@{ + #include "QF/qtypes.h" void CRC_Init(unsigned short *crcvalue); @@ -37,4 +44,7 @@ void CRC_ProcessBlock (byte *start, unsigned short *crcvalue, int count); unsigned short CRC_Value(unsigned short crcvalue); unsigned short CRC_Block (byte *start, int count); +//@} +//@} + #endif // __crc_h diff --git a/include/QF/cvar.h b/include/QF/cvar.h index 658a675dc..ad33588cb 100644 --- a/include/QF/cvar.h +++ b/include/QF/cvar.h @@ -29,6 +29,13 @@ #ifndef __cvar_h #define __cvar_h +/** \addtogroup utils */ +//@{ + +/** \defgroup cvar Configuration variables +*/ +//@{ + #include "QF/qtypes.h" #include "QF/quakeio.h" @@ -126,4 +133,7 @@ void Cvar_Shutdown (void); extern cvar_t *cvar_vars; +//@} +//@} + #endif // __cvar_h diff --git a/include/QF/dstring.h b/include/QF/dstring.h index 51672e335..b29926aa1 100644 --- a/include/QF/dstring.h +++ b/include/QF/dstring.h @@ -29,6 +29,13 @@ #ifndef __dstring_h #define __dstring_h +/** \addtogroup utils */ +//@{ + +/** \defgroup dstring Dynamic Strings +*/ +//@{ + #include #include @@ -150,4 +157,7 @@ int davsprintf (dstring_t *dstr, const char *fmt, va_list args); int dasprintf (dstring_t *dstr, const char *fmt, ...) __attribute__((format(printf,2,3))); //@} +//@} +//@} + #endif // __dstring_h diff --git a/include/QF/hash.h b/include/QF/hash.h index 90923c0a0..21ae14b64 100644 --- a/include/QF/hash.h +++ b/include/QF/hash.h @@ -29,6 +29,13 @@ #ifndef __hash_h #define __hash_h +/** \addtogroup utils */ +//@{ + +/** \defgroup hash Hash tables +*/ +//@{ + typedef struct hashtab_s hashtab_t; /** create a new hash table: @@ -154,4 +161,7 @@ void **Hash_GetList (hashtab_t *tab); */ void Hash_Stats (hashtab_t *tab); +//@} +//@} + #endif // __hash_h diff --git a/include/QF/idparse.h b/include/QF/idparse.h index b2faa27c1..c4c916134 100644 --- a/include/QF/idparse.h +++ b/include/QF/idparse.h @@ -31,6 +31,13 @@ #ifndef __QF_idparse_h #define __QF_idparse_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup cbuf +*/ +//@{ + extern const char *com_token; struct cbuf_args_s; @@ -40,4 +47,7 @@ void COM_TokenizeString (const char *str, struct cbuf_args_s *args); extern struct cbuf_interpreter_s id_interp; +//@} +//@} + #endif//__QF_idparse_h diff --git a/include/QF/info.h b/include/QF/info.h index 5e94bb746..0c1db0028 100644 --- a/include/QF/info.h +++ b/include/QF/info.h @@ -29,6 +29,13 @@ #ifndef _INFO_H #define _INFO_H +/** \addtogroup utils */ +//@{ + +/** \defgroup info Info Keys +*/ +//@{ + #include // for size_t. sys/types.h SHOULD be used, but can't :(bc) #include @@ -60,4 +67,7 @@ void Info_Destroy (info_t *info); char *Info_MakeString (info_t *info, int (*filter)(const char *)); void Info_AddKeys (info_t *info, info_t *keys); +//@} +//@} + #endif // _INFO_H diff --git a/include/QF/mathlib.h b/include/QF/mathlib.h index 809225211..68522b5f5 100644 --- a/include/QF/mathlib.h +++ b/include/QF/mathlib.h @@ -29,6 +29,13 @@ #ifndef __mathlib_h #define __mathlib_h +/** \addtogroup utils */ +//@{ + +/** \defgroup mathlib Vector and matrix functions +*/ +//@{ + #include #include "QF/qtypes.h" @@ -312,4 +319,7 @@ VectorNormalize (vec3_t v) return length; } +//@} +//@} + #endif // __mathlib_h diff --git a/include/QF/mdfour.h b/include/QF/mdfour.h index 372f24a96..5e34f9841 100644 --- a/include/QF/mdfour.h +++ b/include/QF/mdfour.h @@ -30,6 +30,13 @@ #ifndef __mdfour_h #define __mdfour_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup crc +*/ +//@{ + #include "QF/uint32.h" #define MDFOUR_DIGEST_BYTES 16 @@ -44,4 +51,7 @@ void mdfour_update(struct mdfour *md, const unsigned char *in, int n); //old: MD void mdfour_result(struct mdfour *md, unsigned char *out); // old: MD4Final void mdfour(unsigned char *out, const unsigned char *in, int n); +//@} +//@} + #endif // __mdfour_h diff --git a/include/QF/msg.h b/include/QF/msg.h index 1bc362d38..4733d8cb7 100644 --- a/include/QF/msg.h +++ b/include/QF/msg.h @@ -28,6 +28,13 @@ #ifndef _MSG_H #define _MSG_H +/** \addtogroup utils */ +//@{ + +/** \defgroup msg Message reading and writing +*/ +//@{ + #include "QF/sizebuf.h" void MSG_WriteByte (sizebuf_t *sb, int c); @@ -66,4 +73,7 @@ void MSG_ReadCoordAngleV (qmsg_t *msg, vec3_t coord, vec3_t angles); void MSG_ReadAngleV (qmsg_t *msg, vec3_t angles); float MSG_ReadAngle16 (qmsg_t *msg); +//@} +//@} + #endif diff --git a/include/QF/pak.h b/include/QF/pak.h index 615417c21..7c4e1d9e7 100644 --- a/include/QF/pak.h +++ b/include/QF/pak.h @@ -32,6 +32,13 @@ #ifndef __qf_pak_h #define __qf_pak_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup pak +*/ +//@{ + #define PAK_PATH_LENGTH 56 typedef struct { @@ -45,4 +52,7 @@ typedef struct { int dirlen; } dpackheader_t; +//@} +//@} + #endif//__qf_pak_h diff --git a/include/QF/pakfile.h b/include/QF/pakfile.h index d172f302d..887a1d800 100644 --- a/include/QF/pakfile.h +++ b/include/QF/pakfile.h @@ -31,6 +31,13 @@ #ifndef __QF_pakfile_h #define __QF_pakfile_h +/** \addtogroup utils */ +//@{ + +/** \defgroup pak pakfile proccessing +*/ +//@{ + #include "QF/hash.h" #include "QF/pak.h" #include "QF/quakeio.h" @@ -60,4 +67,7 @@ int pack_add (pack_t *pack, const char *filename); int pack_extract (pack_t *pack, dpackfile_t *pf); dpackfile_t *pack_find_file (pack_t *pack, const char *filename); +//@} +//@} + #endif//__QF_pakfile_h diff --git a/include/QF/plugin.h b/include/QF/plugin.h index 0e5237968..ae5e3767b 100644 --- a/include/QF/plugin.h +++ b/include/QF/plugin.h @@ -29,6 +29,13 @@ #ifndef __QF_plugin_h_ #define __QF_plugin_h_ +/** \addtogroup utils */ +//@{ + +/** \defgroup plugin Plugins +*/ +//@{ + #define QFPLUGIN_VERSION "1.0" #include @@ -110,4 +117,7 @@ void PI_Shutdown (void); // FIXME: we need a generic function to initialize unused fields +//@} +//@} + #endif // __QF_plugin_h_ diff --git a/include/QF/progs.h b/include/QF/progs.h index 7adc05644..f6284eb87 100644 --- a/include/QF/progs.h +++ b/include/QF/progs.h @@ -29,60 +29,212 @@ #ifndef __QF_progs_h #define __QF_progs_h +/** \defgroup progs QuakeC Virtual Machine (VM) + \image html vm-mem.png + \image latex vm-mem.eps "VM memory map" +*/ +//@{ + #include "QF/link.h" #include "QF/pr_comp.h" #include "QF/pr_debug.h" #include "QF/quakeio.h" +typedef struct progs_s progs_t; +typedef struct pr_resource_s pr_resource_t; +typedef struct edict_s edict_t; + +//============================================================================ + +/** \defgroup progs_misc Miscelaneous functions +*/ +//@{ + +/** Initialize the progs engine. +*/ +void PR_Init (void); + +/** Initialize the Cvars for the progs engine. Call before calling PR_Init(). +*/ +void PR_Init_Cvars (void); + +void PR_Error (progs_t *pr, const char *error, ...) __attribute__((format(printf,2,3), noreturn)); +void PR_RunError (progs_t *pr, const char *error, ...) __attribute__((format(printf,2,3), noreturn)); + +//@} + +/** \defgroup progs_execution Execution +*/ +//@{ + +/** Push an execution frame onto the VM stack. Saves current execution state + \param pr pointer to progs_t VM struct +*/ +void PR_PushFrame (progs_t *pr); + +/** Pop an execution frame from the VM stack. Restores execution state. Also + frees any temporary strings allocated in this frame (via + PR_FreeTempStrings()). + \param pr pointer to progs_t VM struct +*/ +void PR_PopFrame (progs_t *pr); + +/** Run a progs function. if \c fnum is a builtin rather than a progs + function, PR_ExecuteProgram() will call the function and then immediately + return to the caller. Nested calls are fully supported. + \param pr pointer to progs_t VM struct + \param fnum number of the function to call +*/ +void PR_ExecuteProgram (progs_t *pr, func_t fnum); + +/** Setup to call a function. If \c fnum is a builtin rather than a progs + function, then the function is called immediately. When called from a + builtin function, the progs function will execute upon return of control + to PR_ExecuteProgram(). + \param pr pointer to progs_t VM struct + \param fnum number of the function to call + \return true if \c fnum was a progs function, false if fnum was a builtin +*/ +int PR_CallFunction (progs_t *pr, func_t fnum); + +//@} + +/** \defgroup progs_load Loading +*/ +//@{ + +/** Type of functions that are called at progs load. + \param pr pointer to progs_t VM struct + \return true for success, false for failure +*/ +typedef int pr_load_func_t (progs_t *pr); + +/** Initialize a progs_t VM struct from an already open file. + \param pr pointer to progs_t VM struct + \param file handle of file to read progs data from + \param size bytes of \c file to read + \param edicts \e number of entities to allocate space for + \param zone minimum size of dynamic memory to allocate space for + + \note \e All runtime strings (permanent or temporary) are allocated from + the VM's dynamic memory space, so be sure \c zone is of sufficient size. + So far, 1MB has proven more than sufficient for Quakeword, even when using + Ruamoko objects. +*/ +void PR_LoadProgsFile (progs_t *pr, QFile *file, int size, int edicts, + int zone); + +/** Convenience wrapper for PR_LoadProgsFile() and PR_RunLoadFuncs(). + Searches for the specified file in the Quake filesystem. + \param pr pointer to progs_t VM struct + \param progsname name of the file to load as progs data + \param edicts \e number of entities to allocate space for + \param zone minimum size of dynamic memory to allocate space for +*/ +void PR_LoadProgs (progs_t *pr, const char *progsname, int edicts, int zone); + +/** Register a primary function to be called after the progs code has been + loaded. These functions are remembered across progs loads. They will be + called in order of registration. + \param pr pointer to progs_t VM struct + \param func function to call +*/ +void PR_AddLoadFunc (progs_t *pr, pr_load_func_t *func); + +/** Register a secondary function to be called after the progs code has been + loaded. These functions will be forgotten after each load. They will be + called in \e reverse order of being registered. + \param pr pointer to progs_t VM struct + \param func function to call +*/ +void PR_AddLoadFinishFunc (progs_t *pr, pr_load_func_t *func); + +/** Run all load functions. Any load function returning false will abort the + load operation. + \param pr pointer to progs_t VM struct + + Calls the first set of internal load functions followed by the supplied + symbol resolution function, if any (progs_t::resolve), the second set of + internal load functions. After that, any primary load functions are called + in order of registration followed by any \c .ctor functions in the progs, + then any secondary load functions the primary load functions registered + in \e reverse order of registration. +*/ +int PR_RunLoadFuncs (progs_t *pr); + +/** Validate the opcodes and statement addresses in the progs. This is an + internal load function. + \param pr pointer to progs_t VM struct + + \todo should this be elsewhere? +*/ +int PR_Check_Opcodes (progs_t *pr); + +//@} + +/** \defgroup progs_edict Edict management +*/ +//@{ + #define MAX_ENT_LEAFS 16 -typedef struct edict_s { +struct edict_s { qboolean free; - link_t area; // linked to a division node or leaf + link_t area; ///< linked to a division node or leaf int num_leafs; short leafnums[MAX_ENT_LEAFS]; - float freetime; // sv.time when the object was freed - void *data; // external per-edict data - pr_type_t v[1]; // fields from progs -} edict_t; + float freetime; ///< sv.time when the object was freed + void *data; ///< external per-edict data + pr_type_t v[1]; ///< fields from progs +}; #define EDICT_FROM_AREA(l) STRUCT_FROM_LINK(l,edict_t,area) -typedef struct progs_s progs_t; -typedef struct pr_resource_s pr_resource_t; - -//============================================================================ - -void PR_Init (void); -void PR_Init_Cvars (void); - -void PR_PrintStatement (progs_t *pr, dstatement_t *s, int contents); -void PR_PushFrame (progs_t *pr); -void PR_PopFrame (progs_t *pr); -void PR_EnterFunction (progs_t *pr, dfunction_t *f); -void PR_ExecuteProgram (progs_t *pr, func_t fnum); -void PR_LoadProgsFile (progs_t *pr, QFile *file, int size, int edicts, - int zone); -void PR_LoadProgs (progs_t *pr, const char *progsname, int edicts, int zone); -int PR_LoadStrings (progs_t *pr); -int PR_LoadDebug (progs_t *pr); -edict_t *PR_InitEdicts (progs_t *pr, int num_edicts); -int PR_Check_Opcodes (progs_t *pr); - -void PR_Profile_f (void); - +// pr_edict.c void ED_ClearEdict (progs_t *pr, edict_t *e, int val); edict_t *ED_Alloc (progs_t *pr); void ED_Free (progs_t *pr, edict_t *ed); +edict_t *ED_EdictNum(progs_t *pr, int n); +int ED_NumForEdict(progs_t *pr, edict_t *e); +void ED_Count (progs_t *pr); +qboolean PR_EdictValid (progs_t *pr, int e); +// pr_debug.c void ED_Print (progs_t *pr, edict_t *ed); +void ED_PrintEdicts (progs_t *pr, const char *fieldval); +void ED_PrintNum (progs_t *pr, int ent); + +// pr_parse.c +struct script_s; +qboolean ED_ParseEpair (progs_t *pr, pr_type_t *base, ddef_t *key, + const char *s); void ED_Write (progs_t *pr, QFile *f, edict_t *ed); -const char *ED_ParseEdict (progs_t *pr, const char *data, edict_t *ent); - void ED_WriteGlobals (progs_t *pr, QFile *f); -void ED_ParseGlobals (progs_t *pr, const char *data); - +void ED_ParseEdict (progs_t *pr, struct script_s *script, edict_t *ent); +void ED_ParseGlobals (progs_t *pr, struct script_s *script); void ED_LoadFromFile (progs_t *pr, const char *data); +void ED_EntityParseFunction (progs_t *pr); + +#define PR_edicts(p) ((byte *) *(p)->edicts) + +#define NEXT_EDICT(p,e) ((edict_t *) ((byte *) e + (p)->pr_edict_size)) +#define EDICT_TO_PROG(p,e) ((byte *) (e) - PR_edicts (p)) +#define PROG_TO_EDICT(p,e) ((edict_t *) (PR_edicts (p) + (e))) +#define NUM_FOR_BAD_EDICT(p,e) (EDICT_TO_PROG (p, e) / (p)->pr_edict_size) +#ifndef PR_PARANOID_PROGS +# define EDICT_NUM(p,n) (PROG_TO_EDICT (p, (n) * (p)->pr_edict_size)) +# define NUM_FOR_EDICT(p,e) NUM_FOR_BAD_EDICT (p, e) +#else +# define EDICT_NUM(p,n) ED_EdictNum (p, n) +# define NUM_FOR_EDICT(p,e) ED_NumForEdict (p, e) +#endif + +//@} + +/** \defgroup pr_symbols Symbol Management + Lookup functions for symbol name resolution. +*/ +//@{ ddef_t *PR_FieldAtOfs (progs_t *pr, int ofs); ddef_t *PR_GlobalAtOfs (progs_t *pr, int ofs); @@ -91,78 +243,490 @@ ddef_t *PR_FindField (progs_t *pr, const char *name); ddef_t *PR_FindGlobal (progs_t *pr, const char *name); dfunction_t *PR_FindFunction (progs_t *pr, const char *name); -#define NEXT_EDICT(p,e) ((edict_t *)( (byte *)e + (p)->pr_edict_size)) +int PR_ResolveGlobals (progs_t *pr); -#define PR_edicts(p) ((byte *)*(p)->edicts) +int PR_AccessField (progs_t *pr, const char *name, etype_t type, + const char *file, int line); -#define EDICT_TO_PROG(p,e) ((byte *)(e) - PR_edicts (p)) -#define PROG_TO_EDICT(p,e) ((edict_t *)(PR_edicts (p) + (e))) +void PR_Undefined (progs_t *pr, const char *type, const char *name) __attribute__((noreturn)); +//@} //============================================================================ +/** \defgroup progs_data_access Data Access + Macros for accessing data in the VM address space +*/ +//@{ +/** \defgroup prda_globals Globals +*/ +//@{ +/** Typed global access for internal use. + \param p pointer to progs_t VM struct + \param o offset into global data space + \param t typename prefix (see pr_type_u) + \return lvalue of the appropriate type +*/ #define G_var(p,o,t) ((p)->pr_globals[o].t##_var) +/** Access a float global. Can be assigned to. + + QC type: \c float + \param p pointer to progs_t VM struct + \param o offset into global data space + \return float lvalue +*/ #define G_FLOAT(p,o) G_var (p, o, float) + +/** Access a integer global. Can be assigned to. + + QC type: \c integer + \param p pointer to progs_t VM struct + \param o offset into global data space + \return int lvalue +*/ #define G_INT(p,o) G_var (p, o, integer) + +/** Access a unsigned integer global. Can be assigned to. + + QC type: \c uinteger + \param p pointer to progs_t VM struct + \param o offset into global data space + \return unsigned int lvalue +*/ #define G_UINT(p,o) G_var (p, o, uinteger) + +/** Access a vector global. Can be assigned to. + + QC type: \c vector + \param p pointer to progs_t VM struct + \param o offset into global data space + \return vec3_t lvalue +*/ #define G_VECTOR(p,o) G_var (p, o, vector) + +/** Access a string index global. Can be assigned to. + + QC type: \c string + \param p pointer to progs_t VM struct + \param o offset into global data space + \return string_t lvalue +*/ #define G_STRING(p,o) G_var (p, o, string) + +/** Access a function global. Can be assigned to. + + QC type: \c void() + \param p pointer to progs_t VM struct + \param o offset into global data space + \return func_t lvalue +*/ #define G_FUNCTION(p,o) G_var (p, o, func) + +/** Access a pointer global. Can be assigned to. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param o offset into global data space + \return pointer_t lvalue +*/ #define G_POINTER(p,o) G_var (p, o, pointer) -#define G_EDICT(p,o) ((edict_t *)(PR_edicts (p) + G_INT (p, o))) -#define G_EDICTNUM(p,o) NUM_FOR_EDICT(p, G_EDICT (p, o)) -#define G_GSTRING(p,o) PR_GetString (p, G_STRING (p, o)) -#define G_DSTRING(p,o) PR_GetDString (p, G_STRING (p, o)) -#define G_GPOINTER(p,o) PR_GetPointer (p, o) -#define G_STRUCT(p,t,o) (*(t *)G_GPOINTER (p, o)) +/** Access an entity global. + + QC type: \c entity + \param p pointer to progs_t VM struct + \param o offset into global data space + \return C pointer to the entity +*/ +#define G_EDICT(p,o) ((edict_t *)(PR_edicts (p) + G_INT (p, o))) + +/** Access an entity global. + + QC type: \c entity + \param p pointer to progs_t VM struct + \param o offset into global data space + \return the entity number +*/ +#define G_EDICTNUM(p,o) NUM_FOR_EDICT(p, G_EDICT (p, o)) + +/** Access a a string global, converting it to a C string. Kills the program + if the string is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param o offset into global data space + \return const char * to the string +*/ +#define G_GSTRING(p,o) PR_GetString (p, G_STRING (p, o)) + +/** Access a dstring global. Kills the program if the dstring is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param o offset into global data space + \return dstring_t * to the dstring +*/ +#define G_DSTRING(p,o) PR_GetMutableString (p, G_STRING (p, o)) + +/** Access a pointer parameter. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param o offset into global data space + \return C pointer represented by the parameter. 0 offset -> NULL +*/ +#define G_GPOINTER(p,o) PR_GetPointer (p, o) + +/** Access an entity global. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param t C type of the structure + \param o offset into global data space + \return structure lvalue. use & to make a pointer of the appropriate type. +*/ +#define G_STRUCT(p,t,o) (*(t *)G_GPOINTER (p, o)) +//@} + +/** \defgroup prda_parameters Parameters +*/ +//@{ +/** Typed parameter access for internal use. + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \param t typename prefix (see pr_type_u) + \return lvalue of the appropriate type +*/ #define P_var(p,n,t) ((p)->pr_params[n]->t##_var) +/** Access a float parameter. Can be assigned to. + + QC type: \c float + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return float lvalue +*/ #define P_FLOAT(p,n) P_var (p, n, float) + +/** Access a integer parameter. Can be assigned to. + + QC type: \c integer + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return int lvalue +*/ #define P_INT(p,n) P_var (p, n, integer) + +/** Access a unsigned integer parameter. Can be assigned to. + + QC type: \c uinteger + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return unsigned int lvalue +*/ #define P_UINT(p,n) P_var (p, n, uinteger) + +/** Access a vector parameter. Can be used any way a vec3_t variable can. + + QC type: \c vector + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return vec3_t lvalue +*/ #define P_VECTOR(p,n) P_var (p, n, vector) + +/** Access a string index parameter. Can be assigned to. + + QC type: \c string + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return string_t lvalue +*/ #define P_STRING(p,n) P_var (p, n, string) + +/** Access a function parameter. Can be assigned to. + + QC type: \c void() + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return func_t lvalue +*/ #define P_FUNCTION(p,n) P_var (p, n, func) + +/** Access a pointer parameter. Can be assigned to. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return pointer_t lvalue +*/ #define P_POINTER(p,n) P_var (p, n, pointer) -#define P_EDICT(p,n) ((edict_t *)(PR_edicts (p) + P_INT (p, n))) -#define P_EDICTNUM(p,n) NUM_FOR_EDICT (p, P_EDICT (p, n)) -#define P_GSTRING(p,n) PR_GetString (p, P_STRING (p, n)) -#define P_DSTRING(p,n) PR_GetDString (p, P_STRING (p, n)) -#define P_GPOINTER(p,n) PR_GetPointer (p, P_POINTER (p, n)) -#define P_STRUCT(p,t,n) (*(t *)P_GPOINTER (p, n)) +/** Access an entity parameter. + + QC type: \c entity + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return C pointer to the entity +*/ +#define P_EDICT(p,n) ((edict_t *)(PR_edicts (p) + P_INT (p, n))) + +/** Access a entity parameter. + + QC type: \c entity + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return the entity number +*/ +#define P_EDICTNUM(p,n) NUM_FOR_EDICT (p, P_EDICT (p, n)) + +/** Access a string parameter, converting it to a C string. Kills the program + if the string is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return const char * to the string +*/ +#define P_GSTRING(p,n) PR_GetString (p, P_STRING (p, n)) + +/** Access a dstring parameter. Kills the program if the dstring is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return dstring_t * to the dstring +*/ +#define P_DSTRING(p,n) PR_GetMutableString (p, P_STRING (p, n)) + +/** Access a pointer parameter. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param n parameter number (0-7) + \return C pointer represented by the parameter. 0 offset -> NULL +*/ +#define P_GPOINTER(p,n) PR_GetPointer (p, P_POINTER (p, n)) + +/** Access a structure pointer parameter. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param t C type of the structure + \param n parameter number (0-7) + \return structure lvalue. use & to make a pointer of the appropriate type. +*/ +#define P_STRUCT(p,t,n) (*(t *)P_GPOINTER (p, n)) +//@} + +/** \defgroup prda_return Return value +*/ +//@{ +/** Typed return value access for internal use. + \param p pointer to progs_t VM struct + \param t typename prefix (see pr_type_u) + \return lvalue of the appropriate type +*/ #define R_var(p,t) ((p)->pr_return->t##_var) +/** Access the float return value. + + QC type: \c float + \param p pointer to progs_t VM struct + \return float lvalue +*/ #define R_FLOAT(p) R_var (p, float) + +/** Access the integer return value. + + QC type: \c integer + \param p pointer to progs_t VM struct + \return int lvalue +*/ #define R_INT(p) R_var (p, integer) + +/** Access the unsigned integer return value. + + QC type: \c uinteger + \param p pointer to progs_t VM struct + \return unsigned int lvalue +*/ #define R_UINT(p) R_var (p, uinteger) + +/** Access the vector return value. + + QC type: \c vector + \param p pointer to progs_t VM struct + \return vec3_t lvalue +*/ #define R_VECTOR(p) R_var (p, vector) + +/** Access the string index return value. + + QC type: \c string + \param p pointer to progs_t VM struct + \return string_t lvalue +*/ #define R_STRING(p) R_var (p, string) + +/** Access the function return value. + + QC type: \c void() + \param p pointer to progs_t VM struct + \return func_t lvalue +*/ #define R_FUNCTION(p) R_var (p, func) + +/** Access the pointer return value. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \return pointer_t lvalue +*/ #define R_POINTER(p) R_var (p, pointer) -#define RETURN_STRING(p, s) (R_STRING (p) = PR_SetReturnString((p), s)) -#define RETURN_EDICT(p, e) (R_STRING (p) = EDICT_TO_PROG(p, e)) -#define RETURN_POINTER(pr,p) (R_POINTER (pr) = POINTER_TO_PROG (pr, p)) -#define RETURN_VECTOR(p, v) VectorCopy (v, R_VECTOR (p)) +/** Set the return value to the given C string. The returned string will + eventually be garbage collected (see PR_SetReturnString()). + + QC type: \c string + \param p pointer to progs_t VM struct + \param s C string to be returned +*/ +#define RETURN_STRING(p,s) (R_STRING (p) = PR_SetReturnString((p), s)) + +/** Set the return value to the given C entity pointer. The pointer is + converted into a progs entity address. + + QC type: \c entity + \param p pointer to progs_t VM struct + \param e C entity pointer to be returned +*/ +#define RETURN_EDICT(p,e) (R_STRING (p) = EDICT_TO_PROG(p, e)) + +/** Set the return value to the given C pointer. NULL is converted to 0. + + QC type: \c void[] + \param p pointer to progs_t VM struct + \param ptr C entity pointer to be returned +*/ +#define RETURN_POINTER(p,ptr) (R_POINTER (p) = PR_SetPointer (p, ptr)) + +/** Set the return value to the given vector. The pointer is + converted into a progs entity address. + + QC type: \c vector + \param p pointer to progs_t VM struct + \param v C entity pointer to be returned +*/ +#define RETURN_VECTOR(p,v) VectorCopy (v, R_VECTOR (p)) +//@} + +/** \defgroup prda_entity_fields Entity Fields +*/ +//@{ +/** Typed entity field access for internal use. + \param e pointer to the entity + \param o field offset into entity data space + \param t typename prefix (see pr_type_u) + \return lvalue of the appropriate type +*/ #define E_var(e,o,t) ((e)->v[o].t##_var) + +/** Access a float entity field. Can be assigned to. + + QC type: \c float + \param e pointer to the entity + \param o field offset into entity data space + \return float lvalue +*/ #define E_FLOAT(e,o) E_var (e, o, float) + +/** Access a integer entity field. Can be assigned to. + + QC type: \c integer + \param e pointer to the entity + \param o field offset into entity data space + \return int lvalue +*/ #define E_INT(e,o) E_var (e, o, integer) + +/** Access a unsigned integer entity field. Can be assigned to. + + QC type: \c uinteger + \param e pointer to the entity + \param o field offset into entity data space + \return unsigned int lvalue +*/ #define E_UINT(e,o) E_var (e, o, uinteger) + +/** Access a vector entity field. Can be used any way a vec3_t variable can. + + QC type: \c vector + \param e pointer to the entity + \param o field offset into entity data space + \return vec3_t lvalue +*/ #define E_VECTOR(e,o) E_var (e, o, vector) + +/** Access a string index entity field. Can be assigned to. + + QC type: \c string + \param e pointer to the entity + \param o field offset into entity data space + \return string_t lvalue +*/ #define E_STRING(e,o) E_var (e, o, string) -#define E_FUNCTION(p,o) E_var (p, o, func) -#define E_POINTER(p,o) E_var (p, o, pointer) +/** Access a function entity field. Can be assigned to. + + QC type: \c void() + \param e pointer to the entity + \param o field offset into entity data space + \return func_t lvalue +*/ +#define E_FUNCTION(e,o) E_var (e, o, func) + +/** Access a pointer entity field. Can be assigned to. + + QC type: \c void[] + \param e pointer to the entity + \param o field offset into entity data space + \return pointer_t lvalue +*/ +#define E_POINTER(e,o) E_var (e, o, pointer) + + +/** Access a string entity field, converting it to a C string. Kills the + program if the string is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param e pointer to the entity + \param o field offset into entity data space + \return const char * to the string +*/ #define E_GSTRING(p,e,o) (PR_GetString (p, E_STRING (e, o))) -#define E_DSTRING(p,e,o) (PR_GetDString (p, E_STRING (e, o))) -/* +/** Access a dstring entity field. Kills the program if the dstring is invalid. + + QC type: \c string + \param p pointer to progs_t VM struct + \param e pointer to the entity + \param o field offset into entity data space + \return dstring_t * to the dstring +*/ +#define E_DSTRING(p,e,o) (PR_GetMutableString (p, E_STRING (e, o))) +//@} + +//@} + +/** \defgroup pr_builtins VM Builtin functions + Builtin management functions. + Builtins are arranged into groups of 65536 to allow for diffent expantion sets. eg, stock id is 0x0000xxxx, quakeforge is 0x000fxxxx. predefined groups go up to 0x0fffxxxx allowing 4096 different groups. Builtins @@ -170,74 +734,265 @@ dfunction_t *PR_FindFunction (progs_t *pr, const char *name); 0x8000000 to 0xffffffff is unavailable due to the builtin number being a negative statement address. */ +//@{ + #define PR_RANGE_SHIFT 16 #define PR_RANGE_MASK 0xffff0000 #define PR_RANGE_QF 0x000f #define PR_RANGE_AUTO 0x1000 +#define PR_RANGE_MAX 0x7fff #define PR_RANGE_NONE 0xffff #define PR_AUTOBUILTIN (PR_RANGE_AUTO << PR_RANGE_SHIFT) typedef void (*builtin_proc) (progs_t *pr); + +/** Create a static array of these and pass the array to PR_RegisterBuiltins() + to register the module's QC builtin functions. +*/ typedef struct { + /// QC name of the builtin. Must be an exact match for automaticly resolved + /// builtins (func = #0 in QC). NULL indicates end of array for + /// PR_RegisterBuiltins() const char *name; + /// Pointer to the C implementation of the builtin function. builtin_proc proc; + /// The number of the builtin for #N in QC. -1 for automatic allocation. + /// 0 or >= ::PR_AUTOBUILTIN is invalid. int binum; } builtin_t; +/** Aliased over the dfunction_t entry for builtin functions. Removes one + level of indirection when calling a builtin function. + \bug alignment access violations on 64 bit architectures (eg, alpha) +*/ typedef struct { int first_statement; builtin_proc func; } bfunction_t; - -pr_type_t *PR_GetGlobalPointer (progs_t *pr, const char *name); -func_t PR_GetFunctionIndex (progs_t *pr, const char *name); -int PR_GetFieldOffset (progs_t *pr, const char *name); - -void PR_Error (progs_t *pr, const char *error, ...) __attribute__((format(printf,2,3), noreturn)); -void PR_RunError (progs_t *pr, const char *error, ...) __attribute__((format(printf,2,3), noreturn)); - -void ED_PrintEdicts (progs_t *pr, const char *fieldval); -void ED_PrintNum (progs_t *pr, int ent); -void ED_Count (progs_t *pr); -void PR_Profile (progs_t *pr); - -pr_type_t *GetEdictFieldValue(progs_t *pr, edict_t *ed, const char *field); - +/** Register a set of builtin functions with the VM. Different VMs within the + same program can have different builtin sets. + \param pr pointer to progs_t VM struct + \param builtins array of builtin_t builtins +*/ void PR_RegisterBuiltins (progs_t *pr, builtin_t *builtins); + +/** Lookup a builtin function referred by name. + \param pr pointer to progs_t VM struct + \param name name of the builtin function to lookup + \return pointer to the builtin function entry, or NULL if not found +*/ builtin_t *PR_FindBuiltin (progs_t *pr, const char *name); + +/** Lookup a builtin function by builtin number. + \param pr pointer to progs_t VM struct + \param num number of the builtin function to lookup + \return pointer to the builtin function entry, or NULL if not found +*/ builtin_t *PR_FindBuiltinNum (progs_t *pr, int num); + +/** Fixup all automaticly resolved builtin functions (func = #0 in QC). + Performs any necessary builtin function number mapping. Also edits the + dfunction_t entry for all builtin functions to contain the function + pointer to the C implementation of the builtin function. Automaticly + during progs load. + \bug alignment access violations on 64 bit architectures (eg, alpha) + \param pr pointer to progs_t VM struct +*/ int PR_RelocateBuiltins (progs_t *pr); -int PR_ResolveGlobals (progs_t *pr); -typedef int pr_load_func_t (progs_t *); -void PR_AddLoadFunc (progs_t *pr, pr_load_func_t *func); -void PR_AddLoadFinishFunc (progs_t *pr, pr_load_func_t *func); -int PR_RunLoadFuncs (progs_t *pr); +//@} -// -// PR Strings stuff -// +/** \defgroup pr_strings String Management + Strings management functions. -qboolean PR_StringValid (progs_t *pr, int num); -const char *PR_GetString(progs_t *pr, int num); -struct dstring_s *PR_GetDString(progs_t *pr, int num); -int PR_SetString(progs_t *pr, const char *s); -int PR_SetReturnString(progs_t *pr, const char *s); -int PR_SetTempString(progs_t *pr, const char *s); -void PR_MakeTempString(progs_t *pr, int str); -int PR_NewString (progs_t *pr); + All strings accessable by the VM are stored within the VM address space. + These functions provide facilities to set permanent, dynamic and + temporary strings, as well as mutable strings using dstrings. + + Permanent strings are either supplied by the progs (+ve string index) or + set the the main program using PR_SetString(). Permanent strings can never + be altered and will exist until the next progs load. + + Dynamic strings can be freed at any time, but not altered. + + Temporary strings are always set by the main program using + PR_SetTempString() or PR_CatStrings(). They will be freed when the + current progs stack frame is cleared. Use PR_PushFrame() and PR_PopFrame() + around the calls to PR_SetTempString() and PR_ExecuteProgram() when using + strings as parameters to a progs function. + + Mutable strings are dstrings (dstring_t) mapped into the VM address space. + They can be created, altered, and destroyed at any time by the main + program (or the progs code via an appropriate builtin function). +*/ +//@{ + +/** Initialize the string tables using the strings supplied by the progs. + Automaticly called at progs load. + \param pr pointer to progs_t VM struct +*/ +int PR_LoadStrings (progs_t *pr); + +/** Check the validity of a string index. + \param pr pointer to progs_t VM struct + \param num string index to be validated + \return true if the index is valid, false otherwise +*/ +qboolean PR_StringValid (progs_t *pr, string_t num); + +/** Convert a string index to a C string. + \param pr pointer to progs_t VM struct + \param num string index to be converted + \return C pointer to the string. +*/ +const char *PR_GetString(progs_t *pr, string_t num); + +/** Retieve the dstring_t associated with a mutable string. + \param pr pointer to progs_t VM struct + \param num string index of the mutable string + \return the dstring implementing the mutable string +*/ +struct dstring_s *PR_GetMutableString(progs_t *pr, string_t num); + +/** Make a permanent progs string from the given C string. Will not create a + duplicate permanent string (temporary and mutable strings are not checked). + \param pr pointer to progs_t VM struct + \param s C string to be made into a permanent progs string + \return string index of the progs string +*/ +string_t PR_SetString(progs_t *pr, const char *s); + +/** Make a temporary progs string that will survive across function returns. + Will not duplicate a permanent string. If a new progs string is created, + it will be freed after ::PR_RS_SLOTS calls to this function. ie, up to + ::PR_RS_SLOTS return strings can exist at a time. + \param pr pointer to progs_t VM struct + \param s C string to be returned to the progs code + \return string index of the progs string +*/ +string_t PR_SetReturnString(progs_t *pr, const char *s); + +/** Make a temporary progs string that will be freed when the current progs + stack frame is exited. Will not duplicate a permantent string. + \param pr pointer to progs_t VM struct + \param s C string + \return string index of the progs string +*/ +string_t PR_SetTempString(progs_t *pr, const char *s); + +/** Make a tempoary progs string that is the concatenation of two C strings. + \param pr pointer to progs_t VM struct + \param a C string + \param b C string + \return string index of the progs string that represents the concatenation + of strings a and b +*/ +string_t PR_CatStrings (progs_t *pr, const char *a, const char *b); + +/** Convert a mutable string to a temporary string. + \param pr pointer to progs_t VM struct + \param str string index of the mutable string to be converted +*/ +void PR_MakeTempString(progs_t *pr, string_t str); + +/** Create a new mutable string. + \param pr pointer to progs_t VM struct + \return string index of the newly created mutable string +*/ +string_t PR_NewMutableString (progs_t *pr); + +/** Make a dynamic progs string from the given C string. Will not create a + duplicate permanent string (temporary, dynamic and mutable strings are + not checked). + \param pr pointer to progs_t VM struct + \param s C string to be made into a permanent progs string + \return string index of the progs string +*/ +string_t PR_SetDynamicString (progs_t *pr, const char *s); + +/** Clear all of the return string slots. Called at progs load. + \param pr pointer to progs_t VM struct +*/ void PR_ClearReturnStrings (progs_t *pr); -void PR_FreeString (progs_t *pr, int str); + +/** Destroy a mutable, dynamic or temporary string. + \param pr pointer to progs_t VM struct + \param str string index of the string to be destroyed +*/ +void PR_FreeString (progs_t *pr, string_t str); + +/** Free all the temporary strings allocated in the current stack frame. + \param pr pointer to progs_t VM struct +*/ void PR_FreeTempStrings (progs_t *pr); + +/** Formatted printing similar to C's vsprintf, but using QC types. + The format string is a string of characters (other than \c \%) to be printed + that includes optional format specifiers, one for each arg to be printed. + A format specifier can be one of two forms: +
    +
  • \c "%%" print a single \c '%' +
  • %[flags][field width][precision](conversion specifier) print a + single arg of the type specified by the conversion specifier using + the given format. +
+ +
    +
  • flags (optional) +
      +
    • \c '#' print the value using an alternat form +
    • \c '0' the value should be zero padded +
    • \c '-' the value is to be left adjusted on the field bondary. +
    • ' ' A blank should be left before a positve number (or + empty string) produced by a signed conversion. +
    • \c '+' A sign (+ or -) will always be placed before a number + produced by a signed conversion. +
    +
  • field width (optional) +
      +
    • a string of decimal digits with nonzero first digit. +
    +
  • precision (optional) +
      +
    • a string consisting of '.' followed by 0 or more decimal + digits. +
    +
  • conversion specifier (mandatory) +
      +
    • \c '@' \c id Not yet implemented. Silently ignored. +
    • \c 'e' \c entity Prints the edict number of the given entity ("%i") +
    • \c 'i' \c integer Print a integer value. ("%i") +
    • \c 'f' \c float Print a float value. ("%f") +
    • \c 'g' \c float Print a float value. ("%f") +
    • \c 'p' \c void[] Print a pointer value. ("%f") +
    • \c 's' \c string Print a string value. ("%f") +
    • \c 'v' \c vector Print a vector value. ("'%g %g %g'") +
    • \c 'q' \c quaternion Print a quaternion value. ("'%g %g %g %g'") +
    • \c 'x' \c uinteger Print an unsigned integer value. ("%x") +
    +
+ + For details, refer to the C documentation for printf. + + \param pr pointer to progs_t VM struct + \param result dstring to which printing is done + \param name name to be reported in case of errors + \param format the format string + \param count number of args + \param args array of pointers to the values to be printed +*/ void PR_Sprintf (progs_t *pr, struct dstring_s *result, const char *name, const char *format, int count, pr_type_t **args); -// -// PR Resources stuff -// +//@} + +/** \defgroup pr_resources Resource Management + Builtin module private data management. +*/ +//@{ void PR_Resources_Init (progs_t *pr); void PR_Resources_Clear (progs_t *pr); @@ -245,21 +1000,88 @@ void PR_Resources_Register (progs_t *pr, const char *name, void *data, void (*clear)(progs_t *, void *)); void *PR_Resources_Find (progs_t *pr, const char *name); -// -// PR Zone stuff -// +#define PR_RESMAP(type) struct { type *_free; type **_map; unsigned _size; } + +#define PR_RESNEW(type,map) \ + type *t; \ + \ + if (!map._free) { \ + int i, size; \ + map._size++; \ + size = map._size * sizeof (type *); \ + map._map = realloc (map._map, size); \ + if (!map._map) \ + return 0; \ + map._free = calloc (1024, sizeof (type)); \ + if (!map._free) \ + return 0; \ + map._map[map._size - 1] = map._free; \ + for (i = 0; i < 1023; i++) \ + *(type **) &map._free[i] = &map._free[i + 1]; \ + *(type **) &map._free[i] = 0; \ + } \ + t = map._free; \ + map._free = *(type **) t; \ + memset (t, 0, sizeof (type)); \ + return t + +#define PR_RESFREE(type,map,t) \ + memset (t, 0, sizeof (type)); \ + *(type **) t = map._free; \ + map._free = t + +#define PR_RESRESET(type,map) \ + unsigned i, j; \ + if (!map._size) \ + return; \ + for (i = 0; i < map._size; i++) { \ + map._free = map._map[i]; \ + for (j = 0; j < 1023; j++) \ + *(type **) &map._free[j] = &map._free[j + 1]; \ + if (i < map._size - 1) \ + *(type **) &map._free[j] = &map._map[i + 1][0]; \ + } \ + map._free = map._map[0]; + +#define PR_RESGET(map,col) \ + unsigned row = ~col / 1024; \ + col = ~col % 1024; \ + if (row >= map._size) \ + return 0; \ + return &map._map[row][col] + +#define PR_RESINDEX(map,ptr) \ + unsigned i; \ + for (i = 0; i < map._size; i++) { \ + long d = ptr - map._map[i]; \ + if (d >= 0 && d < 1024) \ + return ~(i * 1024 + d); \ + } \ + return 0 + +//@} + +/** \defgroup pr_zone VM memory management. + + Used to allocate and free memory in the VM address space. +*/ +//@{ void PR_Zone_Init (progs_t *pr); void PR_Zone_Free (progs_t *pr, void *ptr); void *PR_Zone_Malloc (progs_t *pr, int size); void *PR_Zone_Realloc (progs_t *pr, void *ptr, int size); -// -// PR Debug stuff -// +//@} + +/** \defgroup debug VM Debugging + Progs debugging support. +*/ +//@{ void PR_Debug_Init (void); void PR_Debug_Init_Cvars (void); +int PR_LoadDebug (progs_t *pr); pr_auxfunction_t *PR_Get_Lineno_Func (progs_t *pr, pr_lineno_t *lineno); unsigned int PR_Get_Lineno_Addr (progs_t *pr, pr_lineno_t *lineno); unsigned int PR_Get_Lineno_Line (progs_t *pr, pr_lineno_t *lineno); @@ -267,8 +1089,10 @@ pr_lineno_t *PR_Find_Lineno (progs_t *pr, unsigned int addr); const char *PR_Get_Source_File (progs_t *pr, pr_lineno_t *lineno); const char *PR_Get_Source_Line (progs_t *pr, unsigned int addr); ddef_t *PR_Get_Local_Def (progs_t *pr, int offs); +void PR_PrintStatement (progs_t *pr, dstatement_t *s, int contents); void PR_DumpState (progs_t *pr); void PR_StackTrace (progs_t *pr); +void PR_Profile (progs_t *pr); extern struct cvar_s *pr_debug; extern struct cvar_s *pr_deadbeef_ents; @@ -276,15 +1100,20 @@ extern struct cvar_s *pr_deadbeef_locals; extern struct cvar_s *pr_boundscheck; extern struct cvar_s *pr_faultchecks; -extern const char *pr_gametype; +//@} -// -// PR Cmds stuff -// +/** \defgroup pr_cmds Quake and Quakeworld common builtins + \todo This doesn't really belong in progs. +*/ +//@{ char *PF_VarString (progs_t *pr, int first); void PR_Cmds_Init (progs_t *pr); +extern const char *pr_gametype; + +//@} + //============================================================================ #define MAX_STACK_DEPTH 64 @@ -300,9 +1129,30 @@ typedef struct { } prstack_t; struct progs_s { + edict_t **edicts; + int *num_edicts; + int *reserved_edicts; ///< alloc will start at reserved_edicts+1 + void (*unlink) (edict_t *ent); + void (*flush) (void); + int (*parse_field) (progs_t *pr, const char *key, const char *value); + int (*prune_edict) (progs_t *pr, edict_t *ent); + void (*free_edict) (progs_t *pr, edict_t *ent); + + int null_bad; + int no_exec_limit; + + void (*file_error) (progs_t *pr, const char *path); + void *(*load_file) (progs_t *pr, const char *path); + void *(*allocate_progs_mem) (progs_t *pr, int size); + void (*free_progs_mem) (progs_t *pr, void *mem); + + int (*resolve) (progs_t *pr); + const char *progs_name; + dprograms_t *progs; int progs_size; + unsigned short crc; struct memzone_s *zone; int zone_size; @@ -320,10 +1170,12 @@ struct progs_s { int max_load_funcs; pr_load_func_t **load_funcs; - // cleared each load + /// cleared each load + //@{ int num_load_finish_funcs; int max_load_finish_funcs; pr_load_func_t **load_finish_funcs; + //@} struct dstring_mem_s *ds_mem; strref_t *free_string_refs; @@ -342,19 +1194,20 @@ struct progs_s { ddef_t *pr_globaldefs; ddef_t *pr_fielddefs; dstatement_t *pr_statements; - pr_type_t *pr_globals; // same as pr_global_struct + pr_type_t *pr_globals; int globals_size; pr_type_t *pr_return; pr_type_t *pr_params[MAX_PARMS]; - int pr_param_size; // covers both params and return + int pr_param_size; ///< covers both params and return - int pr_edict_size; // in bytes - int pr_edictareasize; // for bounds checking, starts at 0 + int pr_edict_size; ///< in bytes + int pr_edictareasize; ///< for bounds checking, starts at 0 int pr_argc; qboolean pr_trace; + int pr_trace_depth; dfunction_t *pr_xfunction; int pr_xstatement; @@ -364,28 +1217,7 @@ struct progs_s { int localstack[LOCALSTACK_SIZE]; int localstack_used; - edict_t **edicts; - int *num_edicts; - int *reserved_edicts; //alloc will start at reserved_edicts+1 double *time; - int null_bad; - int no_exec_limit; - - unsigned short crc; - - void (*unlink) (edict_t *ent); - void (*flush) (void); - - int (*parse_field) (progs_t *pr, const char *key, const char *value); - int (*prune_edict) (progs_t *pr, edict_t *ent); - void (*free_edict) (progs_t *pr, edict_t *ent); - - void (*file_error) (progs_t *pr, const char *path); - void *(*load_file) (progs_t *pr, const char *path); - void *(*allocate_progs_mem) (progs_t *pr, int size); - void (*free_progs_mem) (progs_t *pr, void *mem); - - int (*resolve) (progs_t *pr); pr_resource_t *resources; struct hashtab_s *resource_hash; @@ -397,19 +1229,23 @@ struct progs_s { struct hashtab_s *protocols; // debug info + + /// debug info + //@{ const char *debugfile; struct pr_debug_header_s *debug; struct pr_auxfunction_s *auxfunctions; struct pr_auxfunction_s **auxfunction_map; struct pr_lineno_s *linenos; ddef_t *local_defs; + //@} - // required globals + /// required globals (for OP_STATE) struct { float *time; int *self; } globals; - // required fields (offsets into the edict) + /// required fields (for OP_STATE) struct { int nextthink; int frame; @@ -418,16 +1254,37 @@ struct progs_s { } fields; }; +/** \addtogroup progs_data_access +*/ +//@{ + +/** Convert a progs offset/pointer to a C pointer. + \param pr pointer to progs_t VM struct + \param o offset into global data space + \return C pointer represented by the parameter. 0 offset -> NULL +*/ static inline pr_type_t * PR_GetPointer (progs_t *pr, int o) { return o ? pr->pr_globals + o : 0; } +/** Convert a C pointer to a progs offset/pointer. + \param pr pointer to progs_t VM struct + \param p C pointer to be converted. + \return Progs offset/pointer represented by \c p. NULL -> 0 offset +*/ static inline pointer_t PR_SetPointer (progs_t *pr, void *p) { return p ? (pr_type_t *) p - pr->pr_globals : 0; } +//@} + +//@} + +/** \example vm-exec.c +*/ + #endif//__QF_progs_h diff --git a/include/QF/qargs.h b/include/QF/qargs.h index 8579fc2bd..fff3ae5d3 100644 --- a/include/QF/qargs.h +++ b/include/QF/qargs.h @@ -31,6 +31,13 @@ #ifndef __qargs_h #define __qargs_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup misc +*/ +//@{ + #include "QF/qtypes.h" extern int com_argc; @@ -44,4 +51,7 @@ void COM_Init (void); void COM_Init_Cvars (void); void COM_InitArgv (int argc, const char **argv); +//@} +//@} + #endif // __qargs_h diff --git a/include/QF/qendian.h b/include/QF/qendian.h index d2e4418f7..70ea44674 100644 --- a/include/QF/qendian.h +++ b/include/QF/qendian.h @@ -31,6 +31,13 @@ #ifndef __qendian_h #define __qendian_h +/** \addtogroup utils */ +//@{ + +/** \defgroup qendian Endian handling functions +*/ +//@{ + #include "QF/qtypes.h" #ifndef NULL @@ -89,4 +96,7 @@ byte ReadByte (struct QFile_s *file); unsigned short ReadShort (struct QFile_s *file); unsigned long ReadLong (struct QFile_s *file); +//@} +//@} + #endif // __qendian_h diff --git a/include/QF/quakefs.h b/include/QF/quakefs.h index 10ed6abd0..4bebb72e7 100644 --- a/include/QF/quakefs.h +++ b/include/QF/quakefs.h @@ -31,6 +31,13 @@ #ifndef __quakefs_h #define __quakefs_h +/** \addtogroup utils */ +//@{ + +/** \defgroup quakefs Quake Filesystem +*/ +//@{ + #include "QF/qtypes.h" #include "QF/quakeio.h" @@ -125,4 +132,7 @@ void QFS_FilelistFree (filelist_t *list); // FIXME: This is here temporarily until fs_usercfg gets sorted out char *expand_squiggle (const char *path); +//@} +//@} + #endif // __quakefs_h diff --git a/include/QF/quakeio.h b/include/QF/quakeio.h index 1207b5bec..d311bc052 100644 --- a/include/QF/quakeio.h +++ b/include/QF/quakeio.h @@ -30,6 +30,13 @@ #ifndef __quakeio_h #define __quakeio_h +/** \addtogroup utils */ +//@{ + +/** \defgroup quakeio File IO +*/ +//@{ + typedef struct QFile_s QFile; int Qrename(const char *old, const char *new); @@ -53,4 +60,7 @@ int Qflush(QFile *file); int Qeof(QFile *file); const char *Qgetline(QFile *file); +//@} +//@} + #endif /*__quakeio_h*/ diff --git a/include/QF/sizebuf.h b/include/QF/sizebuf.h index fad7debb9..994363cb6 100644 --- a/include/QF/sizebuf.h +++ b/include/QF/sizebuf.h @@ -28,6 +28,14 @@ #ifndef __sizebuf_h #define __sizebuf_h +/** \addtogroup utils */ +//@{ + +/** \defgroup sizebuf Fixed Size Buffers + Fixed size buffer management +*/ +//@{ + #include "QF/qtypes.h" typedef struct sizebuf_s @@ -46,4 +54,7 @@ void *SZ_GetSpace (sizebuf_t *buf, int length); void SZ_Write (sizebuf_t *buf, const void *data, int length); void SZ_Print (sizebuf_t *buf, const char *data); // strcats onto the sizebuf +//@} +//@} + #endif // __sizebuf_h diff --git a/include/QF/sys.h b/include/QF/sys.h index 15e543bc3..47bc6c401 100644 --- a/include/QF/sys.h +++ b/include/QF/sys.h @@ -29,6 +29,14 @@ #ifndef __sys_h #define __sys_h +/** \addtogroup utils */ +//@{ + +/** \defgroup sys Portability + Non-portable functions +*/ +//@{ + #include #include @@ -105,4 +113,7 @@ void Sys_DebugLog(const char *file, const char *fmt, ...) __attribute__((format( Sys_Error ("%s: Failed to allocate memory.", __FUNCTION__); \ } while (0) +//@} +//@} + #endif // __sys_h diff --git a/include/QF/va.h b/include/QF/va.h index 845b8bb5a..1c24796c9 100644 --- a/include/QF/va.h +++ b/include/QF/va.h @@ -30,9 +30,20 @@ #ifndef __va_h #define __va_h +/** \addtogroup utils */ +//@{ + +/** \addtogroup misc + Formatted printing. +*/ +//@{ + // does a varargs printf into a temp buffer char *va(const char *format, ...) __attribute__((format(printf,1,2))); // does a varargs printf into a malloced buffer char *nva(const char *format, ...) __attribute__((format(printf,1,2))); +//@} +//@} + #endif // __va_h diff --git a/include/QF/ver_check.h b/include/QF/ver_check.h index 030bb1e3e..5f4cce3fb 100644 --- a/include/QF/ver_check.h +++ b/include/QF/ver_check.h @@ -29,6 +29,13 @@ #ifndef __ver_check_h_ #define __ver_check_h_ +/** \addtogroup utils */ +//@{ + +/** \addtogroup misc +*/ +//@{ + /* ver_compare @@ -38,4 +45,7 @@ */ int ver_compare (const char *, const char *); +//@} +//@} + #endif // __ver_check_h_ diff --git a/include/QF/wad.h b/include/QF/wad.h index b3d7b463f..e3c847d43 100644 --- a/include/QF/wad.h +++ b/include/QF/wad.h @@ -30,6 +30,14 @@ #ifndef _WAD_H #define _WAD_H +/** \addtogroup utils */ +//@{ + +/** \addtogroup wad + Wad Files +*/ +//@{ + #include "QF/wadfile.h" @@ -42,4 +50,7 @@ void *W_GetLumpName (const char *name); void SwapPic (qpic_t *pic); +//@} +//@} + #endif // _WAD_H diff --git a/include/QF/wadfile.h b/include/QF/wadfile.h index 0569bcbd9..0bf14fbaf 100644 --- a/include/QF/wadfile.h +++ b/include/QF/wadfile.h @@ -25,7 +25,14 @@ $Id$ */ -// wadfile.h + +/** \addtogroup utils */ +//@{ + +/** \defgroup wad Wad Files + Wadfile processing +*/ +//@{ #ifndef __QF_wadfile_h #define __QF_wadfile_h @@ -98,4 +105,7 @@ int wad_add_data (wad_t *wad, const char *lumpname, byte type, const void *data, int bytes); lumpinfo_t *wad_find_lump (wad_t *wad, const char *filename); +//@} +//@} + #endif//__QF_wadfile_h diff --git a/include/QF/zone.h b/include/QF/zone.h index 1507cb1ee..6973c2530 100644 --- a/include/QF/zone.h +++ b/include/QF/zone.h @@ -28,71 +28,70 @@ #ifndef __zone_h #define __zone_h -/* - memory allocation +/** \addtogroup utils */ +//@{ + +/** \defgroup zone Memory Management + + H_??? The hunk manages the entire memory block given to quake. It must be + contiguous. Memory can be allocated from either the low or high end in a + stack fashion. The only way memory is released is by resetting one of the + pointers. + + Hunk allocations should be given a name, so the Hunk_Print () function + can display usage. + + Hunk allocations are guaranteed to be 16 byte aligned. + + The video buffers are allocated high to avoid leaving a hole underneath + server allocations when changing to a higher video mode. -H_??? The hunk manages the entire memory block given to quake. It must be -contiguous. Memory can be allocated from either the low or high end in a -stack fashion. The only way memory is released is by resetting one of the -pointers. + Z_??? Zone memory functions used for small, dynamic allocations like text + strings from command input. There is only about 48K for it, allocated at + the very bottom of the hunk. -Hunk allocations should be given a name, so the Hunk_Print () function -can display usage. + Cache_??? Cache memory is for objects that can be dynamically loaded and + can usefully stay persistant between levels. The size of the cache + fluctuates from level to level. -Hunk allocations are guaranteed to be 16 byte aligned. - -The video buffers are allocated high to avoid leaving a hole underneath -server allocations when changing to a higher video mode. + To allocate a cachable object -Z_??? Zone memory functions used for small, dynamic allocations like text -strings from command input. There is only about 48K for it, allocated at -the very bottom of the hunk. - -Cache_??? Cache memory is for objects that can be dynamically loaded and -can usefully stay persistant between levels. The size of the cache -fluctuates from level to level. - -To allocate a cachable object + Temp_??? Temp memory is used for file loading and surface caching. The + size of the cache memory is adjusted so that there is a minimum of 512k + remaining for temp memory. -Temp_??? Temp memory is used for file loading and surface caching. The size -of the cache memory is adjusted so that there is a minimum of 512k remaining -for temp memory. + ------ Top of Memory ------- + high hunk allocations ------- Top of Memory ------- + <--- high hunk reset point held by vid -high hunk allocations + video buffer -<--- high hunk reset point held by vid + z buffer -video buffer + surface cache -z buffer + <--- high hunk used -surface cache + cachable memory -<--- high hunk used + <--- low hunk used -cachable memory + client and server low hunk allocations -<--- low hunk used - -client and server low hunk allocations - -<-- low hunk reset point held by host - -startup hunk allocations - -Zone block - ------ Bottom of Memory ----- + <-- low hunk reset point held by host + startup hunk allocations + Zone block + ----- Bottom of Memory ----- */ +//@{ typedef struct memzone_s memzone_t; @@ -174,4 +173,7 @@ void *QA_realloc (void *ptr, size_t size); void QA_free (void *ptr); char *QA_strdup (const char *s); +//@} +//@} + #endif // __zone_h diff --git a/libs/gamecode/builtins/bi_gib.c b/libs/gamecode/builtins/bi_gib.c index 87e91d468..fcd5bcef2 100644 --- a/libs/gamecode/builtins/bi_gib.c +++ b/libs/gamecode/builtins/bi_gib.c @@ -92,7 +92,7 @@ bi_gib_builtin_f (void) pr_list[i].integer_var = PR_SetTempString (builtin->pr, GIB_Argv(i)); P_INT (builtin->pr, 0) = GIB_Argc(); - P_INT (builtin->pr, 1) = POINTER_TO_PROG (builtin->pr, pr_list); + P_INT (builtin->pr, 1) = PR_SetPointer (builtin->pr, pr_list); PR_ExecuteProgram (builtin->pr, builtin->func); PR_PopFrame (builtin->pr); PR_Zone_Free (builtin->pr, pr_list); diff --git a/qw/include/sv_pr_cmds.h b/qw/include/sv_pr_cmds.h index 97b6d6191..6c55dc8c6 100644 --- a/qw/include/sv_pr_cmds.h +++ b/qw/include/sv_pr_cmds.h @@ -29,11 +29,6 @@ #ifndef __sv_pr_cmds_h #define __sv_pr_cmds_h -#ifndef PROGS_T -typedef struct progs_s progs_t; -# define PROGS_T -#endif - void PF_error (progs_t *pr); void PF_objerror (progs_t *pr); void PF_makevectors (progs_t *pr); diff --git a/qw/source/sv_pr_cmds.c b/qw/source/sv_pr_cmds.c index 57bf679e7..74d7e02c3 100644 --- a/qw/source/sv_pr_cmds.c +++ b/qw/source/sv_pr_cmds.c @@ -44,6 +44,7 @@ static __attribute__ ((unused)) const char rcsid[] = #include "QF/cmd.h" #include "QF/cvar.h" #include "QF/msg.h" +#include "QF/progs.h" #include "QF/ruamoko.h" #include "QF/sys.h" #include "QF/va.h" diff --git a/tools/qfcc/source/modules.c b/tools/qfcc/source/modules.c index f436e2022..61f055e8c 100644 --- a/tools/qfcc/source/modules.c +++ b/tools/qfcc/source/modules.c @@ -55,16 +55,22 @@ dump_methods (progs_t *pr, pr_method_list_t *methods, int class) int i; char mark = class ? '+' : '-'; const char *sel_id; + const char *types; while (methods) { pr_method_t *method = methods->method_list; for (i = 0; i < methods->method_count; i++) { - if (PR_StringValid (pr, method->method_name.sel_id)) - sel_id = PR_GetString (pr, method->method_name.sel_id); + if (PR_StringValid (pr, method->method_name)) + sel_id = PR_GetString (pr, method->method_name); else sel_id = ""; - printf (" %c%s %d @ %d\n", mark, sel_id, method->method_imp, - POINTER_TO_PROG (pr, method)); + if (PR_StringValid (pr, method->method_types)) + types = PR_GetString (pr, method->method_types); + else + types = ""; + printf (" %c%s %d @ %d %s\n", mark, sel_id, + method->method_imp, + PR_SetPointer (pr, method), types); method++; } methods = &G_STRUCT (pr, pr_method_list_t, methods->method_next); @@ -98,10 +104,10 @@ dump_class (progs_t *pr, pr_class_t *class) if (class->super_class) { if (PR_StringValid (pr, class->super_class)) super_class_name = PR_GetString (pr, class->super_class); - printf (" %s @ %d : %s\n", class_name, POINTER_TO_PROG (pr, class), + printf (" %s @ %d : %s\n", class_name, PR_SetPointer (pr, class), super_class_name); } else { - printf (" %s @ %d\n", class_name, POINTER_TO_PROG (pr, class)); + printf (" %s @ %d\n", class_name, PR_SetPointer (pr, class)); } printf (" %d %d %u %d\n", class->class_pointer, class->version, class->info, class->instance_size); @@ -120,7 +126,7 @@ dump_category (progs_t *pr, pr_category_t *category) if (PR_StringValid (pr, category->class_name)) class_name = PR_GetString (pr, category->class_name); printf (" %s (%s) @ %d\n", class_name, category_name, - POINTER_TO_PROG (pr, category)); + PR_SetPointer (pr, category)); dump_methods (pr, &G_STRUCT (pr, pr_method_list_t, category->instance_methods), 0); diff --git a/tools/qwaq/main.c b/tools/qwaq/main.c index ef538b1f0..9d8789ea9 100644 --- a/tools/qwaq/main.c +++ b/tools/qwaq/main.c @@ -117,6 +117,7 @@ init_qf (void) pr.load_file = load_file; pr.allocate_progs_mem = allocate_progs_mem; pr.free_progs_mem = free_progs_mem; + pr.no_exec_limit = 1; PR_Init_Cvars (); PR_Init ();