diff options
Diffstat (limited to 'src/server/sv_world.cpp')
-rw-r--r-- | src/server/sv_world.cpp | 745 |
1 files changed, 745 insertions, 0 deletions
diff --git a/src/server/sv_world.cpp b/src/server/sv_world.cpp new file mode 100644 index 0000000..fd0710e --- /dev/null +++ b/src/server/sv_world.cpp @@ -0,0 +1,745 @@ +/* +=========================================================================== +Copyright (C) 1999-2005 Id Software, Inc. +Copyright (C) 2000-2013 Darklegion Development +Copyright (C) 2015-2019 GrangerHub + +This file is part of Tremulous. + +Tremulous 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 3 of the License, +or (at your option) any later version. + +Tremulous 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 Tremulous; if not, see <https://www.gnu.org/licenses/> + +=========================================================================== +*/ + +// world.c -- world query functions + +#include "server.h" + +/* +================ +SV_ClipHandleForEntity + +Returns a headnode that can be used for testing or clipping to a +given entity. If the entity is a bsp model, the headnode will +be returned, otherwise a custom box tree will be constructed. +================ +*/ +clipHandle_t SV_ClipHandleForEntity(const sharedEntity_t *ent) +{ + if (ent->r.bmodel) + { + // explicit hulls in the BSP model + return CM_InlineModel(ent->s.modelindex); + } + if (ent->r.svFlags & SVF_CAPSULE) + { + // create a temp capsule from bounding box sizes + return CM_TempBoxModel(ent->r.mins, ent->r.maxs, true); + } + + // create a temp tree from bounding box sizes + return CM_TempBoxModel(ent->r.mins, ent->r.maxs, qfalse); +} + +/* +=============================================================================== + +ENTITY CHECKING + +To avoid linearly searching through lists of entities during environment testing, +the world is carved up with an evenly spaced, axially aligned bsp tree. Entities +are kept in chains either at the final leafs, or at the first node that splits +them, which prevents having to deal with multiple fragments of a single entity. + +=============================================================================== +*/ + +struct worldSector_t { + int axis; // -1 = leaf node + float dist; + worldSector_t *children[2]; + svEntity_t *entities; +}; + +#define AREA_DEPTH 4 +#define AREA_NODES 64 + +worldSector_t sv_worldSectors[AREA_NODES]; +int sv_numworldSectors; + +/* +=============== +SV_SectorList_f +=============== +*/ +void SV_SectorList_f(void) +{ + int i, c; + worldSector_t *sec; + svEntity_t *ent; + + for (i = 0; i < AREA_NODES; i++) + { + sec = &sv_worldSectors[i]; + + c = 0; + for (ent = sec->entities; ent; ent = ent->nextEntityInWorldSector) + { + c++; + } + Com_Printf("sector %i: %i entities\n", i, c); + } +} + +/* +=============== +SV_CreateworldSector + +Builds a uniformly subdivided tree for the given world size +=============== +*/ +static worldSector_t *SV_CreateworldSector(int depth, vec3_t mins, vec3_t maxs) +{ + worldSector_t *anode; + vec3_t size; + vec3_t mins1, maxs1, mins2, maxs2; + + anode = &sv_worldSectors[sv_numworldSectors]; + sv_numworldSectors++; + + if (depth == AREA_DEPTH) + { + anode->axis = -1; + anode->children[0] = anode->children[1] = NULL; + return anode; + } + + VectorSubtract(maxs, mins, size); + if (size[0] > size[1]) + { + anode->axis = 0; + } + else + { + anode->axis = 1; + } + + anode->dist = 0.5 * (maxs[anode->axis] + mins[anode->axis]); + VectorCopy(mins, mins1); + VectorCopy(mins, mins2); + VectorCopy(maxs, maxs1); + VectorCopy(maxs, maxs2); + + maxs1[anode->axis] = mins2[anode->axis] = anode->dist; + + anode->children[0] = SV_CreateworldSector(depth + 1, mins2, maxs2); + anode->children[1] = SV_CreateworldSector(depth + 1, mins1, maxs1); + + return anode; +} + +/* +=============== +SV_ClearWorld + +=============== +*/ +void SV_ClearWorld(void) +{ + clipHandle_t h; + vec3_t mins, maxs; + + ::memset(sv_worldSectors, 0, sizeof(sv_worldSectors)); + sv_numworldSectors = 0; + + // get world map bounds + h = CM_InlineModel(0); + CM_ModelBounds(h, mins, maxs); + SV_CreateworldSector(0, mins, maxs); +} + +/* +=============== +SV_UnlinkEntity + +=============== +*/ +void SV_UnlinkEntity(sharedEntity_t *gEnt) +{ + svEntity_t *ent; + svEntity_t *scan; + worldSector_t *ws; + + ent = SV_SvEntityForGentity(gEnt); + + gEnt->r.linked = qfalse; + + ws = ent->worldSector; + if (!ws) + { + return; // not linked in anywhere + } + ent->worldSector = NULL; + + if (ws->entities == ent) + { + ws->entities = ent->nextEntityInWorldSector; + return; + } + + for (scan = ws->entities; scan; scan = scan->nextEntityInWorldSector) + { + if (scan->nextEntityInWorldSector == ent) + { + scan->nextEntityInWorldSector = ent->nextEntityInWorldSector; + return; + } + } + + Com_Printf("WARNING: SV_UnlinkEntity: not found in worldSector\n"); +} + +/* +=============== +SV_LinkEntity + +=============== +*/ +#define MAX_TOTAL_ENT_LEAFS 128 +void SV_LinkEntity(sharedEntity_t *gEnt) +{ + worldSector_t *node; + int leafs[MAX_TOTAL_ENT_LEAFS]; + int cluster; + int num_leafs; + int i, j, k; + int area; + int lastLeaf; + float *origin, *angles; + svEntity_t *ent; + + ent = SV_SvEntityForGentity(gEnt); + + if (ent->worldSector) + { + SV_UnlinkEntity(gEnt); // unlink from old position + } + + // encode the size into the entityState_t for client prediction + if (gEnt->r.bmodel) + { + gEnt->s.solid = SOLID_BMODEL; // a solid_box will never create this value + } + else if (gEnt->r.contents & (CONTENTS_SOLID | CONTENTS_BODY)) + { + // assume that x/y are equal and symetric + i = gEnt->r.maxs[0]; + if (i < 1) i = 1; + if (i > 255) i = 255; + + // z is not symetric + j = (-gEnt->r.mins[2]); + if (j < 1) j = 1; + if (j > 255) j = 255; + + // and z maxs can be negative... + k = (gEnt->r.maxs[2] + 32); + if (k < 1) k = 1; + if (k > 255) k = 255; + + gEnt->s.solid = (k << 16) | (j << 8) | i; + } + else + { + gEnt->s.solid = 0; + } + + // get the position + origin = gEnt->r.currentOrigin; + angles = gEnt->r.currentAngles; + + // set the abs box + if (gEnt->r.bmodel && (angles[0] || angles[1] || angles[2])) + { + // expand for rotation + float max; + + max = RadiusFromBounds(gEnt->r.mins, gEnt->r.maxs); + for (i = 0; i < 3; i++) + { + gEnt->r.absmin[i] = origin[i] - max; + gEnt->r.absmax[i] = origin[i] + max; + } + } + else + { + // normal + VectorAdd(origin, gEnt->r.mins, gEnt->r.absmin); + VectorAdd(origin, gEnt->r.maxs, gEnt->r.absmax); + } + + // because movement is clipped an epsilon away from an actual edge, + // we must fully check even when bounding boxes don't quite touch + gEnt->r.absmin[0] -= 1; + gEnt->r.absmin[1] -= 1; + gEnt->r.absmin[2] -= 1; + gEnt->r.absmax[0] += 1; + gEnt->r.absmax[1] += 1; + gEnt->r.absmax[2] += 1; + + // link to PVS leafs + ent->numClusters = 0; + ent->lastCluster = 0; + ent->areanum = -1; + ent->areanum2 = -1; + + // get all leafs, including solids + num_leafs = CM_BoxLeafnums(gEnt->r.absmin, gEnt->r.absmax, leafs, MAX_TOTAL_ENT_LEAFS, &lastLeaf); + + // if none of the leafs were inside the map, the + // entity is outside the world and can be considered unlinked + if (!num_leafs) + { + return; + } + + // set areas, even from clusters that don't fit in the entity array + for (i = 0; i < num_leafs; i++) + { + area = CM_LeafArea(leafs[i]); + if (area != -1) + { + // doors may legally straggle two areas, + // but nothing should evern need more than that + if (ent->areanum != -1 && ent->areanum != area) + { + if (ent->areanum2 != -1 && ent->areanum2 != area && sv.state == SS_LOADING) + { + Com_DPrintf("Object %i touching 3 areas at %f %f %f\n", gEnt->s.number, gEnt->r.absmin[0], + gEnt->r.absmin[1], gEnt->r.absmin[2]); + } + ent->areanum2 = area; + } + else + { + ent->areanum = area; + } + } + } + + // store as many explicit clusters as we can + ent->numClusters = 0; + for (i = 0; i < num_leafs; i++) + { + cluster = CM_LeafCluster(leafs[i]); + if (cluster != -1) + { + ent->clusternums[ent->numClusters++] = cluster; + if (ent->numClusters == MAX_ENT_CLUSTERS) + { + break; + } + } + } + + // store off a last cluster if we need to + if (i != num_leafs) + { + ent->lastCluster = CM_LeafCluster(lastLeaf); + } + + gEnt->r.linkcount++; + + // find the first world sector node that the ent's box crosses + node = sv_worldSectors; + for ( ;; ) + { + if (node->axis == -1) + break; + + if (gEnt->r.absmin[node->axis] > node->dist) + node = node->children[0]; + else if (gEnt->r.absmax[node->axis] < node->dist) + node = node->children[1]; + else + break; // crosses the node + } + + // link it in + ent->worldSector = node; + ent->nextEntityInWorldSector = node->entities; + node->entities = ent; + + gEnt->r.linked = qtrue; +} + +/* +============================================================================ + +AREA QUERY + +Fills in a list of all entities who's absmin / absmax intersects the given +bounds. This does NOT mean that they actually touch in the case of bmodels. +============================================================================ +*/ + +struct areaParms_t { + const float *mins; + const float *maxs; + int *list; + int count; + int maxcount; +}; + +/* +==================== +SV_AreaEntities_r + +==================== +*/ +static void SV_AreaEntities_r(worldSector_t *node, areaParms_t *ap) +{ + svEntity_t *check, *next; + sharedEntity_t *gcheck; + + for (check = node->entities; check; check = next) + { + next = check->nextEntityInWorldSector; + + gcheck = SV_GEntityForSvEntity(check); + + if (gcheck->r.absmin[0] > ap->maxs[0] || gcheck->r.absmin[1] > ap->maxs[1] || + gcheck->r.absmin[2] > ap->maxs[2] || gcheck->r.absmax[0] < ap->mins[0] || + gcheck->r.absmax[1] < ap->mins[1] || gcheck->r.absmax[2] < ap->mins[2]) + { + continue; + } + + if (ap->count == ap->maxcount) + { + Com_Printf("SV_AreaEntities: MAXCOUNT\n"); + return; + } + + ap->list[ap->count] = check - sv.svEntities; + ap->count++; + } + + if (node->axis == -1) + { + return; // terminal node + } + + // recurse down both sides + if (ap->maxs[node->axis] > node->dist) + { + SV_AreaEntities_r(node->children[0], ap); + } + if (ap->mins[node->axis] < node->dist) + { + SV_AreaEntities_r(node->children[1], ap); + } +} + +/* +================ +SV_AreaEntities +================ +*/ +int SV_AreaEntities(const vec3_t mins, const vec3_t maxs, int *entityList, int maxcount) +{ + areaParms_t ap; + + ap.mins = mins; + ap.maxs = maxs; + ap.list = entityList; + ap.count = 0; + ap.maxcount = maxcount; + + SV_AreaEntities_r(sv_worldSectors, &ap); + + return ap.count; +} + +//=========================================================================== + +struct moveclip_t { + vec3_t boxmins; + vec3_t boxmaxs; // enclose the test object along entire move + const float *mins; + const float *maxs; // size of the moving object + const float *start; + vec3_t end; + trace_t trace; + int passEntityNum; + int contentmask; + traceType_t collisionType; +}; + +/* +==================== +SV_ClipToEntity + +==================== +*/ +void SV_ClipToEntity(trace_t *trace, const vec3_t start, const vec3_t mins, const vec3_t maxs, const vec3_t end, + int entityNum, int contentmask, traceType_t type) +{ + sharedEntity_t *touch; + clipHandle_t clipHandle; + float *origin, *angles; + + touch = SV_GentityNum(entityNum); + + ::memset(trace, 0, sizeof(trace_t)); + + // if it doesn't have any brushes of a type we + // are looking for, ignore it + if (!(contentmask & touch->r.contents)) + { + trace->fraction = 1.0; + return; + } + + // might intersect, so do an exact clip + clipHandle = SV_ClipHandleForEntity(touch); + + origin = touch->r.currentOrigin; + angles = touch->r.currentAngles; + + if (!touch->r.bmodel) + { + angles = vec3_origin; // boxes don't rotate + } + + CM_TransformedBoxTrace(trace, (float *)start, (float *)end, (float *)mins, (float *)maxs, clipHandle, contentmask, + origin, angles, type); + + if (trace->fraction < 1) + { + trace->entityNum = touch->s.number; + } +} + +/* +==================== +SV_ClipMoveToEntities + +==================== +*/ +static void SV_ClipMoveToEntities(moveclip_t *clip) +{ + int i, num; + int touchlist[MAX_GENTITIES]; + sharedEntity_t *touch; + int passOwnerNum; + trace_t trace; + clipHandle_t clipHandle; + float *origin, *angles; + + num = SV_AreaEntities(clip->boxmins, clip->boxmaxs, touchlist, MAX_GENTITIES); + + if (clip->passEntityNum != ENTITYNUM_NONE) + { + passOwnerNum = (SV_GentityNum(clip->passEntityNum))->r.ownerNum; + if (passOwnerNum == ENTITYNUM_NONE) + { + passOwnerNum = -1; + } + } + else + { + passOwnerNum = -1; + } + + for (i = 0; i < num; i++) + { + if (clip->trace.allsolid) + { + return; + } + touch = SV_GentityNum(touchlist[i]); + + // see if we should ignore this entity + if (clip->passEntityNum != ENTITYNUM_NONE) + { + if (touchlist[i] == clip->passEntityNum) + { + continue; // don't clip against the pass entity + } + if (touch->r.ownerNum == clip->passEntityNum) + { + continue; // don't clip against own missiles + } + if (touch->r.ownerNum == passOwnerNum) + { + continue; // don't clip against other missiles from our owner + } + } + + // if it doesn't have any brushes of a type we + // are looking for, ignore it + if (!(clip->contentmask & touch->r.contents)) + { + continue; + } + + // might intersect, so do an exact clip + clipHandle = SV_ClipHandleForEntity(touch); + + origin = touch->r.currentOrigin; + angles = touch->r.currentAngles; + + if (!touch->r.bmodel) + { + angles = vec3_origin; // boxes don't rotate + } + + CM_TransformedBoxTrace(&trace, (float *)clip->start, (float *)clip->end, (float *)clip->mins, + (float *)clip->maxs, clipHandle, clip->contentmask, origin, angles, clip->collisionType); + + if (trace.allsolid) + { + clip->trace.allsolid = qtrue; + trace.entityNum = touch->s.number; + } + else if (trace.startsolid) + { + clip->trace.startsolid = qtrue; + trace.entityNum = touch->s.number; + } + + if (trace.fraction < clip->trace.fraction) + { + int oldStart; + + // make sure we keep a startsolid from a previous trace + oldStart = clip->trace.startsolid; + + trace.entityNum = touch->s.number; + clip->trace = trace; + clip->trace.startsolid |= oldStart; + } + } +} + +/* +================== +SV_Trace + +Moves the given mins/maxs volume through the world from start to end. +passEntityNum and entities owned by passEntityNum are explicitly not checked. +================== +*/ +void SV_Trace(trace_t *results, const vec3_t start, vec3_t mins, vec3_t maxs, const vec3_t end, int passEntityNum, + int contentmask, traceType_t type) +{ + moveclip_t clip; + int i; + + if (!mins) + { + mins = vec3_origin; + } + if (!maxs) + { + maxs = vec3_origin; + } + + ::memset(&clip, 0, sizeof(moveclip_t)); + + // clip to world + CM_BoxTrace(&clip.trace, start, end, mins, maxs, 0, contentmask, type); + clip.trace.entityNum = clip.trace.fraction != 1.0 ? ENTITYNUM_WORLD : ENTITYNUM_NONE; + if (clip.trace.fraction == 0) + { + *results = clip.trace; + return; // blocked immediately by the world + } + + clip.contentmask = contentmask; + clip.start = start; + // VectorCopy( clip.trace.endpos, clip.end ); + VectorCopy(end, clip.end); + clip.mins = mins; + clip.maxs = maxs; + clip.passEntityNum = passEntityNum; + clip.collisionType = type; + + // create the bounding box of the entire move + // we can limit it to the part of the move not + // already clipped off by the world, which can be + // a significant savings for line of sight and shot traces + for (i = 0; i < 3; i++) + { + if (end[i] > start[i]) + { + clip.boxmins[i] = clip.start[i] + clip.mins[i] - 1; + clip.boxmaxs[i] = clip.end[i] + clip.maxs[i] + 1; + } + else + { + clip.boxmins[i] = clip.end[i] + clip.mins[i] - 1; + clip.boxmaxs[i] = clip.start[i] + clip.maxs[i] + 1; + } + } + + // clip to other solid entities + SV_ClipMoveToEntities(&clip); + + *results = clip.trace; +} + +/* +============= +SV_PointContents +============= +*/ +int SV_PointContents(const vec3_t p, int passEntityNum) +{ + int touch[MAX_GENTITIES]; + sharedEntity_t *hit; + int i, num; + int contents, c2; + clipHandle_t clipHandle; + float *angles; + + // get base contents from world + contents = CM_PointContents(p, 0); + + // or in contents from all the other entities + num = SV_AreaEntities(p, p, touch, MAX_GENTITIES); + + for (i = 0; i < num; i++) + { + if (touch[i] == passEntityNum) + { + continue; + } + hit = SV_GentityNum(touch[i]); + // might intersect, so do an exact clip + clipHandle = SV_ClipHandleForEntity(hit); + angles = hit->r.currentAngles; + if (!hit->r.bmodel) + { + angles = vec3_origin; // boxes don't rotate + } + + c2 = CM_TransformedPointContents(p, clipHandle, hit->r.currentOrigin, angles); + + contents |= c2; + } + + return contents; +} |