2005-01-17 17:43:13 +00:00
# include "quakedef.h"
# ifdef CSQC_DAT
2005-01-24 05:10:11 +00:00
# ifdef RGLQUAKE
# include "glquake.h" //evil to include this
# endif
2005-04-26 16:04:12 +00:00
static progfuncs_t * csqcprogs ;
2005-01-24 05:10:11 +00:00
2005-04-26 16:04:12 +00:00
static unsigned int csqcchecksum ;
static qboolean csqcwantskeys ;
2005-03-10 03:55:18 +00:00
2005-03-20 02:57:11 +00:00
cvar_t pr_csmaxedicts = { " pr_csmaxedicts " , " 3072 " } ;
2005-03-24 18:13:37 +00:00
cvar_t cl_csqcdebug = { " cl_csqcdebug " , " 0 " } ; //prints entity numbers which arrive (so I can tell people not to apply it to players...)
2005-03-20 02:57:11 +00:00
2005-05-08 06:01:25 +00:00
//If I do it like this, I'll never forget to register something...
2005-03-10 03:55:18 +00:00
# define csqcglobals \
globalfunction ( init_function , " CSQC_Init " ) ; \
globalfunction ( shutdown_function , " CSQC_Shutdown " ) ; \
globalfunction ( draw_function , " CSQC_UpdateView " ) ; \
globalfunction ( parse_stuffcmd , " CSQC_Parse_StuffCmd " ) ; \
globalfunction ( parse_centerprint , " CSQC_Parse_CenterPrint " ) ; \
2005-04-26 16:04:12 +00:00
globalfunction ( input_event , " CSQC_InputEvent " ) ; \
2005-05-15 18:49:04 +00:00
globalfunction ( console_command , " CSQC_ConsoleCommand " ) ; \
2005-03-10 03:55:18 +00:00
\
globalfunction ( ent_update , " CSQC_Ent_Update " ) ; \
globalfunction ( ent_remove , " CSQC_Ent_Remove " ) ; \
\
/*These are pointers to the csqc's globals.*/ \
globalfloat ( time , " time " ) ; /*float Written before entering most qc functions*/ \
globalentity ( self , " self " ) ; /*entity Written before entering most qc functions*/ \
\
2005-04-26 16:04:12 +00:00
globalfloat ( maxclients , " maxclients " ) ; /*float */ \
\
2005-03-10 03:55:18 +00:00
globalvector ( forward , " v_forward " ) ; /*vector written by anglevectors*/ \
globalvector ( right , " v_right " ) ; /*vector written by anglevectors*/ \
globalvector ( up , " v_up " ) ; /*vector written by anglevectors*/ \
\
globalfloat ( trace_allsolid , " trace_allsolid " ) ; /*bool written by traceline*/ \
globalfloat ( trace_startsolid , " trace_startsolid " ) ; /*bool written by traceline*/ \
globalfloat ( trace_fraction , " trace_fraction " ) ; /*float written by traceline*/ \
globalfloat ( trace_inwater , " trace_inwater " ) ; /*bool written by traceline*/ \
globalfloat ( trace_inopen , " trace_inopen " ) ; /*bool written by traceline*/ \
globalvector ( trace_endpos , " trace_endpos " ) ; /*vector written by traceline*/ \
globalvector ( trace_plane_normal , " trace_plane_normal " ) ; /*vector written by traceline*/ \
globalfloat ( trace_plane_dist , " trace_plane_dist " ) ; /*float written by traceline*/ \
globalentity ( trace_ent , " trace_ent " ) ; /*entity written by traceline*/ \
2005-04-16 16:21:27 +00:00
\
2005-04-26 16:04:12 +00:00
globalfloat ( clientcommandframe , " clientcommandframe " ) ; \
globalfloat ( servercommandframe , " servercommandframe " ) ; \
\
globalfloat ( player_localentnum , " player_localentnum " ) ; /*float the entity number of the local player*/ \
\
globalvector ( pmove_org , " pmove_org " ) ; \
globalvector ( pmove_vel , " pmove_vel " ) ; \
globalvector ( pmove_mins , " pmove_mins " ) ; \
globalvector ( pmove_maxs , " pmove_maxs " ) ; \
globalfloat ( input_timelength , " input_timelength " ) ; \
globalvector ( input_angles , " input_angles " ) ; \
globalvector ( input_movevalues , " input_movevalues " ) ; \
globalfloat ( input_buttons , " input_buttons " ) ; \
\
globalfloat ( movevar_gravity , " movevar_gravity " ) ; \
globalfloat ( movevar_stopspeed , " movevar_stopspeed " ) ; \
globalfloat ( movevar_maxspeed , " movevar_maxspeed " ) ; \
globalfloat ( movevar_spectatormaxspeed , " movevar_spectatormaxspeed " ) ; \
globalfloat ( movevar_accelerate , " movevar_accelerate " ) ; \
globalfloat ( movevar_airaccelerate , " movevar_airaccelerate " ) ; \
globalfloat ( movevar_wateraccelerate , " movevar_wateraccelerate " ) ; \
globalfloat ( movevar_friction , " movevar_friction " ) ; \
globalfloat ( movevar_waterfriction , " movevar_waterfriction " ) ; \
globalfloat ( movevar_entgravity , " movevar_entgravity " ) ; \
2005-04-16 16:21:27 +00:00
2005-03-10 03:55:18 +00:00
2005-02-28 07:16:19 +00:00
typedef struct {
2005-03-10 03:55:18 +00:00
# define globalfloat(name,qcname) float *name
# define globalvector(name,qcname) float *name
# define globalentity(name,qcname) int *name
# define globalstring(name,qcname) string_t *name
# define globalfunction(name,qcname) func_t name
2005-02-28 07:16:19 +00:00
//These are the functions the engine will call to, found by name.
2005-03-10 03:55:18 +00:00
csqcglobals
# undef globalfloat
# undef globalvector
# undef globalentity
# undef globalstring
# undef globalfunction
2005-02-28 07:16:19 +00:00
} csqcglobals_t ;
static csqcglobals_t csqcg ;
2005-04-26 16:04:12 +00:00
# define plnum 0
2005-02-28 07:16:19 +00:00
2005-04-26 16:04:12 +00:00
static void CSQC_FindGlobals ( void )
2005-02-28 07:16:19 +00:00
{
2005-03-10 03:55:18 +00:00
# define globalfloat(name,qcname) csqcg.name = (float*)PR_FindGlobal(csqcprogs, qcname, 0);
# define globalvector(name,qcname) csqcg.name = (float*)PR_FindGlobal(csqcprogs, qcname, 0);
# define globalentity(name,qcname) csqcg.name = (int*)PR_FindGlobal(csqcprogs, qcname, 0);
# define globalstring(name,qcname) csqcg.name = (string_t*)PR_FindGlobal(csqcprogs, qcname, 0);
# define globalfunction(name,qcname) csqcg.name = PR_FindFunction(csqcprogs,qcname,PR_ANY);
csqcglobals
# undef globalfloat
# undef globalvector
# undef globalentity
# undef globalstring
# undef globalfunction
2005-04-16 16:21:27 +00:00
2005-02-28 07:16:19 +00:00
if ( csqcg . time )
* csqcg . time = Sys_DoubleTime ( ) ;
2005-04-16 16:21:27 +00:00
if ( csqcg . player_localentnum )
2005-04-26 16:04:12 +00:00
* csqcg . player_localentnum = cl . playernum [ plnum ] + 1 ;
if ( csqcg . maxclients )
* csqcg . maxclients = MAX_CLIENTS ;
2005-02-28 07:16:19 +00:00
}
//this is the list for all the csqc fields.
//(the #define is so the list always matches the ones pulled out)
# define csqcfields \
2005-04-26 16:04:12 +00:00
fieldfloat ( entnum ) ; \
2005-02-28 07:16:19 +00:00
fieldfloat ( modelindex ) ; \
fieldvector ( origin ) ; \
fieldvector ( angles ) ; \
fieldfloat ( alpha ) ; /*transparency*/ \
fieldfloat ( scale ) ; /*model scale*/ \
fieldfloat ( fatness ) ; /*expand models X units along thier normals.*/ \
fieldfloat ( skin ) ; \
fieldfloat ( colormap ) ; \
fieldfloat ( frame ) ; \
fieldfloat ( oldframe ) ; \
fieldfloat ( lerpfrac ) ; \
\
fieldfloat ( drawmask ) ; /*So that the qc can specify all rockets at once or all bannanas at once*/ \
fieldfunction ( predraw ) ; /*If present, is called just before it's drawn.*/ \
\
fieldstring ( model ) ;
2005-02-09 19:32:09 +00:00
//note: doesn't even have to match the clprogs.dat :)
2005-01-24 05:10:11 +00:00
typedef struct {
2005-02-28 07:16:19 +00:00
# define fieldfloat(name) float name
# define fieldvector(name) vec3_t name
# define fieldentity(name) int name
# define fieldstring(name) string_t name
# define fieldfunction(name) func_t name
csqcfields
# undef fieldfloat
# undef fieldvector
# undef fieldentity
# undef fieldstring
# undef fieldfunction
2005-01-24 05:10:11 +00:00
} csqcentvars_t ;
2005-02-09 19:32:09 +00:00
typedef struct csqcedict_s
2005-01-17 17:43:13 +00:00
{
qboolean isfree ;
float freetime ; // sv.time when the object was freed
int entnum ;
qboolean readonly ; //world
2005-03-28 00:11:59 +00:00
csqcentvars_t * v ;
//add whatever you wish here
2005-01-17 17:43:13 +00:00
} csqcedict_t ;
2005-04-26 16:04:12 +00:00
static csqcedict_t * csqc_edicts ; //consider this 'world'
2005-02-28 07:16:19 +00:00
2005-04-26 16:04:12 +00:00
static void CSQC_InitFields ( void )
2005-01-24 05:10:11 +00:00
{ //CHANGING THIS FUNCTION REQUIRES CHANGES TO csqcentvars_t
2005-03-28 07:05:38 +00:00
# define fieldfloat(name) PR_RegisterFieldVar(csqcprogs, ev_float, #name, (int)&((csqcentvars_t*)0)->name, -1)
# define fieldvector(name) PR_RegisterFieldVar(csqcprogs, ev_vector, #name, (int)&((csqcentvars_t*)0)->name, -1)
# define fieldentity(name) PR_RegisterFieldVar(csqcprogs, ev_entity, #name, (int)&((csqcentvars_t*)0)->name, -1)
# define fieldstring(name) PR_RegisterFieldVar(csqcprogs, ev_string, #name, (int)&((csqcentvars_t*)0)->name, -1)
# define fieldfunction(name) PR_RegisterFieldVar(csqcprogs, ev_function, #name, (int)&((csqcentvars_t*)0)->name, -1)
2005-05-17 02:36:54 +00:00
csqcfields //any *64->int32 casts are erroneous, it's biased off NULL.
2005-02-28 07:16:19 +00:00
# undef fieldfloat
# undef fieldvector
# undef fieldentity
# undef fieldstring
# undef fieldfunction
2005-01-24 05:10:11 +00:00
}
2005-04-26 16:04:12 +00:00
static csqcedict_t * csqcent [ MAX_EDICTS ] ;
2005-02-28 07:16:19 +00:00
2005-05-17 02:36:54 +00:00
# define RETURN_SSTRING(s) (((string_t *)pr_globals)[OFS_RETURN] = PR_SetString(prinst, s)) //static - exe will not change it.
2005-05-20 03:32:53 +00:00
char * PF_TempStr ( progfuncs_t * prinst ) ;
2005-01-17 17:43:13 +00:00
2005-04-26 16:04:12 +00:00
static int csqcentsize ;
2005-01-17 17:43:13 +00:00
//pr_cmds.c builtins that need to be moved to a common.
void VARGS PR_BIError ( progfuncs_t * progfuncs , char * format , . . . ) ;
void PF_cvar_string ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_cvar_set ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-03-07 08:55:30 +00:00
void PF_print ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-01-17 17:43:13 +00:00
void PF_dprint ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_error ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_rint ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_floor ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_ceil ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_Tokenize ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_ArgV ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_FindString ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_FindFloat ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_nextent ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_randomvec ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_Sin ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_Cos ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_Sqrt ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_bound ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strlen ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strcat ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_ftos ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_fabs ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_vtos ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_etos ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_stof ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_mod ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_substring ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_stov ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_dupstring ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_forgetstring ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_Spawn ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_min ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_max ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_registercvar ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_pow ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_chr2str ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_localcmd ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_random ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_fopen ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_fclose ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_fputs ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_fgets ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_normalize ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_vlen ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_vectoyaw ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_vectoangles ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_findchain ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_findchainfloat ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_coredump ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_traceon ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_traceoff ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_eprint ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-04-26 16:04:12 +00:00
void PF_registercvar ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-02-10 00:14:10 +00:00
void PF_strstrofs ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_str2chr ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_chr2str ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strconv ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_infoadd ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_infoget ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strncmp ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strcasecmp ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_strncasecmp ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-02-09 19:32:09 +00:00
//these functions are from pr_menu.dat
void PF_CL_is_cached_pic ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_precache_pic ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_free_pic ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawcharacter ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawstring ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawpic ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawfill ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawsetcliparea ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawresetcliparea ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
void PF_CL_drawgetimagesize ( progfuncs_t * prinst , struct globalvars_s * pr_globals ) ;
2005-04-26 16:04:12 +00:00
2005-02-12 18:56:04 +00:00
# define MAXTEMPBUFFERLEN 1024
2005-01-17 17:43:13 +00:00
void PF_fclose_progs ( progfuncs_t * prinst ) ;
char * PF_VarString ( progfuncs_t * prinst , int first , struct globalvars_s * pr_globals ) ;
2005-02-28 07:16:19 +00:00
static void PF_Remove_ ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
csqcedict_t * ed ;
ed = ( csqcedict_t * ) G_EDICT ( prinst , OFS_PARM0 ) ;
if ( ed - > isfree )
{
Con_DPrintf ( " CSQC Tried removing free entity \n " ) ;
return ;
}
ED_Free ( prinst , ( void * ) ed ) ;
}
2005-01-17 17:43:13 +00:00
static void PF_cvar ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
cvar_t * var ;
char * str ;
str = PR_GetStringOfs ( prinst , OFS_PARM0 ) ;
{
var = Cvar_Get ( str , " " , 0 , " csqc cvars " ) ;
if ( var )
G_FLOAT ( OFS_RETURN ) = var - > value ;
else
G_FLOAT ( OFS_RETURN ) = 0 ;
}
}
//too specific to the prinst's builtins.
static void PF_Fixme ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
Con_Printf ( " \n " ) ;
2005-02-28 07:16:19 +00:00
prinst - > RunError ( prinst , " \n Builtin %i not implemented. \n Menu is not compatable. " , prinst - > lastcalledbuiltinnumber ) ;
2005-01-17 17:43:13 +00:00
PR_BIError ( prinst , " bulitin not implemented " ) ;
}
2005-03-07 08:40:03 +00:00
static void PF_csqc_centerprint ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * str = PF_VarString ( prinst , 0 , pr_globals ) ;
SCR_CenterPrint ( 0 , str ) ;
}
2005-01-17 17:43:13 +00:00
static void PF_makevectors ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-02-28 07:16:19 +00:00
if ( ! csqcg . forward | | ! csqcg . right | | ! csqcg . up )
Host_EndGame ( " PF_makevectors: one of v_forward, v_right or v_up was not defined \n " ) ;
AngleVectors ( G_VECTOR ( OFS_PARM0 ) , csqcg . forward , csqcg . right , csqcg . up ) ;
2005-01-17 17:43:13 +00:00
}
2005-06-04 04:20:20 +00:00
void QuaternainToAngleMatrix ( float * quat , vec3_t * mat )
{
float xx = quat [ 0 ] * quat [ 0 ] ;
float xy = quat [ 0 ] * quat [ 1 ] ;
float xz = quat [ 0 ] * quat [ 2 ] ;
float xw = quat [ 0 ] * quat [ 3 ] ;
float yy = quat [ 1 ] * quat [ 1 ] ;
float yz = quat [ 1 ] * quat [ 2 ] ;
float yw = quat [ 1 ] * quat [ 3 ] ;
float zz = quat [ 2 ] * quat [ 2 ] ;
float zw = quat [ 2 ] * quat [ 3 ] ;
mat [ 0 ] [ 0 ] = 1 - 2 * ( yy + zz ) ;
mat [ 0 ] [ 1 ] = 2 * ( xy - zw ) ;
mat [ 0 ] [ 2 ] = 2 * ( xz + yw ) ;
mat [ 1 ] [ 0 ] = 2 * ( xy + zw ) ;
mat [ 1 ] [ 1 ] = 1 - 2 * ( xx + zz ) ;
mat [ 1 ] [ 2 ] = 2 * ( yz - xw ) ;
mat [ 2 ] [ 0 ] = 2 * ( xz - yw ) ;
mat [ 2 ] [ 1 ] = 2 * ( yz + xw ) ;
mat [ 2 ] [ 2 ] = 1 - 2 * ( xx + yy ) ;
}
void quaternion_multiply ( float * a , float * b , float * c )
{
# define x1 a[0]
# define y1 a[1]
# define z1 a[2]
# define w1 a[3]
# define x2 b[0]
# define y2 b[1]
# define z2 b[2]
# define w2 b[3]
c [ 0 ] = w1 * x2 + x1 * w2 + y1 * z2 - z1 * y2 ;
c [ 1 ] = w1 * y2 + y1 * w2 + z1 * x2 - x1 * z2 ;
c [ 2 ] = w1 * z2 + z1 * w2 + x1 * y2 - y1 * x2 ;
c [ 3 ] = w1 * w2 - x1 * x2 - y1 * y2 - z1 * z2 ;
}
void quaternion_rotation ( float pitch , float roll , float yaw , float angle , float * quat )
{
float sin_a , cos_a ;
sin_a = sin ( angle / 360 ) ;
cos_a = cos ( angle / 360 ) ;
quat [ 0 ] = pitch * sin_a ;
quat [ 1 ] = yaw * sin_a ;
quat [ 2 ] = roll * sin_a ;
quat [ 3 ] = cos_a ;
}
void EularToQuaternian ( vec3_t angles , float * quat )
{
float x [ 4 ] = { sin ( angles [ 2 ] / 360 ) , 0 , 0 , cos ( angles [ 2 ] / 360 ) } ;
float y [ 4 ] = { 0 , sin ( angles [ 1 ] / 360 ) , 0 , cos ( angles [ 1 ] / 360 ) } ;
float z [ 4 ] = { 0 , 0 , sin ( angles [ 0 ] / 360 ) , cos ( angles [ 0 ] / 360 ) } ;
float t [ 4 ] ;
quaternion_multiply ( x , y , t ) ;
quaternion_multiply ( t , z , quat ) ;
}
void _PF_R_AddEntity ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
2005-01-24 05:10:11 +00:00
{
csqcedict_t * in = ( void * ) G_EDICT ( prinst , OFS_PARM0 ) ;
entity_t ent ;
int i ;
2005-02-28 07:16:19 +00:00
model_t * model ;
2005-03-28 00:11:59 +00:00
if ( in - > v - > predraw )
2005-02-28 07:16:19 +00:00
{
int oldself = * csqcg . self ;
* csqcg . self = EDICT_TO_PROG ( prinst , ( void * ) in ) ;
2005-03-28 00:11:59 +00:00
PR_ExecuteProgram ( prinst , in - > v - > predraw ) ;
2005-02-28 07:16:19 +00:00
* csqcg . self = oldself ;
}
2005-01-24 05:10:11 +00:00
2005-03-28 00:11:59 +00:00
i = in - > v - > modelindex ;
2005-02-12 18:56:04 +00:00
if ( i = = 0 )
return ;
else if ( i > 0 & & i < MAX_MODELS )
2005-02-28 07:16:19 +00:00
model = cl . model_precache [ i ] ;
2005-02-12 18:56:04 +00:00
else if ( i < 0 & & i > - MAX_CSQCMODELS )
2005-02-28 07:16:19 +00:00
model = cl . model_csqcprecache [ - i ] ;
2005-02-12 18:56:04 +00:00
else
2005-01-24 05:10:11 +00:00
return ; //there might be other ent types later as an extension that stop this.
2005-02-28 07:16:19 +00:00
memset ( & ent , 0 , sizeof ( ent ) ) ;
ent . model = model ;
2005-02-12 18:56:04 +00:00
if ( ! ent . model )
{
Con_Printf ( " PF_R_AddEntity: model wasn't precached! \n " ) ;
return ;
}
2005-01-24 05:10:11 +00:00
2005-03-28 00:11:59 +00:00
ent . frame = in - > v - > frame ;
ent . oldframe = in - > v - > oldframe ;
ent . lerpfrac = in - > v - > lerpfrac ;
ent . angles [ 0 ] = in - > v - > angles [ 0 ] ;
ent . angles [ 1 ] = in - > v - > angles [ 1 ] ;
ent . angles [ 2 ] = in - > v - > angles [ 2 ] ;
2005-06-04 04:20:20 +00:00
2005-03-28 00:11:59 +00:00
memcpy ( ent . origin , in - > v - > origin , sizeof ( vec3_t ) ) ;
2005-01-24 05:10:11 +00:00
AngleVectors ( ent . angles , ent . axis [ 0 ] , ent . axis [ 1 ] , ent . axis [ 2 ] ) ;
VectorInverse ( ent . axis [ 1 ] ) ;
2005-03-28 00:11:59 +00:00
ent . alpha = in - > v - > alpha ;
ent . scale = in - > v - > scale ;
2005-03-28 07:05:38 +00:00
ent . skinnum = in - > v - > skin ;
2005-01-24 05:10:11 +00:00
V_AddEntity ( & ent ) ;
2005-06-04 04:20:20 +00:00
/*
{
float a [ 4 ] ;
float q [ 4 ] ;
float r [ 4 ] ;
EularToQuaternian ( ent . angles , a ) ;
QuaternainToAngleMatrix ( a , ent . axis ) ;
ent . origin [ 0 ] + = 16 ;
V_AddEntity ( & ent ) ;
quaternion_rotation ( 0 , 0 , 1 , cl . time * 360 , r ) ;
quaternion_multiply ( a , r , q ) ;
QuaternainToAngleMatrix ( q , ent . axis ) ;
ent . origin [ 0 ] - = 32 ;
ent . angles [ 1 ] = cl . time ;
V_AddEntity ( & ent ) ;
}
*/
}
void PF_R_AddEntity ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
_PF_R_AddEntity ( prinst , pr_globals ) ;
2005-01-24 05:10:11 +00:00
}
2005-02-28 07:16:19 +00:00
static void PF_R_AddDynamicLight ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
float * org = G_VECTOR ( OFS_PARM0 ) ;
float radius = G_FLOAT ( OFS_PARM1 ) ;
float * rgb = G_VECTOR ( OFS_PARM2 ) ;
V_AddLight ( org , radius , rgb [ 0 ] / 5 , rgb [ 1 ] / 5 , rgb [ 2 ] / 5 ) ;
}
2005-02-09 19:32:09 +00:00
# define MASK_ENGINE 1
static void PF_R_AddEntityMask ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int mask = G_FLOAT ( OFS_PARM0 ) ;
csqcedict_t * ent ;
int e ;
for ( e = 1 ; e < * prinst - > parms - > sv_num_edicts ; e + + )
{
ent = ( void * ) EDICT_NUM ( prinst , e ) ;
if ( ent - > isfree )
continue ;
2005-03-28 00:11:59 +00:00
if ( ( int ) ent - > v - > drawmask & mask )
2005-02-09 19:32:09 +00:00
{
G_INT ( OFS_PARM0 ) = EDICT_TO_PROG ( prinst , ( void * ) ent ) ;
PF_R_AddEntity ( prinst , pr_globals ) ;
}
}
2005-02-12 18:56:04 +00:00
if ( mask & MASK_ENGINE & & cl . worldmodel )
2005-02-09 19:32:09 +00:00
{
2005-02-12 18:56:04 +00:00
CL_LinkViewModel ( ) ;
CL_LinkPlayers ( ) ;
CL_LinkPacketEntities ( ) ;
CL_LinkProjectiles ( ) ;
CL_UpdateTEnts ( ) ;
2005-02-09 19:32:09 +00:00
}
}
2005-02-12 18:56:04 +00:00
//float CalcFov (float fov_x, float width, float height);
2005-01-24 05:10:11 +00:00
//clear scene, and set up the default stuff.
static void PF_R_ClearScene ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-02-09 19:32:09 +00:00
extern frame_t * view_frame ;
extern player_state_t * view_message ;
2005-02-12 18:56:04 +00:00
CL_DecayLights ( ) ;
if ( cl . worldmodel )
{
//work out which packet entities are solid
CL_SetSolidEntities ( ) ;
// Set up prediction for other players
CL_SetUpPlayerPrediction ( false ) ;
// do client side motion prediction
CL_PredictMove ( ) ;
// Set up prediction for other players
CL_SetUpPlayerPrediction ( true ) ;
}
CL_SwapEntityLists ( ) ;
2005-02-09 19:32:09 +00:00
2005-04-26 16:04:12 +00:00
view_frame = NULL ; //&cl.frames[cls.netchan.incoming_sequence & UPDATE_MASK];
view_message = NULL ; //&view_frame->playerstate[cl.playernum[plnum]];
2005-02-09 19:32:09 +00:00
V_CalcRefdef ( 0 ) ; //set up the defaults (for player 0)
/*
2005-01-24 05:10:11 +00:00
VectorCopy ( cl . simangles [ 0 ] , r_refdef . viewangles ) ;
2005-02-09 19:32:09 +00:00
VectorCopy ( cl . simorg [ 0 ] , r_refdef . vieworg ) ;
2005-01-24 05:10:11 +00:00
r_refdef . flags = 0 ;
r_refdef . vrect . x = 0 ;
r_refdef . vrect . y = 0 ;
r_refdef . vrect . width = vid . width ;
r_refdef . vrect . height = vid . height ;
2005-02-09 19:32:09 +00:00
r_refdef . fov_x = scr_fov . value ;
r_refdef . fov_y = CalcFov ( r_refdef . fov_x , r_refdef . vrect . width , r_refdef . vrect . height ) ;
*/
2005-01-24 05:10:11 +00:00
}
static void PF_R_SetViewFlag ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * s = PR_GetStringOfs ( prinst , OFS_PARM0 ) ;
float * p = G_VECTOR ( OFS_PARM1 ) ;
switch ( * s )
{
case ' F ' :
if ( ! strcmp ( s , " FOV " ) ) //set both fov numbers
{
r_refdef . fov_x = p [ 0 ] ;
r_refdef . fov_y = p [ 1 ] ;
return ;
}
if ( ! strcmp ( s , " FOV_X " ) )
{
r_refdef . fov_x = * p ;
return ;
}
if ( ! strcmp ( s , " FOV_Y " ) )
{
r_refdef . fov_y = * p ;
return ;
}
break ;
case ' O ' :
if ( ! strcmp ( s , " ORIGIN " ) )
{
VectorCopy ( p , r_refdef . vieworg ) ;
return ;
}
if ( ! strcmp ( s , " ORIGIN_X " ) )
{
r_refdef . vieworg [ 0 ] = * p ;
return ;
}
if ( ! strcmp ( s , " ORIGIN_Y " ) )
{
r_refdef . vieworg [ 1 ] = * p ;
return ;
}
if ( ! strcmp ( s , " ORIGIN_Z " ) )
{
r_refdef . vieworg [ 2 ] = * p ;
return ;
}
break ;
case ' A ' :
if ( ! strcmp ( s , " ANGLES " ) )
{
VectorCopy ( p , r_refdef . viewangles ) ;
return ;
}
if ( ! strcmp ( s , " ANGLES_X " ) )
{
r_refdef . viewangles [ 0 ] = * p ;
return ;
}
if ( ! strcmp ( s , " ANGLES_Y " ) )
{
r_refdef . viewangles [ 1 ] = * p ;
return ;
}
if ( ! strcmp ( s , " ANGLES_Z " ) )
{
r_refdef . viewangles [ 2 ] = * p ;
return ;
}
break ;
case ' W ' :
if ( ! strcmp ( s , " WIDTH " ) )
{
r_refdef . vrect . width = * p ;
return ;
}
break ;
case ' H ' :
if ( ! strcmp ( s , " HEIGHT " ) )
{
r_refdef . vrect . height = * p ;
return ;
}
break ;
case ' S ' :
if ( ! strcmp ( s , " SIZE " ) )
{
r_refdef . vrect . width = p [ 0 ] ;
r_refdef . vrect . height = p [ 1 ] ;
return ;
}
break ;
case ' M ' :
if ( ! strcmp ( s , " MIN_X " ) )
{
r_refdef . vrect . x = * p ;
return ;
}
if ( ! strcmp ( s , " MIN_Y " ) )
{
r_refdef . vrect . y = * p ;
return ;
}
if ( ! strcmp ( s , " MIN " ) )
{
r_refdef . vrect . x = p [ 0 ] ;
r_refdef . vrect . y = p [ 1 ] ;
return ;
}
break ;
default :
break ;
}
Con_DPrintf ( " SetViewFlag: %s not recognised \n " , s ) ;
}
static void PF_R_RenderScene ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-02-28 07:16:19 +00:00
if ( cl . worldmodel )
R_PushDlights ( ) ;
2005-04-26 16:04:12 +00:00
/*
if ( cl_csqcdebug . value )
Con_Printf ( " %f %f %f \n " , r_refdef . vieworg [ 0 ] ,
r_refdef . vieworg [ 1 ] ,
r_refdef . vieworg [ 2 ] ) ;
*/
2005-02-09 19:32:09 +00:00
2005-01-24 05:10:11 +00:00
# ifdef RGLQUAKE
if ( qrenderer = = QR_OPENGL )
{
gl_ztrickdisabled | = 16 ;
qglDisable ( GL_ALPHA_TEST ) ;
qglDisable ( GL_BLEND ) ;
}
# endif
2005-04-26 16:04:12 +00:00
VectorCopy ( r_refdef . vieworg , cl . viewent [ 0 ] . origin ) ;
CalcGunAngle ( 0 ) ;
2005-01-24 05:10:11 +00:00
R_RenderView ( ) ;
2005-05-08 06:01:25 +00:00
2005-01-24 05:10:11 +00:00
# ifdef RGLQUAKE
if ( qrenderer = = QR_OPENGL )
{
gl_ztrickdisabled & = ~ 16 ;
GL_Set2D ( ) ;
qglBlendFunc ( GL_SRC_ALPHA , GL_ONE_MINUS_SRC_ALPHA ) ;
GL_TexEnv ( GL_MODULATE ) ;
}
# endif
# ifdef RGLQUAKE
if ( qrenderer = = QR_OPENGL )
{
qglDisable ( GL_ALPHA_TEST ) ;
qglEnable ( GL_BLEND ) ;
}
# endif
vid . recalc_refdef = 1 ;
}
2005-01-17 17:43:13 +00:00
2005-02-10 00:14:10 +00:00
static void PF_cs_getstatf ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int stnum = G_FLOAT ( OFS_PARM0 ) ;
2005-04-26 16:04:12 +00:00
float val = * ( float * ) & cl . stats [ plnum ] [ stnum ] ; //copy float into the stat
2005-02-10 00:14:10 +00:00
G_FLOAT ( OFS_RETURN ) = val ;
}
static void PF_cs_getstati ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
2005-02-12 18:56:04 +00:00
{ //convert an int stat into a qc float.
2005-02-10 00:14:10 +00:00
int stnum = G_FLOAT ( OFS_PARM0 ) ;
2005-04-26 16:04:12 +00:00
int val = cl . stats [ plnum ] [ stnum ] ;
2005-02-12 18:56:04 +00:00
if ( * prinst - > callargc > 1 )
{
int first , count ;
first = G_FLOAT ( OFS_PARM1 ) ;
count = G_FLOAT ( OFS_PARM2 ) ;
G_FLOAT ( OFS_RETURN ) = ( ( ( unsigned int ) val ) & ( ( ( 1 < < count ) - 1 ) < < first ) ) > > first ;
}
2005-02-10 00:14:10 +00:00
else
2005-02-12 18:56:04 +00:00
G_FLOAT ( OFS_RETURN ) = val ;
2005-02-10 00:14:10 +00:00
}
static void PF_cs_getstats ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int stnum = G_FLOAT ( OFS_PARM0 ) ;
char * out ;
2005-05-20 03:32:53 +00:00
out = PF_TempStr ( prinst ) ;
2005-02-10 00:14:10 +00:00
//the network protocol byteswaps
( ( unsigned int * ) out ) [ 0 ] = LittleLong ( cl . stats [ 0 ] [ stnum + 0 ] ) ;
( ( unsigned int * ) out ) [ 1 ] = LittleLong ( cl . stats [ 0 ] [ stnum + 1 ] ) ;
( ( unsigned int * ) out ) [ 2 ] = LittleLong ( cl . stats [ 0 ] [ stnum + 2 ] ) ;
( ( unsigned int * ) out ) [ 3 ] = LittleLong ( cl . stats [ 0 ] [ stnum + 3 ] ) ;
( ( unsigned int * ) out ) [ 4 ] = 0 ; //make sure it's null terminated
RETURN_SSTRING ( out ) ;
}
2005-02-12 18:56:04 +00:00
static void PF_CSQC_SetOrigin ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
csqcedict_t * ent = ( void * ) G_EDICT ( prinst , OFS_PARM0 ) ;
float * org = G_VECTOR ( OFS_PARM1 ) ;
2005-03-28 00:11:59 +00:00
VectorCopy ( org , ent - > v - > origin ) ;
2005-02-12 18:56:04 +00:00
//fixme: add some sort of fast area grid
}
2005-03-10 03:55:18 +00:00
static void PF_CSQC_SetSize ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
csqcedict_t * ent = ( void * ) G_EDICT ( prinst , OFS_PARM0 ) ;
float * mins = G_VECTOR ( OFS_PARM1 ) ;
float * maxs = G_VECTOR ( OFS_PARM1 ) ;
//fixme: set the size. :p
//fixme: add some sort of fast area grid
}
2005-02-28 07:16:19 +00:00
//FIXME: Not fully functional
static void PF_CSQC_traceline ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
float * v1 , * v2 , * mins , * maxs ;
trace_t trace ;
int nomonsters ;
edict_t * ent ;
// int savedhull;
v1 = G_VECTOR ( OFS_PARM0 ) ;
v2 = G_VECTOR ( OFS_PARM1 ) ;
nomonsters = G_FLOAT ( OFS_PARM2 ) ;
ent = G_EDICT ( prinst , OFS_PARM3 ) ;
// if (*prinst->callargc == 6)
// {
// mins = G_VECTOR(OFS_PARM4);
// maxs = G_VECTOR(OFS_PARM5);
// }
// else
{
mins = vec3_origin ;
maxs = vec3_origin ;
}
/*
savedhull = ent - > v . hull ;
ent - > v . hull = 0 ;
trace = SV_Move ( v1 , mins , maxs , v2 , nomonsters , ent ) ;
ent - > v . hull = savedhull ;
*/
memset ( & trace , 0 , sizeof ( trace ) ) ;
trace . fraction = 1 ;
cl . worldmodel - > hulls - > funcs . RecursiveHullCheck ( cl . worldmodel - > hulls , 0 , 0 , 1 , v1 , v2 , & trace ) ;
* csqcg . trace_allsolid = trace . allsolid ;
* csqcg . trace_startsolid = trace . startsolid ;
* csqcg . trace_fraction = trace . fraction ;
* csqcg . trace_inwater = trace . inwater ;
* csqcg . trace_inopen = trace . inopen ;
VectorCopy ( trace . endpos , csqcg . trace_endpos ) ;
VectorCopy ( trace . plane . normal , csqcg . trace_plane_normal ) ;
* csqcg . trace_plane_dist = trace . plane . dist ;
// if (trace.ent)
// *csqcg.trace_ent = EDICT_TO_PROG(prinst, trace.ent);
// else
* csqcg . trace_ent = EDICT_TO_PROG ( prinst , ( void * ) csqc_edicts ) ;
}
static void PF_CSQC_pointcontents ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
float * v ;
int cont ;
v = G_VECTOR ( OFS_PARM0 ) ;
cont = cl . worldmodel - > hulls [ 0 ] . funcs . HullPointContents ( & cl . worldmodel - > hulls [ 0 ] , v ) ;
if ( cont & FTECONTENTS_SOLID )
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_SOLID ;
else if ( cont & FTECONTENTS_SKY )
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_SKY ;
else if ( cont & FTECONTENTS_LAVA )
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_LAVA ;
else if ( cont & FTECONTENTS_SLIME )
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_SLIME ;
else if ( cont & FTECONTENTS_WATER )
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_WATER ;
else
G_FLOAT ( OFS_RETURN ) = Q1CONTENTS_EMPTY ;
}
2005-02-12 18:56:04 +00:00
static int FindModel ( char * name , int * free )
{
int i ;
* free = 0 ;
for ( i = 1 ; i < MAX_CSQCMODELS ; i + + )
{
if ( ! * cl . model_csqcname [ i ] )
{
* free = - i ;
break ;
}
if ( ! strcmp ( cl . model_csqcname [ i ] , name ) )
return - i ;
}
for ( i = 1 ; i < MAX_MODELS ; i + + )
{
if ( ! strcmp ( cl . model_name [ i ] , name ) )
return i ;
}
return 0 ;
}
static void PF_CSQC_SetModel ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
csqcedict_t * ent = ( void * ) G_EDICT ( prinst , OFS_PARM0 ) ;
char * modelname = PR_GetStringOfs ( prinst , OFS_PARM1 ) ;
int freei ;
int modelindex = FindModel ( modelname , & freei ) ;
if ( ! modelindex )
{
if ( ! freei )
Host_EndGame ( " CSQC ran out of model slots \n " ) ;
Con_DPrintf ( " Late caching model \" %s \" \n " , modelname ) ;
Q_strncpyz ( cl . model_csqcname [ - freei ] , modelname , sizeof ( cl . model_csqcname [ - freei ] ) ) ; //allocate a slot now
modelindex = freei ;
cl . model_csqcprecache [ - freei ] = Mod_ForName ( cl . model_csqcname [ - freei ] , false ) ;
}
2005-03-28 00:11:59 +00:00
ent - > v - > modelindex = modelindex ;
2005-02-12 18:56:04 +00:00
if ( modelindex < 0 )
2005-03-28 00:11:59 +00:00
ent - > v - > model = PR_SetString ( prinst , cl . model_csqcname [ - modelindex ] ) ;
2005-02-12 18:56:04 +00:00
else
2005-03-28 00:11:59 +00:00
ent - > v - > model = PR_SetString ( prinst , cl . model_name [ modelindex ] ) ;
2005-02-12 18:56:04 +00:00
}
static void PF_CSQC_SetModelIndex ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
csqcedict_t * ent = ( void * ) G_EDICT ( prinst , OFS_PARM0 ) ;
int modelindex = G_FLOAT ( OFS_PARM1 ) ;
2005-03-28 00:11:59 +00:00
ent - > v - > modelindex = modelindex ;
2005-02-12 18:56:04 +00:00
if ( modelindex < 0 )
2005-03-28 00:11:59 +00:00
ent - > v - > model = PR_SetString ( prinst , cl . model_csqcname [ - modelindex ] ) ;
2005-02-12 18:56:04 +00:00
else
2005-03-28 00:11:59 +00:00
ent - > v - > model = PR_SetString ( prinst , cl . model_name [ modelindex ] ) ;
2005-02-12 18:56:04 +00:00
}
static void PF_CSQC_PrecacheModel ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-02-28 07:16:19 +00:00
int modelindex , freei ;
2005-02-12 18:56:04 +00:00
char * modelname = PR_GetStringOfs ( prinst , OFS_PARM0 ) ;
int i ;
2005-02-28 07:16:19 +00:00
for ( i = 1 ; i < MAX_MODELS ; i + + ) //Make sure that the server specified model is loaded..
2005-02-12 18:56:04 +00:00
{
if ( ! * cl . model_name [ i ] )
break ;
if ( ! strcmp ( cl . model_name [ i ] , modelname ) )
{
cl . model_precache [ i ] = Mod_ForName ( cl . model_name [ i ] , false ) ;
break ;
}
}
2005-02-28 07:16:19 +00:00
modelindex = FindModel ( modelname , & freei ) ; //now load it
if ( ! modelindex )
{
if ( ! freei )
Host_EndGame ( " CSQC ran out of model slots \n " ) ;
Q_strncpyz ( cl . model_csqcname [ - freei ] , modelname , sizeof ( cl . model_csqcname [ - freei ] ) ) ; //allocate a slot now
modelindex = freei ;
cl . model_csqcprecache [ - freei ] = Mod_ForName ( cl . model_csqcname [ - freei ] , false ) ;
}
}
static void PF_CSQC_PrecacheSound ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * soundname = PR_GetStringOfs ( prinst , OFS_PARM0 ) ;
S_PrecacheSound ( soundname ) ;
2005-02-12 18:56:04 +00:00
}
2005-02-28 07:16:19 +00:00
2005-02-12 18:56:04 +00:00
static void PF_CSQC_ModelnameForIndex ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int modelindex = G_FLOAT ( OFS_PARM0 ) ;
if ( modelindex < 0 )
G_INT ( OFS_RETURN ) = ( int ) PR_SetString ( prinst , cl . model_csqcname [ - modelindex ] ) ;
else
G_INT ( OFS_RETURN ) = ( int ) PR_SetString ( prinst , cl . model_name [ modelindex ] ) ;
}
2005-01-17 17:43:13 +00:00
2005-02-28 07:16:19 +00:00
static void PF_ReadByte ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadByte ( ) ;
}
static void PF_ReadChar ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadChar ( ) ;
}
static void PF_ReadShort ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadShort ( ) ;
}
static void PF_ReadLong ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadLong ( ) ;
}
static void PF_ReadCoord ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadCoord ( ) ;
}
static void PF_ReadString ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-05-20 03:32:53 +00:00
char * str = PF_TempStr ( prinst ) ;
2005-02-28 07:16:19 +00:00
char * read = MSG_ReadString ( ) ;
Q_strncpyz ( str , read , MAXTEMPBUFFERLEN ) ;
}
static void PF_ReadAngle ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = MSG_ReadAngle ( ) ;
}
static void PF_objerror ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * s ;
edict_t * ed ;
s = PF_VarString ( prinst , 0 , pr_globals ) ;
/* Con_Printf ("======OBJECT ERROR in %s:\n%s\n", PR_GetString(pr_xfunction->s_name),s);
*/ ed = PROG_TO_EDICT ( prinst , pr_global_struct - > self ) ;
/* ED_Print (ed);
*/
ED_Print ( prinst , ed ) ;
Con_Printf ( " %s " , s ) ;
if ( developer . value )
( * prinst - > pr_trace ) = 2 ;
else
{
ED_Free ( prinst , ed ) ;
prinst - > AbortStack ( prinst ) ;
PR_BIError ( prinst , " Program error: %s " , s ) ;
if ( sv . time > 10 )
Cbuf_AddText ( " restart \n " , RESTRICT_LOCAL ) ;
}
}
static void PF_cs_setsensativityscaler ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
in_sensitivityscale = G_FLOAT ( OFS_PARM0 ) ;
}
2005-03-10 03:55:18 +00:00
static void PF_cs_pointparticles ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
2005-06-04 04:20:20 +00:00
int effectnum = G_FLOAT ( OFS_PARM0 ) ;
2005-03-10 03:55:18 +00:00
float * org = G_VECTOR ( OFS_PARM1 ) ;
float * vel = G_VECTOR ( OFS_PARM2 ) ;
float count = G_FLOAT ( OFS_PARM3 ) ;
if ( * prinst - > callargc < 3 )
vel = vec3_origin ;
if ( * prinst - > callargc < 4 )
count = 1 ;
P_RunParticleEffectType ( org , vel , count , effectnum ) ;
}
static void PF_cs_particlesloaded ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * effectname = PR_GetStringOfs ( prinst , OFS_PARM0 ) ;
2005-06-04 04:20:20 +00:00
G_FLOAT ( OFS_RETURN ) = P_ParticleTypeForName ( effectname ) ;
2005-03-10 03:55:18 +00:00
}
2005-04-26 16:04:12 +00:00
//get the input commands, and stuff them into some globals.
static void PF_cs_getinputstate ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int f ;
usercmd_t * cmd ;
f = G_FLOAT ( OFS_PARM0 ) ;
if ( f > cls . netchan . outgoing_sequence )
{
G_FLOAT ( OFS_RETURN ) = false ;
return ;
}
if ( f < cls . netchan . outgoing_sequence - UPDATE_MASK | | f < 0 )
{
G_FLOAT ( OFS_RETURN ) = false ;
return ;
}
// save this command off for prediction
cmd = & cl . frames [ f & UPDATE_MASK ] . cmd [ plnum ] ;
if ( csqcg . input_timelength )
* csqcg . input_timelength = cmd - > msec / 1000.0f ;
if ( csqcg . input_angles )
{
csqcg . input_angles [ 0 ] = SHORT2ANGLE ( cmd - > angles [ 0 ] ) ;
csqcg . input_angles [ 1 ] = SHORT2ANGLE ( cmd - > angles [ 1 ] ) ;
csqcg . input_angles [ 2 ] = SHORT2ANGLE ( cmd - > angles [ 2 ] ) ;
}
if ( csqcg . input_movevalues )
{
csqcg . input_movevalues [ 0 ] = cmd - > forwardmove ;
csqcg . input_movevalues [ 1 ] = cmd - > sidemove ;
csqcg . input_movevalues [ 2 ] = cmd - > upmove ;
}
if ( csqcg . input_buttons )
* csqcg . input_buttons = cmd - > buttons ;
G_FLOAT ( OFS_RETURN ) = true ;
}
# define ANGLE2SHORT(x) ((x / 360.0)*65535)
//read lots of globals, run the default player physics, write lots of globals.
static void PF_cs_runplayerphysics ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
int msecs ;
extern vec3_t player_mins ;
extern vec3_t player_maxs ;
/*
int sequence ; // just for debugging prints
// player state
vec3_t origin ;
vec3_t angles ;
vec3_t velocity ;
qboolean jump_held ;
int jump_msec ; // msec since last jump
float waterjumptime ;
int pm_type ;
int hullnum ;
// world state
int numphysent ;
physent_t physents [ MAX_PHYSENTS ] ; // 0 should be the world
// input
usercmd_t cmd ;
qboolean onladder ;
// results
int numtouch ;
int touchindex [ MAX_PHYSENTS ] ;
qboolean onground ;
int groundent ; // index in physents array, only valid
// when onground is true
int waterlevel ;
int watertype ;
} playermove_t ;
typedef struct {
float gravity ;
float stopspeed ;
float maxspeed ;
float spectatormaxspeed ;
float accelerate ;
float airaccelerate ;
float wateraccelerate ;
float friction ;
float waterfriction ;
float entgravity ;
float bunnyspeedcap ;
float ktjump ;
qboolean slidefix ;
qboolean airstep ;
qboolean walljump ;
*/
pmove . sequence = * csqcg . clientcommandframe ;
pmove . pm_type = PM_NORMAL ;
//set up the movement command
msecs = * csqcg . input_timelength * 1000 + 0.5f ;
//precision inaccuracies. :(
pmove . angles [ 0 ] = ANGLE2SHORT ( csqcg . input_angles [ 0 ] ) ;
pmove . angles [ 1 ] = ANGLE2SHORT ( csqcg . input_angles [ 1 ] ) ;
pmove . angles [ 2 ] = ANGLE2SHORT ( csqcg . input_angles [ 2 ] ) ;
pmove . cmd . forwardmove = csqcg . input_movevalues [ 0 ] ;
pmove . cmd . sidemove = csqcg . input_movevalues [ 1 ] ;
pmove . cmd . upmove = csqcg . input_movevalues [ 2 ] ;
VectorCopy ( csqcg . pmove_org , pmove . origin ) ;
VectorCopy ( csqcg . pmove_vel , pmove . velocity ) ;
VectorCopy ( csqcg . pmove_maxs , player_maxs ) ;
VectorCopy ( csqcg . pmove_mins , player_mins ) ;
pmove . hullnum = 1 ;
while ( msecs )
{
pmove . cmd . msec = msecs ;
if ( pmove . cmd . msec > 50 )
pmove . cmd . msec = 50 ;
msecs - = pmove . cmd . msec ;
PM_PlayerMove ( 1 ) ;
}
VectorCopy ( pmove . origin , csqcg . pmove_org ) ;
VectorCopy ( pmove . velocity , csqcg . pmove_vel ) ;
}
static void CheckSendPings ( void )
{ //quakeworld sends a 'pings' client command to retrieve the frequently updating stuff
if ( realtime - cl . last_ping_request > 2 )
{
cl . last_ping_request = realtime ;
CL_SendClientCommand ( false , " pings " ) ;
}
}
//string(float pnum, string keyname)
static void PF_cs_getplayerkey ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * ret ;
int pnum = G_FLOAT ( OFS_PARM0 ) ;
char * keyname = PR_GetStringOfs ( prinst , OFS_PARM1 ) ;
if ( pnum < 0 )
{
Sbar_SortFrags ( false ) ;
if ( pnum > = - scoreboardlines )
{ //sort by
pnum = fragsort [ - ( pnum + 1 ) ] ;
}
}
if ( pnum < 0 | | pnum > = MAX_CLIENTS )
ret = " " ;
else if ( ! * cl . players [ pnum ] . userinfo )
ret = " " ;
else if ( ! strcmp ( keyname , " ping " ) )
{
CheckSendPings ( ) ;
2005-05-20 03:32:53 +00:00
ret = PF_TempStr ( prinst ) ;
2005-04-26 16:04:12 +00:00
sprintf ( ret , " %i " , cl . players [ pnum ] . ping ) ;
}
else if ( ! strcmp ( keyname , " frags " ) )
{
2005-05-20 03:32:53 +00:00
ret = PF_TempStr ( prinst ) ;
2005-04-26 16:04:12 +00:00
sprintf ( ret , " %i " , cl . players [ pnum ] . frags ) ;
}
else if ( ! strcmp ( keyname , " pl " ) ) //packet loss
{
CheckSendPings ( ) ;
2005-05-20 03:32:53 +00:00
ret = PF_TempStr ( prinst ) ;
2005-04-26 16:04:12 +00:00
sprintf ( ret , " %i " , cl . players [ pnum ] . pl ) ;
}
else if ( ! strcmp ( keyname , " entertime " ) ) //packet loss
{
2005-05-20 03:32:53 +00:00
ret = PF_TempStr ( prinst ) ;
2005-04-26 16:04:12 +00:00
sprintf ( ret , " %i " , cl . players [ pnum ] . entertime ) ;
}
else
{
ret = Info_ValueForKey ( cl . players [ pnum ] . userinfo , keyname ) ;
}
if ( * ret )
RETURN_SSTRING ( ret ) ;
2005-05-08 06:01:25 +00:00
else
G_INT ( OFS_RETURN ) = 0 ;
2005-04-26 16:04:12 +00:00
}
extern int mouseusedforgui , mousecursor_x , mousecursor_y ;
static void PF_cs_setwantskeys ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
qboolean wants = G_FLOAT ( OFS_PARM0 ) ;
csqcwantskeys = wants ;
mouseusedforgui = wants ;
}
static void PF_cs_getmousepos ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN + 0 ) = mousecursor_x ;
G_FLOAT ( OFS_RETURN + 1 ) = mousecursor_y ;
G_FLOAT ( OFS_RETURN + 2 ) = 0 ;
}
2005-05-19 02:53:03 +00:00
static void PF_checkextension ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
G_FLOAT ( OFS_RETURN ) = 0 ;
}
2005-06-04 04:20:20 +00:00
void PF_cs_sound ( progfuncs_t * prinst , struct globalvars_s * pr_globals )
{
char * sample ;
int channel ;
csqcedict_t * entity ;
int volume ;
float attenuation ;
sfx_t * sfx ;
2005-06-14 04:52:10 +00:00
entity = ( csqcedict_t * ) G_EDICT ( prinst , OFS_PARM0 ) ;
2005-06-04 04:20:20 +00:00
channel = G_FLOAT ( OFS_PARM1 ) ;
sample = PR_GetStringOfs ( prinst , OFS_PARM2 ) ;
volume = G_FLOAT ( OFS_PARM3 ) * 255 ;
attenuation = G_FLOAT ( OFS_PARM4 ) ;
sfx = S_PrecacheSound ( sample ) ;
if ( sfx )
S_StartSound ( - entity - > entnum , channel , sfx , entity - > v - > origin , volume , attenuation ) ;
} ;
2005-04-26 16:04:12 +00:00
2005-03-07 08:40:03 +00:00
# define PF_FixTen PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme,PF_Fixme
2005-01-17 17:43:13 +00:00
//warning: functions that depend on globals are bad, mkay?
2005-04-26 16:04:12 +00:00
static builtin_t csqc_builtins [ ] = {
2005-01-17 17:43:13 +00:00
//0
PF_Fixme ,
PF_makevectors ,
2005-02-12 18:56:04 +00:00
PF_CSQC_SetOrigin , //PF_setorigin
PF_CSQC_SetModel , //PF_setmodel
2005-03-10 03:55:18 +00:00
PF_CSQC_SetSize , //PF_setsize
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_Fixme , //PF_break,
PF_random ,
2005-06-04 04:20:20 +00:00
PF_cs_sound , //PF_sound,
2005-01-17 17:43:13 +00:00
PF_normalize ,
//10
PF_error ,
2005-02-28 07:16:19 +00:00
PF_objerror ,
2005-01-17 17:43:13 +00:00
PF_vlen ,
PF_vectoyaw ,
PF_Spawn ,
2005-02-28 07:16:19 +00:00
PF_Remove_ , //PF_Remove,
PF_CSQC_traceline , //PF_traceline,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_checkclient, (don't support)
2005-01-17 17:43:13 +00:00
PF_FindString ,
2005-02-28 07:16:19 +00:00
PF_CSQC_PrecacheSound , //PF_precache_sound,
2005-01-17 17:43:13 +00:00
//20
2005-02-12 18:56:04 +00:00
PF_CSQC_PrecacheModel , //PF_precache_model,
PF_Fixme , //PF_stuffcmd, (don't support)
2005-01-17 17:43:13 +00:00
PF_Fixme , //PF_findradius,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_bprint, (don't support)
2005-01-17 17:43:13 +00:00
PF_Fixme , //PF_sprint,
PF_dprint ,
PF_ftos ,
PF_vtos ,
PF_coredump ,
PF_traceon ,
//30
PF_traceoff ,
PF_eprint ,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_walkmove, (don't support yet)
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_Fixme , //PF_droptofloor,
PF_Fixme , //PF_lightstyle,
PF_rint ,
PF_floor ,
PF_ceil ,
PF_Fixme ,
//40
PF_Fixme , //PF_checkbottom,
2005-02-28 07:16:19 +00:00
PF_CSQC_pointcontents , //PF_pointcontents,
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_fabs ,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_aim, hehehe... (don't support)
2005-01-17 17:43:13 +00:00
PF_cvar ,
PF_localcmd ,
PF_nextent ,
PF_Fixme , //PF_particle,
PF_Fixme , //PF_changeyaw,
//50
PF_Fixme ,
PF_vectoangles ,
2005-02-28 07:16:19 +00:00
PF_ReadByte ,
PF_ReadChar ,
PF_ReadShort ,
PF_ReadLong ,
PF_ReadCoord ,
PF_ReadAngle ,
PF_ReadString ,
PF_Fixme , //PF_ReadEntity,
2005-01-17 17:43:13 +00:00
//60
PF_Fixme ,
PF_Sin ,
PF_Cos ,
PF_Sqrt ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
SV_MoveToGoal ,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_precache_file, (don't support)
2005-01-17 17:43:13 +00:00
PF_Fixme , //PF_makestatic,
//70
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_changelevel, (don't support)
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_cvar_set ,
PF_Fixme , //PF_centerprint,
PF_Fixme , //PF_ambientsound,
2005-02-28 07:16:19 +00:00
PF_CSQC_PrecacheModel , //PF_precache_model,
2005-01-17 17:43:13 +00:00
PF_Fixme , //PF_precache_sound,
PF_Fixme , //PF_precache_file,
PF_Fixme , //PF_setspawnparms,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_logfrag, (don't support)
2005-01-17 17:43:13 +00:00
//80
PF_Fixme , //PF_infokey,
PF_stof ,
2005-02-12 18:56:04 +00:00
PF_Fixme , //PF_multicast, (don't support)
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
//90
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
2005-04-26 16:04:12 +00:00
PF_registercvar ,
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
//100
2005-05-19 02:53:03 +00:00
PF_checkextension ,
2005-01-17 17:43:13 +00:00
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
//110
2005-02-10 00:14:10 +00:00
PF_fopen ,
PF_fclose ,
PF_fgets ,
PF_fputs ,
PF_strlen ,
2005-01-17 17:43:13 +00:00
2005-02-10 00:14:10 +00:00
PF_strcat ,
PF_substring ,
PF_stov ,
PF_dupstring ,
PF_forgetstring ,
2005-01-17 17:43:13 +00:00
2005-02-09 19:32:09 +00:00
//120
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
//130
PF_R_ClearScene ,
PF_R_AddEntityMask ,
PF_R_AddEntity ,
PF_R_SetViewFlag ,
PF_R_RenderScene ,
2005-02-28 07:16:19 +00:00
PF_R_AddDynamicLight ,
2005-02-09 19:32:09 +00:00
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
//140
PF_CL_is_cached_pic , //0
PF_CL_precache_pic , //1
PF_CL_free_pic , //2
PF_CL_drawcharacter , //3
PF_CL_drawstring , //4
PF_CL_drawpic , //5
PF_CL_drawfill , //6
PF_CL_drawsetcliparea , //7
PF_CL_drawresetcliparea , //8
PF_CL_drawgetimagesize , //9
2005-01-17 17:43:13 +00:00
2005-02-10 00:14:10 +00:00
//150
PF_cs_getstatf ,
PF_cs_getstati ,
PF_cs_getstats ,
2005-02-28 07:16:19 +00:00
PF_CSQC_SetModelIndex ,
PF_CSQC_ModelnameForIndex ,
2005-02-10 00:14:10 +00:00
2005-02-28 07:16:19 +00:00
PF_cs_setsensativityscaler ,
2005-03-07 08:40:03 +00:00
PF_csqc_centerprint ,
PF_print ,
2005-03-10 03:55:18 +00:00
PF_cs_pointparticles ,
PF_cs_particlesloaded ,
2005-02-10 00:14:10 +00:00
//160
2005-04-26 16:04:12 +00:00
PF_cs_getinputstate ,
PF_cs_runplayerphysics ,
PF_cs_getplayerkey ,
PF_cs_setwantskeys ,
PF_cs_getmousepos ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
2005-02-10 00:14:10 +00:00
//170
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//180
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//190
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//200
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//210
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//220
PF_Fixme ,
PF_strstrofs ,
PF_str2chr ,
PF_chr2str ,
PF_strconv ,
PF_infoadd ,
PF_infoget ,
PF_strncmp ,
PF_strcasecmp ,
PF_strncasecmp ,
//230
2005-03-07 08:40:03 +00:00
PF_FixTen ,
2005-02-10 00:14:10 +00:00
//240
2005-03-07 08:40:03 +00:00
PF_FixTen ,
//250
PF_FixTen ,
//260
PF_FixTen ,
//270
PF_FixTen ,
//280
PF_FixTen ,
//290
PF_FixTen ,
//300
PF_FixTen ,
//310
PF_FixTen ,
//320
PF_FixTen ,
//330
PF_FixTen ,
//340
PF_FixTen ,
//350
PF_FixTen ,
//360
PF_FixTen ,
//370
PF_FixTen ,
//380
PF_FixTen ,
//390
PF_FixTen ,
//400
PF_FixTen ,
//410
PF_FixTen ,
//420
PF_FixTen ,
//430
PF_FixTen ,
//440
2005-02-10 00:14:10 +00:00
PF_Fixme ,
2005-03-07 08:40:03 +00:00
PF_Tokenize ,
PF_ArgV ,
2005-02-10 00:14:10 +00:00
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
PF_Fixme ,
2005-03-07 08:40:03 +00:00
PF_Fixme ,
2005-02-09 19:32:09 +00:00
} ;
2005-04-26 16:04:12 +00:00
static int csqc_numbuiltins = sizeof ( csqc_builtins ) / sizeof ( csqc_builtins [ 0 ] ) ;
2005-01-17 17:43:13 +00:00
2005-04-26 16:04:12 +00:00
static jmp_buf csqc_abort ;
static progparms_t csqcprogparms ;
static int num_csqc_edicts ;
2005-01-17 17:43:13 +00:00
int COM_FileSize ( char * path ) ;
pbool QC_WriteFile ( char * name , void * data , int len ) ;
2005-04-19 21:09:29 +00:00
void * VARGS PR_CB_Malloc ( int size ) ; //these functions should be tracked by the library reliably, so there should be no need to track them ourselves.
void VARGS PR_CB_Free ( void * mem ) ;
2005-01-17 17:43:13 +00:00
//Any menu builtin error or anything like that will come here.
2005-02-09 19:32:09 +00:00
void VARGS CSQC_Abort ( char * format , . . . ) //an error occured.
2005-01-17 17:43:13 +00:00
{
va_list argptr ;
char string [ 1024 ] ;
va_start ( argptr , format ) ;
_vsnprintf ( string , sizeof ( string ) - 1 , format , argptr ) ;
va_end ( argptr ) ;
Con_Printf ( " CSQC_Abort: %s \n Shutting down csqc \n " , string ) ;
{
2005-05-19 02:53:03 +00:00
int size = 1024 * 1024 * 8 ;
char * buffer = BZ_Malloc ( size ) ;
2005-01-17 17:43:13 +00:00
csqcprogs - > save_ents ( csqcprogs , buffer , & size , 3 ) ;
COM_WriteFile ( " csqccore.txt " , buffer , size ) ;
2005-05-19 02:53:03 +00:00
BZ_Free ( buffer ) ;
2005-01-17 17:43:13 +00:00
}
Host_EndGame ( " csqc error " ) ;
}
2005-02-09 19:32:09 +00:00
void CSQC_Shutdown ( void )
{
if ( csqcprogs )
{
CloseProgs ( csqcprogs ) ;
Con_Printf ( " Closed csqc \n " ) ;
}
csqcprogs = NULL ;
2005-02-28 07:16:19 +00:00
in_sensitivityscale = 1 ;
2005-01-24 05:10:11 +00:00
}
2005-03-10 03:55:18 +00:00
//when the qclib needs a file, it calls out to this function.
qbyte * CSQC_PRLoadFile ( char * path , void * buffer , int bufsize )
{
qbyte * file ;
//pretend it doesn't
file = COM_LoadStackFile ( path , buffer , bufsize ) ;
2005-05-19 02:53:03 +00:00
if ( ! cls . demoplayback ) //allow any csqc when playing a demo
if ( ! strcmp ( path , " csprogs.dat " ) ) //Fail to load any csprogs who's checksum doesn't match.
if ( Com_BlockChecksum ( buffer , com_filesize ) ! = csqcchecksum )
return NULL ;
2005-03-10 03:55:18 +00:00
return file ;
}
2005-01-17 17:43:13 +00:00
double csqctime ;
2005-03-10 03:55:18 +00:00
qboolean CSQC_Init ( unsigned int checksum )
2005-01-17 17:43:13 +00:00
{
2005-03-10 03:55:18 +00:00
csqcchecksum = checksum ;
2005-02-09 19:32:09 +00:00
CSQC_Shutdown ( ) ;
2005-01-17 17:43:13 +00:00
if ( ! qrenderer )
{
2005-03-10 03:55:18 +00:00
return false ;
2005-01-17 17:43:13 +00:00
}
2005-02-12 18:56:04 +00:00
memset ( cl . model_csqcname , 0 , sizeof ( cl . model_csqcname ) ) ;
memset ( cl . model_csqcprecache , 0 , sizeof ( cl . model_csqcprecache ) ) ;
2005-01-17 17:43:13 +00:00
csqcprogparms . progsversion = PROGSTRUCT_VERSION ;
2005-03-10 03:55:18 +00:00
csqcprogparms . ReadFile = CSQC_PRLoadFile ; //char *(*ReadFile) (char *fname, void *buffer, int *len);
2005-01-17 17:43:13 +00:00
csqcprogparms . FileSize = COM_FileSize ; //int (*FileSize) (char *fname); //-1 if file does not exist
csqcprogparms . WriteFile = QC_WriteFile ; //bool (*WriteFile) (char *name, void *data, int len);
csqcprogparms . printf = ( void * ) Con_Printf ; //Con_Printf;//void (*printf) (char *, ...);
csqcprogparms . Sys_Error = Sys_Error ;
csqcprogparms . Abort = CSQC_Abort ;
2005-03-28 07:05:38 +00:00
csqcprogparms . edictsize = sizeof ( csqcedict_t ) ;
2005-01-17 17:43:13 +00:00
csqcprogparms . entspawn = NULL ; //void (*entspawn) (struct edict_s *ent); //ent has been spawned, but may not have all the extra variables (that may need to be set) set
csqcprogparms . entcanfree = NULL ; //bool (*entcanfree) (struct edict_s *ent); //return true to stop ent from being freed
csqcprogparms . stateop = NULL ; //StateOp;//void (*stateop) (float var, func_t func);
csqcprogparms . cstateop = NULL ; //CStateOp;
csqcprogparms . cwstateop = NULL ; //CWStateOp;
csqcprogparms . thinktimeop = NULL ; //ThinkTimeOp;
//used when loading a game
csqcprogparms . builtinsfor = NULL ; //builtin_t *(*builtinsfor) (int num); //must return a pointer to the builtins that were used before the state was saved.
csqcprogparms . loadcompleate = NULL ; //void (*loadcompleate) (int edictsize); //notification to reset any pointers.
2005-04-19 21:09:29 +00:00
csqcprogparms . memalloc = PR_CB_Malloc ; //void *(*memalloc) (int size); //small string allocation malloced and freed randomly
csqcprogparms . memfree = PR_CB_Free ; //void (*memfree) (void * mem);
2005-01-17 17:43:13 +00:00
csqcprogparms . globalbuiltins = csqc_builtins ; //builtin_t *globalbuiltins; //these are available to all progs
csqcprogparms . numglobalbuiltins = csqc_numbuiltins ;
csqcprogparms . autocompile = PR_NOCOMPILE ; //enum {PR_NOCOMPILE, PR_COMPILENEXIST, PR_COMPILECHANGED, PR_COMPILEALWAYS} autocompile;
csqcprogparms . gametime = & csqctime ;
csqcprogparms . sv_edicts = ( edict_t * * ) & csqc_edicts ;
csqcprogparms . sv_num_edicts = & num_csqc_edicts ;
csqcprogparms . useeditor = NULL ; //sorry... QCEditor;//void (*useeditor) (char *filename, int line, int nump, char **parms);
csqctime = Sys_DoubleTime ( ) ;
if ( ! csqcprogs )
{
2005-02-28 07:16:19 +00:00
in_sensitivityscale = 1 ;
2005-01-17 17:43:13 +00:00
csqcprogs = InitProgs ( & csqcprogparms ) ;
2005-03-28 00:11:59 +00:00
PR_Configure ( csqcprogs , - 1 , 1 ) ;
2005-01-24 05:10:11 +00:00
CSQC_InitFields ( ) ; //let the qclib know the field order that the engine needs.
2005-02-06 02:47:36 +00:00
if ( PR_LoadProgs ( csqcprogs , " csprogs.dat " , 0 , NULL , 0 ) < 0 ) //no per-progs builtins.
2005-01-17 17:43:13 +00:00
{
2005-02-09 19:32:09 +00:00
CSQC_Shutdown ( ) ;
2005-01-17 17:43:13 +00:00
//failed to load or something
2005-03-10 03:55:18 +00:00
return false ;
2005-01-17 17:43:13 +00:00
}
if ( setjmp ( csqc_abort ) )
{
2005-02-09 19:32:09 +00:00
CSQC_Shutdown ( ) ;
2005-03-10 03:55:18 +00:00
return false ;
2005-01-17 17:43:13 +00:00
}
2005-02-28 07:16:19 +00:00
2005-05-20 03:32:53 +00:00
PF_InitTempStrings ( csqcprogs ) ;
2005-02-28 07:16:19 +00:00
memset ( csqcent , 0 , sizeof ( csqcent ) ) ;
2005-01-24 05:10:11 +00:00
2005-03-20 02:57:11 +00:00
csqcentsize = PR_InitEnts ( csqcprogs , pr_csmaxedicts . value ) ;
2005-01-24 05:10:11 +00:00
CSQC_FindGlobals ( ) ;
2005-01-17 17:43:13 +00:00
2005-02-28 07:16:19 +00:00
ED_Alloc ( csqcprogs ) ; //we need a word entity.
2005-01-24 05:10:11 +00:00
//world edict becomes readonly
2005-01-17 17:43:13 +00:00
EDICT_NUM ( csqcprogs , 0 ) - > readonly = true ;
EDICT_NUM ( csqcprogs , 0 ) - > isfree = false ;
2005-02-28 07:16:19 +00:00
if ( csqcg . init_function )
PR_ExecuteProgram ( csqcprogs , csqcg . init_function ) ;
2005-02-09 19:32:09 +00:00
Con_Printf ( " Loaded csqc \n " ) ;
2005-01-17 17:43:13 +00:00
}
2005-03-10 03:55:18 +00:00
return true ; //success!
2005-01-17 17:43:13 +00:00
}
2005-03-20 02:57:11 +00:00
void CSQC_RegisterCvarsAndThings ( void )
{
Cvar_Register ( & pr_csmaxedicts , " csqc " ) ;
2005-03-24 18:13:37 +00:00
Cvar_Register ( & cl_csqcdebug , " csqc " ) ;
2005-03-20 02:57:11 +00:00
}
2005-01-17 17:43:13 +00:00
qboolean CSQC_DrawView ( void )
{
2005-03-10 03:55:18 +00:00
if ( ! csqcg . draw_function | | ! csqcprogs | | ! cl . worldmodel )
2005-01-17 17:43:13 +00:00
return false ;
2005-02-12 18:56:04 +00:00
r_secondaryview = 0 ;
2005-05-17 02:36:54 +00:00
DropPunchAngle ( 0 ) ;
2005-02-12 18:56:04 +00:00
if ( cl . worldmodel )
R_LessenStains ( ) ;
2005-02-09 19:32:09 +00:00
2005-04-26 16:04:12 +00:00
if ( csqcg . clientcommandframe )
* csqcg . clientcommandframe = cls . netchan . outgoing_sequence ;
if ( csqcg . servercommandframe )
* csqcg . servercommandframe = cls . netchan . incoming_sequence ;
2005-02-28 07:16:19 +00:00
if ( csqcg . time )
* csqcg . time = Sys_DoubleTime ( ) ;
PR_ExecuteProgram ( csqcprogs , csqcg . draw_function ) ;
2005-01-17 17:43:13 +00:00
return true ;
}
2005-04-26 16:04:12 +00:00
qboolean CSQC_KeyPress ( int key , qboolean down )
{
void * pr_globals ;
if ( ! csqcprogs | | ! csqcwantskeys )
return false ;
pr_globals = PR_globals ( csqcprogs , PR_CURRENT ) ;
2005-04-27 15:52:15 +00:00
G_FLOAT ( OFS_PARM0 ) = ! down ;
2005-05-08 06:01:25 +00:00
G_FLOAT ( OFS_PARM1 ) = MP_TranslateFTEtoDPCodes ( key ) ;
2005-04-26 16:04:12 +00:00
G_FLOAT ( OFS_PARM2 ) = 0 ;
PR_ExecuteProgram ( csqcprogs , csqcg . input_event ) ;
return true ;
}
2005-05-15 18:49:04 +00:00
qboolean CSQC_ConsoleCommand ( char * cmd )
{
void * pr_globals ;
char * str ;
if ( ! csqcprogs | | ! csqcg . console_command )
return false ;
2005-05-20 03:32:53 +00:00
str = PF_TempStr ( csqcprogs ) ;
2005-05-15 18:49:04 +00:00
Q_strncpyz ( str , cmd , MAXTEMPBUFFERLEN ) ;
pr_globals = PR_globals ( csqcprogs , PR_CURRENT ) ;
( ( ( string_t * ) pr_globals ) [ OFS_PARM0 ] = PR_SetString ( csqcprogs , str ) ) ;
PR_ExecuteProgram ( csqcprogs , csqcg . console_command ) ;
return true ;
}
2005-02-12 18:56:04 +00:00
qboolean CSQC_StuffCmd ( char * cmd )
{
void * pr_globals ;
char * str ;
2005-02-28 07:16:19 +00:00
if ( ! csqcprogs | | ! csqcg . parse_stuffcmd )
2005-02-12 18:56:04 +00:00
return false ;
2005-05-20 03:32:53 +00:00
str = PF_TempStr ( csqcprogs ) ;
2005-02-12 18:56:04 +00:00
Q_strncpyz ( str , cmd , MAXTEMPBUFFERLEN ) ;
pr_globals = PR_globals ( csqcprogs , PR_CURRENT ) ;
2005-05-15 18:49:04 +00:00
( ( ( string_t * ) pr_globals ) [ OFS_PARM0 ] = PR_SetString ( csqcprogs , str ) ) ;
2005-02-12 18:56:04 +00:00
2005-02-28 07:16:19 +00:00
PR_ExecuteProgram ( csqcprogs , csqcg . parse_stuffcmd ) ;
2005-02-12 18:56:04 +00:00
return true ;
}
qboolean CSQC_CenterPrint ( char * cmd )
{
void * pr_globals ;
char * str ;
2005-02-28 07:16:19 +00:00
if ( ! csqcprogs | | ! csqcg . parse_centerprint )
2005-02-12 18:56:04 +00:00
return false ;
2005-05-20 03:32:53 +00:00
str = PF_TempStr ( csqcprogs ) ;
2005-02-12 18:56:04 +00:00
Q_strncpyz ( str , cmd , MAXTEMPBUFFERLEN ) ;
pr_globals = PR_globals ( csqcprogs , PR_CURRENT ) ;
2005-05-15 18:49:04 +00:00
( ( ( string_t * ) pr_globals ) [ OFS_PARM0 ] = PR_SetString ( csqcprogs , str ) ) ;
2005-02-12 18:56:04 +00:00
2005-02-28 07:16:19 +00:00
PR_ExecuteProgram ( csqcprogs , csqcg . parse_centerprint ) ;
2005-02-12 18:56:04 +00:00
return G_FLOAT ( OFS_RETURN ) ;
}
//this protocol allows up to 32767 edicts.
2005-02-28 07:16:19 +00:00
# ifdef PEXT_CSQC
2005-02-09 19:32:09 +00:00
void CSQC_ParseEntities ( void )
{
2005-02-12 18:56:04 +00:00
csqcedict_t * ent ;
unsigned short entnum ;
2005-02-28 07:16:19 +00:00
void * pr_globals ;
2005-03-20 02:57:11 +00:00
int packetsize ;
int packetstart ;
2005-02-12 18:56:04 +00:00
if ( ! csqcprogs )
2005-03-01 15:36:23 +00:00
Host_EndGame ( " CSQC needs to be initialized for this server. \n " ) ;
2005-04-26 16:04:12 +00:00
if ( ! csqcg . ent_update | | ! csqcg . self )
2005-03-01 15:36:23 +00:00
Host_EndGame ( " CSQC is unable to parse entities \n " ) ;
2005-02-28 07:16:19 +00:00
pr_globals = PR_globals ( csqcprogs , PR_CURRENT ) ;
if ( csqcg . time )
* csqcg . time = Sys_DoubleTime ( ) ;
2005-04-26 16:04:12 +00:00
if ( csqcg . servercommandframe )
* csqcg . servercommandframe = cls . netchan . incoming_sequence ;
2005-02-12 18:56:04 +00:00
for ( ; ; )
{
entnum = MSG_ReadShort ( ) ;
if ( ! entnum )
break ;
if ( entnum & 0x8000 )
{ //remove
entnum & = ~ 0x8000 ;
if ( ! entnum )
2005-02-28 07:16:19 +00:00
Host_EndGame ( " CSQC cannot remove world! \n " ) ;
if ( entnum > = MAX_EDICTS )
Host_EndGame ( " CSQC recieved too many edicts! \n " ) ;
2005-03-24 18:13:37 +00:00
if ( cl_csqcdebug . value )
Con_Printf ( " Remove %i \n " , entnum ) ;
2005-02-12 18:56:04 +00:00
ent = csqcent [ entnum ] ;
2005-04-26 16:04:12 +00:00
csqcent [ entnum ] = NULL ;
2005-02-12 18:56:04 +00:00
if ( ! ent ) //hrm.
continue ;
2005-02-28 07:16:19 +00:00
* csqcg . self = EDICT_TO_PROG ( csqcprogs , ( void * ) ent ) ;
PR_ExecuteProgram ( csqcprogs , csqcg . ent_remove ) ;
2005-02-12 18:56:04 +00:00
//the csqc is expected to call the remove builtin.
}
else
{
2005-02-28 07:16:19 +00:00
if ( entnum > = MAX_EDICTS )
Host_EndGame ( " CSQC recieved too many edicts! \n " ) ;
2005-03-20 02:57:11 +00:00
if ( cl . csqcdebug )
{
packetsize = MSG_ReadShort ( ) ;
packetstart = msg_readcount ;
}
2005-06-14 04:52:10 +00:00
else
{
packetsize = 0 ;
packetstart = 0 ;
}
2005-03-20 02:57:11 +00:00
2005-02-12 18:56:04 +00:00
ent = csqcent [ entnum ] ;
if ( ! ent )
{
2005-02-28 07:16:19 +00:00
ent = ( csqcedict_t * ) ED_Alloc ( csqcprogs ) ;
csqcent [ entnum ] = ent ;
2005-04-26 16:04:12 +00:00
ent - > v - > entnum = entnum ;
2005-02-12 18:56:04 +00:00
G_FLOAT ( OFS_PARM0 ) = true ;
2005-03-24 18:13:37 +00:00
if ( cl_csqcdebug . value )
Con_Printf ( " Add %i \n " , entnum ) ;
2005-02-12 18:56:04 +00:00
}
else
2005-03-24 18:13:37 +00:00
{
2005-02-12 18:56:04 +00:00
G_FLOAT ( OFS_PARM0 ) = false ;
2005-03-24 18:13:37 +00:00
if ( cl_csqcdebug . value )
Con_Printf ( " Update %i \n " , entnum ) ;
}
2005-02-12 18:56:04 +00:00
2005-02-28 07:16:19 +00:00
* csqcg . self = EDICT_TO_PROG ( csqcprogs , ( void * ) ent ) ;
PR_ExecuteProgram ( csqcprogs , csqcg . ent_update ) ;
2005-03-20 02:57:11 +00:00
if ( cl . csqcdebug )
{
if ( msg_readcount ! = packetstart + packetsize )
{
if ( msg_readcount > packetstart + packetsize )
Con_Printf ( " CSQC overread entity %i. Size %i, read %i \n " , entnum , packetsize , msg_readcount - packetsize ) ;
else
Con_Printf ( " CSQC underread entity %i. Size %i, read %i \n " , entnum , packetsize , msg_readcount - packetsize ) ;
Con_Printf ( " First byte is %i \n " , net_message . data [ msg_readcount ] ) ;
# ifndef CLIENTONLY
if ( sv . state )
{
2005-03-28 00:11:59 +00:00
Con_Printf ( " Server classname: \" %s \" \n " , svprogfuncs - > stringtable + EDICT_NUM ( svprogfuncs , entnum ) - > v - > classname ) ;
2005-03-20 02:57:11 +00:00
}
# endif
}
msg_readcount = packetstart + packetsize ; //leetism.
}
2005-02-12 18:56:04 +00:00
}
2005-02-28 07:16:19 +00:00
}
2005-02-09 19:32:09 +00:00
}
2005-02-28 07:16:19 +00:00
# endif
2005-02-09 19:32:09 +00:00
2005-01-17 17:43:13 +00:00
# endif