quakeforge/include/QF/progs.h
Bill Currie 98f773b0ed [gamecode] Expose value_string as PR_Debug_ValueString
PR_Debug_ValueString prints the value at the given offset using the
provided type to format the string. The formatted string is appended to
the provided dstring.
2022-09-15 00:30:21 +09:00

2260 lines
65 KiB
C

/*
progs.h
Progs VM Engine interface.
Copyright (C) 1996-1997 Id Software, Inc.
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
*/
#ifndef __QF_progs_h
#define __QF_progs_h
/** \defgroup progs QuakeC Virtual Machine (VM)
\image html vm-mem.svg
\image latex vm-mem.eps "VM memory map"
*/
#include "QF/math/bitop.h"
#include "QF/progs/pr_comp.h"
#include "QF/progs/pr_debug.h"
struct QFile_s;
/** \ingroup progs */
///@{
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
\ingroup progs
*/
///@{
/** Initialize the progs engine.
The first call will initialize subsystems common to all progs instances.
\param pr The progs engine instance to initialize.
*/
void PR_Init (progs_t *pr);
/** Shut down the progs engine.
Shuts down only the specified progs engine. Frees resources allocated
during progs init.
\param pr The progs engine instance to shut down.
*/
void PR_Shutdown (progs_t *pr);
/** 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
\ingroup progs
*/
///@{
/** Ensure P_* macros point to the right place for passing parameters to progs
functions.
\param pr pointer to ::progs_t VM struct
\warning Failure to use this macro before assigning to the P_* macros can
cause corruption of the VM data due to "register" based calling. Can be
safely ignored for parameterless functions, or forwarding parameters
though a builtin. However, it is ok (and encouraged) to call
PR_SetupParams instead, as this macro calls PR_SetupParams with
PR_MAX_PARAMS and 1 for the alignment.
\hideinitializer
*/
#define PR_RESET_PARAMS(pr) PR_SetupParams (pr, PR_MAX_PARAMS, 1)
/** \name Detouring Function Calls
These functions allow a builtin function that uses PR_CallFunction() to
safely insert a call to another VM function. The +initialize diversion
required by Objective-QuakeC uses this.
PR_PushFrame (pr);
__auto_type params = PR_SaveParams (pr);
... set up parameters to detour_function
PR_ExecuteProgram (pr, detour_function)
PR_RestoreParams (pr, params);
PR_PopFrame (pr);
*/
///@{
typedef struct pr_stashed_params_s {
pr_type_t *param_ptrs[2];
int argc;
pr_type_t params[1];
} pr_stashed_params_t;
/** Save the current parameters to the provided stash.
\warning The memory for the parameter stash is allocated using
alloca().
\param pr pointer to ::progs_t VM struct
\return Pointer to a newly allocated and initialized parameter
stash that has the current parameters saved to it.
\hideinitializer
*/
#define PR_SaveParams(pr) \
_PR_SaveParams((pr), \
alloca (field_offset (pr_stashed_params_t, \
params[(pr)->pr_argc \
* (pr)->pr_param_size])))
/** [INTERNAL] Save the current parameters to the provided stash.
\warning Requires \a params to be correctly allocated. Use
PR_SaveParams instead as it will create a suitable stash for
saving the parameters.
\param pr pointer to ::progs_t VM struct
\param params location to save the parameters, must be of adequade size
to hold \a pr_argc * \a pr_param_size words in \a params
\return \a params Allows the likes of:
__auto_type params = PR_SaveParams (pr);
*/
pr_stashed_params_t *_PR_SaveParams (progs_t *pr, pr_stashed_params_t *params);
/** Restore the parameters saved by PR_SaveParams().
\param pr pointer to ::progs_t VM struct
\param params pointer to stash created by PR_SaveParams()
*/
void PR_RestoreParams (progs_t *pr, pr_stashed_params_t *params);
///@}
/** 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);
/** Reserve space on the data stack and set up the param pointers.
For v6p progs, this only sets up the param pointers as v6p progs do not
have a data stack.
For Ruamoko progs, space for at least \a num_params (each being 4 words)
is created on the stack, with a minimum alignment of min_alignment words,
or 4, whichever is larger.
\param pr pointer to ::progs_t VM struct
\param num_params Number of parameter slots needed for the function call.
Each slot is 4 words. dvec4 and lvec4 parameters require
8 words and must be 8-word aligned. dvec3 and lvec3
also require 8 words due to the minimum 4 word alignment,
but have no alignment requirements themselves. Be sure to
take this into account in size calculations.
\param min_alignment Minimum number of words to which the stack will be
aligned. Must be a power of two. Note that when passing
dvec4 or lvec4 parameters, they have a hardware-enforced
requirement of 8 word alignment. This means that for
something like (int, lvec4), there will be an unused
parameter slot between the int and the lvec4.
Ignored for v6p progs.
\return Pointer to the base of the created parameter area. For
v6p progs, this is just .param_0, but for Ruamoko progs
this will be the current top of the the data stack after
adjustment for the parameter space.
\note Attempting to pass more than PR_MAX_PARAMS parameters to v6p progs
is a hard error.
*/
pr_type_t *PR_SetupParams (progs_t *pr, int num_params, int min_alignment);
/** 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 \p 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
\note Calls PR_CallFunction()
*/
void PR_ExecuteProgram (progs_t *pr, pr_func_t fnum);
/** Setup to call a function. If \p fnum is a builtin rather than a progs
function, then the function is called immediately. When called from a
builtin function, and \p fnum is not a builtin, 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
\param return_ptr pointer to location in which return values will be
written
\return true if \p fnum was a progs function, false if \p fnum was
a builtin
\note Called by PR_ExecuteProgram, so the only time this should be called
is in a builtin function that calls a progs function and returns
immediately (eg, to implement `return progsfunc();`).
*/
int PR_CallFunction (progs_t *pr, pr_func_t fnum, pr_type_t *return_ptr);
///@}
/** \defgroup progs_load Loading
\ingroup progs
*/
///@{
/** 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 from which to read progs data
\param size bytes of \p file to read
\note \e All runtime strings (permanent or temporary) are allocated from
the VM's dynamic memory space, so be sure \p zone is of sufficient size
(by setting pr->zone_size prior to calling). So far, 1MB has proven more
than sufficient for Quakeword, even when using Ruamoko objects.
\note If entities are used, ensure pr->max_edicts is set appropriately
prior to calling.
*/
void PR_LoadProgsFile (progs_t *pr, struct QFile_s *file, int size);
/** 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
*/
void PR_LoadProgs (progs_t *pr, const char *progsname);
/** 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
\return true for success, false for failure
Calls the first set of internal load functions followed by the supplied
symbol resolution function, if any (progs_t::resolve), then the second set
of internal load functions. After that, any primary load functions are
called in order of registration, and if there is no debug handler,
PR_RunPostLoadFuncs() is called.
*/
int PR_RunLoadFuncs (progs_t *pr);
/** Run any progs-dependent load functions.
\param pr pointer to ::progs_t VM struct
\return true for success, false for failure
This means any \c .ctor functions in the progs, followed by any secondary
load functions registered by either the primary load functions or the
\.c ctor functions in \e reverse order of registration. This is called
automatically by PR_RunLoadFuncs() if there is no debug handler, otherwise
it is up to the host to call this function.
*/
int PR_RunPostLoadFuncs (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
\return true for success, false for failure
\todo should this be elsewhere?
*/
int PR_Check_Opcodes (progs_t *pr);
int PR_Check_v6p_Opcodes (progs_t *pr);
void PR_BoundsCheckSize (progs_t *pr, pr_ptr_t addr, unsigned size);
void PR_BoundsCheck (progs_t *pr, int addr, etype_t type);
///@}
/** \defgroup progs_edict Edict management
\ingroup progs
*/
///@{
struct edict_s {
qboolean free;
progs_t *pr; ///< progs owning this edict
pr_uint_t entnum; ///< number of this entity
pr_uint_t edict; ///< offset of this entity in pr_edict_area
float freetime; ///< sv.time when the object was freed
void *edata; ///< external per-edict data
};
// 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, pr_uint_t n) __attribute__((pure));
pr_uint_t ED_NumForEdict(progs_t *pr, edict_t *e) __attribute__((pure));
void ED_Count (progs_t *pr);
qboolean PR_EdictValid (progs_t *pr, pr_uint_t e) __attribute__((pure));
// pr_debug.c
void ED_Print (progs_t *pr, edict_t *ed, const char *fieldname);
void ED_PrintEdicts (progs_t *pr, const char *fieldval);
void ED_PrintNum (progs_t *pr, pr_int_t ent, const char *fieldname);
// pr_parse.c
struct script_s;
struct plitem_s;
struct hashctx_s;
qboolean ED_ParseEpair (progs_t *pr, pr_type_t *base, pr_def_t *key,
const char *s);
struct plitem_s *ED_EntityDict (progs_t *pr, edict_t *ed);
struct plitem_s *ED_GlobalsDict (progs_t *pr);
void ED_InitGlobals (progs_t *pr, struct plitem_s *globals);
void ED_InitEntity (progs_t *pr, struct plitem_s *entity, edict_t *ent);
struct plitem_s *ED_ConvertToPlist (struct script_s *script, int nohack,
struct hashctx_s **hashctx);
struct plitem_s *ED_Parse (progs_t *pr, const char *data);
void ED_LoadFromFile (progs_t *pr, const char *data);
void ED_EntityParseFunction (progs_t *pr, void *data);
#define PR_edicts(p) (*(p)->pr_edicts)
#define NEXT_EDICT(p,e) ((e) ? (e) + 1 : 0)
#define EDICT_TO_PROG(p,e) ((e) ? (e)->entnum * (p)->pr_edict_size : 0)
#define PROG_TO_EDICT(p,e) (&PR_edicts(p)[(e) / (p)->pr_edict_size])
#define NUM_FOR_BAD_EDICT(p,e) ((e) ? (e)->entnum : 0)
#ifndef PR_PARANOID_PROGS
# define EDICT_NUM(p,n) (PR_edicts (p) + (n))
# 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
\ingroup progs
Lookup functions for symbol name resolution.
*/
///@{
pr_def_t *PR_SearchDefs (pr_def_t *defs, unsigned num_defs, pr_ptr_t offset)
__attribute__((pure));
pr_def_t *PR_FieldAtOfs (progs_t *pr, pr_ptr_t ofs) __attribute__((pure));
pr_def_t *PR_GlobalAtOfs (progs_t *pr, pr_ptr_t ofs) __attribute__((pure));
pr_def_t *PR_FindField (progs_t *pr, const char *name);
pr_def_t *PR_FindGlobal (progs_t *pr, const char *name);
dfunction_t *PR_FindFunction (progs_t *pr, const char *name);
int PR_ResolveGlobals (progs_t *pr);
int PR_AccessField (progs_t *pr, const char *name, etype_t type,
const char *file, int line);
void PR_Undefined (progs_t *pr, const char *type, const char *name) __attribute__((noreturn));
///@}
//============================================================================
/** \defgroup progs_data_access Data Access
\ingroup progs
Macros for accessing data in the VM address space
*/
/** \defgroup prda_globals Globals
\ingroup progs_data_access
Typed global access macros. No checking is done against the QC type, but
the appropriate C type will be used.
*/
///@{
/** Access a global as an arbitray type.
More direct than G_STRUCT
\par QC type:
\c struct etc small enough to fit in a single parameter
\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.
\hideinitializer
*/
#define G_PACKED(p,t,o) (*(t *) &(p)->pr_globals[o])
/** \internal
\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
\hideinitializer
*/
#define G_var(p,o,t) G_PACKED(p, pr_##t##_t, o)
/** Access a float global. Can be assigned to.
\par QC type:
\c float
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return float lvalue
\hideinitializer
*/
#define G_FLOAT(p,o) G_var (p, o, float)
/** Access a double global. Can be assigned to.
\par QC type:
\c double
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return double lvalue
\hideinitializer
*/
#define G_DOUBLE(p,o) G_var (p, o, double)
/** Access an int global. Can be assigned to.
\par QC type:
\c int
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return int lvalue
\hideinitializer
*/
#define G_INT(p,o) G_var (p, o, int)
/** Access an unsigned int global. Can be assigned to.
\par QC type:
\c uint
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return unsigned int lvalue
\hideinitializer
*/
#define G_UINT(p,o) G_var (p, o, uint)
/** Access a long global. Can be assigned to.
\par QC type:
\c long
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return long lvalue
\hideinitializer
*/
#define G_LONG(p,o) G_var (p, o, long)
/** Access an unsigned long global. Can be assigned to.
\par QC type:
\c ulong
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return unsigned long lvalue
\hideinitializer
*/
#define G_ULONG(p,o) G_var (p, o, ulong)
/** Access a vector global. Can be assigned to.
\par QC type:
\c vector
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return vec3_t lvalue
\hideinitializer
*/
#define G_VECTOR(p,o) (&G_var (p, o, float))
/** Access a quaternion global. Can be assigned to.
\par QC type:
\c quaternion
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return quat_t lvalue
\hideinitializer
*/
#define G_QUAT(p,o) (&G_var (p, o, float))
/** Access a string index global. Can be assigned to.
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return pr_string_t lvalue
\hideinitializer
*/
#define G_STRING(p,o) G_var (p, o, string)
/** Access a function global. Can be assigned to.
\par QC type:
\c void()
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return pr_func_t lvalue
\hideinitializer
*/
#define G_FUNCTION(p,o) G_var (p, o, func)
/** Access a pointer global. Can be assigned to.
\par QC type:
\c void *
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return pr_ptr_t lvalue
\hideinitializer
*/
#define G_POINTER(p,o) G_var (p, o, ptr)
/** Access a field global.
\par QC type:
\c field
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return field offset
\hideinitializer
*/
#define G_FIELD(p,o) G_var (p, o, field)
/** Access an entity global.
\par QC type:
\c entity
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return entity "pointer"
\hideinitializer
*/
#define G_ENTITY(p,o) G_var (p, o, entity)
/** Access an entity global.
\par 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
\hideinitializer
*/
#define G_EDICT(p,o) PROG_TO_EDICT ((p), G_INT (p, o))
/** Access an entity global.
\par QC type:
\c entity
\param p pointer to ::progs_t VM struct
\param o offset into global data space
\return the entity number
\hideinitializer
*/
#define G_EDICTNUM(p,o) NUM_FOR_EDICT(p, G_EDICT (p, o))
/** Access a string global, converting it to a C string. Kills the program
if the string is invalid.
\par 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
\hideinitializer
*/
#define G_GSTRING(p,o) PR_GetString (p, G_STRING (p, o))
/** Access a dstring global. Kills the program if the dstring is invalid.
\par 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
\hideinitializer
*/
#define G_DSTRING(p,o) PR_GetMutableString (p, G_STRING (p, o))
/** Access a pointer parameter.
\par 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
\hideinitializer
*/
#define G_GPOINTER(p,o) PR_GetPointer (p, o)
/** Access a structure global. Can be assigned to.
\par 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.
\hideinitializer
*/
#define G_STRUCT(p,t,o) (*(t *)G_GPOINTER (p, o))
///@}
/** \defgroup prda_parameters Parameters
\ingroup progs_data_access
Typed parameter access macros. No checking is done against the QC type, but
the appropriate C type will be used.
*/
///@{
/** Access a parameter as an arbitray type.
\par QC type:
\c struct etc small enough to fit in a single parameter
\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.
\hideinitializer
*/
#define P_PACKED(p,t,n) (*(t *) (p)->pr_params[n])
/** \internal
\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
\hideinitializer
*/
#define P_var(p,n,t) P_PACKED(p, pr_##t##_t, n)
/** Access a float parameter. Can be assigned to.
\par QC type:
\c float
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return float lvalue
\hideinitializer
*/
#define P_FLOAT(p,n) P_var (p, n, float)
/** Access a double parameter. Can be assigned to.
\par QC type:
\c double
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return double lvalue
\hideinitializer
*/
#define P_DOUBLE(p,n) P_var (p, n, double)
/** Access an int parameter. Can be assigned to.
\par QC type:
\c int
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return int lvalue
\hideinitializer
*/
#define P_INT(p,n) P_var (p, n, int)
/** Access an unsigned int parameter. Can be assigned to.
\par QC type:
\c uint
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return unsigned int lvalue
\hideinitializer
*/
#define P_UINT(p,n) P_var (p, n, uint)
/** Access a long parameter. Can be assigned to.
\par QC type:
\c long
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return long lvalue
\hideinitializer
*/
#define P_LONG(p,n) P_var (p, n, long)
/** Access an unsigned long parameter. Can be assigned to.
\par QC type:
\c ulong
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return unsigned long lvalue
\hideinitializer
*/
#define P_ULONG(p,n) P_var (p, n, ulong)
/** Access a vector parameter. Can be used any way a vec3_t variable can.
\par QC type:
\c vector
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return vec3_t lvalue
\hideinitializer
*/
#define P_VECTOR(p,n) (&P_var (p, n, float))
/** Access a quaterion parameter. Can be used any way a quat_t variable can.
\par QC type:
\c quaterion
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return quat_t lvalue
\hideinitializer
*/
#define P_QUAT(p,n) (&P_var (p, n, float))
/** Access a string index parameter. Can be assigned to.
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return pr_string_t lvalue
\hideinitializer
*/
#define P_STRING(p,n) P_var (p, n, string)
/** Access a function parameter. Can be assigned to.
\par QC type:
\c void()
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return pr_func_t lvalue
\hideinitializer
*/
#define P_FUNCTION(p,n) P_var (p, n, func)
/** Access a pointer parameter. Can be assigned to.
\par QC type:
\c void *
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return pr_ptr_t lvalue
\hideinitializer
*/
#define P_POINTER(p,n) P_var (p, n, ptr)
/** Access an entity parameter.
\par QC type:
\c entity
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return C pointer to the entity
\hideinitializer
*/
#define P_EDICT(p,n) PROG_TO_EDICT ((p), P_INT (p, n))
/** Access an entity parameter.
\par QC type:
\c entity
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return the entity number
\hideinitializer
*/
#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.
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return const char * to the string
\hideinitializer
*/
#define P_GSTRING(p,n) PR_GetString (p, P_STRING (p, n))
/** Access a dstring parameter. Kills the program if the dstring is invalid.
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\param n parameter number (0-7)
\return dstring_t * to the dstring
\hideinitializer
*/
#define P_DSTRING(p,n) PR_GetMutableString (p, P_STRING (p, n))
/** Access a pointer parameter.
\par 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
\hideinitializer
*/
#define P_GPOINTER(p,n) PR_GetPointer (p, P_POINTER (p, n))
/** Access a structure pointer parameter.
\par 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.
\hideinitializer
*/
#define P_STRUCT(p,t,n) (*(t *)P_GPOINTER (p, n))
///@}
/** \defgroup prda_return Return Values
\ingroup progs_data_access
Typed return value access. These macros are used to access the value
returned by an interpreted VM function, and to return values from engine
functions into progs space (that is, builtins).
\warning No checking is performed against progs types; for example, if you
ask for an \c int from a function that returned a \c float, you're asking
for trouble.
*/
///@{
/** Access the VM function return value as an arbitray type.
\par QC type:
\c struct etc small enough to fit in the return slot
\param p pointer to ::progs_t VM struct
\param t C type of the structure
\return structure lvalue. use & to make a pointer of the
appropriate type.
\hideinitializer
*/
#define R_PACKED(p,t) (*(t *) (p)->pr_return)
/** \internal
\param p pointer to ::progs_t VM struct
\param t typename prefix (see pr_type_u)
\return lvalue of the appropriate type
\hideinitializer
*/
#define R_var(p,t) R_PACKED(p, pr_##t##_t)
/** Access the VM function return value as a \c float
\par QC type:
\c float
\param p pointer to ::progs_t VM struct
\return float lvalue
\hideinitializer
*/
#define R_FLOAT(p) R_var (p, float)
/** Access the VM function return value as a \c double
\par QC type:
\c double
\param p pointer to ::progs_t VM struct
\return double lvalue
\hideinitializer
*/
#define R_DOUBLE(p) R_var (p, double)
/** Access the VM function return value as a \c ::pr_int_t (AKA int32_t)
\par QC type:
\c int
\param p pointer to ::progs_t VM struct
\return ::pr_int_t lvalue
\hideinitializer
*/
#define R_INT(p) R_var (p, int)
/** Access the VM function return value as a \c ::pr_uint_t (AKA uint32_t)
\par QC type:
\c uint
\param p pointer to ::progs_t VM struct
\return ::pr_int_t lvalue
\hideinitializer
*/
#define R_UINT(p) R_var (p, uint)
/** Access the VM function return value as a \c ::pr_long_t (AKA int32_t)
\par QC type:
\c long
\param p pointer to ::progs_t VM struct
\return ::pr_long_t lvalue
\hideinitializer
*/
#define R_LONG(p) R_var (p, long)
/** Access the VM function return value as a \c ::pr_ulong_t (AKA uint32_t)
\par QC type:
\c ulong
\param p pointer to ::progs_t VM struct
\return ::pr_long_t lvalue
\hideinitializer
*/
#define R_ULONG(p) R_var (p, ulong)
/** Access the VM function return value as a \c ::vec3_t vector.
\par QC type:
\c vector
\param p pointer to ::progs_t VM struct
\return ::vec3_t lvalue
\hideinitializer
*/
#define R_VECTOR(p) (&R_var (p, float))
/** Access the VM function return value as a \c ::quat_t quaternion.
\par QC type:
\c quaternion
\param p pointer to ::progs_t VM struct
\return ::quat_t lvalue
\hideinitializer
*/
#define R_QUAT(p) (&R_var (p, float))
/** Access the VM function return value as a ::pr_string_t (a VM string reference).
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\return ::pr_string_t lvalue
\hideinitializer
*/
#define R_STRING(p) R_var (p, string)
/** Access the VM function return value as a ::pr_func_t (a VM function reference)
\par QC type:
\c void()
\param p pointer to ::progs_t VM struct
\return ::pr_func_t lvalue
\hideinitializer
*/
#define R_FUNCTION(p) R_var (p, func)
/** Access the VM function return value as a ::pr_ptr_t (a VM "pointer")
\par QC type:
\c void *
\param p pointer to ::progs_t VM struct
\return ::pr_ptr_t lvalue
\hideinitializer
*/
#define R_POINTER(p) R_var (p, ptr)
/** Set the return value to the given C string. The returned string will
eventually be garbage collected (see PR_SetReturnString()).
\par QC type:
\c string
\param p pointer to ::progs_t VM struct
\param s C string to be returned
\hideinitializer
*/
#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.
\par QC type:
\c entity
\param p pointer to ::progs_t VM struct
\param e C entity pointer to be returned
\hideinitializer
*/
#define RETURN_EDICT(p,e) (R_INT (p) = EDICT_TO_PROG(p, e))
/** Set the return value to the given C pointer. NULL is converted to 0.
\par QC type:
\c void *
\param p pointer to ::progs_t VM struct
\param ptr C entity pointer to be returned
\hideinitializer
*/
#define RETURN_POINTER(p,ptr) (R_POINTER (p) = PR_SetPointer (p, ptr))
/** Set the return value to the given vector.
\par QC type:
\c vector
\param p pointer to ::progs_t VM struct
\param v vector to be returned
\hideinitializer
*/
#define RETURN_VECTOR(p,v) VectorCopy (v, R_VECTOR (p))
/** Set the return value to the given quaterion.
\par QC type:
\c vector
\param p pointer to ::progs_t VM struct
\param q quaternion to be returned
\hideinitializer
*/
#define RETURN_QUAT(p,q) QuatCopy (q, R_QUAT (p))
///@}
/** \defgroup prda_entity_fields Entity Fields
\ingroup progs_data_access
Typed entity field access macros. No checking is done against the QC type,
but the appropriate C type will be used.
*/
///@{
/** \internal
\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
\hideinitializer
*/
#define E_fld(e,o) ((e)->pr->pr_edict_area[(e)->edict + (o)])
/** Access an entity field as an arbitray type.
\par QC type:
\c struct etc small enough to fit in a single parameter
\param e pointer to the entity
\param t C type of the structure
\param o field offset into entity data space
\return structure lvalue. use & to make a pointer of the
appropriate type.
\hideinitializer
*/
#define E_PACKED(e,t,o) (*(t *) &E_fld (e, o))
/** \internal
\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
\hideinitializer
*/
#define E_var(e,o,t) E_PACKED (e, pr_##t##_t,o)
/** Access a float entity field. Can be assigned to.
\par QC type:
\c float
\param e pointer to the entity
\param o field offset into entity data space
\return float lvalue
\hideinitializer
*/
#define E_FLOAT(e,o) E_var (e, o, float)
/** Access a double entity field. Can be assigned to.
\par QC type:
\c double
\param e pointer to the entity
\param o field offset into entity data space
\return double lvalue
\hideinitializer
*/
#define E_DOUBLE(e,o) E_var (e, o, double)
/** Access an int entity field. Can be assigned to.
\par QC type:
\c int
\param e pointer to the entity
\param o field offset into entity data space
\return int lvalue
\hideinitializer
*/
#define E_INT(e,o) E_var (e, o, int)
/** Access an unsigned int entity field. Can be assigned to.
\par QC type:
\c uint
\param e pointer to the entity
\param o field offset into entity data space
\return unsigned int lvalue
\hideinitializer
*/
#define E_UINT(e,o) E_var (e, o, uint)
/** Access a long entity field. Can be assigned to.
\par QC type:
\c long
\param e pointer to the entity
\param o field offset into entity data space
\return long lvalue
\hideinitializer
*/
#define E_LONG(e,o) E_var (e, o, long)
/** Access an unsigned long entity field. Can be assigned to.
\par QC type:
\c ulong
\param e pointer to the entity
\param o field offset into entity data space
\return unsigned long lvalue
\hideinitializer
*/
#define E_ULONG(e,o) E_var (e, o, ulong)
/** Access a vector entity field. Can be used any way a vec3_t variable can.
\par QC type:
\c vector
\param e pointer to the entity
\param o field offset into entity data space
\return vec3_t lvalue
\hideinitializer
*/
#define E_VECTOR(e,o) (&E_var (e, o, float))
/** Access a quaternion entity field. Can be used any way a quat_t variable
can.
\par QC type:
\c quaterion
\param e pointer to the entity
\param o field offset into entity data space
\return quat_t lvalue
\hideinitializer
*/
#define E_QUAT(e,o) E_var (e, o, float)
/** Access a string index entity field. Can be assigned to.
\par QC type:
\c string
\param e pointer to the entity
\param o field offset into entity data space
\return pr_string_t lvalue
\hideinitializer
*/
#define E_STRING(e,o) E_var (e, o, string)
/** Access a function entity field. Can be assigned to.
\par QC type:
\c void()
\param e pointer to the entity
\param o field offset into entity data space
\return pr_func_t lvalue
\hideinitializer
*/
#define E_FUNCTION(e,o) E_var (e, o, func)
/** Access a pointer entity field. Can be assigned to.
\par QC type:
\c void *
\param e pointer to the entity
\param o field offset into entity data space
\return pr_ptr_t lvalue
\hideinitializer
*/
#define E_POINTER(e,o) E_var (e, o, ptr)
/** Access a string entity field, converting it to a C string. Kills the
program if the string is invalid.
\par 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
\hideinitializer
*/
#define E_GSTRING(p,e,o) (PR_GetString (p, E_STRING (e, o)))
/** Access a dstring entity field. Kills the program if the dstring is invalid.
\par 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
\hideinitializer
*/
#define E_DSTRING(p,e,o) (PR_GetMutableString (p, E_STRING (e, o)))
///@}
/** \defgroup pr_builtins VM Builtin functions
\ingroup progs
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
0x10000000 to 0x7fffffff are reserved for auto-allocation. The range
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, void *data);
/** 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 automatically
/// 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.
pr_int_t binum;
/// The number of parameters the builtin takes. Negative numbers mean
/// varargs with ~num_params (-num_params - 1) being the number of real
/// parameters.
pr_int_t num_params;
/// Parameter size specificiation.
///
/// Up to 8 parameters are supported for automatic parameter setup because
/// that's all that v6p progs can pass. Builtins taking more than 8
/// parameters are already Ruamoko-only and thus know how to deal with the
/// parameters being on the data stack.
///
/// The encoding is the same as for progs functions, with 3:5 for
/// alignment:size (size 0 means 32 words).
dparmsize_t params[PR_MAX_PARAMS];
/// Data passed to builtin functions. Set by PR_RegisterBuiltins
void *data;
} builtin_t;
#define PR_PARAM(type) { \
.size = PR_SIZEOF(type) & 0x1f, \
.alignment = BITOP_LOG2(PR_ALIGNOF(type)), \
}
/** Duplicate the dfunction_t descriptor with the addition of a pointer to the
builtin function. Avoids a level of indirection when calling a builtin
function.
*/
typedef struct {
pr_int_t first_statement;
pr_int_t numparams;
pr_ulong_t profile;
union {
struct {
dparmsize_t param_size[PR_MAX_PARAMS];
dfunction_t *descriptor;
pr_uint_t params_start;
pr_uint_t locals;
};
struct {
// although Ruamoko progs support more than PR_MAX_PARAMS
// arguments, only the first PR_MAX_PARAMS parameter pointers
// are initialized. This keeps builtins meant for both ISAs
// simple as they either will never accept more tha PR_MAX_PARAMS
// arugments, or they'll be modified to do the right thing.
pr_ushort_t param_offsets[PR_MAX_PARAMS];
builtin_proc func;
void *data; ///< extra data passed to the builtin
};
};
} bfunction_t;
/** Register a set of builtin functions with the VM. Different VMs within the
same program can have different builtin sets. May be called multiple times
for the same VM, but redefining a builtin is an error.
\param pr pointer to ::progs_t VM struct
\param builtins array of builtin_t builtins
\param data pointer to builtin-specific data. Usually the resources
struct registered with PR_Resources_Register
*/
void PR_RegisterBuiltins (progs_t *pr, builtin_t *builtins, void *data);
/** 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, pr_int_t num);
/** Fixup all automatically resolved builtin functions (func = #0 in QC).
Performs any necessary builtin function number mapping. Also builds the
bfunction_t table. Called automatically during progs load.
\param pr pointer to ::progs_t VM struct
\return true for success, false for failure
*/
int PR_RelocateBuiltins (progs_t *pr);
///@}
/** \defgroup pr_strings String Management
\ingroup progs
Strings management functions.
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 by 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 management subsystem.
\param pr The VM of which the string management subsystem will be
initialized;
*/
void PR_Strings_Init (progs_t *pr);
/** Initialize the string tables using the strings supplied by the progs.
Called automatically during progs load.
\param pr pointer to ::progs_t VM struct
\return true for success, false for failure
*/
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, pr_string_t num) __attribute__((pure));
/** Check if a string is valid and mutable.
\param pr pointer to ::progs_t VM struct
\param num string index to be checked
\return true if the string is valid and mutable, false otherwise
*/
qboolean PR_StringMutable (progs_t *pr, pr_string_t num) __attribute__((pure));
/** 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, pr_string_t num) __attribute__((pure));
/** Retrieve 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, pr_string_t num) __attribute__((pure));
/** 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
*/
pr_string_t PR_SetString(progs_t *pr, const char *s);
/** Get the progs string if it exists.
Only static strings are searched.
\param pr pointer to ::progs_t VM struct
\param s C string to be found
\return string index of the progs string if it exists, otherwise
0 (ambiguous with "").
*/
pr_string_t PR_FindString(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
*/
pr_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
*/
pr_string_t PR_SetTempString(progs_t *pr, const char *s);
/** Make a temporary memory block that will be freed when the current progs
stack frame is exited. The contents may be anything and a new block is
returned every time, and the block is in VM addressible space. To access
the contents of the block (for reading, writing, etc), use PR_GetString()
and cast the pointer as necessary.
\param pr pointer to ::progs_t VM struct
\param size size of block in bytes
\return string index of the block
*/
pr_string_t PR_AllocTempBlock (progs_t *pr, size_t size);
/** Push a temporary string to the callee stack frame
This is for when the temp string needs to be freed when the called function
returns rather than the calling function. It is an error to push a non-temp
string.
\param pr pointer to ::progs_t VM struct
\param num string index of the temp string
*/
void PR_PushTempString (progs_t *pr, pr_string_t num);
/** Make a temporary 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
*/
pr_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, 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
*/
pr_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
*/
pr_string_t PR_SetDynamicString (progs_t *pr, const char *s);
/** Convert an ephemeral string to a dynamic string.
Valid strings that are not ephemeral (static, dynamic, mutable) will not
be affected, but temp and return strings will be marked dynamic, requiring
a call to PR_FreeString to return their memory.
\param pr pointer to ::progs_t VM struct
\param str The string to be "held" (made non-ephemeral). Safe to call
on any valid string, but affects only ephemeral strings.
*/
void PR_HoldString (progs_t *pr, pr_string_t 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, 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);
/** Callback for handling %@ in PR_Sprintf format strings.
\param pr pointer to ::progs_t VM struct
\param at_param Parameter to be converted to a string. Usually a progs
pointer value. It is the callee's responsibility to
interpret the value as a pointer.
\param data User data pointer as passed to PR_Sprintf_SetAtHandler
\return String to be printed in place of %@
*/
typedef const char *(*prstr_at_handler_t) (progs_t *pr, pr_ptr_t at_param,
void *data);
/** Set the at_handler callback and its user data pointer.
The at_handler callback defaults to null resulting in [ptr] in the output
string. Setting the callback allows the at_handler to interpret the value
(nominally pr_ptr_t, but always a 32-bit value) as it chooses.
\param pr pointer to ::progs_t VM struct
\param at_handler Function pointer for callback to interpret the parameter
corresponding to the %@ format specifier.
\param data User data pointer, passed to \a at_handler.
*/
void PR_Sprintf_SetAtHandler (progs_t *pr, prstr_at_handler_t at_handler,
void *data);
/** 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:
<ul>
<li>\c "%%" print a single \c '\%'
<li><tt>%[flags][field width][precision](conversion specifier)</tt> print a
single arg of the type specified by the conversion specifier using
the given format.
</ul>
<ul>
<li>flags (optional)
<ul>
<li>\c '#' print the value using an alternat form
<li>\c '0' the value should be zero padded
<li>\c '-' the value is to be left adjusted on the field bondary.
<li><tt>' '</tt> A blank should be left before a positve number (or
empty string) produced by a signed conversion.
<li>\c '+' A sign (+ or -) will always be placed before a number
produced by a signed conversion.
</ul>
<li>field width (optional)
<ul>
<li>a string of decimal digits with nonzero first digit.
</ul>
<li>precision (optional)
<ul>
<li>a string consisting of <tt>'.'</tt> followed by 0 or more decimal
digits.
</ul>
<li>conversion specifier (mandatory)
<ul>
<li>\c '@' \c id Not yet implemented. Silently ignored.
<li>\c 'e' \c entity Prints the edict number of the given entity ("%i")
<li>\c 'i' \c int Print a int value. ("%i")
<li>\c 'f' \c float Print a float value. ("%f")
<li>\c 'g' \c float Print a float value. ("%f")
<li>\c 'p' \c void * Print a pointer value. ("%#x")
<li>\c 's' \c string Print a string value. ("%s")
<li>\c 'v' \c vector Print a vector value. ("'%g %g %g'")
<li>\c 'q' \c quaternion Print a quaternion value. ("'%g %g %g %g'")
<li>\c 'x' \c uint Print an unsigned int value. ("%x")
</ul>
</ul>
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);
///@}
/** \defgroup pr_resources Resource Management
\ingroup progs
Builtin module private data management.
*/
///@{
/** Initialize the resource management fields.
\param pr The VM of which the resource fields will be initialized.
*/
void PR_Resources_Init (progs_t *pr);
void PR_Resources_Shutdown (progs_t *pr);
void PR_Builtins_Shutdown (progs_t *pr);
/** Clear all resources before loading a new progs.
Calls the clear() callback of all registered resources.
\param pr The VM of which the resources will be cleared.
*/
void PR_Resources_Clear (progs_t *pr);
/** Register a resource with a VM.
\param pr The VM to which the resource will be associated.
\param name The name of the resource. Used for retrieving the
resource.
\param data The resource data.
\param clear Callback for performing any necessary cleanup before
VM reuse. Called by PR_Resources_Clear(). The parameters
are the current VM (\p pr) and \p data.
\param destroy Callback for when the resource is about to be destryed
due to final VM shutdown. The parameters are the current
VM (\p pr) and \p data.
\note The name should be unique to the VM, but no checking is
done. If the name is not unique, the previous registered
resource will break. The name of the sub-system
registering the resource is a suitable name, and will
probably be unique.
\note During VM shutdown, \a clear is called (for all resources)
before \a destroy is called.
*/
void PR_Resources_Register (progs_t *pr, const char *name, void *data,
void (*clear)(progs_t *, void *),
void (*destroy)(progs_t *, void *));
/** Retrieve a resource registered with the VM.
\param pr The VM from which the resource will be retrieved.
\param name The name of the resource.
\return The resource, or NULL if \p name does not match any
resource registered with the VM.
*/
void *PR_Resources_Find (progs_t *pr, const char *name);
/** \name Resource Map support
These macros can be used to create functions for mapping C resources
to QuakeC int handles.
Valid handles are always negative.
\note \p map is the resource map itself, not a pointer to the
resource map.
*/
///@{
/** Type delcaration for the resource map.
\param type The type of the resource. The size must be at least
as large as \c sizeof(type *).
\note \a _size is <em>NOT</em> the number of objects in the
map. It is the number of rows in the map array (each row
has multiple objects).
*/
#define PR_RESMAP(type) struct { type *_free; type **_map; unsigned _size; }
/** Allocate a new resource from the resource map.
\param map The resource map.
\return A pointer to the new resource, or null if no more could
be allocated.
*/
#define PR_RESNEW_NC(map) \
({ \
__auto_type ret = (map)._free; \
do { \
if (!(map)._free) { \
int i, size; \
(map)._size++; \
size = (map)._size * sizeof ((map)._free); \
(map)._map = realloc ((map)._map, size); \
if (!(map)._map) { \
ret = 0; \
break; \
} \
(map)._free = malloc (1024 * sizeof (*(map)._free)); \
if (!(map)._free) { \
ret = 0; \
break; \
} \
(map)._map[(map)._size - 1] = (map)._free; \
for (i = 0; i < 1023; i++) { \
*(typeof ((map)._free) *) &(map)._free[i] \
= &(map)._free[i + 1]; \
} \
*(typeof ((map)._free) *) &(map)._free[i] = 0; \
} \
ret = (map)._free; \
(map)._free = *(typeof ((map)._free) *) ret; \
} while (0); \
ret; \
})
#define PR_RESNEW(map) \
({ \
__auto_type t = PR_RESNEW_NC (map); \
if (t) { \
memset (t, 0, sizeof (*(map)._free)); \
} \
t; \
})
/** Free a resource returning it to the resource map.
\param map The resource map.
\param t Pointer to the resource to be freed.
*/
#define PR_RESFREE(map,t) \
do { \
memset (t, 0, sizeof (*(map)._free)); \
*(typeof ((map)._free) *) t = (map)._free; \
(map)._free = t; \
} while (0)
/** Free all resources in the resource map.
Any memory allocated to the resource must be freed before freeing the
resource.
A reset resource map is guaranteed to allocate elements sequentially.
\param map The resource map.
*/
#define PR_RESRESET(map) \
do { \
unsigned i, j; \
if (!(map)._size) { \
break; \
} \
for (i = 0; i < (map)._size; i++) { \
(map)._free = (map)._map[i]; \
for (j = 0; j < 1023; j++) { \
*(typeof ((map)._free) *) &(map)._free[j] \
= &(map)._free[j + 1]; \
} \
if (i < (map)._size - 1) { \
*(typeof ((map)._free) *) &(map)._free[j] \
= &(map)._map[i + 1][0]; \
} else { \
*(typeof ((map)._free) *) &(map)._free[j] = 0; \
} \
} \
(map)._free = (map)._map[0]; \
} while (0)
/** Retrieve a resource from the resource map using a handle.
\param map The resource map.
\param ind The handle.
\return A pointer to the resource, or NULL if the handle is
invalid.
*/
#define PR_RESGET(map,ind) \
({ \
__auto_type ret = (map)._free; \
unsigned row = ~ind / 1024; \
unsigned col = ~ind % 1024; \
if (row >= (map)._size) { \
ret = 0; \
} else { \
ret = &(map)._map[row][col]; \
} \
ret; \
})
/** Convert a resource pointer to a handle.
\param map The resource map.
\param ptr The resource pointer.
\return The handle or 0 if the pointer is invalid.
*/
#define PR_RESINDEX(map,ptr) \
({ \
unsigned i; \
unsigned index = 0; \
for (i = 0; i < (map)._size; i++) { \
long d = ptr - (map)._map[i]; \
if (d >= 0 && d < 1024) { \
index = ~(i * 1024 + d); \
break; \
} \
} \
index; \
})
///@}
///@}
/** \defgroup pr_zone VM memory management.
\ingroup progs
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, pr_int_t size);
void *PR_Zone_Realloc (progs_t *pr, void *ptr, pr_int_t size);
void *PR_Zone_TagMalloc (progs_t *pr, int size, int tag);
///@}
/** \defgroup debug VM Debugging
\ingroup progs
Progs debugging support.
*/
/// \addtogroup debug
///@{
struct qfot_type_s;
/** Callback for viewing progs data
\param type C pointer to the type definition by which to view the
data.
\param value C pointer to the data to be viewed.
\param data User data.
*/
typedef void (*type_view_func) (struct qfot_type_s *type, pr_type_t *value,
void *data);
/** Set of callbacks for viewing progs data
Each possible type has its own callback. Basic types (those for which the
VM has specific instructions) all have separate callbacks, one for each
type, but the callbacks for compound types are expected to some
interpretation on their own, such as displaying a simple identifier or
the entire contents of the data.
*/
typedef struct type_view_s {
type_view_func struct_view;
type_view_func union_view;
type_view_func enum_view;
type_view_func array_view;
type_view_func class_view;
#define EV_TYPE(type) type_view_func type##_view;
#include "QF/progs/pr_type_names.h"
} type_view_t;
void PR_Debug_Init (progs_t *pr);
void PR_Debug_Init_Cvars (void);
void PR_DebugSetSym (progs_t *pr, pr_debug_header_t *debug);
int PR_LoadDebug (progs_t *pr);
const char *PR_Debug_GetBaseDirectory (progs_t *pr, const char *file);
void PR_Debug_Watch (progs_t *pr, const char *expr);
void PR_Debug_Print (progs_t *pr, const char *expr);
const char *PR_Debug_ValueString (progs_t *pr, pr_ptr_t offset,
struct qfot_type_s *type,
struct dstring_s *dstr);
pr_auxfunction_t *PR_Debug_AuxFunction (progs_t *pr, pr_uint_t func) __attribute__((pure));
pr_auxfunction_t *PR_Debug_MappedAuxFunction (progs_t *pr, pr_uint_t func) __attribute__((pure));
pr_def_t *PR_Debug_LocalDefs (progs_t *pr, pr_auxfunction_t *aux_function) __attribute__((pure));
pr_lineno_t *PR_Debug_Linenos (progs_t *pr, pr_auxfunction_t *aux_function,
pr_uint_t *num_linenos);
pr_auxfunction_t *PR_Get_Lineno_Func (progs_t *pr, pr_lineno_t *lineno) __attribute__((pure));
pr_uint_t PR_Get_Lineno_Addr (progs_t *pr, pr_lineno_t *lineno) __attribute__((pure));
pr_uint_t PR_Get_Lineno_Line (progs_t *pr, pr_lineno_t *lineno) __attribute__((pure));
pr_lineno_t *PR_Find_Lineno (progs_t *pr, pr_uint_t addr) __attribute__((pure));
pr_uint_t PR_FindSourceLineAddr (progs_t *pr, const char *file, pr_uint_t line) __attribute__((pure));
const char *PR_Get_Source_File (progs_t *pr, pr_lineno_t *lineno) __attribute__((pure));
const char *PR_Get_Source_Line (progs_t *pr, pr_uint_t addr);
pr_def_t *PR_Get_Param_Def (progs_t *pr, dfunction_t *func, unsigned parm) __attribute__((pure));
pr_def_t *PR_Get_Local_Def (progs_t *pr, pr_ptr_t *offs) __attribute__((pure));
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 int pr_debug;
extern char *pr_source_path;
extern int pr_deadbeef_ents;
extern int pr_deadbeef_locals;
extern int pr_boundscheck;
extern int pr_faultchecks;
///@}
/** \defgroup pr_cmds Quake and Quakeworld common builtins
\ingroup progs
\todo This really doesn't belong in progs.
*/
///@{
char *PF_VarString (progs_t *pr, int first, int count);
void PR_Cmds_Init (progs_t *pr);
extern const char *pr_gametype;
///@}
//============================================================================
#define PR_MAX_STACK_DEPTH 64
#define PR_LOCAL_STACK_SIZE 4096
#define PR_RS_SLOTS 16
#define PR_BASE_IND(o, b) (((o) & OP_##b##_BASE) >> OP_##b##_SHIFT)
#define PR_BASE(p, s, b) (p->pr_bases[PR_BASE_IND(s->op, b)])
typedef struct strref_s strref_t;
typedef struct {
pr_uivec4_t bases; ///< base registers on entry to function
pr_uint_t staddr; ///< Return statement.
pr_uint_t stack_ptr; ///< data stack on entry to function
bfunction_t *func; ///< Calling function.
strref_t *tstr; ///< Linked list of temporary strings.
pr_type_t *return_ptr; ///< Saved return address
} prstack_t;
struct progs_s {
int (*parse_field) (progs_t *pr, const char *key, const char *value);
int null_bad;
int no_exec_limit;
struct hashctx_s **hashctx;
void (*file_error) (progs_t *pr, const char *path);
void *(*load_file) (progs_t *pr, const char *path, off_t *size);
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;
int denorm_found;
struct memzone_s *zone;
int zone_size; ///< set by user
/// \name builtin functions
///@{
struct hashtab_s *builtin_hash;
struct hashtab_s *builtin_num_hash;
struct biblock_s *builtin_blocks;
builtin_t *bi_no_function;
unsigned bi_next;
unsigned (*bi_map) (progs_t *pr, unsigned binum);
///@}
/// \name symbol management
///@{
struct hashtab_s *function_hash;
struct hashtab_s *global_hash;
struct hashtab_s *field_hash;
///@}
/// \name type encodings
///@{
struct hashtab_s *type_hash;
pr_ptr_t type_encodings;
///@}
/// \name load hooks
///@{
int num_load_funcs;
int max_load_funcs;
pr_load_func_t **load_funcs;
/// cleared each load
///@{
int num_load_finish_funcs;
int max_load_finish_funcs;
pr_load_func_t **load_finish_funcs;
///@}
///@}
/// \name string management
///@{
struct prstr_resources_s *pr_string_resources;
strref_t *pr_xtstr;
strref_t *pr_pushtstr;
int float_promoted; ///< for PR_Sprintf
///@}
/// \name memory map
///@{
dfunction_t *pr_functions;
bfunction_t *function_table;
char *pr_strings;
int pr_stringsize;
pr_def_t *pr_globaldefs;
pr_def_t *pr_fielddefs;
dstatement_t *pr_statements;
pr_type_t *pr_globals;
pr_uint_t globals_size;
pr_uint_t null_size; ///< size of block considered null page
pr_uivec4_t pr_bases; ///< base registers (index in opcode)
///@}
/// \name parameter block
///@{
pr_type_t *pr_return;
pr_type_t *pr_params[PR_MAX_PARAMS];
pr_type_t *pr_real_params[PR_MAX_PARAMS];
int pr_param_size; ///< covers both params and return
int pr_param_alignment; ///< covers both params and return
pr_type_t *pr_return_buffer; ///< for discarded return values
///< or returning values to C
///@}
/// \name edicts
/// \todo FIXME should this be outside the VM?
///@{
edict_t **pr_edicts;
pr_uint_t max_edicts; ///< set by user
pr_uint_t *num_edicts;
pr_uint_t *reserved_edicts; ///< alloc will start at reserved_edicts+1
void (*unlink) (edict_t *ent);
void (*flush) (void);
int (*prune_edict) (progs_t *pr, edict_t *ent);
void (*free_edict) (progs_t *pr, edict_t *ent);
pr_type_t *pr_edict_area;
int pr_edict_size; ///< # of pr_type_t slots
pr_uint_t pr_edict_area_size; ///< for bounds checking, starts at 0
pr_func_t edict_parse;
///@}
/// \name execution state
///@{
int pr_argc; //FIXME need a good way to ensure it is correct
qboolean pr_trace;
int pr_trace_depth;
bfunction_t *pr_xfunction;
int pr_xstatement;
prstack_t pr_stack[PR_MAX_STACK_DEPTH];
int pr_depth;
/// \name progs visible stack
/// Usable by the progs for any purpose. Will not be accessible unless
/// a .stack global is found. Space is allocated from the top of the stack
/// (as is common for hardware). The push and pop instructions will not
/// be considered valid if there is no .stack global.
/// \note The return address and saved locals will not ever be on this
/// stack.
///@{
pr_type_t *stack;
pr_ptr_t stack_bottom;
int stack_size; ///< set by user
///@}
int localstack[PR_LOCAL_STACK_SIZE];
int localstack_used;
///@}
/// \name resources
///@{
pr_resource_t *resources;
struct hashtab_s *resource_hash;
///@}
/// \name obj info
///@{
struct probj_resources_s *pr_objective_resources;
///@}
/// \name debugging
///@{
struct prdeb_resources_s *pr_debug_resources;
void (*debug_handler) (prdebug_t event, void *param, void *data);
void *debug_data;
pr_type_t *watch;
int wp_conditional;
pr_type_t wp_val;
///@}
/// \name globals and fields needed by the VM
///@{
struct {
double *dtime; ///< required for OP_STATE d
float *ftime; ///< required for OP_STATE f
pr_uint_t *self; ///< required for OP_STATE
pr_ptr_t *stack; ///< required for OP_(PUSH|POP)*
} globals;
struct {
pr_int_t nextthink; ///< required for OP_STATE
pr_int_t frame; ///< required for OP_STATE
pr_int_t think; ///< required for OP_STATE
pr_int_t this; ///< optional for entity<->object linking
} 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 (const progs_t *pr, pr_ptr_t 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 pr_ptr_t
PR_SetPointer (const progs_t *pr, const void *p)
{
return p ? (const pr_type_t *) p - pr->pr_globals : 0;
}
///@}
/** \example vm-exec.c
*/
#endif//__QF_progs_h