/* Copyright (C) 1996-2001 Id Software, Inc. Copyright (C) 2002-2009 John Fitzgibbons and others Copyright (C) 2010-2014 QuakeSpasm developers This program is free software; you can redistribute it and/or modify it under the terms of the GNU General Public License as published by the Free Software Foundation; either version 2 of the License, or (at your option) any later version. This program is distributed in the hope that it will be useful, but WITHOUT ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for more details. You should have received a copy of the GNU General Public License along with this program; if not, write to the Free Software Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA 02111-1307, USA. */ #include "q_defs.h" #define STRINGTEMP_BUFFERS 16 #define STRINGTEMP_LENGTH 1024 static char pr_string_temp[STRINGTEMP_BUFFERS][STRINGTEMP_LENGTH]; static byte pr_string_tempindex = 0; static char *PR_GetTempString(void) { return pr_string_temp[(STRINGTEMP_BUFFERS - 1) & ++pr_string_tempindex]; } #define RETURN_EDICT(e) (G_PEdict(GBL_RETURN) = EdictProg(e)) #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 /* =============================================================================== BUILT-IN FUNCTIONS =============================================================================== */ static char *PF_VarString(int32_t first) { int32_t i; static char out[1024]; size_t s; out[0] = 0; s = 0; for(i = first; i < pr_argc; i++) { s = q_strlcat(out, G_String((GBL_PARM0 + i * 3)), sizeof(out)); if(s >= sizeof(out)) { Con_Warning("PF_VarString: overflow (string truncated)\n"); return out; } } if(s > 255) { if(!dev_overflows.varstring || dev_overflows.varstring + CONSOLE_RESPAM_TIME < realtime) { Con_DWarning("PF_VarString: %" PRIi32 " characters exceeds standard limit of 255 (max = %" PRIi32 ").\n", (int32_t)s, (int32_t)strsizeof(out)); dev_overflows.varstring = realtime; } } return out; } /* ================= PF_error This is a TERMINAL error, which will kill off the entire server. Dumps self. error(value) ================= */ static void PF_error(void) { char *s; edict_t *ed; s = PF_VarString(0); Con_Printf("======SERVER ERROR in %s:\n%s\n", PR_GetString(pr_xfunction->s_name), s); ed = ProgEdict(G_PEdict(GBL_self)); ED_Print(ed); Host_Error("Program error"); } /* ================= PF_objerror Dumps out self, then an error message. The program is aborted and self is removed, but the level can continue. objerror(value) ================= */ static void PF_objerror(void) { char *s; edict_t *ed; s = PF_VarString(0); Con_Printf("======OBJECT ERROR in %s:\n%s\n", PR_GetString(pr_xfunction->s_name), s); ed = ProgEdict(G_PEdict(GBL_self)); ED_Print(ed); ED_Free(ed); //Host_Error ("Program error"); //johnfitz -- by design, this should not be fatal } /* ============== PF_makevectors Writes new values for v_forward, v_up, and v_right based on angles makevectors(vector) ============== */ static void PF_makevectors(void) { AngleVectors(G_Vector(GBL_PARM0), G_Vector(GBL_v_forward), G_Vector(GBL_v_right), G_Vector(GBL_v_up)); } /* ================= PF_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 PF_setorigin(void) { edict_t *e; float *org; e = G_Edict(GBL_PARM0); org = G_Vector(GBL_PARM1); VectorCopy(org, ED_Vector(e, ED_origin)); SV_LinkEdict(e, false); } static void SetMinMaxSize(edict_t *e, float *minvec, float *maxvec, bool rotate) { float *angles; vec3_t rmin, rmax; float bounds[2][3]; float xvector[2], yvector[2]; float a; vec3_t base, transformed; int32_t i, j, k, l; for(i = 0; i < 3; i++) if(minvec[i] > maxvec[i]) PR_RunError("backwards mins/maxs"); rotate = false; // FIXME: implement rotation properly again if(!rotate) { VectorCopy(minvec, rmin); VectorCopy(maxvec, rmax); } else { // find min / max for rotations angles = ED_Vector(e, ED_angles); a = angles[1] / 180 * PI; xvector[0] = cos(a); xvector[1] = sin(a); yvector[0] = -sin(a); yvector[1] = cos(a); VectorCopy(minvec, bounds[0]); VectorCopy(maxvec, bounds[1]); rmin[0] = rmin[1] = rmin[2] = 9999; rmax[0] = rmax[1] = rmax[2] = -9999; for(i = 0; i <= 1; i++) { base[0] = bounds[i][0]; for(j = 0; j <= 1; j++) { base[1] = bounds[j][1]; for(k = 0; k <= 1; k++) { base[2] = bounds[k][2]; // transform the point transformed[0] = xvector[0] * base[0] + yvector[0] * base[1]; transformed[1] = xvector[1] * base[0] + yvector[1] * base[1]; transformed[2] = base[2]; for(l = 0; l < 3; l++) { if(transformed[l] < rmin[l]) rmin[l] = transformed[l]; if(transformed[l] > rmax[l]) rmax[l] = transformed[l]; } } } } } // set derived values VectorCopy(rmin, ED_Vector(e, ED_mins)); VectorCopy(rmax, ED_Vector(e, ED_maxs)); VectorSubtract(maxvec, minvec, ED_Vector(e, ED_size)); SV_LinkEdict(e, false); } /* ================= PF_setsize the size box is rotated by the current angle setsize (entity, minvector, maxvector) ================= */ static void PF_setsize(void) { edict_t *e; float *minvec, *maxvec; e = G_Edict(GBL_PARM0); minvec = G_Vector(GBL_PARM1); maxvec = G_Vector(GBL_PARM2); SetMinMaxSize(e, minvec, maxvec, false); } /* ================= PF_setmodel setmodel(entity, model) ================= */ static void PF_setmodel(void) { int32_t i; const char *m, **check; qmodel_t *mod; edict_t *e; e = G_Edict(GBL_PARM0); m = G_String(GBL_PARM1); // check to see if model was properly precached for(i = 0, check = sv.model_precache; *check; i++, check++) { if(!strcmp(*check, m)) break; } if(!*check) { PR_RunError("no precache: %s", m); } ED_RString(e, ED_model) = PR_SetEngineString(*check); ED_Float(e, ED_modelindex) = i; //SV_ModelIndex (m); mod = sv.models[(int32_t)ED_Float(e, ED_modelindex)]; // Mod_ForName (m, true); if(mod) //johnfitz -- correct physics cullboxes for bmodels { if(mod->type == mod_brush) SetMinMaxSize(e, mod->clipmins, mod->clipmaxs, true); else SetMinMaxSize(e, mod->mins, mod->maxs, true); } //johnfitz else SetMinMaxSize(e, vec3_origin, vec3_origin, true); } /* ================= PF_bprint broadcast print to everyone on server bprint(value) ================= */ static void PF_bprint(void) { char *s; s = PF_VarString(0); SV_BroadcastPrintf("%s", s); } /* ================= PF_sprint single print to a specific client sprint(clientent, value) ================= */ static void PF_sprint(void) { char *s; client_t *client; int32_t entnum; entnum = G_EdictNum(GBL_PARM0); s = PF_VarString(1); if(entnum < 1 || entnum > svs.maxclients) { Con_Printf("tried to sprint to a non-client\n"); return; } client = &svs.clients[entnum - 1]; MSG_WriteChar(&client->message, svc_print); MSG_WriteString(&client->message, s); } /* ================= PF_centerprint single print to a specific client centerprint(clientent, value) ================= */ static void PF_centerprint(void) { char *s; client_t *client; int32_t entnum; entnum = G_EdictNum(GBL_PARM0); s = PF_VarString(1); if(entnum < 1 || entnum > svs.maxclients) { Con_Printf("tried to sprint to a non-client\n"); return; } client = &svs.clients[entnum - 1]; MSG_WriteChar(&client->message, svc_centerprint); MSG_WriteString(&client->message, s); } /* ================= PF_normalize vector normalize(vector) ================= */ static void PF_normalize(void) { float *value1; vec3_t newvalue; double new_temp; value1 = G_Vector(GBL_PARM0); new_temp = (double)value1[0] * value1[0] + (double)value1[1] * value1[1] + (double)value1[2] * value1[2]; new_temp = sqrt(new_temp); if(new_temp == 0) newvalue[0] = newvalue[1] = newvalue[2] = 0; else { new_temp = 1 / new_temp; newvalue[0] = value1[0] * new_temp; newvalue[1] = value1[1] * new_temp; newvalue[2] = value1[2] * new_temp; } VectorCopy(newvalue, G_Vector(GBL_RETURN)); } /* ================= PF_vlen scalar vlen(vector) ================= */ static void PF_vlen(void) { float *value1; double new_temp; value1 = G_Vector(GBL_PARM0); new_temp = (double)value1[0] * value1[0] + (double)value1[1] * value1[1] + (double)value1[2] * value1[2]; new_temp = sqrt(new_temp); G_Float(GBL_RETURN) = new_temp; } /* ================= PF_vectoyaw float vectoyaw(vector) ================= */ static void PF_vectoyaw(void) { float *value1; float yaw; value1 = G_Vector(GBL_PARM0); if(value1[1] == 0 && value1[0] == 0) yaw = 0; else { yaw = (int32_t)(atan2(value1[1], value1[0]) * 180 / PI); if(yaw < 0) yaw += 360; } G_Float(GBL_RETURN) = yaw; } /* ================= PF_vectoangles vector vectoangles(vector) ================= */ static void PF_vectoangles(void) { float *value1; float forward; float yaw, pitch; value1 = G_Vector(GBL_PARM0); if(value1[1] == 0 && value1[0] == 0) { yaw = 0; if(value1[2] > 0) pitch = 90; else pitch = 270; } else { yaw = (int32_t)(atan2(value1[1], value1[0]) * 180 / PI); if(yaw < 0) yaw += 360; forward = sqrt(value1[0] * value1[0] + value1[1] * value1[1]); pitch = (int32_t)(atan2(value1[2], forward) * 180 / PI); if(pitch < 0) pitch += 360; } G_Float(GBL_RETURN + 0) = pitch; G_Float(GBL_RETURN + 1) = yaw; G_Float(GBL_RETURN + 2) = 0; } /* ================= PF_Random Returns a number from 0 <= num < 1 random() ================= */ static void PF_random(void) { float num; num = (rand() & 0x7fff) / ((float)0x7fff); G_Float(GBL_RETURN) = num; } /* ================= PF_particle particle(origin, color, count) ================= */ static void PF_particle(void) { float *org, *dir; float color; float count; org = G_Vector(GBL_PARM0); dir = G_Vector(GBL_PARM1); color = G_Float(GBL_PARM2); count = G_Float(GBL_PARM3); SV_StartParticle(org, dir, color, count); } /* ================= PF_ambientsound ================= */ static void PF_ambientsound(void) { const char *samp, **check; float *pos; float vol, attenuation; int32_t i, soundnum; int32_t large = false; //johnfitz -- PROTOCOL_FITZQUAKE pos = G_Vector(GBL_PARM0); samp = G_String(GBL_PARM1); vol = G_Float(GBL_PARM2); attenuation = G_Float(GBL_PARM3); // check to see if samp was properly precached for(soundnum = 0, check = sv.sound_precache; *check; check++, soundnum++) { if(!strcmp(*check, samp)) break; } if(!*check) { Con_Printf("no precache: %s\n", samp); return; } //johnfitz -- PROTOCOL_FITZQUAKE if(soundnum > 255) { if(sv.protocol == PROTOCOL_NETQUAKE) return; //don't send any info protocol can't support else large = true; } //johnfitz // add an svc_spawnambient command to the level signon packet //johnfitz -- PROTOCOL_FITZQUAKE if(large) MSG_WriteByte(&sv.signon, svc_spawnstaticsound2); else MSG_WriteByte(&sv.signon, svc_spawnstaticsound); //johnfitz for(i = 0; i < 3; i++) MSG_WriteCoord(&sv.signon, pos[i], sv.protocolflags); //johnfitz -- PROTOCOL_FITZQUAKE if(large) MSG_WriteShort(&sv.signon, soundnum); else MSG_WriteByte(&sv.signon, soundnum); //johnfitz MSG_WriteByte(&sv.signon, vol * 255); MSG_WriteByte(&sv.signon, attenuation * 64); } /* ================= PF_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 PF_sound(void) { const char *sample; int32_t channel; edict_t *entity; int32_t volume; float attenuation; entity = G_Edict(GBL_PARM0); channel = G_Float(GBL_PARM1); sample = G_String(GBL_PARM2); volume = G_Float(GBL_PARM3) * 255; attenuation = G_Float(GBL_PARM4); if(volume < 0 || volume > 255) Host_Error("SV_StartSound: volume = %" PRIi32, volume); if(attenuation < 0 || attenuation > 4) Host_Error("SV_StartSound: attenuation = %f", attenuation); if(channel < 0 || channel > 7) Host_Error("SV_StartSound: channel = %" PRIi32, channel); SV_StartSound(entity, channel, sample, volume, attenuation); } /* ================= PF_break break() ================= */ static void PF_break(void) { Con_Printf("break statement\n"); *(int32_t *) -4 = 0; // dump to debugger // PR_RunError ("break statement"); } /* ================= PF_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, tryents) ================= */ static void PF_traceline(void) { float *v1, *v2; trace_t trace; int32_t nomonsters; edict_t *ent; v1 = G_Vector(GBL_PARM0); v2 = G_Vector(GBL_PARM1); nomonsters = G_Float(GBL_PARM2); ent = G_Edict(GBL_PARM3); /* FIXME FIXME FIXME: Why do we hit this?? */ if(developer.value) { if(isnan(v1[0]) || isnan(v1[1]) || isnan(v1[2]) || isnan(v2[0]) || isnan(v2[1]) || isnan(v2[2])) { Con_Warning("NAN in traceline:\nv1(%f %f %f) v2(%f %f %f)\nentity %" PRIi32 "\n", v1[0], v1[1], v1[2], v2[0], v2[1], v2[2], NumForEdict(ent)); } } if(isnan(v1[0]) || isnan(v1[1]) || isnan(v1[2])) v1[0] = v1[1] = v1[2] = 0; if(isnan(v2[0]) || isnan(v2[1]) || isnan(v2[2])) v2[0] = v2[1] = v2[2] = 0; trace = SV_Move(v1, vec3_origin, vec3_origin, v2, nomonsters, ent); G_Float(GBL_trace_allsolid) = trace.allsolid; G_Float(GBL_trace_startsolid) = trace.startsolid; G_Float(GBL_trace_fraction) = trace.fraction; G_Float(GBL_trace_inwater) = trace.inwater; G_Float(GBL_trace_inopen) = trace.inopen; VectorCopy(trace.endpos, G_Vector(GBL_trace_endpos)); VectorCopy(trace.plane.normal, G_Vector(GBL_trace_plane_normal)); G_Float(GBL_trace_plane_dist) = trace.plane.dist; if(trace.ent) G_PEdict(GBL_trace_ent) = EdictProg(trace.ent); else G_PEdict(GBL_trace_ent) = EdictProg(sv.edicts); } //============================================================================ static byte *checkpvs; //ericw -- changed to malloc static int32_t checkpvs_capacity; static int32_t PF_newcheckclient(int32_t check) { int32_t i; byte *pvs; edict_t *ent; mleaf_t *leaf; vec3_t org; int32_t pvsbytes; // cycle to the next one if(check < 1) check = 1; if(check > svs.maxclients) check = svs.maxclients; if(check == svs.maxclients) i = 1; else i = check + 1; for(; ; i++) { if(i == svs.maxclients + 1) i = 1; ent = EdictNum(i); if(i == check) break; // didn't find anything else if(ent->free) continue; if(ED_Float(ent, ED_health) <= 0) continue; if((int32_t)ED_Float(ent, ED_flags) & FL_NOTARGET) continue; // anything that is a client, or has a client as an enemy break; } // get the PVS for the entity VectorAdd(ED_Vector(ent, ED_origin), ED_Vector(ent, ED_view_ofs), org); leaf = Mod_PointInLeaf(org, sv.worldmodel); pvs = Mod_LeafPVS(leaf, sv.worldmodel); pvsbytes = (sv.worldmodel->numleafs + 7) >> 3; if(checkpvs == NULL || pvsbytes > checkpvs_capacity) { checkpvs_capacity = pvsbytes; checkpvs = (byte *) realloc(checkpvs, checkpvs_capacity); if(!checkpvs) Sys_Error("PF_newcheckclient: realloc() failed on %" PRIi32 " bytes", checkpvs_capacity); } memcpy(checkpvs, pvs, pvsbytes); return i; } /* ================= PF_checkclient Returns a client (or object that has a client enemy) that would be a valid target. If there are more than one valid options, 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 () ================= */ #define MAX_CHECK 16 static int32_t c_invis, c_notvis; static void PF_checkclient(void) { edict_t *ent, *self; mleaf_t *leaf; int32_t l; vec3_t view; // find a new check if on a new frame if(sv.time - sv.lastchecktime >= 0.1) { sv.lastcheck = PF_newcheckclient(sv.lastcheck); sv.lastchecktime = sv.time; } // return check if it might be visible ent = EdictNum(sv.lastcheck); if(ent->free || ED_Float(ent, ED_health) <= 0) { RETURN_EDICT(sv.edicts); return; } // if current entity can't possibly see the check entity, return 0 self = ProgEdict(G_PEdict(GBL_self)); VectorAdd(ED_Vector(self, ED_origin), ED_Vector(self, ED_view_ofs), view); leaf = Mod_PointInLeaf(view, sv.worldmodel); l = (leaf - sv.worldmodel->leafs) - 1; if((l < 0) || !(checkpvs[l >> 3] & (1 << (l & 7)))) { c_notvis++; RETURN_EDICT(sv.edicts); return; } // might be able to see it c_invis++; RETURN_EDICT(ent); } //============================================================================ /* ================= PF_stuffcmd Sends text over to the client's execution buffer stuffcmd (clientent, value) ================= */ static void PF_stuffcmd(void) { int32_t entnum; const char *str; client_t *old; entnum = G_EdictNum(GBL_PARM0); if(entnum < 1 || entnum > svs.maxclients) PR_RunError("Parm 0 not a client"); str = G_String(GBL_PARM1); old = host_client; host_client = &svs.clients[entnum - 1]; Host_ClientCommands("%s", str); host_client = old; } /* ================= PF_localcmd Sends text over to the client's execution buffer localcmd (string) ================= */ static void PF_localcmd(void) { const char *str; str = G_String(GBL_PARM0); Cbuf_AddText(str); } /* ================= PF_cvar float cvar (string) ================= */ static void PF_cvar(void) { const char *str; str = G_String(GBL_PARM0); G_Float(GBL_RETURN) = Cvar_VariableValue(str); } /* ================= PF_cvar_set float cvar (string) ================= */ static void PF_cvar_set(void) { const char *var, *val; var = G_String(GBL_PARM0); val = G_String(GBL_PARM1); Cvar_Set(var, val); } /* ================= PF_findradius Returns a chain of entities that have origins within a spherical area findradius (origin, radius) ================= */ static void PF_findradius(void) { edict_t *ent, *chain; float rad; float *org; vec3_t eorg; int32_t i, j; chain = (edict_t *)sv.edicts; org = G_Vector(GBL_PARM0); rad = G_Float(GBL_PARM1); ent = NextEdict(sv.edicts); for(i = 1; i < sv.num_edicts; i++, ent = NextEdict(ent)) { if(ent->free) continue; if(ED_Float(ent, ED_solid) == SOLID_NOT) continue; for(j = 0; j < 3; j++) eorg[j] = org[j] - (ED_Vector(ent, ED_origin)[j] + (ED_Vector(ent, ED_mins)[j] + ED_Vector(ent, ED_maxs)[j]) * 0.5); if(VectorLength(eorg) > rad) continue; ED_PEdict(ent, ED_chain) = EdictProg(chain); chain = ent; } RETURN_EDICT(chain); } /* ========= PF_dprint ========= */ static void PF_dprint(void) { Con_DPrintf("%s", PF_VarString(0)); } static void PF_ftos(void) { float v; char *s; v = G_Float(GBL_PARM0); s = PR_GetTempString(); if(v == (int32_t)v) sprintf(s, "%" PRIi32, (int32_t)v); else sprintf(s, "%5.1f", v); G_RString(GBL_RETURN) = PR_SetEngineString(s); } static void PF_fabs(void) { float v; v = G_Float(GBL_PARM0); G_Float(GBL_RETURN) = fabs(v); } static void PF_vtos(void) { char *s; s = PR_GetTempString(); sprintf(s, "'%5.1f %5.1f %5.1f'", G_Vector(GBL_PARM0)[0], G_Vector(GBL_PARM0)[1], G_Vector(GBL_PARM0)[2]); G_RString(GBL_RETURN) = PR_SetEngineString(s); } static void PF_spawn(void) { edict_t *ed; ed = ED_Alloc(); RETURN_EDICT(ed); } static void PF_remove(void) { edict_t *ed; ed = G_Edict(GBL_PARM0); ED_Free(ed); } // entity (entity start, .string field, string match) find = #5; static void PF_find(void) { int32_t e; pfield_t f; const char *s, *t; edict_t *ed; e = G_EdictNum(GBL_PARM0); f = G_PField(GBL_PARM1); s = G_String(GBL_PARM2); if(!s) PR_RunError("PF_find: bad search string"); for(e++ ; e < sv.num_edicts ; e++) { ed = EdictNum(e); if(ed->free) continue; t = ED_String(ed, f); if(!t) continue; if(!strcmp(t, s)) { RETURN_EDICT(ed); return; } } RETURN_EDICT(sv.edicts); } static void PR_CheckEmptyString(const char *s) { if(s[0] <= ' ') PR_RunError("Bad string"); } static void PF_precache_file(void) { // precache_file is only used to copy files with qcc, it does nothing G_RString(GBL_RETURN) = G_RString(GBL_PARM0); } static void PF_precache_sound(void) { const char *s; int32_t i; if(sv.state != ss_loading) PR_RunError("PF_Precache_*: Precache can only be done in spawn functions"); s = G_String(GBL_PARM0); G_RString(GBL_RETURN) = G_RString(GBL_PARM0); PR_CheckEmptyString(s); for(i = 0; i < MAX_SOUNDS; i++) { if(!sv.sound_precache[i]) { sv.sound_precache[i] = s; return; } if(!strcmp(sv.sound_precache[i], s)) return; } PR_RunError("PF_precache_sound: overflow"); } static void PF_precache_model(void) { const char *s; int32_t i; if(sv.state != ss_loading) PR_RunError("PF_Precache_*: Precache can only be done in spawn functions"); s = G_String(GBL_PARM0); G_RString(GBL_RETURN) = G_RString(GBL_PARM0); PR_CheckEmptyString(s); for(i = 0; i < MAX_MODELS; i++) { if(!sv.model_precache[i]) { sv.model_precache[i] = s; sv.models[i] = Mod_ForName(s, true); return; } if(!strcmp(sv.model_precache[i], s)) return; } PR_RunError("PF_precache_model: overflow"); } static void PF_coredump(void) { ED_PrintEdicts(); } static void PF_traceon(void) { pr_trace = true; } static void PF_traceoff(void) { pr_trace = false; } static void PF_eprint(void) { ED_PrintNum(G_EdictNum(GBL_PARM0)); } /* =============== PF_walkmove float(float yaw, float dist) walkmove =============== */ static void PF_walkmove(void) { edict_t *ent; float yaw, dist; vec3_t move; dfunction_t *oldf; int32_t oldself; ent = ProgEdict(G_PEdict(GBL_self)); yaw = G_Float(GBL_PARM0); dist = G_Float(GBL_PARM1); if(!((int32_t)ED_Float(ent, ED_flags) & (FL_ONGROUND | FL_FLY | FL_SWIM))) { G_Float(GBL_RETURN) = 0; return; } yaw = yaw * 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 functions oldf = pr_xfunction; oldself = G_PEdict(GBL_self); G_Float(GBL_RETURN) = SV_movestep(ent, move, true); // restore program state pr_xfunction = oldf; G_PEdict(GBL_self) = oldself; } /* =============== PF_droptofloor void() droptofloor =============== */ static void PF_droptofloor(void) { edict_t *ent; vec3_t end; trace_t trace; ent = ProgEdict(G_PEdict(GBL_self)); VectorCopy(ED_Vector(ent, ED_origin), end); end[2] -= 256; trace = SV_Move(ED_Vector(ent, ED_origin), ED_Vector(ent, ED_mins), ED_Vector(ent, ED_maxs), end, false, ent); if(trace.fraction == 1 || trace.allsolid) G_Float(GBL_RETURN) = 0; else { VectorCopy(trace.endpos, ED_Vector(ent, ED_origin)); SV_LinkEdict(ent, false); ED_Float(ent, ED_flags) = (int32_t)ED_Float(ent, ED_flags) | FL_ONGROUND; ED_PEdict(ent, ED_groundentity) = EdictProg(trace.ent); G_Float(GBL_RETURN) = 1; } } /* =============== PF_lightstyle void(float style, string value) lightstyle =============== */ static void PF_lightstyle(void) { int32_t style; const char *val; client_t *client; int32_t j; style = G_Float(GBL_PARM0); val = G_String(GBL_PARM1); // bounds check to avoid clobbering sv struct if(style < 0 || style >= MAX_LIGHTSTYLES) { Con_DWarning("PF_lightstyle: invalid style %" PRIi32 "\n", style); return; } // change the string in sv sv.lightstyles[style] = val; // 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->spawned) { MSG_WriteChar(&client->message, svc_lightstyle); MSG_WriteChar(&client->message, style); MSG_WriteString(&client->message, val); } } } static void PF_rint(void) { float f; f = G_Float(GBL_PARM0); if(f > 0) G_Float(GBL_RETURN) = (int32_t)(f + 0.5); else G_Float(GBL_RETURN) = (int32_t)(f - 0.5); } static void PF_floor(void) { G_Float(GBL_RETURN) = floor(G_Float(GBL_PARM0)); } static void PF_ceil(void) { G_Float(GBL_RETURN) = ceil(G_Float(GBL_PARM0)); } /* ============= PF_checkbottom ============= */ static void PF_checkbottom(void) { edict_t *ent; ent = G_Edict(GBL_PARM0); G_Float(GBL_RETURN) = SV_CheckBottom(ent); } /* ============= PF_pointcontents ============= */ static void PF_pointcontents(void) { float *v; v = G_Vector(GBL_PARM0); G_Float(GBL_RETURN) = SV_PointContents(v); } /* ============= PF_nextent entity nextent(entity) ============= */ static void PF_nextent(void) { int32_t i; edict_t *ent; i = G_EdictNum(GBL_PARM0); while(1) { i++; if(i == sv.num_edicts) { RETURN_EDICT(sv.edicts); return; } ent = EdictNum(i); if(!ent->free) { RETURN_EDICT(ent); return; } } } /* ============= PF_aim Pick a vector for the player to shoot along vector aim(entity, missilespeed) ============= */ cvar_t sv_aim = {"sv_aim", "1", CVAR_NONE}; // ericw -- turn autoaim off by default. was 0.93 static void PF_aim(void) { edict_t *ent, *check, *bestent; vec3_t start, dir, end, bestdir; int32_t i, j; trace_t tr; float dist, bestdist; float speed; ent = G_Edict(GBL_PARM0); speed = G_Float(GBL_PARM1); (void) speed; /* variable set but not used */ VectorCopy(ED_Vector(ent, ED_origin), start); start[2] += 20; // try sending a trace straight VectorCopy(G_Vector(GBL_v_forward), dir); VectorMA(start, 2048, dir, end); tr = SV_Move(start, vec3_origin, vec3_origin, end, false, ent); if(tr.ent && ED_Float(tr.ent, ED_takedamage) == DAMAGE_AIM && (!teamplay.value || ED_Float(ent, ED_team) <= 0 || ED_Float(ent, ED_team) != ED_Float(tr.ent, ED_team))) { VectorCopy(G_Vector(GBL_v_forward), G_Vector(GBL_RETURN)); return; } // try all possible entities VectorCopy(dir, bestdir); bestdist = sv_aim.value; bestent = NULL; check = NextEdict(sv.edicts); for(i = 1; i < sv.num_edicts; i++, check = NextEdict(check)) { if(ED_Float(check, ED_takedamage) != DAMAGE_AIM) continue; if(check == ent) continue; if(teamplay.value && ED_Float(ent, ED_team) > 0 && ED_Float(ent, ED_team) == ED_Float(check, ED_team)) continue; // don't aim at teammate for(j = 0; j < 3; j++) end[j] = ED_Vector(check, ED_origin)[j] + 0.5 * (ED_Vector(check, ED_mins)[j] + ED_Vector(check, ED_maxs)[j]); VectorSubtract(end, start, dir); VectorNormalize(dir); dist = DotProduct(dir, G_Vector(GBL_v_forward)); if(dist < bestdist) continue; // to far to turn tr = SV_Move(start, vec3_origin, vec3_origin, end, false, ent); if(tr.ent == check) { // can shoot at this one bestdist = dist; bestent = check; } } if(bestent) { VectorSubtract(ED_Vector(bestent, ED_origin), ED_Vector(ent, ED_origin), dir); dist = DotProduct(dir, G_Vector(GBL_v_forward)); VectorScale(G_Vector(GBL_v_forward), dist, end); end[2] = dir[2]; VectorNormalize(end); VectorCopy(end, G_Vector(GBL_RETURN)); } else { VectorCopy(bestdir, G_Vector(GBL_RETURN)); } } /* ============== PF_changeyaw This was a major timewaster so it was converted to C ============== */ void PF_changeyaw(void) { edict_t *ent; float ideal, current, move, speed; ent = ProgEdict(G_PEdict(GBL_self)); current = anglemod(ED_Vector(ent, ED_angles)[1]); ideal = ED_Float(ent, ED_ideal_yaw); speed = ED_Float(ent, ED_yaw_speed); if(current == ideal) return; move = ideal - current; if(ideal > current) { if(move >= 180) move = move - 360; } else { if(move <= -180) move = move + 360; } if(move > 0) { if(move > speed) move = speed; } else { if(move < -speed) move = -speed; } ED_Vector(ent, ED_angles)[1] = anglemod(current + move); } /* =============================================================================== MESSAGE WRITING =============================================================================== */ static sizebuf_t *WriteDest(void) { int32_t entnum; int32_t dest; edict_t *ent; dest = G_Float(GBL_PARM0); switch(dest) { case MSG_BROADCAST: return &sv.datagram; case MSG_ONE: ent = ProgEdict(G_PEdict(GBL_msg_entity)); entnum = NumForEdict(ent); if(entnum < 1 || entnum > svs.maxclients) PR_RunError("WriteDest: not a client"); return &svs.clients[entnum - 1].message; case MSG_ALL: return &sv.reliable_datagram; case MSG_INIT: return &sv.signon; default: PR_RunError("WriteDest: bad destination"); break; } return NULL; } static void PF_WriteByte(void) { MSG_WriteByte(WriteDest(), G_Float(GBL_PARM1)); } static void PF_WriteChar(void) { MSG_WriteChar(WriteDest(), G_Float(GBL_PARM1)); } static void PF_WriteShort(void) { MSG_WriteShort(WriteDest(), G_Float(GBL_PARM1)); } static void PF_WriteLong(void) { MSG_WriteLong(WriteDest(), G_Float(GBL_PARM1)); } static void PF_WriteAngle(void) { MSG_WriteAngle(WriteDest(), G_Float(GBL_PARM1), sv.protocolflags); } static void PF_WriteCoord(void) { MSG_WriteCoord(WriteDest(), G_Float(GBL_PARM1), sv.protocolflags); } static void PF_WriteString(void) { MSG_WriteString(WriteDest(), G_String(GBL_PARM1)); } static void PF_WriteEntity(void) { MSG_WriteShort(WriteDest(), G_EdictNum(GBL_PARM1)); } //============================================================================= static void PF_makestatic(void) { edict_t *ent; int32_t i; int32_t bits = 0; //johnfitz -- PROTOCOL_FITZQUAKE ent = G_Edict(GBL_PARM0); //johnfitz -- don't send invisible static entities if(ent->alpha == ENTALPHA_ZERO) { ED_Free(ent); return; } //johnfitz //johnfitz -- PROTOCOL_FITZQUAKE if(sv.protocol == PROTOCOL_NETQUAKE) { if(SV_ModelIndex(ED_String(ent, ED_model)) & 0xFF00 || (int32_t)(ED_Float(ent, ED_frame)) & 0xFF00) { ED_Free(ent); return; //can't display the correct model & frame, so don't show it at all } } else { if(SV_ModelIndex(ED_String(ent, ED_model)) & 0xFF00) bits |= B_LARGEMODEL; if((int32_t)(ED_Float(ent, ED_frame)) & 0xFF00) bits |= B_LARGEFRAME; if(ent->alpha != ENTALPHA_DEFAULT) bits |= B_ALPHA; } if(bits) { MSG_WriteByte(&sv.signon, svc_spawnstatic2); MSG_WriteByte(&sv.signon, bits); } else MSG_WriteByte(&sv.signon, svc_spawnstatic); if(bits & B_LARGEMODEL) MSG_WriteShort(&sv.signon, SV_ModelIndex(ED_String(ent, ED_model))); else MSG_WriteByte(&sv.signon, SV_ModelIndex(ED_String(ent, ED_model))); if(bits & B_LARGEFRAME) MSG_WriteShort(&sv.signon, ED_Float(ent, ED_frame)); else MSG_WriteByte(&sv.signon, ED_Float(ent, ED_frame)); //johnfitz MSG_WriteByte(&sv.signon, ED_Float(ent, ED_colormap)); MSG_WriteByte(&sv.signon, ED_Float(ent, ED_skin)); for(i = 0; i < 3; i++) { MSG_WriteCoord(&sv.signon, ED_Vector(ent, ED_origin)[i], sv.protocolflags); MSG_WriteAngle(&sv.signon, ED_Vector(ent, ED_angles)[i], sv.protocolflags); } //johnfitz -- PROTOCOL_FITZQUAKE if(bits & B_ALPHA) MSG_WriteByte(&sv.signon, ent->alpha); //johnfitz // throw the entity away now ED_Free(ent); } //============================================================================= /* ============== PF_setspawnparms ============== */ static void PF_setspawnparms(void) { edict_t *ent; int32_t i; client_t *client; ent = G_Edict(GBL_PARM0); i = NumForEdict(ent); if(i < 1 || i > svs.maxclients) PR_RunError("Entity is not a client"); // copy spawn parms out of the client_t client = svs.clients + (i - 1); for(i = 0; i < NUM_SPAWN_PARMS; i++) (&G_Float(GBL_parm1))[i] = client->spawn_parms[i]; } /* ============== PF_changelevel ============== */ static void PF_changelevel(void) { const char *s; // make sure we don't issue two changelevels if(svs.changelevel_issued) return; svs.changelevel_issued = true; s = G_String(GBL_PARM0); Cbuf_AddText(va("changelevel %s\n", s)); } static void PF_fixme(void) { PR_RunError("unimplemented builtin"); } builtin_t pr_builtins[] = { PF_fixme, PF_makevectors, // void(entity e) makevectors = #1 PF_setorigin, // void(entity e, vector o) setorigin = #2 PF_setmodel, // void(entity e, string m) setmodel = #3 PF_setsize, // void(entity e, vector min, vector max) setsize = #4 PF_fixme, // void(entity e, vector min, vector max) setabssize = #5 PF_break, // void() break = #6 PF_random, // float() random = #7 PF_sound, // void(entity e, float chan, string samp) sound = #8 PF_normalize, // vector(vector v) normalize = #9 PF_error, // void(string e) error = #10 PF_objerror, // void(string e) objerror = #11 PF_vlen, // float(vector v) vlen = #12 PF_vectoyaw, // float(vector v) vectoyaw = #13 PF_spawn, // entity() spawn = #14 PF_remove, // void(entity e) remove = #15 PF_traceline, // float(vector v1, vector v2, float tryents) traceline = #16 PF_checkclient, // entity() clientlist = #17 PF_find, // entity(entity start, .string fld, string match) find = #18 PF_precache_sound, // void(string s) precache_sound = #19 PF_precache_model, // void(string s) precache_model = #20 PF_stuffcmd, // void(entity client, string s)stuffcmd = #21 PF_findradius, // entity(vector org, float rad) findradius = #22 PF_bprint, // void(string s) bprint = #23 PF_sprint, // void(entity client, string s) sprint = #24 PF_dprint, // void(string s) dprint = #25 PF_ftos, // void(string s) ftos = #26 PF_vtos, // void(string s) vtos = #27 PF_coredump, PF_traceon, PF_traceoff, PF_eprint, // void(entity e) debug print an entire entity PF_walkmove, // float(float yaw, float dist) walkmove PF_fixme, // float(float yaw, float dist) walkmove PF_droptofloor, PF_lightstyle, PF_rint, PF_floor, PF_ceil, PF_fixme, PF_checkbottom, PF_pointcontents, PF_fixme, PF_fabs, PF_aim, PF_cvar, PF_localcmd, PF_nextent, PF_particle, PF_changeyaw, PF_fixme, PF_vectoangles, PF_WriteByte, PF_WriteChar, PF_WriteShort, PF_WriteLong, PF_WriteCoord, PF_WriteAngle, PF_WriteString, PF_WriteEntity, PF_fixme, PF_fixme, PF_fixme, PF_fixme, PF_fixme, PF_fixme, PF_fixme, SV_MoveToGoal, PF_precache_file, PF_makestatic, PF_changelevel, PF_fixme, PF_cvar_set, PF_centerprint, PF_ambientsound, PF_precache_model, PF_precache_sound, // precache_sound2 is different only for qcc PF_precache_file, PF_setspawnparms }; int32_t pr_numbuiltins = arraysizeof(pr_builtins);