/* * * 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 * * In addition, as a special exception, the author gives permission to * link the code of this program with the Half-Life Game Engine ("HL * Engine") and Modified Game Libraries ("MODs") developed by Valve, * L.L.C ("Valve"). You must obey the GNU General Public License in all * respects for all of the code used other than the HL Engine and MODs * from Valve. If you modify this file, you may extend this exception * to your version of the file, but you are not obligated to do so. If * you do not wish to do so, delete this exception statement from your * version. * */ #include "precompiled.h" vec_t gHullMins[4][3] = { { 0.0f, 0.0f, 0.0f }, { -16.0f, -16.0f, -36.0f }, { -32.0f, -32.0f, -32.0f }, { -16.0f, -16.0f, -18.0f }, }; vec_t gHullMaxs[4][3] = { { 0.0f, 0.0f, 0.0f }, { 16.0f, 16.0f, 36.0f }, { 32.0f, 32.0f, 32.0f }, { 16.0f, 16.0f, 18.0f }, }; unsigned char gMsgData[512]; edict_t *gMsgEntity; int gMsgDest; int gMsgType; qboolean gMsgStarted; vec3_t gMsgOrigin; int32 idum; int g_groupop; int g_groupmask; unsigned char checkpvs[1024]; int c_invis; int c_notvis; // TODO: Move to sv_phys.cpp vec3_t vec_origin; int r_visframecount; /* * Globals initialization */ #ifndef HOOK_ENGINE sizebuf_t gMsgBuffer = { "MessageBegin/End", 0, gMsgData, sizeof(gMsgData), 0 }; #else // HOOK_ENGINE sizebuf_t gMsgBuffer; #endif // HOOK_ENGINE void EXT_FUNC PF_makevectors_I(const float *rgflVector) { AngleVectors(rgflVector, gGlobalVariables.v_forward, gGlobalVariables.v_right, gGlobalVariables.v_up); } float EXT_FUNC PF_Time(void) { return Sys_FloatTime(); } void EXT_FUNC PF_setorigin_I(edict_t *e, const float *org) { if (!e) return; e->v.origin[0] = org[0]; e->v.origin[1] = org[1]; e->v.origin[2] = org[2]; SV_LinkEdict(e, FALSE); } void EXT_FUNC SetMinMaxSize(edict_t *e, const float *min, const float *max, qboolean rotate) { for (int i = 0; i < 3; i++) { if (min[i] > max[i]) Host_Error("%s: backwards mins/maxs", __func__); } e->v.mins[0] = min[0]; e->v.mins[1] = min[1]; e->v.mins[2] = min[2]; e->v.maxs[0] = max[0]; e->v.maxs[1] = max[1]; e->v.maxs[2] = max[2]; e->v.size[0] = max[0] - min[0]; e->v.size[1] = max[1] - min[1]; e->v.size[2] = max[2] - min[2]; SV_LinkEdict(e, 0); } void EXT_FUNC PF_setsize_I(edict_t *e, const float *rgflMin, const float *rgflMax) { SetMinMaxSize(e, rgflMin, rgflMax, 0); } void EXT_FUNC PF_setmodel_I(edict_t *e, const char *m) { const char** check = &g_psv.model_precache[0]; int i = 0; #ifdef REHLDS_CHECKS for (; *check && i < MAX_MODELS; i++, check++) #else for (; *check; i++, check++) #endif { //use case-sensitive names to increase performance #ifdef REHLDS_FIXES if (!Q_strcmp(*check, m)) #else if (!Q_stricmp(*check, m)) #endif { e->v.modelindex = i; model_t *mod = g_psv.models[i]; #ifdef REHLDS_FIXES e->v.model = *check - pr_strings; #else // REHLDS_FIXES e->v.model = m - pr_strings; #endif // REHLDS_FIXES if (mod) { SetMinMaxSize(e, mod->mins, mod->maxs, 1); } else { SetMinMaxSize(e, vec3_origin, vec3_origin, 1); } return; } } Host_Error("%s: no precache: %s\n", __func__, m); } int EXT_FUNC PF_modelindex(const char *pstr) { return SV_ModelIndex(pstr); } int EXT_FUNC ModelFrames(int modelIndex) { if (modelIndex <= 0 || modelIndex >= MAX_MODELS) { Con_DPrintf("Bad sprite index!\n"); return 1; } return ModelFrameCount(g_psv.models[modelIndex]); } void EXT_FUNC PF_bprint(char *s) { SV_BroadcastPrintf("%s", s); } void EXT_FUNC PF_sprint(char *s, int entnum) { if (entnum <= 0 || entnum > g_psvs.maxclients) { Con_Printf("tried to sprint to a non-client\n"); return; } client_t* client = &g_psvs.clients[entnum - 1]; if (!client->fakeclient) { #ifdef REHLDS_FIXES MSG_WriteByte(&client->netchan.message, svc_print); #else // REHLDS_FIXES MSG_WriteChar(&client->netchan.message, svc_print); #endif // REHLDS_FIXES MSG_WriteString(&client->netchan.message, s); } } void EXT_FUNC ServerPrint(const char *szMsg) { Con_Printf("%s", szMsg); } void EXT_FUNC ClientPrintf(edict_t *pEdict, PRINT_TYPE ptype, const char *szMsg) { client_t *client; int entnum; entnum = NUM_FOR_EDICT(pEdict); if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("tried to sprint to a non-client\n"); return; } client = &g_psvs.clients[entnum - 1]; if (client->fakeclient) return; switch (ptype) { case print_center: MSG_WriteChar(&client->netchan.message, svc_centerprint); MSG_WriteString(&client->netchan.message, szMsg); break; case print_chat: case print_console: MSG_WriteByte(&client->netchan.message, svc_print); MSG_WriteString(&client->netchan.message, szMsg); break; default: Con_Printf("invalid PRINT_TYPE %i\n", ptype); break; } } float EXT_FUNC PF_vectoyaw_I(const float *rgflVector) { float yaw = 0.0f; if (rgflVector[1] == 0.0f && rgflVector[0] == 0.0f) return 0.0f; yaw = (float)(int)floor(atan2((double)rgflVector[1], (double)rgflVector[0]) * 180.0 / M_PI); if (yaw < 0.0) yaw = yaw + 360.0; return yaw; } void EXT_FUNC PF_vectoangles_I(const float *rgflVectorIn, float *rgflVectorOut) { VectorAngles(rgflVectorIn, rgflVectorOut); } void EXT_FUNC PF_particle_I(const float *org, const float *dir, float color, float count) { SV_StartParticle(org, dir, color, count); } void EXT_FUNC PF_ambientsound_I(edict_t *entity, float *pos, const char *samp, float vol, float attenuation, int fFlags, int pitch) { int i; int soundnum; int ent; sizebuf_t *pout; if (samp[0] == '!') { fFlags |= SND_FL_SENTENCE; soundnum = Q_atoi(samp + 1); if (soundnum >= CVOXFILESENTENCEMAX) { Con_Printf("invalid sentence number: %s", &samp[1]); return; } } else { for (i = 0; i < MAX_SOUNDS; i++) { if (g_psv.sound_precache[i] && !Q_stricmp(g_psv.sound_precache[i], samp)) { soundnum = i; break; } } if (i == MAX_SOUNDS) { Con_Printf("no precache: %s\n", samp); return; } } ent = NUM_FOR_EDICT(entity); pout = &g_psv.signon; if (!(fFlags & SND_FL_SPAWNING)) pout = &g_psv.datagram; MSG_WriteByte(pout, svc_spawnstaticsound); MSG_WriteCoord(pout, pos[0]); MSG_WriteCoord(pout, pos[1]); MSG_WriteCoord(pout, pos[2]); MSG_WriteShort(pout, soundnum); MSG_WriteByte(pout, (vol * 255.0)); MSG_WriteByte(pout, (attenuation * 64.0)); MSG_WriteShort(pout, ent); MSG_WriteByte(pout, pitch); MSG_WriteByte(pout, fFlags); } void EXT_FUNC PF_sound_I(edict_t *entity, int channel, const char *sample, float volume, float attenuation, int fFlags, int pitch) { if (volume < 0.0 || volume > 255.0) Sys_Error("%s: volume = %i", __func__, volume); if (attenuation < 0.0 || attenuation > 4.0) Sys_Error("%s: attenuation = %f", __func__, attenuation); if (channel < 0 || channel > 7) Sys_Error("%s: channel = %i", __func__, channel); if (pitch < 0 || pitch > 255) Sys_Error("%s: pitch = %i", __func__, pitch); SV_StartSound(0, entity, channel, sample, (int)(volume * 255), attenuation, fFlags, pitch); } void EXT_FUNC PF_traceline_Shared(const float *v1, const float *v2, int nomonsters, edict_t *ent) { #ifdef REHLDS_OPT_PEDANTIC trace_t trace = SV_Move_Point(v1, v2, nomonsters, ent); #else // REHLDS_OPT_PEDANTIC trace_t trace = SV_Move(v1, vec3_origin, vec3_origin, v2, nomonsters, ent, 0); #endif // REHLDS_OPT_PEDANTIC gGlobalVariables.trace_flags = 0; SV_SetGlobalTrace(&trace); } void EXT_FUNC PF_traceline_DLL(const float *v1, const float *v2, int fNoMonsters, edict_t *pentToSkip, TraceResult *ptr) { PF_traceline_Shared(v1, v2, fNoMonsters, pentToSkip ? pentToSkip : &g_psv.edicts[0]); ptr->fAllSolid = (int)gGlobalVariables.trace_allsolid; ptr->fStartSolid = (int) gGlobalVariables.trace_startsolid; ptr->fInOpen = (int)gGlobalVariables.trace_inopen; ptr->fInWater = (int)gGlobalVariables.trace_inwater; ptr->flFraction = gGlobalVariables.trace_fraction; ptr->flPlaneDist = gGlobalVariables.trace_plane_dist; ptr->pHit = gGlobalVariables.trace_ent; ptr->vecEndPos[0] = gGlobalVariables.trace_endpos[0]; ptr->vecEndPos[1] = gGlobalVariables.trace_endpos[1]; ptr->vecEndPos[2] = gGlobalVariables.trace_endpos[2]; ptr->vecPlaneNormal[0] = gGlobalVariables.trace_plane_normal[0]; ptr->vecPlaneNormal[1] = gGlobalVariables.trace_plane_normal[1]; ptr->vecPlaneNormal[2] = gGlobalVariables.trace_plane_normal[2]; ptr->iHitgroup = gGlobalVariables.trace_hitgroup; } void EXT_FUNC TraceHull(const float *v1, const float *v2, int fNoMonsters, int hullNumber, edict_t *pentToSkip, TraceResult *ptr) { hullNumber = hullNumber; if (hullNumber < 0 || hullNumber > 3) hullNumber = 0; trace_t trace = SV_Move(v1, gHullMins[hullNumber], gHullMaxs[hullNumber], v2, fNoMonsters, pentToSkip, 0); ptr->fAllSolid = trace.allsolid; ptr->fStartSolid = trace.startsolid; ptr->fInOpen = trace.inopen; ptr->fInWater = trace.inwater; ptr->flFraction = trace.fraction; ptr->flPlaneDist = trace.plane.dist; ptr->pHit = trace.ent; ptr->iHitgroup = trace.hitgroup; ptr->vecEndPos[0] = trace.endpos[0]; ptr->vecEndPos[1] = trace.endpos[1]; ptr->vecEndPos[2] = trace.endpos[2]; ptr->vecPlaneNormal[0] = trace.plane.normal[0]; ptr->vecPlaneNormal[1] = trace.plane.normal[1]; ptr->vecPlaneNormal[2] = trace.plane.normal[2]; } void EXT_FUNC TraceSphere(const float *v1, const float *v2, int fNoMonsters, float radius, edict_t *pentToSkip, TraceResult *ptr) { Sys_Error("%s: TraceSphere not yet implemented!\n", __func__); } void EXT_FUNC TraceModel(const float *v1, const float *v2, int hullNumber, edict_t *pent, TraceResult *ptr) { int oldMovetype, oldSolid; if (hullNumber < 0 || hullNumber > 3) hullNumber = 0; model_t* pmodel = g_psv.models[pent->v.modelindex]; if (pmodel && pmodel->type == mod_brush) { oldMovetype = pent->v.movetype; oldSolid = pent->v.solid; pent->v.solid = SOLID_BSP; pent->v.movetype = MOVETYPE_PUSH; } trace_t trace = SV_ClipMoveToEntity(pent, v1, gHullMins[hullNumber], gHullMaxs[hullNumber], v2); if (pmodel && pmodel->type == mod_brush) { pent->v.solid = oldSolid; pent->v.movetype = oldMovetype; } ptr->fAllSolid = trace.allsolid; ptr->fStartSolid = trace.startsolid; ptr->fInOpen = trace.inopen; ptr->fInWater = trace.inwater; ptr->flFraction = trace.fraction; ptr->flPlaneDist = trace.plane.dist; ptr->pHit = trace.ent; ptr->iHitgroup = trace.hitgroup; ptr->vecEndPos[0] = trace.endpos[0]; ptr->vecEndPos[1] = trace.endpos[1]; ptr->vecEndPos[2] = trace.endpos[2]; ptr->vecPlaneNormal[0] = trace.plane.normal[0]; ptr->vecPlaneNormal[1] = trace.plane.normal[1]; ptr->vecPlaneNormal[2] = trace.plane.normal[2]; } msurface_t* EXT_FUNC SurfaceAtPoint(model_t *pModel, mnode_t *node, vec_t *start, vec_t *end) { mplane_t *plane; int s; int t; msurface_t *surf; mtexinfo_t *tex; int ds; int dt; vec3_t mid; float back; float front; float frac; if (node->contents < 0) return 0; plane = node->plane; front = _DotProduct(start, plane->normal) - plane->dist; back = _DotProduct(end, plane->normal) - plane->dist; s = (front < 0.0f) ? 1 : 0; t = (back < 0.0f) ? 1 : 0; if (t == s) return SurfaceAtPoint(pModel, node->children[s], start, end); frac = front / (front - back); mid[0] = (end[0] - start[0]) * frac + start[0]; mid[1] = (end[1] - start[1]) * frac + start[1]; mid[2] = (end[2] - start[2]) * frac + start[2]; surf = SurfaceAtPoint(pModel, node->children[s], start, mid); if (surf) return surf; /* Unreachable code if (t == s) return NULL; */ for (int i = 0; i < node->numsurfaces; i++) { surf = &pModel->surfaces[node->firstsurface + i]; tex = surf->texinfo; ds = (int)(_DotProduct(mid, tex->vecs[0]) + tex->vecs[0][3]); dt = (int)(_DotProduct(mid, tex->vecs[1]) + tex->vecs[1][3]); if (ds >= surf->texturemins[0]) { if (dt >= surf->texturemins[1]) { if (ds - surf->texturemins[0] <= surf->extents[0] && dt - surf->texturemins[1] <= surf->extents[1]) return surf; } } } return SurfaceAtPoint(pModel, node->children[s ^ 1], mid, end); } const char* EXT_FUNC TraceTexture(edict_t *pTextureEntity, const float *v1, const float *v2) { int firstnode; model_t *pmodel; hull_t *phull; msurface_t *psurf; vec3_t up; vec3_t right; vec3_t forward; vec3_t offset; vec3_t temp; vec3_t start; vec3_t end; firstnode = 0; if (pTextureEntity) { pmodel = g_psv.models[pTextureEntity->v.modelindex]; if (!pmodel || pmodel->type) return NULL; phull = SV_HullForBsp(pTextureEntity, vec3_origin, vec3_origin, offset); start[0] = v1[0] - offset[0]; start[1] = v1[1] - offset[1]; start[2] = v1[2] - offset[2]; end[0] = v2[0] - offset[0]; end[1] = v2[1] - offset[1]; end[2] = v2[2] - offset[2]; firstnode = phull->firstclipnode; if (pTextureEntity->v.angles[0] != 0.0 || pTextureEntity->v.angles[1] != 0.0 || pTextureEntity->v.angles[2] != 0.0) { AngleVectors(pTextureEntity->v.angles, forward, right, up); temp[0] = start[0]; temp[1] = start[1]; temp[2] = start[2]; start[0] = _DotProduct(forward, temp); start[1] = -_DotProduct(right, temp); start[2] = _DotProduct(up, temp); temp[0] = end[0]; temp[1] = end[1]; temp[2] = end[2]; end[0] = _DotProduct(forward, temp); end[1] = -_DotProduct(right, temp); end[2] = _DotProduct(up, temp); } } else { pmodel = g_psv.worldmodel; start[0] = v1[0]; start[1] = v1[1]; start[2] = v1[2]; end[0] = v2[0]; end[1] = v2[1]; end[2] = v2[2]; } if (!pmodel || pmodel->type != mod_brush || !pmodel->nodes) return NULL; psurf = SurfaceAtPoint(pmodel, &pmodel->nodes[firstnode], start, end); if (psurf) return psurf->texinfo->texture->name; return NULL; } void EXT_FUNC PF_TraceToss_Shared(edict_t *ent, edict_t *ignore) { trace_t trace = SV_Trace_Toss(ent, ignore); SV_SetGlobalTrace(&trace); } void EXT_FUNC SV_SetGlobalTrace(trace_t *ptrace) { gGlobalVariables.trace_fraction = ptrace->fraction; gGlobalVariables.trace_allsolid = (float)ptrace->allsolid; gGlobalVariables.trace_startsolid = (float)ptrace->startsolid; gGlobalVariables.trace_endpos[0] = ptrace->endpos[0]; gGlobalVariables.trace_endpos[1] = ptrace->endpos[1]; gGlobalVariables.trace_endpos[2] = ptrace->endpos[2]; gGlobalVariables.trace_plane_normal[0] = ptrace->plane.normal[0]; gGlobalVariables.trace_plane_normal[2] = ptrace->plane.normal[2]; gGlobalVariables.trace_plane_normal[1] = ptrace->plane.normal[1]; gGlobalVariables.trace_inwater = (float)ptrace->inwater; gGlobalVariables.trace_inopen = (float)ptrace->inopen; gGlobalVariables.trace_plane_dist = ptrace->plane.dist; if (ptrace->ent) { gGlobalVariables.trace_ent = ptrace->ent; gGlobalVariables.trace_hitgroup = ptrace->hitgroup; } else { gGlobalVariables.trace_hitgroup = ptrace->hitgroup; gGlobalVariables.trace_ent = &g_psv.edicts[0]; } } void EXT_FUNC PF_TraceToss_DLL(edict_t *pent, edict_t *pentToIgnore, TraceResult *ptr) { PF_TraceToss_Shared(pent, pentToIgnore ? pentToIgnore : &g_psv.edicts[0]); ptr->fAllSolid = (int) gGlobalVariables.trace_allsolid; ptr->fStartSolid = (int)gGlobalVariables.trace_startsolid; ptr->fInOpen = (int)gGlobalVariables.trace_inopen; ptr->fInWater = (int)gGlobalVariables.trace_inwater; ptr->flFraction = gGlobalVariables.trace_fraction; ptr->flPlaneDist = gGlobalVariables.trace_plane_dist; ptr->pHit = gGlobalVariables.trace_ent; ptr->vecEndPos[0] = gGlobalVariables.trace_endpos[0]; ptr->vecEndPos[1] = gGlobalVariables.trace_endpos[1]; ptr->vecEndPos[2] = gGlobalVariables.trace_endpos[2]; ptr->vecPlaneNormal[0] = gGlobalVariables.trace_plane_normal[0]; ptr->vecPlaneNormal[1] = gGlobalVariables.trace_plane_normal[1]; ptr->vecPlaneNormal[2] = gGlobalVariables.trace_plane_normal[2]; ptr->iHitgroup = gGlobalVariables.trace_hitgroup; } int EXT_FUNC TraceMonsterHull(edict_t *pEdict, const float *v1, const float *v2, int fNoMonsters, edict_t *pentToSkip, TraceResult *ptr) { qboolean monsterClip = (pEdict->v.flags & FL_MONSTERCLIP) ? 1 : 0; trace_t trace = SV_Move(v1, pEdict->v.mins, pEdict->v.maxs, v2, fNoMonsters, pentToSkip, monsterClip); if (ptr) { ptr->fAllSolid = trace.allsolid; ptr->fStartSolid = trace.startsolid; ptr->fInOpen = trace.inopen; ptr->fInWater = trace.inwater; ptr->flPlaneDist = trace.plane.dist; ptr->pHit = trace.ent; ptr->iHitgroup = trace.hitgroup; ptr->vecEndPos[0] = trace.endpos[0]; ptr->vecEndPos[1] = trace.endpos[1]; ptr->vecEndPos[2] = trace.endpos[2]; ptr->vecPlaneNormal[0] = trace.plane.normal[0]; ptr->vecPlaneNormal[1] = trace.plane.normal[1]; ptr->flFraction = trace.fraction; ptr->vecPlaneNormal[2] = trace.plane.normal[2]; } return trace.allsolid || trace.fraction != 1.0f; } int EXT_FUNC PF_newcheckclient(int check) { int i; unsigned char *pvs; edict_t *ent; mleaf_t *leaf; vec3_t org; if (check < 1) check = 1; if (check > g_psvs.maxclients) check = g_psvs.maxclients; i = 1; if (check != g_psvs.maxclients) i = check + 1; while (1) { if (i == g_psvs.maxclients + 1) i = 1; ent = &g_psv.edicts[i]; if (i == check) break; if (!ent->free && ent->pvPrivateData && !(ent->v.flags & FL_NOTARGET)) break; ++i; } org[0] = ent->v.view_ofs[0] + ent->v.origin[0]; org[1] = ent->v.view_ofs[1] + ent->v.origin[1]; org[2] = ent->v.view_ofs[2] + ent->v.origin[2]; leaf = Mod_PointInLeaf(org, g_psv.worldmodel); pvs = Mod_LeafPVS(leaf, g_psv.worldmodel); Q_memcpy(checkpvs, pvs, (g_psv.worldmodel->numleafs + 7) >> 3); return i; } edict_t* EXT_FUNC PF_checkclient_I(edict_t *pEdict) { edict_t *ent; mleaf_t *leaf; int l; vec3_t view; if (g_psv.time - g_psv.lastchecktime >= 0.1) { g_psv.lastcheck = PF_newcheckclient(g_psv.lastcheck); g_psv.lastchecktime = g_psv.time; } ent = &g_psv.edicts[g_psv.lastcheck]; if (!ent->free && ent->pvPrivateData) { view[0] = pEdict->v.view_ofs[0] + pEdict->v.origin[0]; view[1] = pEdict->v.view_ofs[1] + pEdict->v.origin[1]; view[2] = pEdict->v.view_ofs[2] + pEdict->v.origin[2]; leaf = Mod_PointInLeaf(view, g_psv.worldmodel); l = (leaf - g_psv.worldmodel->leafs) - 1; if (l >= 0 && ((1 << (l & 7)) & checkpvs[l >> 3])) { ++c_invis; return ent; } else { ++c_notvis; return &g_psv.edicts[0]; } } return &g_psv.edicts[0]; } mnode_t* EXT_FUNC PVSNode(mnode_t *node, vec_t *emins, vec_t *emaxs) { mplane_t *splitplane; int sides; mnode_t *splitNode; if (node->visframe != r_visframecount) return NULL; if (node->contents < 0) return node->contents != CONTENT_SOLID ? node : NULL; splitplane = node->plane; if (splitplane->type >= 3u) { sides = BoxOnPlaneSide(emins, emaxs, splitplane); } else { if (splitplane->dist > emins[splitplane->type]) { if (splitplane->dist < emaxs[splitplane->type]) sides = 3; else sides = 2; } else { sides = 1; } } if (sides & 1) { splitNode = PVSNode(node->children[0], emins, emaxs); if (splitNode) return splitNode; } if (sides & 2) return PVSNode(node->children[1], emins, emaxs); return NULL; } void EXT_FUNC PVSMark(model_t *pmodel, unsigned char *ppvs) { ++r_visframecount; for (int i = 0; i < pmodel->numleafs; i++) { if ((1 << (i & 7)) & ppvs[i >> 3]) { mnode_t *node = (mnode_t *) &pmodel->leafs[i + 1]; do { if (node->visframe == r_visframecount) break; node->visframe = r_visframecount; node = node->parent; } while (node); } } } edict_t* EXT_FUNC PVSFindEntities(edict_t *pplayer) { edict_t *pent; edict_t *pchain; edict_t *pentPVS; vec3_t org; unsigned char *ppvs; mleaf_t *pleaf; org[0] = pplayer->v.view_ofs[0] + pplayer->v.origin[0]; org[1] = pplayer->v.view_ofs[1] + pplayer->v.origin[1]; org[2] = pplayer->v.view_ofs[2] + pplayer->v.origin[2]; pleaf = Mod_PointInLeaf(org, g_psv.worldmodel); ppvs = Mod_LeafPVS(pleaf, g_psv.worldmodel); PVSMark(g_psv.worldmodel, ppvs); pchain = g_psv.edicts; for (int i = 1; i < g_psv.num_edicts; i++) { pent = &g_psv.edicts[i]; if (pent->free) continue; pentPVS = pent; if (pent->v.movetype == MOVETYPE_FOLLOW && pent->v.aiment) pentPVS = pent->v.aiment; if (PVSNode(g_psv.worldmodel->nodes, pentPVS->v.absmin, pentPVS->v.absmax)) { pent->v.chain = pchain; pchain = pent; } } if (g_pcl.worldmodel) { //r_oldviewleaf = NULL; //clientside only R_MarkLeaves(); } return pchain; } qboolean EXT_FUNC ValidCmd(const char *pCmd) { int len = Q_strlen(pCmd); return len && (pCmd[len - 1] == '\n' || pCmd[len - 1] == ';'); } void EXT_FUNC PF_stuffcmd_I(edict_t *pEdict, const char *szFmt, ...) { int entnum; client_t *old; va_list argptr; static char szOut[1024]; va_start(argptr, szFmt); entnum = NUM_FOR_EDICT(pEdict); Q_vsnprintf(szOut, sizeof(szOut), szFmt, argptr); va_end(argptr); szOut[1023] = 0; if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("\n!!!\n\nStuffCmd: Some entity tried to stuff '%s' to console " "buffer of entity %i when maxclients was set to %i, ignoring\n\n", szOut, entnum, g_psvs.maxclients); } else { if (ValidCmd(szOut)) { old = host_client; host_client = &g_psvs.clients[entnum - 1]; Host_ClientCommands("%s", szOut); host_client = old; } else { Con_Printf("Tried to stuff bad command %s\n", szOut); } } } void EXT_FUNC PF_localcmd_I(const char *str) { if (ValidCmd(str)) Cbuf_AddText(str); else Con_Printf("Error, bad server command %s\n", str); } void EXT_FUNC PF_localexec_I(void) { Cbuf_Execute(); } edict_t* EXT_FUNC FindEntityInSphere(edict_t *pEdictStartSearchAfter, const float *org, float rad) { int e = pEdictStartSearchAfter ? NUM_FOR_EDICT(pEdictStartSearchAfter) : 0; for (int i = e + 1; i < g_psv.num_edicts; i++) { edict_t* ent = &g_psv.edicts[i]; if (ent->free || !ent->v.classname) continue; if (i <= g_psvs.maxclients && !g_psvs.clients[i - 1].active) continue; float distSquared = 0.0; for (int j = 0; j < 3 && distSquared <= (rad * rad); j++) { float eorg; if (org[j] >= ent->v.absmin[j]) eorg = (org[j] <= ent->v.absmax[j]) ? 0.0f : org[j] - ent->v.absmax[j]; else eorg = org[j] - ent->v.absmin[j]; distSquared = eorg * eorg + distSquared; } if (distSquared <= ((rad * rad))) return ent; } return &g_psv.edicts[0]; } edict_t* EXT_FUNC PF_Spawn_I(void) { return ED_Alloc(); } edict_t* EXT_FUNC CreateNamedEntity(int className) { edict_t *pedict; ENTITYINIT pEntityInit; if (!className) Sys_Error("%s: Spawned a NULL entity!", __func__); pedict = ED_Alloc(); pedict->v.classname = className; pEntityInit = GetEntityInit(&pr_strings[className]); if (pEntityInit) { pEntityInit(&pedict->v); return pedict; } else { ED_Free(pedict); Con_DPrintf("Can't create entity: %s\n", &pr_strings[className]); return NULL; } } void EXT_FUNC PF_Remove_I(edict_t *ed) { g_RehldsHookchains.m_PF_Remove_I.callChain(PF_Remove_I_internal, ed); } void EXT_FUNC PF_Remove_I_internal(edict_t *ed) { ED_Free(ed); } edict_t* EXT_FUNC PF_find_Shared(int eStartSearchAfter, int iFieldToMatch, const char *szValueToFind) { for (int e = eStartSearchAfter + 1; e < g_psv.num_edicts; e++) { edict_t* ed = &g_psv.edicts[e]; if (ed->free) continue; char* t = &pr_strings[*(string_t*)((size_t)&ed->v + iFieldToMatch)]; if (t == 0 || t == &pr_strings[0]) continue; if (!Q_strcmp(t, szValueToFind)) return ed; } return &g_psv.edicts[0]; } int EXT_FUNC iGetIndex(const char *pszField) { char sz[512]; Q_strncpy(sz, pszField, sizeof(sz) - 1); sz[sizeof(sz) - 1] = 0; Q_strlwr(sz); #define IGETINDEX_CHECK_FIELD(f) if (!Q_strcmp(sz, #f)) return offsetof(entvars_t, f); IGETINDEX_CHECK_FIELD(classname); IGETINDEX_CHECK_FIELD(model); IGETINDEX_CHECK_FIELD(viewmodel); IGETINDEX_CHECK_FIELD(weaponmodel); IGETINDEX_CHECK_FIELD(netname); IGETINDEX_CHECK_FIELD(target); IGETINDEX_CHECK_FIELD(targetname); IGETINDEX_CHECK_FIELD(message); IGETINDEX_CHECK_FIELD(noise); IGETINDEX_CHECK_FIELD(noise1); IGETINDEX_CHECK_FIELD(noise2); IGETINDEX_CHECK_FIELD(noise3); IGETINDEX_CHECK_FIELD(globalname); return -1; } edict_t* EXT_FUNC FindEntityByString(edict_t *pEdictStartSearchAfter, const char *pszField, const char *pszValue) { if (!pszValue) return NULL; int iField = iGetIndex(pszField); if (iField == -1) return NULL; return PF_find_Shared(pEdictStartSearchAfter ? NUM_FOR_EDICT(pEdictStartSearchAfter) : 0, iField, pszValue); } int EXT_FUNC GetEntityIllum(edict_t *pEnt) { if (!pEnt) return -1; if (NUM_FOR_EDICT(pEnt) <= g_psvs.maxclients) { return pEnt->v.light_level; } else { if (g_pcls.state == ca_connected || g_pcls.state == ca_uninitialized || g_pcls.state == ca_active) return 0x80; else return 0; } } qboolean EXT_FUNC PR_IsEmptyString(const char *s) { return s[0] < ' '; } int EXT_FUNC PF_precache_sound_I(const char *s) { int i; if (!s) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(s)) Host_Error("%s: Bad string '%s'", __func__, s); if (s[0] == '!') Host_Error("%s: '%s' do not precache sentence names!", __func__, s); if (g_psv.state == ss_loading) { g_psv.sound_precache_hashedlookup_built = 0; for (i = 0; i < MAX_SOUNDS; i++) { if (!g_psv.sound_precache[i]) { #ifdef REHLDS_FIXES // For more information, see EV_Precache g_psv.sound_precache[i] = ED_NewString(s); #else g_psv.sound_precache[i] = s; #endif // REHLDS_FIXES return i; } if (!Q_stricmp(g_psv.sound_precache[i], s)) return i; } Host_Error("%s: Sound '%s' failed to precache because the item count is over the %d limit.\n" "Reduce the number of brush models and/or regular models in the map to correct this.", __func__, s, MAX_SOUNDS); } else { // precaching not enabled. check if already exists. for (i = 0; i < MAX_SOUNDS; i++) { if (g_psv.sound_precache[i] && !Q_stricmp(g_psv.sound_precache[i], s)) return i; } Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, s); } // unreach return -1; } unsigned short EXT_FUNC EV_Precache(int type, const char *psz) { if (!psz) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(psz)) Host_Error("%s: Bad string '%s'", __func__, psz); if (g_psv.state == ss_loading) { for (int i = 1; i < MAX_EVENTS; i++) { struct event_s* ev = &g_psv.event_precache[i]; if (!ev->filename) { if (type != 1) Host_Error("%s: only file type 1 supported currently\n", __func__); char szpath[MAX_PATH]; Q_snprintf(szpath, sizeof(szpath), "%s", psz); COM_FixSlashes(szpath); int scriptSize = 0; char* evScript = (char*) COM_LoadFile(szpath, 5, &scriptSize); if (!evScript) Host_Error("%s: file %s missing from server\n", __func__, psz); #ifdef REHLDS_FIXES // Many modders don't know that the resource names passed to precache functions must be a static strings. // Also some metamod modules can be unloaded during the server running. // Thereafter the pointers to resource names, precached by this modules becomes invalid and access to them will cause segfault error. // Reallocating strings in the rehlds temporary hunk memory helps to avoid these problems. g_psv.event_precache[i].filename = ED_NewString(psz); #else g_psv.event_precache[i].filename = psz; #endif // REHLDS_FIXES g_psv.event_precache[i].filesize = scriptSize; g_psv.event_precache[i].pszScript = evScript; g_psv.event_precache[i].index = i; return i; } if (!Q_stricmp(ev->filename, psz)) return i; } Host_Error("%s: '%s' overflow", __func__, psz); } else { for (int i = 1; i < MAX_EVENTS; i++) { struct event_s* ev = &g_psv.event_precache[i]; if (!Q_stricmp(ev->filename, psz)) return i; } Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, psz); } } void EXT_FUNC EV_PlayReliableEvent_api(IGameClient *cl, int entindex, unsigned short eventindex, float delay, event_args_t *pargs) { EV_PlayReliableEvent_internal(cl->GetClient(), entindex, eventindex, delay, pargs); } void EV_PlayReliableEvent(client_t *cl, int entindex, unsigned short eventindex, float delay, event_args_t *pargs) { g_RehldsHookchains.m_EV_PlayReliableEvent.callChain(EV_PlayReliableEvent_api, GetRehldsApiClient(cl), entindex, eventindex, delay, pargs); } void EV_PlayReliableEvent_internal(client_t *cl, int entindex, unsigned short eventindex, float delay, event_args_t *pargs) { unsigned char data[1024]; event_args_t from; event_args_t to; sizebuf_t msg; if (cl->fakeclient) return; Q_memset(&msg, 0, sizeof(msg)); msg.buffername = "Reliable Event"; msg.data = data; msg.cursize = 0; msg.maxsize = sizeof(data); Q_memset(&from, 0, sizeof(from)); to = *pargs; to.entindex = entindex; MSG_WriteByte(&msg, svc_event_reliable); MSG_StartBitWriting(&msg); MSG_WriteBits(eventindex, 10); DELTA_WriteDelta((unsigned char *)&from, (unsigned char *)&to, 1, g_peventdelta, 0); if (delay == 0.0) { MSG_WriteBits(0, 1); } else { MSG_WriteBits(1, 1); MSG_WriteBits((int)(delay * 100.0f), 16); } MSG_EndBitWriting(&msg); if (msg.cursize + cl->netchan.message.cursize <= cl->netchan.message.maxsize) SZ_Write(&cl->netchan.message, msg.data, msg.cursize); else Netchan_CreateFragments(1, &cl->netchan, &msg); } void EXT_FUNC EV_Playback(int flags, const edict_t *pInvoker, unsigned short eventindex, float delay, float *origin, float *angles, float fparam1, float fparam2, int iparam1, int iparam2, int bparam1, int bparam2) { client_t *cl; signed int j; event_args_t eargs; vec3_t event_origin; int invoker; int slot; int leafnum; if (flags & FEV_CLIENT) return; invoker = -1; Q_memset(&eargs, 0, sizeof(eargs)); if (!VectorCompare(origin, vec3_origin)) { eargs.origin[2] = origin[2]; eargs.origin[0] = origin[0]; eargs.origin[1] = origin[1]; eargs.flags |= FEVENT_ORIGIN; } if (!VectorCompare(angles, vec3_origin)) { eargs.angles[2] = angles[2]; eargs.angles[0] = angles[0]; eargs.angles[1] = angles[1]; eargs.flags |= FEVENT_ANGLES; } eargs.fparam1 = fparam1; eargs.fparam2 = fparam2; eargs.iparam1 = iparam1; eargs.iparam2 = iparam2; eargs.bparam1 = bparam1; eargs.bparam2 = bparam2; if (eventindex < 1u || eventindex >= MAX_EVENTS) { Con_DPrintf("%s: index out of range %i\n", __func__, eventindex); return; } if (!g_psv.event_precache[eventindex].pszScript) { Con_DPrintf("%s: no event for index %i\n", __func__, eventindex); return; } if (pInvoker) { event_origin[0] = pInvoker->v.origin[0]; event_origin[1] = pInvoker->v.origin[1]; event_origin[2] = pInvoker->v.origin[2]; invoker = NUM_FOR_EDICT(pInvoker); if (invoker >= 1) { if (invoker <= g_psvs.maxclients) { if (pInvoker->v.flags & FL_DUCKING) eargs.ducking = 1; } } if (!(eargs.flags & FEVENT_ORIGIN)) { eargs.origin[0] = pInvoker->v.origin[0]; eargs.origin[1] = pInvoker->v.origin[1]; eargs.origin[2] = pInvoker->v.origin[2]; } if (!(eargs.flags & FEVENT_ANGLES)) { eargs.angles[0] = pInvoker->v.angles[0]; eargs.angles[1] = pInvoker->v.angles[1]; eargs.angles[2] = pInvoker->v.angles[2]; } } else { event_origin[0] = eargs.origin[0]; event_origin[1] = eargs.origin[1]; event_origin[2] = eargs.origin[2]; } leafnum = SV_PointLeafnum(event_origin); for (slot = 0; slot < g_psvs.maxclients; slot++) { cl = &g_psvs.clients[slot]; if (!cl->active || !cl->spawned || !cl->connected || !cl->fully_connected || cl->fakeclient) continue; if (pInvoker) { if (pInvoker->v.groupinfo) { if (cl->edict->v.groupinfo) { if (g_groupop) { if (g_groupop == GROUP_OP_NAND && (cl->edict->v.groupinfo & pInvoker->v.groupinfo)) continue; } else { if (!(cl->edict->v.groupinfo & pInvoker->v.groupinfo)) continue; } } } } if (pInvoker && !(flags & FEV_GLOBAL)) { if (!SV_ValidClientMulticast(cl, leafnum, 4)) continue; } if (cl == host_client && (flags & FEV_NOTHOST) && cl->lw || (flags & FEV_HOSTONLY) && cl->edict != pInvoker) continue; if (flags & FEV_RELIABLE) { EV_PlayReliableEvent(cl, pInvoker ? NUM_FOR_EDICT(pInvoker) : 0, eventindex, delay, &eargs); continue; } if (flags & FEV_UPDATE) { for (j = 0; j < MAX_EVENT_QUEUE; j++) { event_info_s *ei = &cl->events.ei[j]; if (ei->index == eventindex && invoker != -1 && ei->entity_index == invoker) break; } if (j < MAX_EVENT_QUEUE) { event_info_s *ei = &cl->events.ei[j]; ei->entity_index = -1; ei->index = eventindex; ei->packet_index = -1; if (pInvoker) ei->entity_index = invoker; Q_memcpy(&ei->args, &eargs, sizeof(ei->args)); ei->fire_time = delay; continue; } } for (j = 0; j < MAX_EVENT_QUEUE; j++) { event_info_s *ei = &cl->events.ei[j]; if (ei->index == 0) break; } if (j < MAX_EVENT_QUEUE) { event_info_s *ei = &cl->events.ei[j]; ei->entity_index = -1; ei->index = eventindex; ei->packet_index = -1; if (pInvoker) ei->entity_index = invoker; Q_memcpy(&ei->args, &eargs, sizeof(ei->args)); ei->fire_time = delay; } } } void EXT_FUNC EV_SV_Playback(int flags, int clientindex, unsigned short eventindex, float delay, float *origin, float *angles, float fparam1, float fparam2, int iparam1, int iparam2, int bparam1, int bparam2) { if (flags & FEV_CLIENT) return; if (clientindex < 0 || clientindex >= g_psvs.maxclients) Host_Error("%s: Client index %i out of range\n", __func__, clientindex); edict_t *pEdict = g_psvs.clients[clientindex].edict; EV_Playback(flags,pEdict, eventindex, delay, origin, angles, fparam1, fparam2, iparam1, iparam2, bparam1, bparam2); } #ifdef REHLDS_FIXES int SV_LookupModelIndex(const char *name) { if (!name || !name[0]) return 0; #ifdef REHLDS_OPT_PEDANTIC auto node = g_rehlds_sv.modelsMap.get(name); if (node) { return node->val; } #else // REHLDS_OPT_PEDANTIC for (int i = 0; i < MAX_MODELS; i++) { if (!g_psv.model_precache[i]) break; if (!Q_stricmp(g_psv.model_precache[i], name)) return i; }; #endif // REHLDS_OPT_PEDANTIC return 0; } #endif // REHLDS_FIXES int EXT_FUNC PF_precache_model_I(const char *s) { int iOptional = 0; if (!s) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(s)) Host_Error("%s: Bad string '%s'", __func__, s); if (*s == '!') { s++; iOptional = 1; } if (g_psv.state == ss_loading) { for (int i = 0; i < MAX_MODELS; i++) { if (!g_psv.model_precache[i]) { #ifdef REHLDS_FIXES // For more information, see EV_Precache g_psv.model_precache[i] = ED_NewString(s); #else g_psv.model_precache[i] = s; #endif // REHLDS_FIXES #ifdef REHLDS_OPT_PEDANTIC g_rehlds_sv.modelsMap.put(g_psv.model_precache[i], i); #endif //REHLDS_OPT_PEDANTIC g_psv.models[i] = Mod_ForName(s, TRUE, TRUE); if (!iOptional) g_psv.model_precache_flags[i] |= RES_FATALIFMISSING; return i; } // use case-sensitive names to increase performance #ifdef REHLDS_FIXES if (!Q_strcmp(g_psv.model_precache[i], s)) return i; #else if (!Q_stricmp(g_psv.model_precache[i], s)) return i; #endif } Host_Error("%s: Model '%s' failed to precache because the item count is over the %d limit.\n" "Reduce the number of brush models and/or regular models in the map to correct this.", __func__, s, MAX_MODELS); } else { for (int i = 0; i < MAX_MODELS; i++) { // use case-sensitive names to increase performance #ifdef REHLDS_FIXES if (!Q_strcmp(g_psv.model_precache[i], s)) return i; #else if (!Q_stricmp(g_psv.model_precache[i], s)) return i; #endif } Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, s); } } #ifdef REHLDS_FIXES int EXT_FUNC PF_precache_generic_I(const char *s) { if (!s) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(s)) { // TODO: Call to Con_Printf is replaced with Host_Error in 6153 Host_Error("%s: Bad string '%s'", __func__, s); } char resName[MAX_QPATH]; Q_strncpy(resName, s, sizeof(resName)); resName[sizeof(resName) - 1] = '\0'; ForwardSlashes(resName); size_t soundPrefixLength = sizeof("sound/") - 1; bool isSoundPrefixed = !Q_strnicmp(resName, "sound/", soundPrefixLength); // TODO: check sound with index 0? // UPD: no, not need, because engine do this: g_psv.sound_precache[0] = pr_strings; if ((isSoundPrefixed && SV_LookupSoundIndex(&resName[soundPrefixLength])) || SV_LookupModelIndex(resName)) return 0; size_t resCount = g_rehlds_sv.precachedGenericResourceCount; for (size_t i = 0; i < resCount; i++) { if (!Q_stricmp(g_rehlds_sv.precachedGenericResourceNames[i], resName)) return i; } if (g_psv.state != ss_loading) Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, resName); if (resCount >= ARRAYSIZE(g_rehlds_sv.precachedGenericResourceNames)) { Host_Error("%s: Generic item '%s' failed to precache because the item count is over the %d limit.\n" "Reduce the number of brush models and/or regular models in the map to correct this.", __func__, resName, ARRAYSIZE(g_rehlds_sv.precachedGenericResourceNames)); } Q_strcpy(g_rehlds_sv.precachedGenericResourceNames[resCount], resName); return g_rehlds_sv.precachedGenericResourceCount++; } #else // REHLDS_FIXES int EXT_FUNC PF_precache_generic_I(const char *s) { if (!s) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(s)) { // TODO: Call to Con_Printf is replaced with Host_Error in 6153 Host_Error("%s: Bad string '%s'", __func__, s); } if (g_psv.state == ss_loading) { for (int i = 0; i < MAX_GENERIC; i++) { if (!g_psv.generic_precache[i]) { g_psv.generic_precache[i] = s; return i; } if (!Q_stricmp(g_psv.generic_precache[i], s)) return i; } Host_Error("%s: Generic item '%s' failed to precache because the item count is over the %d limit.\n" "Reduce the number of brush models and/or regular models in the map to correct this.", __func__, s, MAX_GENERIC); } else { for (int i = 0; i < MAX_GENERIC; i++) { if (!Q_stricmp(g_psv.generic_precache[i], s)) return i; } Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, s); } } #endif // REHLDS_FIXES int EXT_FUNC PF_IsMapValid_I(const char *mapname) { char cBuf[260]; if (!mapname || Q_strlen(mapname) == 0) return 0; Q_snprintf(cBuf, sizeof(cBuf), "maps/%.32s.bsp", mapname); return FS_FileExists(cBuf); } int EXT_FUNC PF_NumberOfEntities_I(void) { int ent_count = 0; for (int i = 1; i < g_psv.num_edicts; i++) { if (!g_psv.edicts[i].free) ++ent_count; } return ent_count; } char* EXT_FUNC PF_GetInfoKeyBuffer_I(edict_t *e) { int e1; char *value; if (e) { e1 = NUM_FOR_EDICT(e); if (e1) { #ifdef REHLDS_FIXES if (e1 <= 0 || e1 > g_psvs.maxclients) // FIXED: Check against correct amount of clients #else // REHLDS_FIXES if (e1 > 32) #endif // REHLDS_FIXES value = (char *)""; else value = (char *)&g_psvs.clients[e1 - 1].userinfo; } else { value = Info_Serverinfo(); } } else { value = localinfo; } return value; } char* EXT_FUNC PF_InfoKeyValue_I(char *infobuffer, const char *key) { return (char *)Info_ValueForKey(infobuffer, key); } void EXT_FUNC PF_SetKeyValue_I(char *infobuffer, const char *key, const char *value) { if (infobuffer == localinfo) { Info_SetValueForKey(infobuffer, key, value, MAX_INFO_STRING * 128); } else { if (infobuffer != Info_Serverinfo()) { Sys_Error("%s: Can't set client keys with SetKeyValue", __func__); } #ifdef REHLDS_FIXES Info_SetValueForKey(infobuffer, key, value, MAX_INFO_STRING); // Use correct length #else // REHLDS_FIXES Info_SetValueForKey(infobuffer, key, value, 512); #endif // REHLDS_FIXES } } void EXT_FUNC PF_RemoveKey_I(char *s, const char *key) { Info_RemoveKey(s, key); } void EXT_FUNC PF_SetClientKeyValue_I(int clientIndex, char *infobuffer, const char *key, const char *value) { client_t *pClient; if (infobuffer == localinfo || infobuffer == Info_Serverinfo() || clientIndex <= 0 || clientIndex > g_psvs.maxclients) { return; } if (Q_strcmp(Info_ValueForKey(infobuffer, key), value)) { Info_SetValueForStarKey(infobuffer, key, value, MAX_INFO_STRING); pClient = &g_psvs.clients[clientIndex - 1]; pClient->sendinfo = TRUE; pClient->sendinfo_time = 0.0f; } } int EXT_FUNC PF_walkmove_I(edict_t *ent, float yaw, float dist, int iMode) { vec3_t move; if (ent->v.flags & (FL_SWIM | FL_FLY | FL_ONGROUND)) { move[0] = cos(yaw * 2.0 * M_PI / 360.0) * dist; move[1] = sin(yaw * 2.0 * M_PI / 360.0) * dist; move[2] = 0; switch (iMode) { case 1: return SV_movetest(ent, move, 1); case 2: return SV_movestep(ent, move, 0); default: return SV_movestep(ent, move, 1); } } return 0; } int EXT_FUNC PF_droptofloor_I(edict_t *ent) { vec3_t end; trace_t trace; qboolean monsterClip = (ent->v.flags & FL_MONSTERCLIP) ? 1 : 0; end[0] = ent->v.origin[0]; end[1] = ent->v.origin[1]; end[2] = ent->v.origin[2] - 256.0; trace = SV_Move(ent->v.origin, ent->v.mins, ent->v.maxs, end, 0, ent, monsterClip); if (trace.allsolid) return -1; if (trace.fraction == 1.0f) return 0; ent->v.origin[0] = trace.endpos[0]; ent->v.origin[1] = trace.endpos[1]; ent->v.origin[2] = trace.endpos[2]; SV_LinkEdict(ent, 0); ent->v.flags |= FL_ONGROUND; ent->v.groundentity = trace.ent; return 1; } int EXT_FUNC PF_DecalIndex(const char *name) { for (int i = 0; i < sv_decalnamecount; i++) { if (!Q_stricmp(sv_decalnames[i].name, name)) return i; } return -1; } void EXT_FUNC PF_lightstyle_I(int style, char *val) { g_psv.lightstyles[style] = val; if (g_psv.state != ss_active) return; for (int i = 0; i < g_psvs.maxclients; i++) { client_t* cl = &g_psvs.clients[i]; if ((cl->active || cl->spawned) && !cl->fakeclient) { MSG_WriteChar(&cl->netchan.message, svc_lightstyle); MSG_WriteChar(&cl->netchan.message, style); MSG_WriteString(&cl->netchan.message, val); } } } int EXT_FUNC PF_checkbottom_I(edict_t *pEdict) { return SV_CheckBottom(pEdict); } int EXT_FUNC PF_pointcontents_I(const float *rgflVector) { return SV_PointContents(rgflVector); } void EXT_FUNC PF_aim_I(edict_t *ent, float speed, float *rgflReturn) { vec3_t start; vec3_t dir; vec3_t end; vec3_t bestdir; int j; trace_t tr; float dist; float bestdist; if (!ent || (ent->v.flags & FL_FAKECLIENT)) { rgflReturn[0] = gGlobalVariables.v_forward[0]; rgflReturn[1] = gGlobalVariables.v_forward[1]; rgflReturn[2] = gGlobalVariables.v_forward[2]; return; } start[0] = ent->v.origin[0]; start[1] = ent->v.origin[1]; start[2] = ent->v.origin[2]; dir[0] = gGlobalVariables.v_forward[0]; dir[1] = gGlobalVariables.v_forward[1]; dir[2] = gGlobalVariables.v_forward[2]; start[0] += ent->v.view_ofs[0]; start[1] += ent->v.view_ofs[1]; start[2] += ent->v.view_ofs[2]; VectorMA(start, 2048.0, dir, end); tr = SV_Move(start, vec3_origin, vec3_origin, end, 0, ent, 0); if (tr.ent && tr.ent->v.takedamage == 2.0f && (ent->v.team <= 0 || ent->v.team != tr.ent->v.team)) { rgflReturn[0] = gGlobalVariables.v_forward[0]; rgflReturn[1] = gGlobalVariables.v_forward[1]; rgflReturn[2] = gGlobalVariables.v_forward[2]; return; } bestdir[1] = dir[1]; bestdir[2] = dir[2]; bestdir[0] = dir[0]; bestdist = sv_aim.value; for (int i = 1; i < g_psv.num_edicts; i++) { edict_t* check = &g_psv.edicts[i]; if (check->v.takedamage != 2.0f || (check->v.flags & FL_FAKECLIENT) || check == ent) continue; if (ent->v.team > 0 && ent->v.team == check->v.team) continue; for (j = 0; j < 3; j++) { end[j] = (check->v.maxs[j] + check->v.mins[j]) * 0.75 + check->v.origin[j] + ent->v.view_ofs[j] * 0.0; } dir[0] = end[0] - start[0]; dir[1] = end[1] - start[1]; dir[2] = end[2] - start[2]; VectorNormalize(dir); dist = gGlobalVariables.v_forward[2] * dir[2] + gGlobalVariables.v_forward[1] * dir[1] + gGlobalVariables.v_forward[0] * dir[0]; if (dist >= bestdist) { tr = SV_Move(start, vec3_origin, vec3_origin, end, 0, ent, 0); if (tr.ent == check) { bestdist = dist; bestdir[0] = dir[0]; bestdir[1] = dir[1]; bestdir[2] = dir[2]; } } } rgflReturn[0] = bestdir[0]; rgflReturn[1] = bestdir[1]; rgflReturn[2] = bestdir[2]; } void EXT_FUNC PF_changeyaw_I(edict_t *ent) { float ideal; float current; float move; float speed; current = anglemod(ent->v.angles[1]); ideal = ent->v.ideal_yaw; speed = ent->v.yaw_speed; if (current == ideal) return; move = ideal - current; if (ideal <= current) { if (move <= -180.0) move += 360.0; } else { if (move >= 180.0) move -= 360.0; } if (move <= 0.0f) { if (move < -speed) move = -speed; } else { if (move > speed) move = speed; } ent->v.angles[1] = anglemod(move + current); } void EXT_FUNC PF_changepitch_I(edict_t *ent) { float ideal; float current; float move; float speed; current = anglemod(ent->v.angles[0]); ideal = ent->v.idealpitch; speed = ent->v.pitch_speed; if (current == ideal) return; move = ideal - current; if (ideal <= (double)current) { if (move <= -180.0) move += 360.0; } else { if (move >= 180.0) move -= 360; } if (move <= 0.0) { if (move < -speed) move = -speed; } else { if (move > speed) move = speed; } ent->v.angles[0] = anglemod(move + current); } void EXT_FUNC PF_setview_I(const edict_t *clientent, const edict_t *viewent) { int clientnum = NUM_FOR_EDICT(clientent); if (clientnum < 1 || clientnum > g_psvs.maxclients) Host_Error("%s: not a client", __func__); client_t *client = &g_psvs.clients[clientnum - 1]; if (!client->fakeclient) { client->pViewEntity = viewent; MSG_WriteByte(&client->netchan.message, svc_setview); MSG_WriteShort(&client->netchan.message, NUM_FOR_EDICT(viewent)); } } void EXT_FUNC PF_crosshairangle_I(const edict_t *clientent, float pitch, float yaw) { int clientnum = NUM_FOR_EDICT(clientent); if (clientnum < 1 || clientnum > g_psvs.maxclients) return; client_t* client = &g_psvs.clients[clientnum - 1]; if (!client->fakeclient) { if (pitch > 180.0) pitch -= 360.0; if (pitch < -180.0) pitch += 360.0; if (yaw > 180.0) yaw -= 360.0; if (yaw < -180.0) yaw += 360.0; MSG_WriteByte(&client->netchan.message, svc_crosshairangle); MSG_WriteChar(&client->netchan.message, (int)(pitch * 5.0)); MSG_WriteChar(&client->netchan.message, (int)(yaw * 5.0)); } } edict_t* EXT_FUNC PF_CreateFakeClient_I(const char *netname) { client_t *fakeclient; edict_t *ent; int i = 0; fakeclient = g_psvs.clients; for (i = 0; i < g_psvs.maxclients; i++, fakeclient++) { if (!fakeclient->active && !fakeclient->spawned && !fakeclient->connected) break; } if (i >= g_psvs.maxclients) return NULL; ent = EDICT_NUM(i + 1); if (fakeclient->frames) SV_ClearFrames(&fakeclient->frames); Q_memset(fakeclient, 0, sizeof(client_t)); fakeclient->resourcesneeded.pPrev = &fakeclient->resourcesneeded; fakeclient->resourcesneeded.pNext = &fakeclient->resourcesneeded; fakeclient->resourcesonhand.pPrev = &fakeclient->resourcesonhand; fakeclient->resourcesonhand.pNext = &fakeclient->resourcesonhand; Q_strncpy(fakeclient->name, netname, sizeof(fakeclient->name) - 1); fakeclient->name[sizeof(fakeclient->name) - 1] = 0; fakeclient->active = TRUE; fakeclient->spawned = TRUE; fakeclient->fully_connected = TRUE; fakeclient->connected = TRUE; fakeclient->fakeclient = TRUE; fakeclient->userid = g_userid++; fakeclient->uploading = FALSE; fakeclient->edict = ent; ent->v.netname = (size_t)fakeclient->name - (size_t)pr_strings; ent->v.pContainingEntity = ent; ent->v.flags = FL_FAKECLIENT | FL_CLIENT; Info_SetValueForKey(fakeclient->userinfo, "name", netname, MAX_INFO_STRING); Info_SetValueForKey(fakeclient->userinfo, "model", "gordon", MAX_INFO_STRING); Info_SetValueForKey(fakeclient->userinfo, "topcolor", "1", MAX_INFO_STRING); Info_SetValueForKey(fakeclient->userinfo, "bottomcolor", "1", MAX_INFO_STRING); fakeclient->sendinfo = TRUE; SV_ExtractFromUserinfo(fakeclient); fakeclient->network_userid.m_SteamID = ISteamGameServer_CreateUnauthenticatedUserConnection(); fakeclient->network_userid.idtype = AUTH_IDTYPE_STEAM; ISteamGameServer_BUpdateUserData(fakeclient->network_userid.m_SteamID, netname, 0); return ent; } void EXT_FUNC PF_RunPlayerMove_I(edict_t *fakeclient, const float *viewangles, float forwardmove, float sidemove, float upmove, unsigned short buttons, unsigned char impulse, unsigned char msec) { usercmd_t cmd; edict_t *oldclient; client_t *old; int entnum; oldclient = sv_player; old = host_client; entnum = NUM_FOR_EDICT(fakeclient); sv_player = fakeclient; host_client = &g_psvs.clients[entnum - 1]; host_client->svtimebase = host_frametime + g_psv.time - msec / 1000.0; Q_memset(&cmd, 0, sizeof(cmd)); cmd.lightlevel = 0; pmove = &g_svmove; cmd.viewangles[0] = viewangles[0]; cmd.viewangles[1] = viewangles[1]; cmd.viewangles[2] = viewangles[2]; cmd.forwardmove = forwardmove; cmd.sidemove = sidemove; cmd.upmove = upmove; cmd.buttons = buttons; cmd.impulse = impulse; cmd.msec = msec; SV_PreRunCmd(); SV_RunCmd(&cmd, 0); Q_memcpy(&host_client->lastcmd, &cmd, sizeof(host_client->lastcmd)); sv_player = oldclient; host_client = old; } sizebuf_t* EXT_FUNC WriteDest_Parm(int dest) { int entnum; switch (dest) { case MSG_BROADCAST: return &g_psv.datagram; case MSG_ONE: case MSG_ONE_UNRELIABLE: entnum = NUM_FOR_EDICT(gMsgEntity); if (entnum <= 0 || entnum > g_psvs.maxclients) { Host_Error("%s: not a client", __func__); } if (dest == MSG_ONE) { return &g_psvs.clients[entnum - 1].netchan.message; } else { return &g_psvs.clients[entnum - 1].datagram; } case MSG_INIT: return &g_psv.signon; case MSG_ALL: return &g_psv.reliable_datagram; case MSG_PVS: case MSG_PAS: return &g_psv.multicast; case MSG_SPEC: return &g_psv.spectator; default: Host_Error("%s: bad destination = %d", __func__, dest); } } void EXT_FUNC PF_MessageBegin_I(int msg_dest, int msg_type, const float *pOrigin, edict_t *ed) { if (msg_dest == MSG_ONE || msg_dest == MSG_ONE_UNRELIABLE) { if (!ed) Sys_Error("%s: with no target entity\n", __func__); } else { if (ed) Sys_Error("%s: Invalid message: Cannot use broadcast message with a target entity", __func__); } if (gMsgStarted) Sys_Error("%s: New message started when msg '%d' has not been sent yet", __func__, gMsgType); if (msg_type == 0) Sys_Error("%s: Tried to create a message with a bogus message type ( 0 )", __func__); gMsgStarted = 1; gMsgType = msg_type; gMsgEntity = ed; gMsgDest = msg_dest; if (msg_dest == MSG_PVS || msg_dest == MSG_PAS) { if (pOrigin) { gMsgOrigin[0] = pOrigin[0]; gMsgOrigin[1] = pOrigin[1]; gMsgOrigin[2] = pOrigin[2]; } //No idea why is it called here //Host_IsSinglePlayerGame(); } gMsgBuffer.flags = SIZEBUF_ALLOW_OVERFLOW; gMsgBuffer.cursize = 0; } void EXT_FUNC PF_MessageEnd_I(void) { qboolean MsgIsVarLength = 0; if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); gMsgStarted = 0; if (gMsgEntity && (gMsgEntity->v.flags & FL_FAKECLIENT)) return; if (gMsgBuffer.flags & SIZEBUF_OVERFLOWED) Sys_Error("%s: called, but message buffer from .dll had overflowed\n", __func__); if (gMsgType > svc_startofusermessages) { UserMsg* pUserMsg = sv_gpUserMsgs; while (pUserMsg && pUserMsg->iMsg != gMsgType) pUserMsg = pUserMsg->next; if (!pUserMsg && gMsgDest == MSG_INIT) { pUserMsg = sv_gpNewUserMsgs; while (pUserMsg && pUserMsg->iMsg != gMsgType) pUserMsg = pUserMsg->next; } if (!pUserMsg) { Con_DPrintf("%s: Unknown User Msg %d\n", __func__, gMsgType); return; } if (pUserMsg->iSize == -1) { MsgIsVarLength = 1; // Limit packet sizes if (gMsgBuffer.cursize > MAX_USER_MSG_DATA) Host_Error("%s: Refusing to send user message %s of %i bytes to client, user message size limit is %i bytes\n", __func__, pUserMsg->szName, gMsgBuffer.cursize, MAX_USER_MSG_DATA); } else { if (pUserMsg->iSize != gMsgBuffer.cursize) Sys_Error("%s: User Msg '%s': %d bytes written, expected %d\n", __func__, pUserMsg->szName, gMsgBuffer.cursize, pUserMsg->iSize); } } #ifdef REHLDS_FIXES auto writer = [MsgIsVarLength] #endif { sizebuf_t * pBuffer = WriteDest_Parm(gMsgDest); if ((gMsgDest == MSG_BROADCAST && gMsgBuffer.cursize + pBuffer->cursize > pBuffer->maxsize) || !pBuffer->data) return; if (gMsgType > svc_startofusermessages && (gMsgDest == MSG_ONE || gMsgDest == MSG_ONE_UNRELIABLE)) { int entnum = NUM_FOR_EDICT((const edict_t *)gMsgEntity); if (entnum < 1 || entnum > g_psvs.maxclients) Host_Error("%s: not a client", __func__); client_t* client = &g_psvs.clients[entnum - 1]; if (client->fakeclient || !client->hasusrmsgs || (!client->active && !client->spawned)) return; } MSG_WriteByte(pBuffer, gMsgType); if (MsgIsVarLength) MSG_WriteByte(pBuffer, gMsgBuffer.cursize); MSG_WriteBuf(pBuffer, gMsgBuffer.cursize, gMsgBuffer.data); } #ifdef REHLDS_FIXES ; if (gMsgDest == MSG_ALL) { gMsgDest = MSG_ONE; for (int i = 0; i < g_psvs.maxclients; i++) { gMsgEntity = g_psvs.clients[i].edict; if (gMsgEntity == nullptr) continue; if (FBitSet(gMsgEntity->v.flags, FL_FAKECLIENT)) continue; writer(); } } else { writer(); } #endif switch (gMsgDest) { case MSG_PVS: SV_Multicast((edict_t *)gMsgEntity, gMsgOrigin, MSG_FL_PVS, 0); break; case MSG_PAS: SV_Multicast((edict_t *)gMsgEntity, gMsgOrigin, MSG_FL_PAS, 0); break; case MSG_PVS_R: SV_Multicast((edict_t *)gMsgEntity, gMsgOrigin, MSG_FL_PAS, 1); // TODO: Should be MSG_FL_PVS, investigation needed break; case MSG_PAS_R: SV_Multicast((edict_t *)gMsgEntity, gMsgOrigin, MSG_FL_PAS, 1); break; default: return; } } void EXT_FUNC PF_WriteByte_I(int iValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteByte(&gMsgBuffer, iValue); } void EXT_FUNC PF_WriteChar_I(int iValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteChar(&gMsgBuffer, iValue); } void EXT_FUNC PF_WriteShort_I(int iValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteShort(&gMsgBuffer, iValue); } void EXT_FUNC PF_WriteLong_I(int iValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteLong(&gMsgBuffer, iValue); } void EXT_FUNC PF_WriteAngle_I(float flValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteAngle(&gMsgBuffer, flValue); } void EXT_FUNC PF_WriteCoord_I(float flValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteShort(&gMsgBuffer, (int)(flValue * 8.0)); } void EXT_FUNC PF_WriteString_I(const char *sz) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteString(&gMsgBuffer, sz); } void EXT_FUNC PF_WriteEntity_I(int iValue) { if (!gMsgStarted) Sys_Error("%s: called with no active message\n", __func__); MSG_WriteShort(&gMsgBuffer, iValue); } void EXT_FUNC PF_makestatic_I(edict_t *ent) { MSG_WriteByte(&g_psv.signon, svc_spawnstatic); MSG_WriteShort(&g_psv.signon, SV_ModelIndex(&pr_strings[ent->v.model])); MSG_WriteByte(&g_psv.signon, ent->v.sequence); MSG_WriteByte(&g_psv.signon, (int)ent->v.frame); MSG_WriteWord(&g_psv.signon, ent->v.colormap); MSG_WriteByte(&g_psv.signon, ent->v.skin); for (int i = 0; i < 3; i++) { MSG_WriteCoord(&g_psv.signon, ent->v.origin[i]); MSG_WriteAngle(&g_psv.signon, ent->v.angles[i]); } MSG_WriteByte(&g_psv.signon, ent->v.rendermode); if (ent->v.rendermode) { MSG_WriteByte(&g_psv.signon, (int)ent->v.renderamt); MSG_WriteByte(&g_psv.signon, (int)ent->v.rendercolor[0]); MSG_WriteByte(&g_psv.signon, (int)ent->v.rendercolor[1]); MSG_WriteByte(&g_psv.signon, (int)ent->v.rendercolor[2]); MSG_WriteByte(&g_psv.signon, ent->v.renderfx); } ED_Free(ent); } void EXT_FUNC PF_StaticDecal(const float *origin, int decalIndex, int entityIndex, int modelIndex) { MSG_WriteByte(&g_psv.signon, svc_temp_entity); MSG_WriteByte(&g_psv.signon, TE_BSPDECAL); MSG_WriteCoord(&g_psv.signon, *origin); MSG_WriteCoord(&g_psv.signon, origin[1]); MSG_WriteCoord(&g_psv.signon, origin[2]); MSG_WriteShort(&g_psv.signon, decalIndex); MSG_WriteShort(&g_psv.signon, entityIndex); if (entityIndex) MSG_WriteShort(&g_psv.signon, modelIndex); } void EXT_FUNC PF_setspawnparms_I(edict_t *ent) { int i = NUM_FOR_EDICT(ent); if (i < 1 || i > g_psvs.maxclients) Host_Error("%s: Entity is not a client", __func__); } void EXT_FUNC PF_changelevel_I(const char *s1, const char *s2) { static int last_spawncount; if (g_psvs.spawncount != last_spawncount) { last_spawncount = g_psvs.spawncount; SV_SkipUpdates(); if (s2) Cbuf_AddText(va("changelevel2 %s %s\n", s1, s2)); else Cbuf_AddText(va("changelevel %s\n", s1)); } } void SeedRandomNumberGenerator(void) { idum = -(int)CRehldsPlatformHolder::get()->time(NULL); if (idum > 1000) { idum = -idum; } else if (idum > -1000) { idum -= 22261048; } } #define IA 16807 #define IM 2147483647 #define IQ 127773 #define IR 2836 #define NTAB 32 #define NDIV (1+(IM-1)/NTAB) int32 ran1(void) { int j; long k; static long iy = 0; static long iv[NTAB]; if (idum <= 0 || !iy) { if (-(idum) < 1) idum = 1; else idum = -(idum); for (j = NTAB + 7; j >= 0; j--) { k = (idum) / IQ; idum = IA*(idum - k*IQ) - IR*k; if (idum < 0) idum += IM; if (j < NTAB) iv[j] = idum; } iy = iv[0]; } k = (idum) / IQ; idum = IA*(idum - k*IQ) - IR*k; if (idum < 0) idum += IM; j = iy / NDIV; iy = iv[j]; iv[j] = idum; return iy; } #define AM (1.0/IM) #define EPS 1.2e-7 #define RNMX (1.0-EPS) float fran1(void) { float temp = (float)AM*ran1(); if (temp > RNMX) return (float)RNMX; else return temp; } float EXT_FUNC RandomFloat(float flLow, float flHigh) { #ifndef SWDS g_engdstAddrs.pfnRandomFloat(&flLow, &flHigh); #endif float fl = fran1(); // float in [0,1) return (fl * (flHigh - flLow)) + flLow; // float in [low,high) } int32 EXT_FUNC RandomLong(int32 lLow, int32 lHigh) { #ifndef SWDS g_engdstAddrs.pfnRandomLong(&lLow, &lHigh); #endif unsigned long maxAcceptable; unsigned long x = lHigh - lLow + 1; unsigned long n; if (x <= 0 || MAX_RANDOM_RANGE < x - 1) { return lLow; } // The following maps a uniform distribution on the interval [0,MAX_RANDOM_RANGE] // to a smaller, client-specified range of [0,x-1] in a way that doesn't bias // the uniform distribution unfavorably. Even for a worst case x, the loop is // guaranteed to be taken no more than half the time, so for that worst case x, // the average number of times through the loop is 2. For cases where x is // much smaller than MAX_RANDOM_RANGE, the average number of times through the // loop is very close to 1. // maxAcceptable = MAX_RANDOM_RANGE - ((MAX_RANDOM_RANGE + 1) % x); do { n = ran1(); } while (n > maxAcceptable); return lLow + (n % x); } void EXT_FUNC PF_FadeVolume(const edict_t *clientent, int fadePercent, int fadeOutSeconds, int holdTime, int fadeInSeconds) { int entnum = NUM_FOR_EDICT(clientent); if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("tried to PF_FadeVolume a non-client\n"); return; } client_t* client = &g_psvs.clients[entnum - 1]; if (client->fakeclient) return; MSG_WriteChar(&client->netchan.message, svc_soundfade); MSG_WriteByte(&client->netchan.message, (uint8)fadePercent); MSG_WriteByte(&client->netchan.message, (uint8)holdTime); MSG_WriteByte(&client->netchan.message, (uint8)fadeOutSeconds); MSG_WriteByte(&client->netchan.message, (uint8)fadeInSeconds); } void EXT_FUNC PF_SetClientMaxspeed(edict_t *clientent, float fNewMaxspeed) { int entnum = NUM_FOR_EDICT(clientent); if (entnum < 1 || entnum > g_psvs.maxclients) Con_Printf("tried to PF_SetClientMaxspeed a non-client\n"); clientent->v.maxspeed = fNewMaxspeed; } int EXT_FUNC PF_GetPlayerUserId(edict_t *e) { if (!g_psv.active || !e) return -1; for (int i = 0; i < g_psvs.maxclients; i++) { if (g_psvs.clients[i].edict == e) { return g_psvs.clients[i].userid; } } return -1; } unsigned int EXT_FUNC PF_GetPlayerWONId(edict_t *e) { return 0xFFFFFFFF; } const char* EXT_FUNC PF_GetPlayerAuthId(edict_t *e) { static char szAuthID[5][64]; static int count = 0; client_t *cl; int i; count = (count + 1) % 5; szAuthID[count][0] = 0; if (!g_psv.active || e == NULL) { return szAuthID[count]; } for (i = 0; i < g_psvs.maxclients; i++) { cl = &g_psvs.clients[i]; if (cl->edict != e) { continue; } if (cl->fakeclient) { Q_strcpy(szAuthID[count], "BOT"); } // AUTH_IDTYPE_LOCAL is handled inside SV_GetIDString(), no need to do it here // else if (cl->network_userid.idtype == AUTH_IDTYPE_LOCAL) // { // Q_strcpy(szAuthID[count], "HLTV"); // } else { Q_snprintf(szAuthID[count], sizeof(szAuthID[count]) - 1, "%s", SV_GetClientIDString(cl)); } break; } return szAuthID[count]; } void EXT_FUNC PF_BuildSoundMsg_I(edict_t *entity, int channel, const char *sample, float volume, float attenuation, int fFlags, int pitch, int msg_dest, int msg_type, const float *pOrigin, edict_t *ed) { g_RehldsHookchains.m_PF_BuildSoundMsg_I.callChain(PF_BuildSoundMsg_I_internal, entity, channel, sample, volume, attenuation, fFlags, pitch, msg_dest, msg_type, pOrigin, ed); } void EXT_FUNC PF_BuildSoundMsg_I_internal(edict_t *entity, int channel, const char *sample, float volume, float attenuation, int fFlags, int pitch, int msg_dest, int msg_type, const float *pOrigin, edict_t *ed) { PF_MessageBegin_I(msg_dest, msg_type, pOrigin, ed); SV_BuildSoundMsg(entity, channel, sample, (int)volume, attenuation, fFlags, pitch, pOrigin, &gMsgBuffer); PF_MessageEnd_I(); } int EXT_FUNC PF_IsDedicatedServer(void) { return g_bIsDedicatedServer; } const char* EXT_FUNC PF_GetPhysicsInfoString(const edict_t *pClient) { int entnum = NUM_FOR_EDICT(pClient); if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("tried to PF_GetPhysicsInfoString a non-client\n"); return ""; } client_t* client = &g_psvs.clients[entnum - 1]; return client->physinfo; } const char* EXT_FUNC PF_GetPhysicsKeyValue(const edict_t *pClient, const char *key) { int entnum = NUM_FOR_EDICT(pClient); if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("tried to PF_GetPhysicsKeyValue a non-client\n"); return ""; } client_t* client = &g_psvs.clients[entnum - 1]; return Info_ValueForKey(client->physinfo, key); } void EXT_FUNC PF_SetPhysicsKeyValue(const edict_t *pClient, const char *key, const char *value) { int entnum = NUM_FOR_EDICT(pClient); if (entnum < 1 || entnum > g_psvs.maxclients) Con_Printf("tried to PF_SetPhysicsKeyValue a non-client\n"); client_t* client = &g_psvs.clients[entnum - 1]; Info_SetValueForKey(client->physinfo, key, value, MAX_INFO_STRING); } int EXT_FUNC PF_GetCurrentPlayer(void) { int idx = host_client - g_psvs.clients; if (idx < 0 || idx >= g_psvs.maxclients) return -1; return idx; } int EXT_FUNC PF_CanSkipPlayer(const edict_t *pClient) { int entnum = NUM_FOR_EDICT(pClient); if (entnum < 1 || entnum > g_psvs.maxclients) { Con_Printf("tried to PF_CanSkipPlayer a non-client\n"); return 0; } client_t* client = &g_psvs.clients[entnum - 1]; return client->lw != 0; } void EXT_FUNC PF_SetGroupMask(int mask, int op) { g_groupmask = mask; g_groupop = op; } int EXT_FUNC PF_CreateInstancedBaseline(int classname, struct entity_state_s *baseline) { extra_baselines_t *bls = g_psv.instance_baselines; if (bls->number >= NUM_BASELINES) return 0; bls->classname[bls->number] = classname; Q_memcpy(&bls->baseline[bls->number], baseline, sizeof(struct entity_state_s)); bls->number += 1; return bls->number; } void EXT_FUNC PF_Cvar_DirectSet(struct cvar_s *var, const char *value) { Cvar_DirectSet(var, value); } void EXT_FUNC PF_ForceUnmodified(FORCE_TYPE type, float *mins, float *maxs, const char *filename) { int i; if (!filename) Host_Error("%s: NULL pointer", __func__); if (PR_IsEmptyString(filename)) Host_Error("%s: Bad string '%s'", __func__, filename); if (g_psv.state == ss_loading) { i = 0; consistency_t* cnode = g_psv.consistency_list; while (cnode->filename) { if (!Q_stricmp(cnode->filename, (char *)filename)) return; ++cnode; ++i; if (i >= 512) Host_Error("%s: '%s' overflow", __func__, filename); } cnode->check_type = type; cnode->filename = (char *)filename; if (mins) { cnode->mins[0] = mins[0]; cnode->mins[1] = mins[1]; cnode->mins[2] = mins[2]; } if (maxs) { cnode->maxs[0] = maxs[0]; cnode->maxs[1] = maxs[1]; cnode->maxs[2] = maxs[2]; } } else { i = 0; consistency_t* cnode = g_psv.consistency_list; while (!cnode->filename || Q_stricmp(cnode->filename, filename)) { ++cnode; ++i; if (i >= 512) Host_Error("%s: '%s' Precache can only be done in spawn functions", __func__, filename); } } } void EXT_FUNC PF_GetPlayerStats(const edict_t *pClient, int *ping, int *packet_loss) { *packet_loss = 0; *ping = 0; int c = NUM_FOR_EDICT(pClient); if (c < 1 || c > g_psvs.maxclients) { Con_Printf("tried to PF_GetPlayerStats a non-client\n"); return; } client_t* client = &g_psvs.clients[c - 1]; *packet_loss = client->packet_loss; *ping = client->latency * 1000.0; } NOXREF void QueryClientCvarValueCmd(void) { NOXREFCHECK; if (Cmd_Argc() <= 1) { Con_Printf("%s \n", Cmd_Argv(0)); return; } for (int i = 0; i < g_psvs.maxclients; i++) { client_t *cl = &g_psvs.clients[i]; if (cl->active || cl->connected) { if (!Q_stricmp(cl->name, Cmd_Argv(1))) { QueryClientCvarValue(cl->edict, Cmd_Argv(2)); break; } } } } NOXREF void QueryClientCvarValueCmd2(void) { NOXREFCHECK; int i; client_t *cl; int requestID; if (Cmd_Argc() < 3) { Con_Printf("%s ", Cmd_Argv(0)); return; } requestID = Q_atoi(Cmd_Argv(3)); for (i = 0; i < g_psvs.maxclients; i++) { cl = &g_psvs.clients[i]; if (cl->active || cl->connected) { if (!Q_stricmp(cl->name, Cmd_Argv(1))) { QueryClientCvarValue2(cl->edict, Cmd_Argv(2), requestID); break; } } } } void EXT_FUNC QueryClientCvarValue(const edict_t *player, const char *cvarName) { int entnum = NUM_FOR_EDICT(player); if (entnum < 1 || entnum > g_psvs.maxclients) { if (gNewDLLFunctions.pfnCvarValue) gNewDLLFunctions.pfnCvarValue(player, "Bad Player"); Con_Printf("tried to QueryClientCvarValue a non-client\n"); return; } client_t *client = &g_psvs.clients[entnum - 1]; MSG_WriteChar(&client->netchan.message, svc_sendcvarvalue); MSG_WriteString(&client->netchan.message, cvarName); } void EXT_FUNC QueryClientCvarValue2(const edict_t *player, const char *cvarName, int requestID) { int entnum = NUM_FOR_EDICT(player); if (entnum < 1 || entnum > g_psvs.maxclients) { if (gNewDLLFunctions.pfnCvarValue2) gNewDLLFunctions.pfnCvarValue2(player, requestID, cvarName, "Bad Player"); Con_Printf("tried to QueryClientCvarValue a non-client\n"); return; } client_t *client = &g_psvs.clients[entnum - 1]; MSG_WriteChar(&client->netchan.message, svc_sendcvarvalue2); MSG_WriteLong(&client->netchan.message, requestID); MSG_WriteString(&client->netchan.message, cvarName); } int hudCheckParm(char *parm, char **ppnext) { #ifndef SWDS g_engdstAddrs.CheckParm(&parm, &ppnext); #endif int i = COM_CheckParm(parm); if (ppnext) { if (i && i < com_argc - 1) *ppnext = com_argv[i + 1]; else *ppnext = 0; } return i; } int EXT_FUNC EngCheckParm(const char *pchCmdLineToken, char **pchNextVal) { return hudCheckParm((char*)pchCmdLineToken, pchNextVal); }