#include "quakedef.h" #include "prvm_cmds.h" #include "jpeg.h" //============================================================================ // Server char *vm_sv_extensions = "BX_WAL_SUPPORT " "DP_BUTTONCHAT " "DP_BUTTONUSE " "DP_CL_LOADSKY " "DP_CON_ALIASPARAMETERS " "DP_CON_BESTWEAPON " "DP_CON_EXPANDCVAR " "DP_CON_SET " "DP_CON_SETA " "DP_CON_STARTMAP " "DP_CSQC_MULTIFRAME_INTERPOLATION " "DP_EF_ADDITIVE " "DP_EF_BLUE " "DP_EF_DOUBLESIDED " "DP_EF_FLAME " "DP_EF_FULLBRIGHT " "DP_EF_NODEPTHTEST " "DP_EF_NODRAW " "DP_EF_NOGUNBOB " "DP_EF_NOSELFSHADOW " "DP_EF_NOSHADOW " "DP_EF_RED " "DP_EF_STARDUST " "DP_EF_TELEPORT_BIT " "DP_ENT_ALPHA " "DP_ENT_COLORMOD " "DP_ENT_CUSTOMCOLORMAP " "DP_ENT_EXTERIORMODELTOCLIENT " "DP_ENT_GLOW " "DP_ENT_LOWPRECISION " "DP_ENT_SCALE " "DP_ENT_VIEWMODEL " "DP_GECKO_SUPPORT " "DP_GFX_EXTERNALTEXTURES " "DP_GFX_EXTERNALTEXTURES_PERMAP " "DP_GFX_FOG " "DP_GFX_QUAKE3MODELTAGS " "DP_GFX_SKINFILES " "DP_GFX_SKYBOX " "DP_GFX_MODEL_INTERPOLATION " "DP_HALFLIFE_MAP " "DP_HALFLIFE_MAP_CVAR " "DP_HALFLIFE_SPRITE " "DP_INPUTBUTTONS " "DP_LITSPRITES " "DP_LITSUPPORT " "DP_MONSTERWALK " "DP_MOVETYPEBOUNCEMISSILE " "DP_MOVETYPEFOLLOW " "DP_NULL_MODEL " "DP_QC_ASINACOSATANATAN2TAN " "DP_QC_CHANGEPITCH " "DP_QC_CMD " "DP_QC_COPYENTITY " "DP_QC_CRC16 " "DP_QC_CVAR_DEFSTRING " "DP_QC_CVAR_DESCRIPTION " "DP_QC_CVAR_STRING " "DP_QC_CVAR_TYPE " "DP_QC_EDICT_NUM " "DP_QC_ENTITYDATA " "DP_QC_ETOS " "DP_QC_EXTRESPONSEPACKET " "DP_QC_FINDCHAIN " "DP_QC_FINDCHAIN_TOFIELD " "DP_QC_FINDCHAINFLAGS " "DP_QC_FINDCHAINFLOAT " "DP_QC_FINDFLAGS " "DP_QC_FINDFLOAT " "DP_QC_FS_SEARCH " "DP_QC_GETLIGHT " "DP_QC_GETSURFACE " "DP_QC_GETSURFACEPOINTATTRIBUTE " "DP_QC_GETTAGINFO " "DP_QC_GETTAGINFO_BONEPROPERTIES " "DP_QC_GETTIME " "DP_QC_GETTIME_CDTRACK " "DP_QC_MINMAXBOUND " "DP_QC_MULTIPLETEMPSTRINGS " "DP_QC_NUM_FOR_EDICT " "DP_QC_RANDOMVEC " "DP_QC_SINCOSSQRTPOW " "DP_QC_STRFTIME " "DP_QC_STRINGBUFFERS " "DP_QC_STRINGBUFFERS_CVARLIST " "DP_QC_STRINGCOLORFUNCTIONS " "DP_QC_STRING_CASE_FUNCTIONS " "DP_QC_STRREPLACE " "DP_QC_TOKENIZEBYSEPARATOR " "DP_QC_TOKENIZE_CONSOLE " "DP_QC_TRACEBOX " "DP_QC_TRACETOSS " "DP_QC_TRACE_MOVETYPE_HITMODEL " "DP_QC_TRACE_MOVETYPE_WORLDONLY " "DP_QC_UNLIMITEDTEMPSTRINGS " "DP_QC_URI_ESCAPE " "DP_QC_URI_GET " "DP_QC_VECTOANGLES_WITH_ROLL " "DP_QC_VECTORVECTORS " "DP_QC_WHICHPACK " "DP_QUAKE2_MODEL " "DP_QUAKE2_SPRITE " "DP_QUAKE3_MAP " "DP_QUAKE3_MODEL " "DP_REGISTERCVAR " "DP_SND_DIRECTIONLESSATTNNONE " "DP_SND_FAKETRACKS " "DP_SND_OGGVORBIS " "DP_SND_STEREOWAV " "DP_SOLIDCORPSE " "DP_SPRITE32 " "DP_SV_BOTCLIENT " "DP_SV_BOUNCEFACTOR " "DP_SV_CLIENTCOLORS " "DP_SV_CLIENTNAME " "DP_SV_CMD " "DP_SV_CUSTOMIZEENTITYFORCLIENT " "DP_SV_DRAWONLYTOCLIENT " "DP_SV_DROPCLIENT " "DP_SV_EFFECT " "DP_SV_ENTITYCONTENTSTRANSITION " "DP_SV_MODELFLAGS_AS_EFFECTS " "DP_SV_MOVETYPESTEP_LANDEVENT " "DP_SV_NETADDRESS " "DP_SV_NODRAWTOCLIENT " "DP_SV_ONENTITYNOSPAWNFUNCTION " "DP_SV_ONENTITYPREPOSTSPAWNFUNCTION " "DP_SV_PING " "DP_SV_PLAYERPHYSICS " "DP_SV_POINTPARTICLES " "DP_SV_POINTSOUND " "DP_SV_PRECACHEANYTIME " "DP_SV_PRINT " "DP_SV_PUNCHVECTOR " "DP_SV_QCSTATUS " "DP_SV_ROTATINGBMODEL " "DP_SV_SETCOLOR " "DP_SV_SHUTDOWN " "DP_SV_SLOWMO " "DP_SV_SPAWNFUNC_PREFIX " "DP_SV_WRITEPICTURE " "DP_SV_WRITEUNTERMINATEDSTRING " "DP_TE_BLOOD " "DP_TE_BLOODSHOWER " "DP_TE_CUSTOMFLASH " "DP_TE_EXPLOSIONRGB " "DP_TE_FLAMEJET " "DP_TE_PARTICLECUBE " "DP_TE_PARTICLERAIN " "DP_TE_PARTICLESNOW " "DP_TE_PLASMABURN " "DP_TE_QUADEFFECTS1 " "DP_TE_SMALLFLASH " "DP_TE_SPARK " "DP_TE_STANDARDEFFECTBUILTINS " "DP_TRACE_HITCONTENTSMASK_SURFACEINFO " "DP_VIEWZOOM " "EXT_BITSHIFT " "FRIK_FILE " "FTE_QC_CHECKPVS " "FTE_STRINGS " "KRIMZON_SV_PARSECLIENTCOMMAND " "NEH_CMD_PLAY2 " "NEH_RESTOREGAME " "NEXUIZ_PLAYERMODEL " "NXQ_GFX_LETTERBOX " "PRYDON_CLIENTCURSOR " "TENEBRAE_GFX_DLIGHTS " "TW_SV_STEPCONTROL " "ZQ_PAUSE " //"EXT_CSQC " // not ready yet ; /* ================= VM_SV_setorigin This is the only valid way to move an object without using the physics of the world (setting velocity and waiting). Directly changing origin will not set internal links correctly, so clipping would be messed up. This should be called when an object is spawned, and then only if it is teleported. setorigin (entity, origin) ================= */ static void VM_SV_setorigin (void) { prvm_edict_t *e; float *org; VM_SAFEPARMCOUNT(2, VM_setorigin); e = PRVM_G_EDICT(OFS_PARM0); if (e == prog->edicts) { VM_Warning("setorigin: can not modify world entity\n"); return; } if (e->priv.server->free) { VM_Warning("setorigin: can not modify free entity\n"); return; } org = PRVM_G_VECTOR(OFS_PARM1); VectorCopy (org, e->fields.server->origin); SV_LinkEdict (e, false); } // TODO: rotate param isnt used.. could be a bug. please check this and remove it if possible [1/10/2008 Black] static void SetMinMaxSize (prvm_edict_t *e, float *min, float *max, qboolean rotate) { int i; for (i=0 ; i<3 ; i++) if (min[i] > max[i]) PRVM_ERROR("SetMinMaxSize: backwards mins/maxs"); // set derived values VectorCopy (min, e->fields.server->mins); VectorCopy (max, e->fields.server->maxs); VectorSubtract (max, min, e->fields.server->size); SV_LinkEdict (e, false); } /* ================= VM_SV_setsize the size box is rotated by the current angle LordHavoc: no it isn't... setsize (entity, minvector, maxvector) ================= */ static void VM_SV_setsize (void) { prvm_edict_t *e; float *min, *max; VM_SAFEPARMCOUNT(3, VM_setsize); e = PRVM_G_EDICT(OFS_PARM0); if (e == prog->edicts) { VM_Warning("setsize: can not modify world entity\n"); return; } if (e->priv.server->free) { VM_Warning("setsize: can not modify free entity\n"); return; } min = PRVM_G_VECTOR(OFS_PARM1); max = PRVM_G_VECTOR(OFS_PARM2); SetMinMaxSize (e, min, max, false); } /* ================= VM_SV_setmodel setmodel(entity, model) ================= */ static vec3_t quakemins = {-16, -16, -16}, quakemaxs = {16, 16, 16}; static void VM_SV_setmodel (void) { prvm_edict_t *e; dp_model_t *mod; int i; VM_SAFEPARMCOUNT(2, VM_setmodel); e = PRVM_G_EDICT(OFS_PARM0); if (e == prog->edicts) { VM_Warning("setmodel: can not modify world entity\n"); return; } if (e->priv.server->free) { VM_Warning("setmodel: can not modify free entity\n"); return; } i = SV_ModelIndex(PRVM_G_STRING(OFS_PARM1), 1); e->fields.server->model = PRVM_SetEngineString(sv.model_precache[i]); e->fields.server->modelindex = i; mod = sv.models[i]; if (mod) { if (mod->type != mod_alias || sv_gameplayfix_setmodelrealbox.integer) SetMinMaxSize (e, mod->normalmins, mod->normalmaxs, true); else SetMinMaxSize (e, quakemins, quakemaxs, true); } else SetMinMaxSize (e, vec3_origin, vec3_origin, true); } /* ================= VM_SV_sprint single print to a specific client sprint(clientent, value) ================= */ static void VM_SV_sprint (void) { client_t *client; int entnum; char string[VM_STRINGTEMP_LENGTH]; VM_VarString(1, string, sizeof(string)); VM_SAFEPARMCOUNTRANGE(2, 8, VM_SV_sprint); entnum = PRVM_G_EDICTNUM(OFS_PARM0); // LordHavoc: div0 requested that sprintto world operate like print if (entnum == 0) { Con_Print(string); return; } if (entnum < 1 || entnum > svs.maxclients || !svs.clients[entnum-1].active) { VM_Warning("tried to centerprint to a non-client\n"); return; } client = svs.clients + entnum-1; if (!client->netconnection) return; MSG_WriteChar(&client->netconnection->message,svc_print); MSG_WriteString(&client->netconnection->message, string); } /* ================= VM_SV_centerprint single print to a specific client centerprint(clientent, value) ================= */ static void VM_SV_centerprint (void) { client_t *client; int entnum; char string[VM_STRINGTEMP_LENGTH]; VM_SAFEPARMCOUNTRANGE(2, 8, VM_SV_centerprint); entnum = PRVM_G_EDICTNUM(OFS_PARM0); if (entnum < 1 || entnum > svs.maxclients || !svs.clients[entnum-1].active) { VM_Warning("tried to centerprint to a non-client\n"); return; } client = svs.clients + entnum-1; if (!client->netconnection) return; VM_VarString(1, string, sizeof(string)); MSG_WriteChar(&client->netconnection->message,svc_centerprint); MSG_WriteString(&client->netconnection->message, string); } /* ================= VM_SV_particle particle(origin, color, count) ================= */ static void VM_SV_particle (void) { float *org, *dir; float color; float count; VM_SAFEPARMCOUNT(4, VM_SV_particle); org = PRVM_G_VECTOR(OFS_PARM0); dir = PRVM_G_VECTOR(OFS_PARM1); color = PRVM_G_FLOAT(OFS_PARM2); count = PRVM_G_FLOAT(OFS_PARM3); SV_StartParticle (org, dir, (int)color, (int)count); } /* ================= VM_SV_ambientsound ================= */ static void VM_SV_ambientsound (void) { const char *samp; float *pos; float vol, attenuation; int soundnum, large; VM_SAFEPARMCOUNT(4, VM_SV_ambientsound); pos = PRVM_G_VECTOR (OFS_PARM0); samp = PRVM_G_STRING(OFS_PARM1); vol = PRVM_G_FLOAT(OFS_PARM2); attenuation = PRVM_G_FLOAT(OFS_PARM3); // check to see if samp was properly precached soundnum = SV_SoundIndex(samp, 1); if (!soundnum) return; large = false; if (soundnum >= 256) large = true; // add an svc_spawnambient command to the level signon packet if (large) MSG_WriteByte (&sv.signon, svc_spawnstaticsound2); else MSG_WriteByte (&sv.signon, svc_spawnstaticsound); MSG_WriteVector(&sv.signon, pos, sv.protocol); if (large || sv.protocol == PROTOCOL_NEHAHRABJP || sv.protocol == PROTOCOL_NEHAHRABJP2 || sv.protocol == PROTOCOL_NEHAHRABJP3) MSG_WriteShort (&sv.signon, soundnum); else MSG_WriteByte (&sv.signon, soundnum); MSG_WriteByte (&sv.signon, (int)(vol*255)); MSG_WriteByte (&sv.signon, (int)(attenuation*64)); } /* ================= VM_SV_sound Each entity can have eight independant sound sources, like voice, weapon, feet, etc. Channel 0 is an auto-allocate channel, the others override anything already running on that entity/channel pair. An attenuation of 0 will play full volume everywhere in the level. Larger attenuations will drop off. ================= */ static void VM_SV_sound (void) { const char *sample; int channel; prvm_edict_t *entity; int volume; float attenuation; VM_SAFEPARMCOUNTRANGE(4, 5, VM_SV_sound); entity = PRVM_G_EDICT(OFS_PARM0); channel = (int)PRVM_G_FLOAT(OFS_PARM1); sample = PRVM_G_STRING(OFS_PARM2); volume = (int)(PRVM_G_FLOAT(OFS_PARM3) * 255); attenuation = PRVM_G_FLOAT(OFS_PARM4); if (prog->argc < 5) { Con_DPrintf("VM_SV_sound: given only 4 parameters, expected 5, assuming attenuation = ATTN_NORMAL\n"); attenuation = 1; } if (volume < 0 || volume > 255) { VM_Warning("SV_StartSound: volume must be in range 0-1\n"); return; } if (attenuation < 0 || attenuation > 4) { VM_Warning("SV_StartSound: attenuation must be in range 0-4\n"); return; } if (channel < 0 || channel > 7) { VM_Warning("SV_StartSound: channel must be in range 0-7\n"); return; } SV_StartSound (entity, channel, sample, volume, attenuation); } /* ================= VM_SV_pointsound Follows the same logic as VM_SV_sound, except instead of an entity, an origin for the sound is provided, and channel is omitted (since no entity is being tracked). ================= */ static void VM_SV_pointsound(void) { const char *sample; int volume; float attenuation; vec3_t org; VM_SAFEPARMCOUNT(4, VM_SV_pointsound); VectorCopy(PRVM_G_VECTOR(OFS_PARM0), org); sample = PRVM_G_STRING(OFS_PARM1); volume = (int)(PRVM_G_FLOAT(OFS_PARM2) * 255); attenuation = PRVM_G_FLOAT(OFS_PARM3); if (volume < 0 || volume > 255) { VM_Warning("SV_StartPointSound: volume must be in range 0-1\n"); return; } if (attenuation < 0 || attenuation > 4) { VM_Warning("SV_StartPointSound: attenuation must be in range 0-4\n"); return; } SV_StartPointSound (org, sample, volume, attenuation); } /* ================= VM_SV_traceline Used for use tracing and shot targeting Traces are blocked by bbox and exact bsp entityes, and also slide box entities if the tryents flag is set. traceline (vector1, vector2, movetype, ignore) ================= */ static void VM_SV_traceline (void) { float *v1, *v2; trace_t trace; int move; prvm_edict_t *ent; VM_SAFEPARMCOUNTRANGE(4, 8, VM_SV_traceline); // allow more parameters for future expansion prog->xfunction->builtinsprofile += 30; v1 = PRVM_G_VECTOR(OFS_PARM0); v2 = PRVM_G_VECTOR(OFS_PARM1); move = (int)PRVM_G_FLOAT(OFS_PARM2); ent = PRVM_G_EDICT(OFS_PARM3); if (IS_NAN(v1[0]) || IS_NAN(v1[1]) || IS_NAN(v1[2]) || IS_NAN(v2[0]) || IS_NAN(v2[1]) || IS_NAN(v2[2])) PRVM_ERROR("%s: NAN errors detected in traceline('%f %f %f', '%f %f %f', %i, entity %i)\n", PRVM_NAME, v1[0], v1[1], v1[2], v2[0], v2[1], v2[2], move, PRVM_EDICT_TO_PROG(ent)); trace = SV_Move (v1, vec3_origin, vec3_origin, v2, move, ent, SV_GenericHitSuperContentsMask(ent)); VM_SetTraceGlobals(&trace); } /* ================= VM_SV_tracebox Used for use tracing and shot targeting Traces are blocked by bbox and exact bsp entityes, and also slide box entities if the tryents flag is set. tracebox (vector1, vector mins, vector maxs, vector2, tryents) ================= */ // LordHavoc: added this for my own use, VERY useful, similar to traceline static void VM_SV_tracebox (void) { float *v1, *v2, *m1, *m2; trace_t trace; int move; prvm_edict_t *ent; VM_SAFEPARMCOUNTRANGE(6, 8, VM_SV_tracebox); // allow more parameters for future expansion prog->xfunction->builtinsprofile += 30; v1 = PRVM_G_VECTOR(OFS_PARM0); m1 = PRVM_G_VECTOR(OFS_PARM1); m2 = PRVM_G_VECTOR(OFS_PARM2); v2 = PRVM_G_VECTOR(OFS_PARM3); move = (int)PRVM_G_FLOAT(OFS_PARM4); ent = PRVM_G_EDICT(OFS_PARM5); if (IS_NAN(v1[0]) || IS_NAN(v1[1]) || IS_NAN(v1[2]) || IS_NAN(v2[0]) || IS_NAN(v2[1]) || IS_NAN(v2[2])) PRVM_ERROR("%s: NAN errors detected in tracebox('%f %f %f', '%f %f %f', '%f %f %f', '%f %f %f', %i, entity %i)\n", PRVM_NAME, v1[0], v1[1], v1[2], m1[0], m1[1], m1[2], m2[0], m2[1], m2[2], v2[0], v2[1], v2[2], move, PRVM_EDICT_TO_PROG(ent)); trace = SV_Move (v1, m1, m2, v2, move, ent, SV_GenericHitSuperContentsMask(ent)); VM_SetTraceGlobals(&trace); } static trace_t SV_Trace_Toss (prvm_edict_t *tossent, prvm_edict_t *ignore) { int i; float gravity; vec3_t move, end; vec3_t original_origin; vec3_t original_velocity; vec3_t original_angles; vec3_t original_avelocity; prvm_eval_t *val; trace_t trace; VectorCopy(tossent->fields.server->origin , original_origin ); VectorCopy(tossent->fields.server->velocity , original_velocity ); VectorCopy(tossent->fields.server->angles , original_angles ); VectorCopy(tossent->fields.server->avelocity, original_avelocity); val = PRVM_EDICTFIELDVALUE(tossent, prog->fieldoffsets.gravity); if (val != NULL && val->_float != 0) gravity = val->_float; else gravity = 1.0; gravity *= sv_gravity.value * 0.025; for (i = 0;i < 200;i++) // LordHavoc: sanity check; never trace more than 10 seconds { SV_CheckVelocity (tossent); tossent->fields.server->velocity[2] -= gravity; VectorMA (tossent->fields.server->angles, 0.05, tossent->fields.server->avelocity, tossent->fields.server->angles); VectorScale (tossent->fields.server->velocity, 0.05, move); VectorAdd (tossent->fields.server->origin, move, end); trace = SV_Move (tossent->fields.server->origin, tossent->fields.server->mins, tossent->fields.server->maxs, end, MOVE_NORMAL, tossent, SV_GenericHitSuperContentsMask(tossent)); VectorCopy (trace.endpos, tossent->fields.server->origin); tossent->fields.server->velocity[2] -= gravity; if (trace.fraction < 1) break; } VectorCopy(original_origin , tossent->fields.server->origin ); VectorCopy(original_velocity , tossent->fields.server->velocity ); VectorCopy(original_angles , tossent->fields.server->angles ); VectorCopy(original_avelocity, tossent->fields.server->avelocity); return trace; } static void VM_SV_tracetoss (void) { trace_t trace; prvm_edict_t *ent; prvm_edict_t *ignore; VM_SAFEPARMCOUNT(2, VM_SV_tracetoss); prog->xfunction->builtinsprofile += 600; ent = PRVM_G_EDICT(OFS_PARM0); if (ent == prog->edicts) { VM_Warning("tracetoss: can not use world entity\n"); return; } ignore = PRVM_G_EDICT(OFS_PARM1); trace = SV_Trace_Toss (ent, ignore); VM_SetTraceGlobals(&trace); } //============================================================================ static int checkpvsbytes; static unsigned char checkpvs[MAX_MAP_LEAFS/8]; static int VM_SV_newcheckclient (int check) { int i; prvm_edict_t *ent; vec3_t org; // cycle to the next one check = bound(1, check, svs.maxclients); if (check == svs.maxclients) i = 1; else i = check + 1; for ( ; ; i++) { // count the cost prog->xfunction->builtinsprofile++; // wrap around if (i == svs.maxclients+1) i = 1; // look up the client's edict ent = PRVM_EDICT_NUM(i); // check if it is to be ignored, but never ignore the one we started on (prevent infinite loop) if (i != check && (ent->priv.server->free || ent->fields.server->health <= 0 || ((int)ent->fields.server->flags & FL_NOTARGET))) continue; // found a valid client (possibly the same one again) break; } // get the PVS for the entity VectorAdd(ent->fields.server->origin, ent->fields.server->view_ofs, org); checkpvsbytes = 0; if (sv.worldmodel && sv.worldmodel->brush.FatPVS) checkpvsbytes = sv.worldmodel->brush.FatPVS(sv.worldmodel, org, 0, checkpvs, sizeof(checkpvs), false); return i; } /* ================= VM_SV_checkclient Returns a client (or object that has a client enemy) that would be a valid target. If there is more than one valid option, they are cycled each frame If (self.origin + self.viewofs) is not in the PVS of the current target, it is not returned at all. name checkclient () ================= */ int c_invis, c_notvis; static void VM_SV_checkclient (void) { prvm_edict_t *ent, *self; vec3_t view; VM_SAFEPARMCOUNT(0, VM_SV_checkclient); // find a new check if on a new frame if (sv.time - sv.lastchecktime >= 0.1) { sv.lastcheck = VM_SV_newcheckclient (sv.lastcheck); sv.lastchecktime = sv.time; } // return check if it might be visible ent = PRVM_EDICT_NUM(sv.lastcheck); if (ent->priv.server->free || ent->fields.server->health <= 0) { VM_RETURN_EDICT(prog->edicts); return; } // if current entity can't possibly see the check entity, return 0 self = PRVM_PROG_TO_EDICT(prog->globals.server->self); VectorAdd(self->fields.server->origin, self->fields.server->view_ofs, view); if (sv.worldmodel && checkpvsbytes && !sv.worldmodel->brush.BoxTouchingPVS(sv.worldmodel, checkpvs, view, view)) { c_notvis++; VM_RETURN_EDICT(prog->edicts); return; } // might be able to see it c_invis++; VM_RETURN_EDICT(ent); } //============================================================================ /* ================= VM_SV_checkpvs Checks if an entity is in a point's PVS. Should be fast but can be inexact. float checkpvs(vector viewpos, entity viewee) = #240; ================= */ static void VM_SV_checkpvs (void) { vec3_t viewpos; prvm_edict_t *viewee; #if 1 unsigned char *pvs; #else static int fatpvsbytes; static unsigned char fatpvs[MAX_MAP_LEAFS/8]; #endif VM_SAFEPARMCOUNT(2, VM_SV_checkpvs); VectorCopy(PRVM_G_VECTOR(OFS_PARM0), viewpos); viewee = PRVM_G_EDICT(OFS_PARM1); if(viewee->priv.server->free) { VM_Warning("checkpvs: can not check free entity\n"); PRVM_G_FLOAT(OFS_RETURN) = 4; return; } #if 1 if(!sv.worldmodel->brush.GetPVS || !sv.worldmodel->brush.BoxTouchingPVS) { // no PVS support on this worldmodel... darn PRVM_G_FLOAT(OFS_RETURN) = 3; return; } pvs = sv.worldmodel->brush.GetPVS(sv.worldmodel, viewpos); if(!pvs) { // viewpos isn't in any PVS... darn PRVM_G_FLOAT(OFS_RETURN) = 2; return; } PRVM_G_FLOAT(OFS_RETURN) = sv.worldmodel->brush.BoxTouchingPVS(sv.worldmodel, pvs, viewee->fields.server->absmin, viewee->fields.server->absmax); #else // using fat PVS like FTEQW does (slow) if(!sv.worldmodel->brush.FatPVS || !sv.worldmodel->brush.BoxTouchingPVS) { // no PVS support on this worldmodel... darn PRVM_G_FLOAT(OFS_RETURN) = 3; return; } fatpvsbytes = sv.worldmodel->brush.FatPVS(sv.worldmodel, viewpos, 8, fatpvs, sizeof(fatpvs), false); if(!fatpvsbytes) { // viewpos isn't in any PVS... darn PRVM_G_FLOAT(OFS_RETURN) = 2; return; } PRVM_G_FLOAT(OFS_RETURN) = sv.worldmodel->brush.BoxTouchingPVS(sv.worldmodel, fatpvs, viewee->fields.server->absmin, viewee->fields.server->absmax); #endif } /* ================= VM_SV_stuffcmd Sends text over to the client's execution buffer stuffcmd (clientent, value, ...) ================= */ static void VM_SV_stuffcmd (void) { int entnum; client_t *old; char string[VM_STRINGTEMP_LENGTH]; VM_SAFEPARMCOUNTRANGE(2, 8, VM_SV_stuffcmd); entnum = PRVM_G_EDICTNUM(OFS_PARM0); if (entnum < 1 || entnum > svs.maxclients || !svs.clients[entnum-1].active) { VM_Warning("Can't stuffcmd to a non-client\n"); return; } VM_VarString(1, string, sizeof(string)); old = host_client; host_client = svs.clients + entnum-1; Host_ClientCommands ("%s", string); host_client = old; } /* ================= VM_SV_findradius Returns a chain of entities that have origins within a spherical area findradius (origin, radius) ================= */ static void VM_SV_findradius (void) { prvm_edict_t *ent, *chain; vec_t radius, radius2; vec3_t org, eorg, mins, maxs; int i; int numtouchedicts; prvm_edict_t *touchedicts[MAX_EDICTS]; int chainfield; VM_SAFEPARMCOUNTRANGE(2, 3, VM_SV_findradius); if(prog->argc == 3) chainfield = PRVM_G_INT(OFS_PARM2); else chainfield = prog->fieldoffsets.chain; if (chainfield < 0) PRVM_ERROR("VM_findchain: %s doesnt have the specified chain field !", PRVM_NAME); chain = (prvm_edict_t *)prog->edicts; VectorCopy(PRVM_G_VECTOR(OFS_PARM0), org); radius = PRVM_G_FLOAT(OFS_PARM1); radius2 = radius * radius; mins[0] = org[0] - (radius + 1); mins[1] = org[1] - (radius + 1); mins[2] = org[2] - (radius + 1); maxs[0] = org[0] + (radius + 1); maxs[1] = org[1] + (radius + 1); maxs[2] = org[2] + (radius + 1); numtouchedicts = World_EntitiesInBox(&sv.world, mins, maxs, MAX_EDICTS, touchedicts); if (numtouchedicts > MAX_EDICTS) { // this never happens Con_Printf("SV_EntitiesInBox returned %i edicts, max was %i\n", numtouchedicts, MAX_EDICTS); numtouchedicts = MAX_EDICTS; } for (i = 0;i < numtouchedicts;i++) { ent = touchedicts[i]; prog->xfunction->builtinsprofile++; // Quake did not return non-solid entities but darkplaces does // (note: this is the reason you can't blow up fallen zombies) if (ent->fields.server->solid == SOLID_NOT && !sv_gameplayfix_blowupfallenzombies.integer) continue; // LordHavoc: compare against bounding box rather than center so it // doesn't miss large objects, and use DotProduct instead of Length // for a major speedup VectorSubtract(org, ent->fields.server->origin, eorg); if (sv_gameplayfix_findradiusdistancetobox.integer) { eorg[0] -= bound(ent->fields.server->mins[0], eorg[0], ent->fields.server->maxs[0]); eorg[1] -= bound(ent->fields.server->mins[1], eorg[1], ent->fields.server->maxs[1]); eorg[2] -= bound(ent->fields.server->mins[2], eorg[2], ent->fields.server->maxs[2]); } else VectorMAMAM(1, eorg, -0.5f, ent->fields.server->mins, -0.5f, ent->fields.server->maxs, eorg); if (DotProduct(eorg, eorg) < radius2) { PRVM_EDICTFIELDVALUE(ent,chainfield)->edict = PRVM_EDICT_TO_PROG(chain); chain = ent; } } VM_RETURN_EDICT(chain); } static void VM_SV_precache_sound (void) { VM_SAFEPARMCOUNT(1, VM_SV_precache_sound); PRVM_G_FLOAT(OFS_RETURN) = SV_SoundIndex(PRVM_G_STRING(OFS_PARM0), 2); } static void VM_SV_precache_model (void) { VM_SAFEPARMCOUNT(1, VM_SV_precache_model); SV_ModelIndex(PRVM_G_STRING(OFS_PARM0), 2); PRVM_G_INT(OFS_RETURN) = PRVM_G_INT(OFS_PARM0); } /* =============== VM_SV_walkmove float(float yaw, float dist[, settrace]) walkmove =============== */ static void VM_SV_walkmove (void) { prvm_edict_t *ent; float yaw, dist; vec3_t move; mfunction_t *oldf; int oldself; qboolean settrace; VM_SAFEPARMCOUNTRANGE(2, 3, VM_SV_walkmove); // assume failure if it returns early PRVM_G_FLOAT(OFS_RETURN) = 0; ent = PRVM_PROG_TO_EDICT(prog->globals.server->self); if (ent == prog->edicts) { VM_Warning("walkmove: can not modify world entity\n"); return; } if (ent->priv.server->free) { VM_Warning("walkmove: can not modify free entity\n"); return; } yaw = PRVM_G_FLOAT(OFS_PARM0); dist = PRVM_G_FLOAT(OFS_PARM1); settrace = prog->argc >= 3 && PRVM_G_FLOAT(OFS_PARM2); if ( !( (int)ent->fields.server->flags & (FL_ONGROUND|FL_FLY|FL_SWIM) ) ) return; yaw = yaw*M_PI*2 / 360; move[0] = cos(yaw)*dist; move[1] = sin(yaw)*dist; move[2] = 0; // save program state, because SV_movestep may call other progs oldf = prog->xfunction; oldself = prog->globals.server->self; PRVM_G_FLOAT(OFS_RETURN) = SV_movestep(ent, move, true, false, settrace); // restore program state prog->xfunction = oldf; prog->globals.server->self = oldself; } /* =============== VM_SV_droptofloor void() droptofloor =============== */ static void VM_SV_droptofloor (void) { prvm_edict_t *ent; vec3_t end; trace_t trace; VM_SAFEPARMCOUNTRANGE(0, 2, VM_SV_droptofloor); // allow 2 parameters because the id1 defs.qc had an incorrect prototype // assume failure if it returns early PRVM_G_FLOAT(OFS_RETURN) = 0; ent = PRVM_PROG_TO_EDICT(prog->globals.server->self); if (ent == prog->edicts) { VM_Warning("droptofloor: can not modify world entity\n"); return; } if (ent->priv.server->free) { VM_Warning("droptofloor: can not modify free entity\n"); return; } VectorCopy (ent->fields.server->origin, end); end[2] -= 256; if (sv_gameplayfix_droptofloorstartsolid_nudgetocorrect.integer) SV_UnstickEntity(ent); trace = SV_Move (ent->fields.server->origin, ent->fields.server->mins, ent->fields.server->maxs, end, MOVE_NORMAL, ent, SV_GenericHitSuperContentsMask(ent)); if (trace.startsolid && sv_gameplayfix_droptofloorstartsolid.integer) { vec3_t offset, org; VectorSet(offset, 0.5f * (ent->fields.server->mins[0] + ent->fields.server->maxs[0]), 0.5f * (ent->fields.server->mins[1] + ent->fields.server->maxs[1]), ent->fields.server->mins[2]); VectorAdd(ent->fields.server->origin, offset, org); trace = SV_Move (org, vec3_origin, vec3_origin, end, MOVE_NORMAL, ent, SV_GenericHitSuperContentsMask(ent)); VectorSubtract(trace.endpos, offset, trace.endpos); if (trace.startsolid) { Con_DPrintf("droptofloor at %f %f %f - COULD NOT FIX BADLY PLACED ENTITY\n", ent->fields.server->origin[0], ent->fields.server->origin[1], ent->fields.server->origin[2]); SV_UnstickEntity(ent); SV_LinkEdict (ent, false); ent->fields.server->flags = (int)ent->fields.server->flags | FL_ONGROUND; ent->fields.server->groundentity = 0; PRVM_G_FLOAT(OFS_RETURN) = 1; } else if (trace.fraction < 1) { Con_DPrintf("droptofloor at %f %f %f - FIXED BADLY PLACED ENTITY\n", ent->fields.server->origin[0], ent->fields.server->origin[1], ent->fields.server->origin[2]); VectorCopy (trace.endpos, ent->fields.server->origin); SV_UnstickEntity(ent); SV_LinkEdict (ent, false); ent->fields.server->flags = (int)ent->fields.server->flags | FL_ONGROUND; ent->fields.server->groundentity = PRVM_EDICT_TO_PROG(trace.ent); PRVM_G_FLOAT(OFS_RETURN) = 1; // if support is destroyed, keep suspended (gross hack for floating items in various maps) ent->priv.server->suspendedinairflag = true; } } else { if (trace.fraction != 1) { if (trace.fraction < 1) VectorCopy (trace.endpos, ent->fields.server->origin); SV_LinkEdict (ent, false); ent->fields.server->flags = (int)ent->fields.server->flags | FL_ONGROUND; ent->fields.server->groundentity = PRVM_EDICT_TO_PROG(trace.ent); PRVM_G_FLOAT(OFS_RETURN) = 1; // if support is destroyed, keep suspended (gross hack for floating items in various maps) ent->priv.server->suspendedinairflag = true; } } } /* =============== VM_SV_lightstyle void(float style, string value) lightstyle =============== */ static void VM_SV_lightstyle (void) { int style; const char *val; client_t *client; int j; VM_SAFEPARMCOUNT(2, VM_SV_lightstyle); style = (int)PRVM_G_FLOAT(OFS_PARM0); val = PRVM_G_STRING(OFS_PARM1); if( (unsigned) style >= MAX_LIGHTSTYLES ) { PRVM_ERROR( "PF_lightstyle: style: %i >= 64", style ); } // change the string in sv strlcpy(sv.lightstyles[style], val, sizeof(sv.lightstyles[style])); // send message to all clients on this server if (sv.state != ss_active) return; for (j = 0, client = svs.clients;j < svs.maxclients;j++, client++) { if (client->active && client->netconnection) { MSG_WriteChar (&client->netconnection->message, svc_lightstyle); MSG_WriteChar (&client->netconnection->message,style); MSG_WriteString (&client->netconnection->message, val); } } } /* ============= VM_SV_checkbottom ============= */ static void VM_SV_checkbottom (void) { VM_SAFEPARMCOUNT(1, VM_SV_checkbottom); PRVM_G_FLOAT(OFS_RETURN) = SV_CheckBottom (PRVM_G_EDICT(OFS_PARM0)); } /* ============= VM_SV_pointcontents ============= */ static void VM_SV_pointcontents (void) { VM_SAFEPARMCOUNT(1, VM_SV_pointcontents); PRVM_G_FLOAT(OFS_RETURN) = Mod_Q1BSP_NativeContentsFromSuperContents(NULL, SV_PointSuperContents(PRVM_G_VECTOR(OFS_PARM0))); } /* ============= VM_SV_aim Pick a vector for the player to shoot along vector aim(entity, missilespeed) ============= */ static void VM_SV_aim (void) { prvm_edict_t *ent, *check, *bestent; vec3_t start, dir, end, bestdir; int i, j; trace_t tr; float dist, bestdist; float speed; VM_SAFEPARMCOUNT(2, VM_SV_aim); // assume failure if it returns early VectorCopy(prog->globals.server->v_forward, PRVM_G_VECTOR(OFS_RETURN)); // if sv_aim is so high it can't possibly accept anything, skip out early if (sv_aim.value >= 1) return; ent = PRVM_G_EDICT(OFS_PARM0); if (ent == prog->edicts) { VM_Warning("aim: can not use world entity\n"); return; } if (ent->priv.server->free) { VM_Warning("aim: can not use free entity\n"); return; } speed = PRVM_G_FLOAT(OFS_PARM1); VectorCopy (ent->fields.server->origin, start); start[2] += 20; // try sending a trace straight VectorCopy (prog->globals.server->v_forward, dir); VectorMA (start, 2048, dir, end); tr = SV_Move (start, vec3_origin, vec3_origin, end, MOVE_NORMAL, ent, SUPERCONTENTS_SOLID | SUPERCONTENTS_BODY); if (tr.ent && ((prvm_edict_t *)tr.ent)->fields.server->takedamage == DAMAGE_AIM && (!teamplay.integer || ent->fields.server->team <=0 || ent->fields.server->team != ((prvm_edict_t *)tr.ent)->fields.server->team) ) { VectorCopy (prog->globals.server->v_forward, PRVM_G_VECTOR(OFS_RETURN)); return; } // try all possible entities VectorCopy (dir, bestdir); bestdist = sv_aim.value; bestent = NULL; check = PRVM_NEXT_EDICT(prog->edicts); for (i=1 ; inum_edicts ; i++, check = PRVM_NEXT_EDICT(check) ) { prog->xfunction->builtinsprofile++; if (check->fields.server->takedamage != DAMAGE_AIM) continue; if (check == ent) continue; if (teamplay.integer && ent->fields.server->team > 0 && ent->fields.server->team == check->fields.server->team) continue; // don't aim at teammate for (j=0 ; j<3 ; j++) end[j] = check->fields.server->origin[j] + 0.5*(check->fields.server->mins[j] + check->fields.server->maxs[j]); VectorSubtract (end, start, dir); VectorNormalize (dir); dist = DotProduct (dir, prog->globals.server->v_forward); if (dist < bestdist) continue; // to far to turn tr = SV_Move (start, vec3_origin, vec3_origin, end, MOVE_NORMAL, ent, SUPERCONTENTS_SOLID | SUPERCONTENTS_BODY); if (tr.ent == check) { // can shoot at this one bestdist = dist; bestent = check; } } if (bestent) { VectorSubtract (bestent->fields.server->origin, ent->fields.server->origin, dir); dist = DotProduct (dir, prog->globals.server->v_forward); VectorScale (prog->globals.server->v_forward, dist, end); end[2] = dir[2]; VectorNormalize (end); VectorCopy (end, PRVM_G_VECTOR(OFS_RETURN)); } else { VectorCopy (bestdir, PRVM_G_VECTOR(OFS_RETURN)); } } /* =============================================================================== MESSAGE WRITING =============================================================================== */ #define MSG_BROADCAST 0 // unreliable to all #define MSG_ONE 1 // reliable to one (msg_entity) #define MSG_ALL 2 // reliable to all #define MSG_INIT 3 // write to the init string #define MSG_ENTITY 5 sizebuf_t *WriteDest (void) { int entnum; int dest; prvm_edict_t *ent; dest = (int)PRVM_G_FLOAT(OFS_PARM0); switch (dest) { case MSG_BROADCAST: return &sv.datagram; case MSG_ONE: ent = PRVM_PROG_TO_EDICT(prog->globals.server->msg_entity); entnum = PRVM_NUM_FOR_EDICT(ent); if (entnum < 1 || entnum > svs.maxclients || !svs.clients[entnum-1].active || !svs.clients[entnum-1].netconnection) { VM_Warning ("WriteDest: tried to write to non-client\n"); return &sv.reliable_datagram; } else return &svs.clients[entnum-1].netconnection->message; default: VM_Warning ("WriteDest: bad destination\n"); case MSG_ALL: return &sv.reliable_datagram; case MSG_INIT: return &sv.signon; case MSG_ENTITY: return sv.writeentitiestoclient_msg; } return NULL; } static void VM_SV_WriteByte (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteByte); MSG_WriteByte (WriteDest(), (int)PRVM_G_FLOAT(OFS_PARM1)); } static void VM_SV_WriteChar (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteChar); MSG_WriteChar (WriteDest(), (int)PRVM_G_FLOAT(OFS_PARM1)); } static void VM_SV_WriteShort (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteShort); MSG_WriteShort (WriteDest(), (int)PRVM_G_FLOAT(OFS_PARM1)); } static void VM_SV_WriteLong (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteLong); MSG_WriteLong (WriteDest(), (int)PRVM_G_FLOAT(OFS_PARM1)); } static void VM_SV_WriteAngle (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteAngle); MSG_WriteAngle (WriteDest(), PRVM_G_FLOAT(OFS_PARM1), sv.protocol); } static void VM_SV_WriteCoord (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteCoord); MSG_WriteCoord (WriteDest(), PRVM_G_FLOAT(OFS_PARM1), sv.protocol); } static void VM_SV_WriteString (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteString); MSG_WriteString (WriteDest(), PRVM_G_STRING(OFS_PARM1)); } static void VM_SV_WriteUnterminatedString (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteUnterminatedString); MSG_WriteUnterminatedString (WriteDest(), PRVM_G_STRING(OFS_PARM1)); } static void VM_SV_WriteEntity (void) { VM_SAFEPARMCOUNT(2, VM_SV_WriteEntity); MSG_WriteShort (WriteDest(), PRVM_G_EDICTNUM(OFS_PARM1)); } // writes a picture as at most size bytes of data // message: // IMGNAME \0 SIZE(short) IMGDATA // if failed to read/compress: // IMGNAME \0 \0 \0 //#501 void(float dest, string name, float maxsize) WritePicture (DP_SV_WRITEPICTURE)) static void VM_SV_WritePicture (void) { const char *imgname; void *buf; size_t size; VM_SAFEPARMCOUNT(3, VM_SV_WritePicture); imgname = PRVM_G_STRING(OFS_PARM1); size = (int) PRVM_G_FLOAT(OFS_PARM2); if(size > 65535) size = 65535; MSG_WriteString(WriteDest(), imgname); if(Image_Compress(imgname, size, &buf, &size)) { // actual picture MSG_WriteShort(WriteDest(), size); SZ_Write(WriteDest(), (unsigned char *) buf, size); } else { // placeholder MSG_WriteShort(WriteDest(), 0); } } ////////////////////////////////////////////////////////// static void VM_SV_makestatic (void) { prvm_edict_t *ent; int i, large; // allow 0 parameters due to an id1 qc bug in which this function is used // with no parameters (but directly after setmodel with self in OFS_PARM0) VM_SAFEPARMCOUNTRANGE(0, 1, VM_SV_makestatic); if (prog->argc >= 1) ent = PRVM_G_EDICT(OFS_PARM0); else ent = PRVM_PROG_TO_EDICT(prog->globals.server->self); if (ent == prog->edicts) { VM_Warning("makestatic: can not modify world entity\n"); return; } if (ent->priv.server->free) { VM_Warning("makestatic: can not modify free entity\n"); return; } large = false; if (ent->fields.server->modelindex >= 256 || ent->fields.server->frame >= 256) large = true; if (large) { MSG_WriteByte (&sv.signon,svc_spawnstatic2); MSG_WriteShort (&sv.signon, (int)ent->fields.server->modelindex); MSG_WriteShort (&sv.signon, (int)ent->fields.server->frame); } else if (sv.protocol == PROTOCOL_NEHAHRABJP || sv.protocol == PROTOCOL_NEHAHRABJP2 || sv.protocol == PROTOCOL_NEHAHRABJP3) { MSG_WriteByte (&sv.signon,svc_spawnstatic); MSG_WriteShort (&sv.signon, (int)ent->fields.server->modelindex); MSG_WriteByte (&sv.signon, (int)ent->fields.server->frame); } else { MSG_WriteByte (&sv.signon,svc_spawnstatic); MSG_WriteByte (&sv.signon, (int)ent->fields.server->modelindex); MSG_WriteByte (&sv.signon, (int)ent->fields.server->frame); } MSG_WriteByte (&sv.signon, (int)ent->fields.server->colormap); MSG_WriteByte (&sv.signon, (int)ent->fields.server->skin); for (i=0 ; i<3 ; i++) { MSG_WriteCoord(&sv.signon, ent->fields.server->origin[i], sv.protocol); MSG_WriteAngle(&sv.signon, ent->fields.server->angles[i], sv.protocol); } // throw the entity away now PRVM_ED_Free (ent); } //============================================================================= /* ============== VM_SV_setspawnparms ============== */ static void VM_SV_setspawnparms (void) { prvm_edict_t *ent; int i; client_t *client; VM_SAFEPARMCOUNT(1, VM_SV_setspawnparms); ent = PRVM_G_EDICT(OFS_PARM0); i = PRVM_NUM_FOR_EDICT(ent); if (i < 1 || i > svs.maxclients || !svs.clients[i-1].active) { Con_Print("tried to setspawnparms on a non-client\n"); return; } // copy spawn parms out of the client_t client = svs.clients + i-1; for (i=0 ; i< NUM_SPAWN_PARMS ; i++) (&prog->globals.server->parm1)[i] = client->spawn_parms[i]; } /* ================= VM_SV_getlight Returns a color vector indicating the lighting at the requested point. (Internal Operation note: actually measures the light beneath the point, just like the model lighting on the client) getlight(vector) ================= */ static void VM_SV_getlight (void) { vec3_t ambientcolor, diffusecolor, diffusenormal; vec_t *p; VM_SAFEPARMCOUNT(1, VM_SV_getlight); p = PRVM_G_VECTOR(OFS_PARM0); VectorClear(ambientcolor); VectorClear(diffusecolor); VectorClear(diffusenormal); if (sv.worldmodel && sv.worldmodel->brush.LightPoint) sv.worldmodel->brush.LightPoint(sv.worldmodel, p, ambientcolor, diffusecolor, diffusenormal); VectorMA(ambientcolor, 0.5, diffusecolor, PRVM_G_VECTOR(OFS_RETURN)); } typedef struct { unsigned char type; // 1/2/8 or other value if isn't used int fieldoffset; }customstat_t; static customstat_t *vm_customstats = NULL; //[515]: it starts from 0, not 32 static int vm_customstats_last; void VM_CustomStats_Clear (void) { if(vm_customstats) { Z_Free(vm_customstats); vm_customstats = NULL; vm_customstats_last = -1; } } void VM_SV_UpdateCustomStats (client_t *client, prvm_edict_t *ent, sizebuf_t *msg, int *stats) { int i; char s[17]; if(!vm_customstats) return; for(i=0; i= (MAX_CL_STATS-32)) { VM_Warning("PF_SV_AddStat: index >= MAX_CL_STATS\n"); return; } if(i > (MAX_CL_STATS-32-4) && type == 1) { VM_Warning("PF_SV_AddStat: index > (MAX_CL_STATS-4) with string\n"); return; } vm_customstats[i].type = type; vm_customstats[i].fieldoffset = off; if(vm_customstats_last < i) vm_customstats_last = i; } /* ================= VM_SV_copyentity copies data from one entity to another copyentity(src, dst) ================= */ static void VM_SV_copyentity (void) { prvm_edict_t *in, *out; VM_SAFEPARMCOUNT(2, VM_SV_copyentity); in = PRVM_G_EDICT(OFS_PARM0); if (in == prog->edicts) { VM_Warning("copyentity: can not read world entity\n"); return; } if (in->priv.server->free) { VM_Warning("copyentity: can not read free entity\n"); return; } out = PRVM_G_EDICT(OFS_PARM1); if (out == prog->edicts) { VM_Warning("copyentity: can not modify world entity\n"); return; } if (out->priv.server->free) { VM_Warning("copyentity: can not modify free entity\n"); return; } memcpy(out->fields.vp, in->fields.vp, prog->progs->entityfields * 4); SV_LinkEdict(out, false); } /* ================= VM_SV_setcolor sets the color of a client and broadcasts the update to all connected clients setcolor(clientent, value) ================= */ static void VM_SV_setcolor (void) { client_t *client; int entnum, i; prvm_eval_t *val; VM_SAFEPARMCOUNT(2, VM_SV_setcolor); entnum = PRVM_G_EDICTNUM(OFS_PARM0); i = (int)PRVM_G_FLOAT(OFS_PARM1); if (entnum < 1 || entnum > svs.maxclients || !svs.clients[entnum-1].active) { Con_Print("tried to setcolor a non-client\n"); return; } client = svs.clients + entnum-1; if (client->edict) { if ((val = PRVM_EDICTFIELDVALUE(client->edict, prog->fieldoffsets.clientcolors))) val->_float = i; client->edict->fields.server->team = (i & 15) + 1; } client->colors = i; if (client->old_colors != client->colors) { client->old_colors = client->colors; // send notification to all clients MSG_WriteByte (&sv.reliable_datagram, svc_updatecolors); MSG_WriteByte (&sv.reliable_datagram, client - svs.clients); MSG_WriteByte (&sv.reliable_datagram, client->colors); } } /* ================= VM_SV_effect effect(origin, modelname, startframe, framecount, framerate) ================= */ static void VM_SV_effect (void) { int i; const char *s; VM_SAFEPARMCOUNT(5, VM_SV_effect); s = PRVM_G_STRING(OFS_PARM1); if (!s[0]) { VM_Warning("effect: no model specified\n"); return; } i = SV_ModelIndex(s, 1); if (!i) { VM_Warning("effect: model not precached\n"); return; } if (PRVM_G_FLOAT(OFS_PARM3) < 1) { VM_Warning("effect: framecount < 1\n"); return; } if (PRVM_G_FLOAT(OFS_PARM4) < 1) { VM_Warning("effect: framerate < 1\n"); return; } SV_StartEffect(PRVM_G_VECTOR(OFS_PARM0), i, (int)PRVM_G_FLOAT(OFS_PARM2), (int)PRVM_G_FLOAT(OFS_PARM3), (int)PRVM_G_FLOAT(OFS_PARM4)); } static void VM_SV_te_blood (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_blood); if (PRVM_G_FLOAT(OFS_PARM2) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_BLOOD); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // velocity MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[0], 127)); MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[1], 127)); MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[2], 127)); // count MSG_WriteByte(&sv.datagram, bound(0, (int) PRVM_G_FLOAT(OFS_PARM2), 255)); SV_FlushBroadcastMessages(); } static void VM_SV_te_bloodshower (void) { VM_SAFEPARMCOUNT(4, VM_SV_te_bloodshower); if (PRVM_G_FLOAT(OFS_PARM3) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_BLOODSHOWER); // min MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // max MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // speed MSG_WriteCoord(&sv.datagram, PRVM_G_FLOAT(OFS_PARM2), sv.protocol); // count MSG_WriteShort(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM3), 65535)); SV_FlushBroadcastMessages(); } static void VM_SV_te_explosionrgb (void) { VM_SAFEPARMCOUNT(2, VM_SV_te_explosionrgb); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_EXPLOSIONRGB); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // color MSG_WriteByte(&sv.datagram, bound(0, (int) (PRVM_G_VECTOR(OFS_PARM1)[0] * 255), 255)); MSG_WriteByte(&sv.datagram, bound(0, (int) (PRVM_G_VECTOR(OFS_PARM1)[1] * 255), 255)); MSG_WriteByte(&sv.datagram, bound(0, (int) (PRVM_G_VECTOR(OFS_PARM1)[2] * 255), 255)); SV_FlushBroadcastMessages(); } static void VM_SV_te_particlecube (void) { VM_SAFEPARMCOUNT(7, VM_SV_te_particlecube); if (PRVM_G_FLOAT(OFS_PARM3) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_PARTICLECUBE); // min MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // max MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // velocity MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); // count MSG_WriteShort(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM3), 65535)); // color MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM4)); // gravity true/false MSG_WriteByte(&sv.datagram, ((int) PRVM_G_FLOAT(OFS_PARM5)) != 0); // randomvel MSG_WriteCoord(&sv.datagram, PRVM_G_FLOAT(OFS_PARM6), sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_particlerain (void) { VM_SAFEPARMCOUNT(5, VM_SV_te_particlerain); if (PRVM_G_FLOAT(OFS_PARM3) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_PARTICLERAIN); // min MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // max MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // velocity MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); // count MSG_WriteShort(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM3), 65535)); // color MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM4)); SV_FlushBroadcastMessages(); } static void VM_SV_te_particlesnow (void) { VM_SAFEPARMCOUNT(5, VM_SV_te_particlesnow); if (PRVM_G_FLOAT(OFS_PARM3) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_PARTICLESNOW); // min MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // max MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // velocity MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); // count MSG_WriteShort(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM3), 65535)); // color MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM4)); SV_FlushBroadcastMessages(); } static void VM_SV_te_spark (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_spark); if (PRVM_G_FLOAT(OFS_PARM2) < 1) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SPARK); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // velocity MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[0], 127)); MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[1], 127)); MSG_WriteChar(&sv.datagram, bound(-128, (int) PRVM_G_VECTOR(OFS_PARM1)[2], 127)); // count MSG_WriteByte(&sv.datagram, bound(0, (int) PRVM_G_FLOAT(OFS_PARM2), 255)); SV_FlushBroadcastMessages(); } static void VM_SV_te_gunshotquad (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_gunshotquad); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_GUNSHOTQUAD); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_spikequad (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_spikequad); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SPIKEQUAD); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_superspikequad (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_superspikequad); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SUPERSPIKEQUAD); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_explosionquad (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_explosionquad); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_EXPLOSIONQUAD); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_smallflash (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_smallflash); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SMALLFLASH); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_customflash (void) { VM_SAFEPARMCOUNT(4, VM_SV_te_customflash); if (PRVM_G_FLOAT(OFS_PARM1) < 8 || PRVM_G_FLOAT(OFS_PARM2) < (1.0 / 256.0)) return; MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_CUSTOMFLASH); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // radius MSG_WriteByte(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM1) / 8 - 1, 255)); // lifetime MSG_WriteByte(&sv.datagram, (int)bound(0, PRVM_G_FLOAT(OFS_PARM2) * 256 - 1, 255)); // color MSG_WriteByte(&sv.datagram, (int)bound(0, PRVM_G_VECTOR(OFS_PARM3)[0] * 255, 255)); MSG_WriteByte(&sv.datagram, (int)bound(0, PRVM_G_VECTOR(OFS_PARM3)[1] * 255, 255)); MSG_WriteByte(&sv.datagram, (int)bound(0, PRVM_G_VECTOR(OFS_PARM3)[2] * 255, 255)); SV_FlushBroadcastMessages(); } static void VM_SV_te_gunshot (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_gunshot); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_GUNSHOT); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_spike (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_spike); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SPIKE); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_superspike (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_superspike); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_SUPERSPIKE); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_explosion (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_explosion); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_EXPLOSION); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_tarexplosion (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_tarexplosion); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_TAREXPLOSION); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_wizspike (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_wizspike); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_WIZSPIKE); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_knightspike (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_knightspike); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_KNIGHTSPIKE); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_lavasplash (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_lavasplash); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_LAVASPLASH); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_teleport (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_teleport); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_TELEPORT); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_explosion2 (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_explosion2); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_EXPLOSION2); // origin MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // color MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM1)); MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM2)); SV_FlushBroadcastMessages(); } static void VM_SV_te_lightning1 (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_lightning1); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_LIGHTNING1); // owner entity MSG_WriteShort(&sv.datagram, PRVM_G_EDICTNUM(OFS_PARM0)); // start MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // end MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_lightning2 (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_lightning2); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_LIGHTNING2); // owner entity MSG_WriteShort(&sv.datagram, PRVM_G_EDICTNUM(OFS_PARM0)); // start MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // end MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_lightning3 (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_lightning3); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_LIGHTNING3); // owner entity MSG_WriteShort(&sv.datagram, PRVM_G_EDICTNUM(OFS_PARM0)); // start MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // end MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_beam (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_beam); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_BEAM); // owner entity MSG_WriteShort(&sv.datagram, PRVM_G_EDICTNUM(OFS_PARM0)); // start MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // end MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_plasmaburn (void) { VM_SAFEPARMCOUNT(1, VM_SV_te_plasmaburn); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_PLASMABURN); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); SV_FlushBroadcastMessages(); } static void VM_SV_te_flamejet (void) { VM_SAFEPARMCOUNT(3, VM_SV_te_flamejet); MSG_WriteByte(&sv.datagram, svc_temp_entity); MSG_WriteByte(&sv.datagram, TE_FLAMEJET); // org MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM0)[2], sv.protocol); // vel MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[0], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[1], sv.protocol); MSG_WriteCoord(&sv.datagram, PRVM_G_VECTOR(OFS_PARM1)[2], sv.protocol); // count MSG_WriteByte(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM2)); SV_FlushBroadcastMessages(); } void clippointtosurface(dp_model_t *model, msurface_t *surface, vec3_t p, vec3_t out) { int i, j, k; float *v[3], facenormal[3], edgenormal[3], sidenormal[3], temp[3], offsetdist, dist, bestdist; const int *e; bestdist = 1000000000; VectorCopy(p, out); for (i = 0, e = (model->surfmesh.data_element3i + 3 * surface->num_firsttriangle);i < surface->num_triangles;i++, e += 3) { // clip original point to each triangle of the surface and find the // triangle that is closest v[0] = model->surfmesh.data_vertex3f + e[0] * 3; v[1] = model->surfmesh.data_vertex3f + e[1] * 3; v[2] = model->surfmesh.data_vertex3f + e[2] * 3; TriangleNormal(v[0], v[1], v[2], facenormal); VectorNormalize(facenormal); offsetdist = DotProduct(v[0], facenormal) - DotProduct(p, facenormal); VectorMA(p, offsetdist, facenormal, temp); for (j = 0, k = 2;j < 3;k = j, j++) { VectorSubtract(v[k], v[j], edgenormal); CrossProduct(edgenormal, facenormal, sidenormal); VectorNormalize(sidenormal); offsetdist = DotProduct(v[k], sidenormal) - DotProduct(temp, sidenormal); if (offsetdist < 0) VectorMA(temp, offsetdist, sidenormal, temp); } dist = VectorDistance2(temp, p); if (bestdist > dist) { bestdist = dist; VectorCopy(temp, out); } } } static dp_model_t *getmodel(prvm_edict_t *ed) { int modelindex; if (!ed || ed->priv.server->free) return NULL; modelindex = (int)ed->fields.server->modelindex; if (modelindex < 1 || modelindex >= MAX_MODELS) return NULL; return sv.models[modelindex]; } static msurface_t *getsurface(dp_model_t *model, int surfacenum) { if (surfacenum < 0 || surfacenum >= model->nummodelsurfaces) return NULL; return model->data_surfaces + surfacenum + model->firstmodelsurface; } //PF_getsurfacenumpoints, // #434 float(entity e, float s) getsurfacenumpoints = #434; static void VM_SV_getsurfacenumpoints(void) { dp_model_t *model; msurface_t *surface; VM_SAFEPARMCOUNT(2, VM_SV_getsurfacenumpoints); // return 0 if no such surface if (!(model = getmodel(PRVM_G_EDICT(OFS_PARM0))) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) { PRVM_G_FLOAT(OFS_RETURN) = 0; return; } // note: this (incorrectly) assumes it is a simple polygon PRVM_G_FLOAT(OFS_RETURN) = surface->num_vertices; } //PF_getsurfacepoint, // #435 vector(entity e, float s, float n) getsurfacepoint = #435; static void VM_SV_getsurfacepoint(void) { prvm_edict_t *ed; dp_model_t *model; msurface_t *surface; int pointnum; VM_SAFEPARMCOUNT(3, VM_SV_getsurfacepoint); VectorClear(PRVM_G_VECTOR(OFS_RETURN)); ed = PRVM_G_EDICT(OFS_PARM0); if (!(model = getmodel(ed)) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) return; // note: this (incorrectly) assumes it is a simple polygon pointnum = (int)PRVM_G_FLOAT(OFS_PARM2); if (pointnum < 0 || pointnum >= surface->num_vertices) return; // FIXME: implement rotation/scaling VectorAdd(&(model->surfmesh.data_vertex3f + 3 * surface->num_firstvertex)[pointnum * 3], ed->fields.server->origin, PRVM_G_VECTOR(OFS_RETURN)); } //PF_getsurfacepointattribute, // #486 vector(entity e, float s, float n, float a) getsurfacepointattribute = #486; // float SPA_POSITION = 0; // float SPA_S_AXIS = 1; // float SPA_T_AXIS = 2; // float SPA_R_AXIS = 3; // same as SPA_NORMAL // float SPA_TEXCOORDS0 = 4; // float SPA_LIGHTMAP0_TEXCOORDS = 5; // float SPA_LIGHTMAP0_COLOR = 6; static void VM_SV_getsurfacepointattribute(void) { prvm_edict_t *ed; dp_model_t *model; msurface_t *surface; int pointnum; int attributetype; VM_SAFEPARMCOUNT(4, VM_SV_getsurfacepoint); VectorClear(PRVM_G_VECTOR(OFS_RETURN)); ed = PRVM_G_EDICT(OFS_PARM0); if (!(model = getmodel(ed)) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) return; // note: this (incorrectly) assumes it is a simple polygon pointnum = (int)PRVM_G_FLOAT(OFS_PARM2); if (pointnum < 0 || pointnum >= surface->num_vertices) return; // FIXME: implement rotation/scaling attributetype = (int) PRVM_G_FLOAT(OFS_PARM3); switch( attributetype ) { // float SPA_POSITION = 0; case 0: VectorAdd(&(model->surfmesh.data_vertex3f + 3 * surface->num_firstvertex)[pointnum * 3], ed->fields.server->origin, PRVM_G_VECTOR(OFS_RETURN)); break; // float SPA_S_AXIS = 1; case 1: VectorCopy(&(model->surfmesh.data_svector3f + 3 * surface->num_firstvertex)[pointnum * 3], PRVM_G_VECTOR(OFS_RETURN)); break; // float SPA_T_AXIS = 2; case 2: VectorCopy(&(model->surfmesh.data_tvector3f + 3 * surface->num_firstvertex)[pointnum * 3], PRVM_G_VECTOR(OFS_RETURN)); break; // float SPA_R_AXIS = 3; // same as SPA_NORMAL case 3: VectorCopy(&(model->surfmesh.data_normal3f + 3 * surface->num_firstvertex)[pointnum * 3], PRVM_G_VECTOR(OFS_RETURN)); break; // float SPA_TEXCOORDS0 = 4; case 4: { float *ret = PRVM_G_VECTOR(OFS_RETURN); float *texcoord = &(model->surfmesh.data_texcoordtexture2f + 2 * surface->num_firstvertex)[pointnum * 2]; ret[0] = texcoord[0]; ret[1] = texcoord[1]; ret[2] = 0.0f; break; } // float SPA_LIGHTMAP0_TEXCOORDS = 5; case 5: { float *ret = PRVM_G_VECTOR(OFS_RETURN); float *texcoord = &(model->surfmesh.data_texcoordlightmap2f + 2 * surface->num_firstvertex)[pointnum * 2]; ret[0] = texcoord[0]; ret[1] = texcoord[1]; ret[2] = 0.0f; break; } // float SPA_LIGHTMAP0_COLOR = 6; case 6: // ignore alpha for now.. VectorCopy( &(model->surfmesh.data_lightmapcolor4f + 4 * surface->num_firstvertex)[pointnum * 4], PRVM_G_VECTOR(OFS_RETURN)); break; default: VectorSet( PRVM_G_VECTOR(OFS_RETURN), 0.0f, 0.0f, 0.0f ); break; } } //PF_getsurfacenormal, // #436 vector(entity e, float s) getsurfacenormal = #436; static void VM_SV_getsurfacenormal(void) { dp_model_t *model; msurface_t *surface; vec3_t normal; VM_SAFEPARMCOUNT(2, VM_SV_getsurfacenormal); VectorClear(PRVM_G_VECTOR(OFS_RETURN)); if (!(model = getmodel(PRVM_G_EDICT(OFS_PARM0))) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) return; // FIXME: implement rotation/scaling // note: this (incorrectly) assumes it is a simple polygon // note: this only returns the first triangle, so it doesn't work very // well for curved surfaces or arbitrary meshes TriangleNormal((model->surfmesh.data_vertex3f + 3 * surface->num_firstvertex), (model->surfmesh.data_vertex3f + 3 * surface->num_firstvertex) + 3, (model->surfmesh.data_vertex3f + 3 * surface->num_firstvertex) + 6, normal); VectorNormalize(normal); VectorCopy(normal, PRVM_G_VECTOR(OFS_RETURN)); } //PF_getsurfacetexture, // #437 string(entity e, float s) getsurfacetexture = #437; static void VM_SV_getsurfacetexture(void) { dp_model_t *model; msurface_t *surface; VM_SAFEPARMCOUNT(2, VM_SV_getsurfacetexture); PRVM_G_INT(OFS_RETURN) = OFS_NULL; if (!(model = getmodel(PRVM_G_EDICT(OFS_PARM0))) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) return; PRVM_G_INT(OFS_RETURN) = PRVM_SetTempString(surface->texture->name); } //PF_getsurfacenearpoint, // #438 float(entity e, vector p) getsurfacenearpoint = #438; static void VM_SV_getsurfacenearpoint(void) { int surfacenum, best; vec3_t clipped, p; vec_t dist, bestdist; prvm_edict_t *ed; dp_model_t *model; msurface_t *surface; vec_t *point; VM_SAFEPARMCOUNT(2, VM_SV_getsurfacenearpoint); PRVM_G_FLOAT(OFS_RETURN) = -1; ed = PRVM_G_EDICT(OFS_PARM0); point = PRVM_G_VECTOR(OFS_PARM1); if (!ed || ed->priv.server->free) return; model = getmodel(ed); if (!model || !model->num_surfaces) return; // FIXME: implement rotation/scaling VectorSubtract(point, ed->fields.server->origin, p); best = -1; bestdist = 1000000000; for (surfacenum = 0;surfacenum < model->nummodelsurfaces;surfacenum++) { surface = model->data_surfaces + surfacenum + model->firstmodelsurface; // first see if the nearest point on the surface's box is closer than the previous match clipped[0] = bound(surface->mins[0], p[0], surface->maxs[0]) - p[0]; clipped[1] = bound(surface->mins[1], p[1], surface->maxs[1]) - p[1]; clipped[2] = bound(surface->mins[2], p[2], surface->maxs[2]) - p[2]; dist = VectorLength2(clipped); if (dist < bestdist) { // it is, check the nearest point on the actual geometry clippointtosurface(model, surface, p, clipped); VectorSubtract(clipped, p, clipped); dist += VectorLength2(clipped); if (dist < bestdist) { // that's closer too, store it as the best match best = surfacenum; bestdist = dist; } } } PRVM_G_FLOAT(OFS_RETURN) = best; } //PF_getsurfaceclippedpoint, // #439 vector(entity e, float s, vector p) getsurfaceclippedpoint = #439; static void VM_SV_getsurfaceclippedpoint(void) { prvm_edict_t *ed; dp_model_t *model; msurface_t *surface; vec3_t p, out; VM_SAFEPARMCOUNT(3, VM_SV_te_getsurfaceclippedpoint); VectorClear(PRVM_G_VECTOR(OFS_RETURN)); ed = PRVM_G_EDICT(OFS_PARM0); if (!(model = getmodel(ed)) || !(surface = getsurface(model, (int)PRVM_G_FLOAT(OFS_PARM1)))) return; // FIXME: implement rotation/scaling VectorSubtract(PRVM_G_VECTOR(OFS_PARM2), ed->fields.server->origin, p); clippointtosurface(model, surface, p, out); // FIXME: implement rotation/scaling VectorAdd(out, ed->fields.server->origin, PRVM_G_VECTOR(OFS_RETURN)); } //void(entity e, string s) clientcommand = #440; // executes a command string as if it came from the specified client //this function originally written by KrimZon, made shorter by LordHavoc static void VM_SV_clientcommand (void) { client_t *temp_client; int i; VM_SAFEPARMCOUNT(2, VM_SV_clientcommand); //find client for this entity i = (PRVM_NUM_FOR_EDICT(PRVM_G_EDICT(OFS_PARM0)) - 1); if (i < 0 || i >= svs.maxclients || !svs.clients[i].active) { Con_Print("PF_clientcommand: entity is not a client\n"); return; } temp_client = host_client; host_client = svs.clients + i; Cmd_ExecuteString (PRVM_G_STRING(OFS_PARM1), src_client); host_client = temp_client; } //void(entity e, entity tagentity, string tagname) setattachment = #443; // attachs e to a tag on tagentity (note: use "" to attach to entity origin/angles instead of a tag) static void VM_SV_setattachment (void) { prvm_edict_t *e = PRVM_G_EDICT(OFS_PARM0); prvm_edict_t *tagentity = PRVM_G_EDICT(OFS_PARM1); const char *tagname = PRVM_G_STRING(OFS_PARM2); prvm_eval_t *v; int modelindex; dp_model_t *model; VM_SAFEPARMCOUNT(3, VM_SV_setattachment); if (e == prog->edicts) { VM_Warning("setattachment: can not modify world entity\n"); return; } if (e->priv.server->free) { VM_Warning("setattachment: can not modify free entity\n"); return; } if (tagentity == NULL) tagentity = prog->edicts; v = PRVM_EDICTFIELDVALUE(e, prog->fieldoffsets.tag_entity); if (v) v->edict = PRVM_EDICT_TO_PROG(tagentity); v = PRVM_EDICTFIELDVALUE(e, prog->fieldoffsets.tag_index); if (v) v->_float = 0; if (tagentity != NULL && tagentity != prog->edicts && tagname && tagname[0]) { modelindex = (int)tagentity->fields.server->modelindex; if (modelindex >= 0 && modelindex < MAX_MODELS && (model = sv.models[modelindex])) { v->_float = Mod_Alias_GetTagIndexForName(model, (int)tagentity->fields.server->skin, tagname); if (v->_float == 0) Con_DPrintf("setattachment(edict %i, edict %i, string \"%s\"): tried to find tag named \"%s\" on entity %i (model \"%s\") but could not find it\n", PRVM_NUM_FOR_EDICT(e), PRVM_NUM_FOR_EDICT(tagentity), tagname, tagname, PRVM_NUM_FOR_EDICT(tagentity), model->name); } else Con_DPrintf("setattachment(edict %i, edict %i, string \"%s\"): tried to find tag named \"%s\" on entity %i but it has no model\n", PRVM_NUM_FOR_EDICT(e), PRVM_NUM_FOR_EDICT(tagentity), tagname, tagname, PRVM_NUM_FOR_EDICT(tagentity)); } } ///////////////////////////////////////// // DP_MD3_TAGINFO extension coded by VorteX int SV_GetTagIndex (prvm_edict_t *e, const char *tagname) { int i; dp_model_t *model; i = (int)e->fields.server->modelindex; if (i < 1 || i >= MAX_MODELS) return -1; model = sv.models[i]; return Mod_Alias_GetTagIndexForName(model, (int)e->fields.server->skin, tagname); } int SV_GetExtendedTagInfo (prvm_edict_t *e, int tagindex, int *parentindex, const char **tagname, matrix4x4_t *tag_localmatrix) { int r; dp_model_t *model; int frame; int modelindex; *tagname = NULL; *parentindex = 0; Matrix4x4_CreateIdentity(tag_localmatrix); if (tagindex >= 0 && (modelindex = (int)e->fields.server->modelindex) >= 1 && modelindex < MAX_MODELS && (model = sv.models[(int)e->fields.server->modelindex]) && model->animscenes) { frame = (int)e->fields.server->frame; if (frame < 0 || frame >= model->numframes) frame = 0; r = Mod_Alias_GetExtendedTagInfoForIndex(model, (int)e->fields.server->skin, model->animscenes[frame].firstframe, tagindex - 1, parentindex, tagname, tag_localmatrix); if(!r) // success? *parentindex += 1; return r; } return 1; } void SV_GetEntityMatrix (prvm_edict_t *ent, matrix4x4_t *out, qboolean viewmatrix) { prvm_eval_t *val; float scale; float pitchsign; int modelindex; dp_model_t *model; scale = 1; val = PRVM_EDICTFIELDVALUE(ent, prog->fieldoffsets.scale); if (val && val->_float != 0) scale = val->_float; if (viewmatrix) Matrix4x4_CreateFromQuakeEntity(out, ent->fields.server->origin[0], ent->fields.server->origin[1], ent->fields.server->origin[2] + ent->fields.server->view_ofs[2], ent->fields.server->v_angle[0], ent->fields.server->v_angle[1], ent->fields.server->v_angle[2], scale * cl_viewmodel_scale.value); else { pitchsign = 1; if ( ((modelindex = (int)ent->fields.server->modelindex) >= 1 && modelindex < MAX_MODELS && (model = sv.models[(int)ent->fields.server->modelindex])) ? model->type == mod_alias : ( (((unsigned char)PRVM_EDICTFIELDVALUE(ent, prog->fieldoffsets.pflags)->_float) & PFLAGS_FULLDYNAMIC) || ((gamemode == GAME_TENEBRAE) && ((unsigned int)ent->fields.server->effects & (16 | 32))) ) ) pitchsign = -1; Matrix4x4_CreateFromQuakeEntity(out, ent->fields.server->origin[0], ent->fields.server->origin[1], ent->fields.server->origin[2], pitchsign * ent->fields.server->angles[0], ent->fields.server->angles[1], ent->fields.server->angles[2], scale); } } int SV_GetEntityLocalTagMatrix(prvm_edict_t *ent, int tagindex, matrix4x4_t *out) { int modelindex; int frame; dp_model_t *model; if (tagindex >= 0 && (modelindex = (int)ent->fields.server->modelindex) >= 1 && modelindex < MAX_MODELS && (model = sv.models[(int)ent->fields.server->modelindex]) && model->animscenes) { // if model has wrong frame, engine automatically switches to model first frame frame = (int)ent->fields.server->frame; if (frame < 0 || frame >= model->numframes) frame = 0; return Mod_Alias_GetTagMatrix(model, model->animscenes[frame].firstframe, tagindex, out); } *out = identitymatrix; return 0; } // Warnings/errors code: // 0 - normal (everything all-right) // 1 - world entity // 2 - free entity // 3 - null or non-precached model // 4 - no tags with requested index // 5 - runaway loop at attachment chain extern cvar_t cl_bob; extern cvar_t cl_bobcycle; extern cvar_t cl_bobup; int SV_GetTagMatrix (matrix4x4_t *out, prvm_edict_t *ent, int tagindex) { int ret; prvm_eval_t *val; int modelindex, attachloop; matrix4x4_t entitymatrix, tagmatrix, attachmatrix; dp_model_t *model; *out = identitymatrix; // warnings and errors return identical matrix if (ent == prog->edicts) return 1; if (ent->priv.server->free) return 2; modelindex = (int)ent->fields.server->modelindex; if (modelindex <= 0 || modelindex >= MAX_MODELS) return 3; model = sv.models[modelindex]; tagmatrix = identitymatrix; // DP_GFX_QUAKE3MODELTAGS, scan all chain and stop on unattached entity attachloop = 0; for (;;) { if (attachloop >= 256) // prevent runaway looping return 5; // apply transformation by child's tagindex on parent entity and then // by parent entity itself ret = SV_GetEntityLocalTagMatrix(ent, tagindex - 1, &attachmatrix); if (ret && attachloop == 0) return ret; SV_GetEntityMatrix(ent, &entitymatrix, false); Matrix4x4_Concat(&tagmatrix, &attachmatrix, out); Matrix4x4_Concat(out, &entitymatrix, &tagmatrix); // next iteration we process the parent entity if ((val = PRVM_EDICTFIELDVALUE(ent, prog->fieldoffsets.tag_entity)) && val->edict) { tagindex = (int)PRVM_EDICTFIELDVALUE(ent, prog->fieldoffsets.tag_index)->_float; ent = PRVM_EDICT_NUM(val->edict); } else break; attachloop++; } // RENDER_VIEWMODEL magic if ((val = PRVM_EDICTFIELDVALUE(ent, prog->fieldoffsets.viewmodelforclient)) && val->edict) { Matrix4x4_Copy(&tagmatrix, out); ent = PRVM_EDICT_NUM(val->edict); SV_GetEntityMatrix(ent, &entitymatrix, true); Matrix4x4_Concat(out, &entitymatrix, &tagmatrix); /* // Cl_bob, ported from rendering code if (ent->fields.server->health > 0 && cl_bob.value && cl_bobcycle.value) { double bob, cycle; // LordHavoc: this code is *weird*, but not replacable (I think it // should be done in QC on the server, but oh well, quake is quake) // LordHavoc: figured out bobup: the time at which the sin is at 180 // degrees (which allows lengthening or squishing the peak or valley) cycle = sv.time/cl_bobcycle.value; cycle -= (int)cycle; if (cycle < cl_bobup.value) cycle = sin(M_PI * cycle / cl_bobup.value); else cycle = sin(M_PI + M_PI * (cycle-cl_bobup.value)/(1.0 - cl_bobup.value)); // bob is proportional to velocity in the xy plane // (don't count Z, or jumping messes it up) bob = sqrt(ent->fields.server->velocity[0]*ent->fields.server->velocity[0] + ent->fields.server->velocity[1]*ent->fields.server->velocity[1])*cl_bob.value; bob = bob*0.3 + bob*0.7*cycle; Matrix4x4_AdjustOrigin(out, 0, 0, bound(-7, bob, 4)); } */ } return 0; } //float(entity ent, string tagname) gettagindex; static void VM_SV_gettagindex (void) { prvm_edict_t *ent; const char *tag_name; int modelindex, tag_index; VM_SAFEPARMCOUNT(2, VM_SV_gettagindex); ent = PRVM_G_EDICT(OFS_PARM0); tag_name = PRVM_G_STRING(OFS_PARM1); if (ent == prog->edicts) { VM_Warning("gettagindex: can't affect world entity\n"); return; } if (ent->priv.server->free) { VM_Warning("gettagindex: can't affect free entity\n"); return; } modelindex = (int)ent->fields.server->modelindex; tag_index = 0; if (modelindex <= 0 || modelindex >= MAX_MODELS) Con_DPrintf("gettagindex(entity #%i): null or non-precached model\n", PRVM_NUM_FOR_EDICT(ent)); else { tag_index = SV_GetTagIndex(ent, tag_name); if (tag_index == 0) if(developer.integer >= 100) Con_Printf("gettagindex(entity #%i): tag \"%s\" not found\n", PRVM_NUM_FOR_EDICT(ent), tag_name); } PRVM_G_FLOAT(OFS_RETURN) = tag_index; }; //vector(entity ent, float tagindex) gettaginfo; static void VM_SV_gettaginfo (void) { prvm_edict_t *e; int tagindex; matrix4x4_t tag_matrix; matrix4x4_t tag_localmatrix; int parentindex; const char *tagname; int returncode; prvm_eval_t *val; vec3_t fo, le, up, trans; VM_SAFEPARMCOUNT(2, VM_SV_gettaginfo); e = PRVM_G_EDICT(OFS_PARM0); tagindex = (int)PRVM_G_FLOAT(OFS_PARM1); returncode = SV_GetTagMatrix(&tag_matrix, e, tagindex); Matrix4x4_ToVectors(&tag_matrix, prog->globals.server->v_forward, le, prog->globals.server->v_up, PRVM_G_VECTOR(OFS_RETURN)); VectorScale(le, -1, prog->globals.server->v_right); SV_GetExtendedTagInfo(e, tagindex, &parentindex, &tagname, &tag_localmatrix); Matrix4x4_ToVectors(&tag_localmatrix, fo, le, up, trans); if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_parent))) val->_float = parentindex; if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_name))) val->string = tagname ? PRVM_SetTempString(tagname) : 0; if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_offset))) VectorCopy(trans, val->vector); if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_forward))) VectorCopy(fo, val->vector); if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_right))) VectorScale(le, -1, val->vector); if((val = PRVM_GLOBALFIELDVALUE(prog->globaloffsets.gettaginfo_up))) VectorCopy(up, val->vector); switch(returncode) { case 1: VM_Warning("gettagindex: can't affect world entity\n"); break; case 2: VM_Warning("gettagindex: can't affect free entity\n"); break; case 3: Con_DPrintf("SV_GetTagMatrix(entity #%i): null or non-precached model\n", PRVM_NUM_FOR_EDICT(e)); break; case 4: Con_DPrintf("SV_GetTagMatrix(entity #%i): model has no tag with requested index %i\n", PRVM_NUM_FOR_EDICT(e), tagindex); break; case 5: Con_DPrintf("SV_GetTagMatrix(entity #%i): runaway loop at attachment chain\n", PRVM_NUM_FOR_EDICT(e)); break; } } //void(entity clent) dropclient (DP_SV_DROPCLIENT) static void VM_SV_dropclient (void) { int clientnum; client_t *oldhostclient; VM_SAFEPARMCOUNT(1, VM_SV_dropclient); clientnum = PRVM_G_EDICTNUM(OFS_PARM0) - 1; if (clientnum < 0 || clientnum >= svs.maxclients) { VM_Warning("dropclient: not a client\n"); return; } if (!svs.clients[clientnum].active) { VM_Warning("dropclient: that client slot is not connected\n"); return; } oldhostclient = host_client; host_client = svs.clients + clientnum; SV_DropClient(false); host_client = oldhostclient; } //entity() spawnclient (DP_SV_BOTCLIENT) static void VM_SV_spawnclient (void) { int i; prvm_edict_t *ed; VM_SAFEPARMCOUNT(0, VM_SV_spawnclient); prog->xfunction->builtinsprofile += 2; ed = prog->edicts; for (i = 0;i < svs.maxclients;i++) { if (!svs.clients[i].active) { prog->xfunction->builtinsprofile += 100; SV_ConnectClient (i, NULL); // this has to be set or else ClientDisconnect won't be called // we assume the qc will call ClientConnect... svs.clients[i].clientconnectcalled = true; ed = PRVM_EDICT_NUM(i + 1); break; } } VM_RETURN_EDICT(ed); } //float(entity clent) clienttype (DP_SV_BOTCLIENT) static void VM_SV_clienttype (void) { int clientnum; VM_SAFEPARMCOUNT(1, VM_SV_clienttype); clientnum = PRVM_G_EDICTNUM(OFS_PARM0) - 1; if (clientnum < 0 || clientnum >= svs.maxclients) PRVM_G_FLOAT(OFS_RETURN) = 3; else if (!svs.clients[clientnum].active) PRVM_G_FLOAT(OFS_RETURN) = 0; else if (svs.clients[clientnum].netconnection) PRVM_G_FLOAT(OFS_RETURN) = 1; else PRVM_G_FLOAT(OFS_RETURN) = 2; } /* =============== VM_SV_serverkey string(string key) serverkey =============== */ void VM_SV_serverkey(void) { char string[VM_STRINGTEMP_LENGTH]; VM_SAFEPARMCOUNT(1, VM_SV_serverkey); InfoString_GetValue(svs.serverinfo, PRVM_G_STRING(OFS_PARM0), string, sizeof(string)); PRVM_G_INT(OFS_RETURN) = PRVM_SetTempString(string); } //#333 void(entity e, float mdlindex) setmodelindex (EXT_CSQC) static void VM_SV_setmodelindex (void) { prvm_edict_t *e; dp_model_t *mod; int i; VM_SAFEPARMCOUNT(2, VM_SV_setmodelindex); e = PRVM_G_EDICT(OFS_PARM0); if (e == prog->edicts) { VM_Warning("setmodelindex: can not modify world entity\n"); return; } if (e->priv.server->free) { VM_Warning("setmodelindex: can not modify free entity\n"); return; } i = (int)PRVM_G_FLOAT(OFS_PARM1); if (i <= 0 || i >= MAX_MODELS) { VM_Warning("setmodelindex: invalid modelindex\n"); return; } if (!sv.model_precache[i][0]) { VM_Warning("setmodelindex: model not precached\n"); return; } e->fields.server->model = PRVM_SetEngineString(sv.model_precache[i]); e->fields.server->modelindex = i; mod = sv.models[i]; if (mod) { if (mod->type != mod_alias || sv_gameplayfix_setmodelrealbox.integer) SetMinMaxSize (e, mod->normalmins, mod->normalmaxs, true); else SetMinMaxSize (e, quakemins, quakemaxs, true); } else SetMinMaxSize (e, vec3_origin, vec3_origin, true); } //#334 string(float mdlindex) modelnameforindex (EXT_CSQC) static void VM_SV_modelnameforindex (void) { int i; VM_SAFEPARMCOUNT(1, VM_SV_modelnameforindex); PRVM_G_INT(OFS_RETURN) = OFS_NULL; i = (int)PRVM_G_FLOAT(OFS_PARM0); if (i <= 0 || i >= MAX_MODELS) { VM_Warning("modelnameforindex: invalid modelindex\n"); return; } if (!sv.model_precache[i][0]) { VM_Warning("modelnameforindex: model not precached\n"); return; } PRVM_G_INT(OFS_RETURN) = PRVM_SetEngineString(sv.model_precache[i]); } //#335 float(string effectname) particleeffectnum (EXT_CSQC) static void VM_SV_particleeffectnum (void) { int i; VM_SAFEPARMCOUNT(1, VM_SV_particleeffectnum); i = SV_ParticleEffectIndex(PRVM_G_STRING(OFS_PARM0)); if (i == 0) i = -1; PRVM_G_FLOAT(OFS_RETURN) = i; } // #336 void(entity ent, float effectnum, vector start, vector end) trailparticles (EXT_CSQC) static void VM_SV_trailparticles (void) { VM_SAFEPARMCOUNT(4, VM_SV_trailparticles); if ((int)PRVM_G_FLOAT(OFS_PARM0) < 0) return; MSG_WriteByte(&sv.datagram, svc_trailparticles); MSG_WriteShort(&sv.datagram, PRVM_G_EDICTNUM(OFS_PARM0)); MSG_WriteShort(&sv.datagram, (int)PRVM_G_FLOAT(OFS_PARM1)); MSG_WriteVector(&sv.datagram, PRVM_G_VECTOR(OFS_PARM2), sv.protocol); MSG_WriteVector(&sv.datagram, PRVM_G_VECTOR(OFS_PARM3), sv.protocol); SV_FlushBroadcastMessages(); } //#337 void(float effectnum, vector origin, vector dir, float count) pointparticles (EXT_CSQC) static void VM_SV_pointparticles (void) { int effectnum, count; vec3_t org, vel; VM_SAFEPARMCOUNTRANGE(4, 8, VM_SV_pointparticles); if ((int)PRVM_G_FLOAT(OFS_PARM0) < 0) return; effectnum = (int)PRVM_G_FLOAT(OFS_PARM0); VectorCopy(PRVM_G_VECTOR(OFS_PARM1), org); VectorCopy(PRVM_G_VECTOR(OFS_PARM2), vel); count = bound(0, (int)PRVM_G_FLOAT(OFS_PARM3), 65535); if (count == 1 && !VectorLength2(vel)) { // 1+2+12=15 bytes MSG_WriteByte(&sv.datagram, svc_pointparticles1); MSG_WriteShort(&sv.datagram, effectnum); MSG_WriteVector(&sv.datagram, org, sv.protocol); } else { // 1+2+12+12+2=29 bytes MSG_WriteByte(&sv.datagram, svc_pointparticles); MSG_WriteShort(&sv.datagram, effectnum); MSG_WriteVector(&sv.datagram, org, sv.protocol); MSG_WriteVector(&sv.datagram, vel, sv.protocol); MSG_WriteShort(&sv.datagram, count); } SV_FlushBroadcastMessages(); } //PF_setpause, // void(float pause) setpause = #531; static void VM_SV_setpause(void) { int pauseValue; pauseValue = (int)PRVM_G_FLOAT(OFS_PARM0); if (pauseValue != 0) { //pause the game sv.paused = 1; sv.pausedstart = Sys_DoubleTime(); } else { //disable pause, in case it was enabled if (sv.paused != 0) { sv.paused = 0; sv.pausedstart = 0; } } // send notification to all clients MSG_WriteByte(&sv.reliable_datagram, svc_setpause); MSG_WriteByte(&sv.reliable_datagram, sv.paused); } prvm_builtin_t vm_sv_builtins[] = { NULL, // #0 NULL function (not callable) (QUAKE) VM_makevectors, // #1 void(vector ang) makevectors (QUAKE) VM_SV_setorigin, // #2 void(entity e, vector o) setorigin (QUAKE) VM_SV_setmodel, // #3 void(entity e, string m) setmodel (QUAKE) VM_SV_setsize, // #4 void(entity e, vector min, vector max) setsize (QUAKE) NULL, // #5 void(entity e, vector min, vector max) setabssize (QUAKE) VM_break, // #6 void() break (QUAKE) VM_random, // #7 float() random (QUAKE) VM_SV_sound, // #8 void(entity e, float chan, string samp) sound (QUAKE) VM_normalize, // #9 vector(vector v) normalize (QUAKE) VM_error, // #10 void(string e) error (QUAKE) VM_objerror, // #11 void(string e) objerror (QUAKE) VM_vlen, // #12 float(vector v) vlen (QUAKE) VM_vectoyaw, // #13 float(vector v) vectoyaw (QUAKE) VM_spawn, // #14 entity() spawn (QUAKE) VM_remove, // #15 void(entity e) remove (QUAKE) VM_SV_traceline, // #16 void(vector v1, vector v2, float tryents) traceline (QUAKE) VM_SV_checkclient, // #17 entity() checkclient (QUAKE) VM_find, // #18 entity(entity start, .string fld, string match) find (QUAKE) VM_SV_precache_sound, // #19 void(string s) precache_sound (QUAKE) VM_SV_precache_model, // #20 void(string s) precache_model (QUAKE) VM_SV_stuffcmd, // #21 void(entity client, string s, ...) stuffcmd (QUAKE) VM_SV_findradius, // #22 entity(vector org, float rad) findradius (QUAKE) VM_bprint, // #23 void(string s, ...) bprint (QUAKE) VM_SV_sprint, // #24 void(entity client, string s, ...) sprint (QUAKE) VM_dprint, // #25 void(string s, ...) dprint (QUAKE) VM_ftos, // #26 string(float f) ftos (QUAKE) VM_vtos, // #27 string(vector v) vtos (QUAKE) VM_coredump, // #28 void() coredump (QUAKE) VM_traceon, // #29 void() traceon (QUAKE) VM_traceoff, // #30 void() traceoff (QUAKE) VM_eprint, // #31 void(entity e) eprint (QUAKE) VM_SV_walkmove, // #32 float(float yaw, float dist) walkmove (QUAKE) NULL, // #33 (QUAKE) VM_SV_droptofloor, // #34 float() droptofloor (QUAKE) VM_SV_lightstyle, // #35 void(float style, string value) lightstyle (QUAKE) VM_rint, // #36 float(float v) rint (QUAKE) VM_floor, // #37 float(float v) floor (QUAKE) VM_ceil, // #38 float(float v) ceil (QUAKE) NULL, // #39 (QUAKE) VM_SV_checkbottom, // #40 float(entity e) checkbottom (QUAKE) VM_SV_pointcontents, // #41 float(vector v) pointcontents (QUAKE) NULL, // #42 (QUAKE) VM_fabs, // #43 float(float f) fabs (QUAKE) VM_SV_aim, // #44 vector(entity e, float speed) aim (QUAKE) VM_cvar, // #45 float(string s) cvar (QUAKE) VM_localcmd, // #46 void(string s) localcmd (QUAKE) VM_nextent, // #47 entity(entity e) nextent (QUAKE) VM_SV_particle, // #48 void(vector o, vector d, float color, float count) particle (QUAKE) VM_changeyaw, // #49 void() ChangeYaw (QUAKE) NULL, // #50 (QUAKE) VM_vectoangles, // #51 vector(vector v) vectoangles (QUAKE) VM_SV_WriteByte, // #52 void(float to, float f) WriteByte (QUAKE) VM_SV_WriteChar, // #53 void(float to, float f) WriteChar (QUAKE) VM_SV_WriteShort, // #54 void(float to, float f) WriteShort (QUAKE) VM_SV_WriteLong, // #55 void(float to, float f) WriteLong (QUAKE) VM_SV_WriteCoord, // #56 void(float to, float f) WriteCoord (QUAKE) VM_SV_WriteAngle, // #57 void(float to, float f) WriteAngle (QUAKE) VM_SV_WriteString, // #58 void(float to, string s) WriteString (QUAKE) VM_SV_WriteEntity, // #59 void(float to, entity e) WriteEntity (QUAKE) VM_sin, // #60 float(float f) sin (DP_QC_SINCOSSQRTPOW) (QUAKE) VM_cos, // #61 float(float f) cos (DP_QC_SINCOSSQRTPOW) (QUAKE) VM_sqrt, // #62 float(float f) sqrt (DP_QC_SINCOSSQRTPOW) (QUAKE) VM_changepitch, // #63 void(entity ent) changepitch (DP_QC_CHANGEPITCH) (QUAKE) VM_SV_tracetoss, // #64 void(entity e, entity ignore) tracetoss (DP_QC_TRACETOSS) (QUAKE) VM_etos, // #65 string(entity ent) etos (DP_QC_ETOS) (QUAKE) NULL, // #66 (QUAKE) SV_MoveToGoal, // #67 void(float step) movetogoal (QUAKE) VM_precache_file, // #68 string(string s) precache_file (QUAKE) VM_SV_makestatic, // #69 void(entity e) makestatic (QUAKE) VM_changelevel, // #70 void(string s) changelevel (QUAKE) NULL, // #71 (QUAKE) VM_cvar_set, // #72 void(string var, string val) cvar_set (QUAKE) VM_SV_centerprint, // #73 void(entity client, strings) centerprint (QUAKE) VM_SV_ambientsound, // #74 void(vector pos, string samp, float vol, float atten) ambientsound (QUAKE) VM_SV_precache_model, // #75 string(string s) precache_model2 (QUAKE) VM_SV_precache_sound, // #76 string(string s) precache_sound2 (QUAKE) VM_precache_file, // #77 string(string s) precache_file2 (QUAKE) VM_SV_setspawnparms, // #78 void(entity e) setspawnparms (QUAKE) NULL, // #79 void(entity killer, entity killee) logfrag (QUAKEWORLD) NULL, // #80 string(entity e, string keyname) infokey (QUAKEWORLD) VM_stof, // #81 float(string s) stof (FRIK_FILE) NULL, // #82 void(vector where, float set) multicast (QUAKEWORLD) NULL, // #83 (QUAKE) NULL, // #84 (QUAKE) NULL, // #85 (QUAKE) NULL, // #86 (QUAKE) NULL, // #87 (QUAKE) NULL, // #88 (QUAKE) NULL, // #89 (QUAKE) VM_SV_tracebox, // #90 void(vector v1, vector min, vector max, vector v2, float nomonsters, entity forent) tracebox (DP_QC_TRACEBOX) VM_randomvec, // #91 vector() randomvec (DP_QC_RANDOMVEC) VM_SV_getlight, // #92 vector(vector org) getlight (DP_QC_GETLIGHT) VM_registercvar, // #93 float(string name, string value) registercvar (DP_REGISTERCVAR) VM_min, // #94 float(float a, floats) min (DP_QC_MINMAXBOUND) VM_max, // #95 float(float a, floats) max (DP_QC_MINMAXBOUND) VM_bound, // #96 float(float minimum, float val, float maximum) bound (DP_QC_MINMAXBOUND) VM_pow, // #97 float(float f, float f) pow (DP_QC_SINCOSSQRTPOW) VM_findfloat, // #98 entity(entity start, .float fld, float match) findfloat (DP_QC_FINDFLOAT) VM_checkextension, // #99 float(string s) checkextension (the basis of the extension system) // FrikaC and Telejano range #100-#199 NULL, // #100 NULL, // #101 NULL, // #102 NULL, // #103 NULL, // #104 NULL, // #105 NULL, // #106 NULL, // #107 NULL, // #108 NULL, // #109 VM_fopen, // #110 float(string filename, float mode) fopen (FRIK_FILE) VM_fclose, // #111 void(float fhandle) fclose (FRIK_FILE) VM_fgets, // #112 string(float fhandle) fgets (FRIK_FILE) VM_fputs, // #113 void(float fhandle, string s) fputs (FRIK_FILE) VM_strlen, // #114 float(string s) strlen (FRIK_FILE) VM_strcat, // #115 string(string s1, string s2, ...) strcat (FRIK_FILE) VM_substring, // #116 string(string s, float start, float length) substring (FRIK_FILE) VM_stov, // #117 vector(string) stov (FRIK_FILE) VM_strzone, // #118 string(string s) strzone (FRIK_FILE) VM_strunzone, // #119 void(string s) strunzone (FRIK_FILE) NULL, // #120 NULL, // #121 NULL, // #122 NULL, // #123 NULL, // #124 NULL, // #125 NULL, // #126 NULL, // #127 NULL, // #128 NULL, // #129 NULL, // #130 NULL, // #131 NULL, // #132 NULL, // #133 NULL, // #134 NULL, // #135 NULL, // #136 NULL, // #137 NULL, // #138 NULL, // #139 NULL, // #140 NULL, // #141 NULL, // #142 NULL, // #143 NULL, // #144 NULL, // #145 NULL, // #146 NULL, // #147 NULL, // #148 NULL, // #149 NULL, // #150 NULL, // #151 NULL, // #152 NULL, // #153 NULL, // #154 NULL, // #155 NULL, // #156 NULL, // #157 NULL, // #158 NULL, // #159 NULL, // #160 NULL, // #161 NULL, // #162 NULL, // #163 NULL, // #164 NULL, // #165 NULL, // #166 NULL, // #167 NULL, // #168 NULL, // #169 NULL, // #170 NULL, // #171 NULL, // #172 NULL, // #173 NULL, // #174 NULL, // #175 NULL, // #176 NULL, // #177 NULL, // #178 NULL, // #179 NULL, // #180 NULL, // #181 NULL, // #182 NULL, // #183 NULL, // #184 NULL, // #185 NULL, // #186 NULL, // #187 NULL, // #188 NULL, // #189 NULL, // #190 NULL, // #191 NULL, // #192 NULL, // #193 NULL, // #194 NULL, // #195 NULL, // #196 NULL, // #197 NULL, // #198 NULL, // #199 // FTEQW range #200-#299 NULL, // #200 NULL, // #201 NULL, // #202 NULL, // #203 NULL, // #204 NULL, // #205 NULL, // #206 NULL, // #207 NULL, // #208 NULL, // #209 NULL, // #210 NULL, // #211 NULL, // #212 NULL, // #213 NULL, // #214 NULL, // #215 NULL, // #216 NULL, // #217 VM_bitshift, // #218 float(float number, float quantity) bitshift (EXT_BITSHIFT) NULL, // #219 NULL, // #220 VM_strstrofs, // #221 float(string str, string sub[, float startpos]) strstrofs (FTE_STRINGS) VM_str2chr, // #222 float(string str, float ofs) str2chr (FTE_STRINGS) VM_chr2str, // #223 string(float c, ...) chr2str (FTE_STRINGS) VM_strconv, // #224 string(float ccase, float calpha, float cnum, string s, ...) strconv (FTE_STRINGS) VM_strpad, // #225 string(float chars, string s, ...) strpad (FTE_STRINGS) VM_infoadd, // #226 string(string info, string key, string value, ...) infoadd (FTE_STRINGS) VM_infoget, // #227 string(string info, string key) infoget (FTE_STRINGS) VM_strncmp, // #228 float(string s1, string s2, float len) strncmp (FTE_STRINGS) VM_strncasecmp, // #229 float(string s1, string s2) strcasecmp (FTE_STRINGS) VM_strncasecmp, // #230 float(string s1, string s2, float len) strncasecmp (FTE_STRINGS) NULL, // #231 VM_SV_AddStat, // #232 void(float index, float type, .void field) SV_AddStat (EXT_CSQC) NULL, // #233 NULL, // #234 NULL, // #235 NULL, // #236 NULL, // #237 NULL, // #238 NULL, // #239 VM_SV_checkpvs, // #240 float(vector viewpos, entity viewee) checkpvs; NULL, // #241 NULL, // #242 NULL, // #243 NULL, // #244 NULL, // #245 NULL, // #246 NULL, // #247 NULL, // #248 NULL, // #249 NULL, // #250 NULL, // #251 NULL, // #252 NULL, // #253 NULL, // #254 NULL, // #255 NULL, // #256 NULL, // #257 NULL, // #258 NULL, // #259 NULL, // #260 NULL, // #261 NULL, // #262 NULL, // #263 NULL, // #264 NULL, // #265 NULL, // #266 NULL, // #267 NULL, // #268 NULL, // #269 NULL, // #270 NULL, // #271 NULL, // #272 NULL, // #273 NULL, // #274 NULL, // #275 NULL, // #276 NULL, // #277 NULL, // #278 NULL, // #279 NULL, // #280 NULL, // #281 NULL, // #282 NULL, // #283 NULL, // #284 NULL, // #285 NULL, // #286 NULL, // #287 NULL, // #288 NULL, // #289 NULL, // #290 NULL, // #291 NULL, // #292 NULL, // #293 NULL, // #294 NULL, // #295 NULL, // #296 NULL, // #297 NULL, // #298 NULL, // #299 // CSQC range #300-#399 NULL, // #300 void() clearscene (EXT_CSQC) NULL, // #301 void(float mask) addentities (EXT_CSQC) NULL, // #302 void(entity ent) addentity (EXT_CSQC) NULL, // #303 float(float property, ...) setproperty (EXT_CSQC) NULL, // #304 void() renderscene (EXT_CSQC) NULL, // #305 void(vector org, float radius, vector lightcolours) adddynamiclight (EXT_CSQC) NULL, // #306 void(string texturename, float flag[, float is2d, float lines]) R_BeginPolygon NULL, // #307 void(vector org, vector texcoords, vector rgb, float alpha) R_PolygonVertex NULL, // #308 void() R_EndPolygon NULL, // #309 NULL, // #310 vector (vector v) cs_unproject (EXT_CSQC) NULL, // #311 vector (vector v) cs_project (EXT_CSQC) NULL, // #312 NULL, // #313 NULL, // #314 NULL, // #315 void(float width, vector pos1, vector pos2, float flag) drawline (EXT_CSQC) NULL, // #316 float(string name) iscachedpic (EXT_CSQC) NULL, // #317 string(string name, float trywad) precache_pic (EXT_CSQC) NULL, // #318 vector(string picname) draw_getimagesize (EXT_CSQC) NULL, // #319 void(string name) freepic (EXT_CSQC) NULL, // #320 float(vector position, float character, vector scale, vector rgb, float alpha, float flag) drawcharacter (EXT_CSQC) NULL, // #321 float(vector position, string text, vector scale, vector rgb, float alpha, float flag) drawstring (EXT_CSQC) NULL, // #322 float(vector position, string pic, vector size, vector rgb, float alpha, float flag) drawpic (EXT_CSQC) NULL, // #323 float(vector position, vector size, vector rgb, float alpha, float flag) drawfill (EXT_CSQC) NULL, // #324 void(float x, float y, float width, float height) drawsetcliparea NULL, // #325 void(void) drawresetcliparea NULL, // #326 NULL, // #327 NULL, // #328 NULL, // #329 NULL, // #330 float(float stnum) getstatf (EXT_CSQC) NULL, // #331 float(float stnum) getstati (EXT_CSQC) NULL, // #332 string(float firststnum) getstats (EXT_CSQC) VM_SV_setmodelindex, // #333 void(entity e, float mdlindex) setmodelindex (EXT_CSQC) VM_SV_modelnameforindex, // #334 string(float mdlindex) modelnameforindex (EXT_CSQC) VM_SV_particleeffectnum, // #335 float(string effectname) particleeffectnum (EXT_CSQC) VM_SV_trailparticles, // #336 void(entity ent, float effectnum, vector start, vector end) trailparticles (EXT_CSQC) VM_SV_pointparticles, // #337 void(float effectnum, vector origin [, vector dir, float count]) pointparticles (EXT_CSQC) NULL, // #338 void(string s, ...) centerprint (EXT_CSQC) VM_print, // #339 void(string s, ...) print (EXT_CSQC, DP_SV_PRINT) NULL, // #340 string(float keynum) keynumtostring (EXT_CSQC) NULL, // #341 float(string keyname) stringtokeynum (EXT_CSQC) NULL, // #342 string(float keynum) getkeybind (EXT_CSQC) NULL, // #343 void(float usecursor) setcursormode (EXT_CSQC) NULL, // #344 vector() getmousepos (EXT_CSQC) NULL, // #345 float(float framenum) getinputstate (EXT_CSQC) NULL, // #346 void(float sens) setsensitivityscaler (EXT_CSQC) NULL, // #347 void() runstandardplayerphysics (EXT_CSQC) NULL, // #348 string(float playernum, string keyname) getplayerkeyvalue (EXT_CSQC) NULL, // #349 float() isdemo (EXT_CSQC) VM_isserver, // #350 float() isserver (EXT_CSQC) NULL, // #351 void(vector origin, vector forward, vector right, vector up) SetListener (EXT_CSQC) NULL, // #352 void(string cmdname) registercommand (EXT_CSQC) VM_wasfreed, // #353 float(entity ent) wasfreed (EXT_CSQC) (should be availabe on server too) VM_SV_serverkey, // #354 string(string key) serverkey (EXT_CSQC) NULL, // #355 NULL, // #356 NULL, // #357 NULL, // #358 NULL, // #359 NULL, // #360 float() readbyte (EXT_CSQC) NULL, // #361 float() readchar (EXT_CSQC) NULL, // #362 float() readshort (EXT_CSQC) NULL, // #363 float() readlong (EXT_CSQC) NULL, // #364 float() readcoord (EXT_CSQC) NULL, // #365 float() readangle (EXT_CSQC) NULL, // #366 string() readstring (EXT_CSQC) NULL, // #367 float() readfloat (EXT_CSQC) NULL, // #368 NULL, // #369 NULL, // #370 NULL, // #371 NULL, // #372 NULL, // #373 NULL, // #374 NULL, // #375 NULL, // #376 NULL, // #377 NULL, // #378 NULL, // #379 NULL, // #380 NULL, // #381 NULL, // #382 NULL, // #383 NULL, // #384 NULL, // #385 NULL, // #386 NULL, // #387 NULL, // #388 NULL, // #389 NULL, // #390 NULL, // #391 NULL, // #392 NULL, // #393 NULL, // #394 NULL, // #395 NULL, // #396 NULL, // #397 NULL, // #398 NULL, // #399 // LordHavoc's range #400-#499 VM_SV_copyentity, // #400 void(entity from, entity to) copyentity (DP_QC_COPYENTITY) VM_SV_setcolor, // #401 void(entity ent, float colors) setcolor (DP_QC_SETCOLOR) VM_findchain, // #402 entity(.string fld, string match) findchain (DP_QC_FINDCHAIN) VM_findchainfloat, // #403 entity(.float fld, float match) findchainfloat (DP_QC_FINDCHAINFLOAT) VM_SV_effect, // #404 void(vector org, string modelname, float startframe, float endframe, float framerate) effect (DP_SV_EFFECT) VM_SV_te_blood, // #405 void(vector org, vector velocity, float howmany) te_blood (DP_TE_BLOOD) VM_SV_te_bloodshower, // #406 void(vector mincorner, vector maxcorner, float explosionspeed, float howmany) te_bloodshower (DP_TE_BLOODSHOWER) VM_SV_te_explosionrgb, // #407 void(vector org, vector color) te_explosionrgb (DP_TE_EXPLOSIONRGB) VM_SV_te_particlecube, // #408 void(vector mincorner, vector maxcorner, vector vel, float howmany, float color, float gravityflag, float randomveljitter) te_particlecube (DP_TE_PARTICLECUBE) VM_SV_te_particlerain, // #409 void(vector mincorner, vector maxcorner, vector vel, float howmany, float color) te_particlerain (DP_TE_PARTICLERAIN) VM_SV_te_particlesnow, // #410 void(vector mincorner, vector maxcorner, vector vel, float howmany, float color) te_particlesnow (DP_TE_PARTICLESNOW) VM_SV_te_spark, // #411 void(vector org, vector vel, float howmany) te_spark (DP_TE_SPARK) VM_SV_te_gunshotquad, // #412 void(vector org) te_gunshotquad (DP_QUADEFFECTS1) VM_SV_te_spikequad, // #413 void(vector org) te_spikequad (DP_QUADEFFECTS1) VM_SV_te_superspikequad, // #414 void(vector org) te_superspikequad (DP_QUADEFFECTS1) VM_SV_te_explosionquad, // #415 void(vector org) te_explosionquad (DP_QUADEFFECTS1) VM_SV_te_smallflash, // #416 void(vector org) te_smallflash (DP_TE_SMALLFLASH) VM_SV_te_customflash, // #417 void(vector org, float radius, float lifetime, vector color) te_customflash (DP_TE_CUSTOMFLASH) VM_SV_te_gunshot, // #418 void(vector org) te_gunshot (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_spike, // #419 void(vector org) te_spike (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_superspike, // #420 void(vector org) te_superspike (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_explosion, // #421 void(vector org) te_explosion (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_tarexplosion, // #422 void(vector org) te_tarexplosion (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_wizspike, // #423 void(vector org) te_wizspike (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_knightspike, // #424 void(vector org) te_knightspike (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_lavasplash, // #425 void(vector org) te_lavasplash (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_teleport, // #426 void(vector org) te_teleport (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_explosion2, // #427 void(vector org, float colorstart, float colorlength) te_explosion2 (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_lightning1, // #428 void(entity own, vector start, vector end) te_lightning1 (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_lightning2, // #429 void(entity own, vector start, vector end) te_lightning2 (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_lightning3, // #430 void(entity own, vector start, vector end) te_lightning3 (DP_TE_STANDARDEFFECTBUILTINS) VM_SV_te_beam, // #431 void(entity own, vector start, vector end) te_beam (DP_TE_STANDARDEFFECTBUILTINS) VM_vectorvectors, // #432 void(vector dir) vectorvectors (DP_QC_VECTORVECTORS) VM_SV_te_plasmaburn, // #433 void(vector org) te_plasmaburn (DP_TE_PLASMABURN) VM_SV_getsurfacenumpoints, // #434 float(entity e, float s) getsurfacenumpoints (DP_QC_GETSURFACE) VM_SV_getsurfacepoint, // #435 vector(entity e, float s, float n) getsurfacepoint (DP_QC_GETSURFACE) VM_SV_getsurfacenormal, // #436 vector(entity e, float s) getsurfacenormal (DP_QC_GETSURFACE) VM_SV_getsurfacetexture, // #437 string(entity e, float s) getsurfacetexture (DP_QC_GETSURFACE) VM_SV_getsurfacenearpoint, // #438 float(entity e, vector p) getsurfacenearpoint (DP_QC_GETSURFACE) VM_SV_getsurfaceclippedpoint, // #439 vector(entity e, float s, vector p) getsurfaceclippedpoint (DP_QC_GETSURFACE) VM_SV_clientcommand, // #440 void(entity e, string s) clientcommand (KRIMZON_SV_PARSECLIENTCOMMAND) VM_tokenize, // #441 float(string s) tokenize (KRIMZON_SV_PARSECLIENTCOMMAND) VM_argv, // #442 string(float n) argv (KRIMZON_SV_PARSECLIENTCOMMAND) VM_SV_setattachment, // #443 void(entity e, entity tagentity, string tagname) setattachment (DP_GFX_QUAKE3MODELTAGS) VM_search_begin, // #444 float(string pattern, float caseinsensitive, float quiet) search_begin (DP_QC_FS_SEARCH) VM_search_end, // #445 void(float handle) search_end (DP_QC_FS_SEARCH) VM_search_getsize, // #446 float(float handle) search_getsize (DP_QC_FS_SEARCH) VM_search_getfilename, // #447 string(float handle, float num) search_getfilename (DP_QC_FS_SEARCH) VM_cvar_string, // #448 string(string s) cvar_string (DP_QC_CVAR_STRING) VM_findflags, // #449 entity(entity start, .float fld, float match) findflags (DP_QC_FINDFLAGS) VM_findchainflags, // #450 entity(.float fld, float match) findchainflags (DP_QC_FINDCHAINFLAGS) VM_SV_gettagindex, // #451 float(entity ent, string tagname) gettagindex (DP_QC_GETTAGINFO) VM_SV_gettaginfo, // #452 vector(entity ent, float tagindex) gettaginfo (DP_QC_GETTAGINFO) VM_SV_dropclient, // #453 void(entity clent) dropclient (DP_SV_DROPCLIENT) VM_SV_spawnclient, // #454 entity() spawnclient (DP_SV_BOTCLIENT) VM_SV_clienttype, // #455 float(entity clent) clienttype (DP_SV_BOTCLIENT) VM_SV_WriteUnterminatedString, // #456 void(float to, string s) WriteUnterminatedString (DP_SV_WRITEUNTERMINATEDSTRING) VM_SV_te_flamejet, // #457 void(vector org, vector vel, float howmany) te_flamejet = #457 (DP_TE_FLAMEJET) NULL, // #458 VM_ftoe, // #459 entity(float num) entitybyindex (DP_QC_EDICT_NUM) VM_buf_create, // #460 float() buf_create (DP_QC_STRINGBUFFERS) VM_buf_del, // #461 void(float bufhandle) buf_del (DP_QC_STRINGBUFFERS) VM_buf_getsize, // #462 float(float bufhandle) buf_getsize (DP_QC_STRINGBUFFERS) VM_buf_copy, // #463 void(float bufhandle_from, float bufhandle_to) buf_copy (DP_QC_STRINGBUFFERS) VM_buf_sort, // #464 void(float bufhandle, float sortpower, float backward) buf_sort (DP_QC_STRINGBUFFERS) VM_buf_implode, // #465 string(float bufhandle, string glue) buf_implode (DP_QC_STRINGBUFFERS) VM_bufstr_get, // #466 string(float bufhandle, float string_index) bufstr_get (DP_QC_STRINGBUFFERS) VM_bufstr_set, // #467 void(float bufhandle, float string_index, string str) bufstr_set (DP_QC_STRINGBUFFERS) VM_bufstr_add, // #468 float(float bufhandle, string str, float order) bufstr_add (DP_QC_STRINGBUFFERS) VM_bufstr_free, // #469 void(float bufhandle, float string_index) bufstr_free (DP_QC_STRINGBUFFERS) NULL, // #470 VM_asin, // #471 float(float s) VM_asin (DP_QC_ASINACOSATANATAN2TAN) VM_acos, // #472 float(float c) VM_acos (DP_QC_ASINACOSATANATAN2TAN) VM_atan, // #473 float(float t) VM_atan (DP_QC_ASINACOSATANATAN2TAN) VM_atan2, // #474 float(float c, float s) VM_atan2 (DP_QC_ASINACOSATANATAN2TAN) VM_tan, // #475 float(float a) VM_tan (DP_QC_ASINACOSATANATAN2TAN) VM_strlennocol, // #476 float(string s) : DRESK - String Length (not counting color codes) (DP_QC_STRINGCOLORFUNCTIONS) VM_strdecolorize, // #477 string(string s) : DRESK - Decolorized String (DP_SV_STRINGCOLORFUNCTIONS) VM_strftime, // #478 string(float uselocaltime, string format, ...) (DP_QC_STRFTIME) VM_tokenizebyseparator, // #479 float(string s) tokenizebyseparator (DP_QC_TOKENIZEBYSEPARATOR) VM_strtolower, // #480 string(string s) VM_strtolower (DP_QC_STRING_CASE_FUNCTIONS) VM_strtoupper, // #481 string(string s) VM_strtoupper (DP_QC_STRING_CASE_FUNCTIONS) VM_cvar_defstring, // #482 string(string s) cvar_defstring (DP_QC_CVAR_DEFSTRING) VM_SV_pointsound, // #483 void(vector origin, string sample, float volume, float attenuation) (DP_SV_POINTSOUND) VM_strreplace, // #484 string(string search, string replace, string subject) strreplace (DP_QC_STRREPLACE) VM_strireplace, // #485 string(string search, string replace, string subject) strireplace (DP_QC_STRREPLACE) VM_SV_getsurfacepointattribute,// #486 vector(entity e, float s, float n, float a) getsurfacepointattribute = #486; NULL, // #487 NULL, // #488 NULL, // #489 NULL, // #490 NULL, // #491 NULL, // #492 NULL, // #493 VM_crc16, // #494 float(float caseinsensitive, string s, ...) crc16 = #494 (DP_QC_CRC16) VM_cvar_type, // #495 float(string name) cvar_type = #495; (DP_QC_CVAR_TYPE) VM_numentityfields, // #496 float() numentityfields = #496; (DP_QC_ENTITYDATA) VM_entityfieldname, // #497 string(float fieldnum) entityfieldname = #497; (DP_QC_ENTITYDATA) VM_entityfieldtype, // #498 float(float fieldnum) entityfieldtype = #498; (DP_QC_ENTITYDATA) VM_getentityfieldstring, // #499 string(float fieldnum, entity ent) getentityfieldstring = #499; (DP_QC_ENTITYDATA) VM_putentityfieldstring, // #500 float(float fieldnum, entity ent, string s) putentityfieldstring = #500; (DP_QC_ENTITYDATA) VM_SV_WritePicture, // #501 NULL, // #502 VM_whichpack, // #503 string(string) whichpack = #503; NULL, // #504 NULL, // #505 NULL, // #506 NULL, // #507 NULL, // #508 NULL, // #509 VM_uri_escape, // #510 string(string in) uri_escape = #510; VM_uri_unescape, // #511 string(string in) uri_unescape = #511; VM_etof, // #512 float(entity ent) num_for_edict = #512 (DP_QC_NUM_FOR_EDICT) VM_uri_get, // #513 float(string uril, float id) uri_get = #513; (DP_QC_URI_GET) VM_tokenize_console, // #514 float(string str) tokenize_console = #514; (DP_QC_TOKENIZE_CONSOLE) VM_argv_start_index, // #515 float(float idx) argv_start_index = #515; (DP_QC_TOKENIZE_CONSOLE) VM_argv_end_index, // #516 float(float idx) argv_end_index = #516; (DP_QC_TOKENIZE_CONSOLE) VM_buf_cvarlist, // #517 void(float buf, string prefix, string antiprefix) buf_cvarlist = #517; (DP_QC_STRINGBUFFERS_CVARLIST) VM_cvar_description, // #518 float(string name) cvar_description = #518; (DP_QC_CVAR_DESCRIPTION) VM_gettime, // #519 float(float timer) gettime = #519; (DP_QC_GETTIME) NULL, // #520 NULL, // #521 NULL, // #522 NULL, // #523 NULL, // #524 NULL, // #525 NULL, // #526 NULL, // #527 NULL, // #528 NULL, // #529 NULL, // #530 VM_SV_setpause, // #531 void(float pause) setpause = #531; NULL, // #532 NULL, // #533 NULL, // #534 NULL, // #535 NULL, // #536 NULL, // #537 NULL, // #538 NULL, // #539 NULL, // #540 NULL, // #541 NULL, // #542 NULL, // #543 NULL, // #544 NULL, // #545 NULL, // #546 NULL, // #547 NULL, // #548 NULL, // #549 NULL, // #550 NULL, // #551 NULL, // #552 NULL, // #553 NULL, // #554 NULL, // #555 NULL, // #556 NULL, // #557 NULL, // #558 NULL, // #559 NULL, // #560 NULL, // #561 NULL, // #562 NULL, // #563 NULL, // #564 NULL, // #565 NULL, // #566 NULL, // #567 NULL, // #568 NULL, // #569 NULL, // #570 NULL, // #571 NULL, // #572 NULL, // #573 NULL, // #574 NULL, // #575 NULL, // #576 NULL, // #577 NULL, // #578 NULL, // #579 NULL, // #580 NULL, // #581 NULL, // #582 NULL, // #583 NULL, // #584 NULL, // #585 NULL, // #586 NULL, // #587 NULL, // #588 NULL, // #589 NULL, // #590 NULL, // #591 NULL, // #592 NULL, // #593 NULL, // #594 NULL, // #595 NULL, // #596 NULL, // #597 NULL, // #598 NULL, // #599 NULL, // #600 NULL, // #601 NULL, // #602 NULL, // #603 NULL, // #604 NULL, // #605 NULL, // #606 NULL, // #607 NULL, // #608 NULL, // #609 NULL, // #610 NULL, // #611 NULL, // #612 NULL, // #613 NULL, // #614 NULL, // #615 NULL, // #616 NULL, // #617 NULL, // #618 NULL, // #619 NULL, // #620 NULL, // #621 NULL, // #622 NULL, // #623 VM_getextresponse, // #624 string getextresponse(void) NULL, // #625 }; const int vm_sv_numbuiltins = sizeof(vm_sv_builtins) / sizeof(prvm_builtin_t); void VM_SV_Cmd_Init(void) { VM_Cmd_Init(); } void VM_SV_Cmd_Reset(void) { if(prog->funcoffsets.SV_Shutdown) { func_t s = prog->funcoffsets.SV_Shutdown; prog->funcoffsets.SV_Shutdown = 0; // prevent it from getting called again PRVM_ExecuteProgram(s,"SV_Shutdown() required"); } VM_Cmd_Reset(); }