]> git.xonotic.org Git - xonotic/darkplaces.git/blobdiff - sv_main.c
fix a couple EDICT_NUM errors that often happened when loading savegames with more...
[xonotic/darkplaces.git] / sv_main.c
index 71ffa7a8ba60b44b24e34a40915824ffa26a30bf..3919dddecde528267262eac769133edf3c2185f0 100644 (file)
--- a/sv_main.c
+++ b/sv_main.c
@@ -20,18 +20,20 @@ Foundation, Inc., 59 Temple Place - Suite 330, Boston, MA  02111-1307, USA.
 // sv_main.c -- server main program
 
 #include "quakedef.h"
+#include "portals.h"
 
 static cvar_t sv_cullentities_pvs = {0, "sv_cullentities_pvs", "0"}; // fast but loose
 static cvar_t sv_cullentities_portal = {0, "sv_cullentities_portal", "0"}; // extremely accurate visibility checking, but too slow
 static cvar_t sv_cullentities_trace = {0, "sv_cullentities_trace", "1"}; // tends to get false negatives, uses a timeout to keep entities visible a short time after becoming hidden
 static cvar_t sv_cullentities_stats = {0, "sv_cullentities_stats", "0"};
+static cvar_t sv_entpatch = {0, "sv_entpatch", "1"};
 
 server_t               sv;
 server_static_t        svs;
 
 static char localmodels[MAX_MODELS][5];                        // inline model names for precache
 
-static mempool_t *sv_edicts_mempool = NULL;
+mempool_t *sv_edicts_mempool = NULL;
 
 //============================================================================
 
@@ -63,6 +65,7 @@ void SV_Init (void)
        Cvar_RegisterVariable (&sv_cullentities_portal);
        Cvar_RegisterVariable (&sv_cullentities_trace);
        Cvar_RegisterVariable (&sv_cullentities_stats);
+       Cvar_RegisterVariable (&sv_entpatch);
 
        SV_Phys_Init();
        SV_World_Init();
@@ -246,7 +249,7 @@ This will be sent on the initial connection and upon each server load.
 void SV_SendServerinfo (client_t *client)
 {
        char                    **s;
-       char                    message[2048];
+       char                    message[128];
 
        // LordHavoc: clear entityframe tracking
        client->entityframenumber = 0;
@@ -265,9 +268,7 @@ void SV_SendServerinfo (client_t *client)
        else
                MSG_WriteByte (&client->message, GAME_COOP);
 
-       sprintf (message, PR_GetString(sv.edicts->v->message));
-
-       MSG_WriteString (&client->message,message);
+       MSG_WriteString (&client->message,PR_GetString(sv.edicts->v->message));
 
        for (s = sv.model_precache+1 ; *s ; s++)
                MSG_WriteString (&client->message, *s);
@@ -303,9 +304,7 @@ once for a player each game, not once for each level change.
 */
 void SV_ConnectClient (int clientnum)
 {
-       edict_t                 *ent;
        client_t                *client;
-       int                             edictnum;
        struct qsocket_s *netconnection;
        int                             i;
        float                   spawn_parms[NUM_SPAWN_PARMS];
@@ -314,10 +313,6 @@ void SV_ConnectClient (int clientnum)
 
        Con_DPrintf ("Client %s connected\n", client->netconnection->address);
 
-       edictnum = clientnum+1;
-
-       ent = EDICT_NUM(edictnum);
-
 // set up the client_t
        netconnection = client->netconnection;
 
@@ -329,7 +324,7 @@ void SV_ConnectClient (int clientnum)
        strcpy (client->name, "unconnected");
        client->active = true;
        client->spawned = false;
-       client->edict = ent;
+       client->edict = EDICT_NUM(clientnum+1);
        client->message.data = client->msgbuf;
        client->message.maxsize = sizeof(client->msgbuf);
        client->message.allowoverflow = true;           // we can catch it
@@ -344,14 +339,7 @@ void SV_ConnectClient (int clientnum)
                        client->spawn_parms[i] = (&pr_global_struct->parm1)[i];
        }
 
-#if NOROUTINGFIX
        SV_SendServerinfo (client);
-#else
-       // send serverinfo on first nop
-       client->waitingforconnect = true;
-       client->sendsignon = true;
-       client->spawned = false;                // need prespawn, spawn, etc
-#endif
 }
 
 
@@ -758,7 +746,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                        bits = bits | U_EXTERIORMODEL;
 
 // send an update
-               baseline = &ent->baseline;
+               baseline = &ent->e->baseline;
 
                if (((int)ent->v->effects & EF_DELTA) && sv_deltacompress.integer)
                {
@@ -766,7 +754,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                        if (realtime < client->nextfullupdate[e])
                        {
                                bits |= U_DELTA;
-                               baseline = &ent->deltabaseline;
+                               baseline = &ent->e->deltabaseline;
                        }
                        else
                                nextfullupdate = realtime + 0.5f;
@@ -806,17 +794,17 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                if (((int) baseline->frame & 0xFF00) != ((int) ent->v->modelindex & 0xFF00))            bits |= U_MODEL2;
 
                // update delta baseline
-               VectorCopy(ent->v->origin, ent->deltabaseline.origin);
-               VectorCopy(ent->v->angles, ent->deltabaseline.angles);
-               ent->deltabaseline.colormap = ent->v->colormap;
-               ent->deltabaseline.skin = ent->v->skin;
-               ent->deltabaseline.frame = ent->v->frame;
-               ent->deltabaseline.effects = ent->v->effects;
-               ent->deltabaseline.modelindex = ent->v->modelindex;
-               ent->deltabaseline.alpha = alpha;
-               ent->deltabaseline.scale = scale;
-               ent->deltabaseline.glowsize = glowsize;
-               ent->deltabaseline.glowcolor = glowcolor;
+               VectorCopy(ent->v->origin, ent->e->deltabaseline.origin);
+               VectorCopy(ent->v->angles, ent->e->deltabaseline.angles);
+               ent->e->deltabaseline.colormap = ent->v->colormap;
+               ent->e->deltabaseline.skin = ent->v->skin;
+               ent->e->deltabaseline.frame = ent->v->frame;
+               ent->e->deltabaseline.effects = ent->v->effects;
+               ent->e->deltabaseline.modelindex = ent->v->modelindex;
+               ent->e->deltabaseline.alpha = alpha;
+               ent->e->deltabaseline.scale = scale;
+               ent->e->deltabaseline.glowsize = glowsize;
+               ent->e->deltabaseline.glowcolor = glowcolor;
 
                // write the message
                if (bits >= 16777216)
@@ -867,6 +855,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                Con_Printf("client \"%s\" entities: %d total, %d visible, %d culled by: %d pvs %d portal %d trace\n", client->name, totalentities, visibleentities, culled_pvs + culled_portal + culled_trace, culled_pvs, culled_portal, culled_trace);
 }
 #else
+static entity_frame_t sv_writeentitiestoclient_entityframe;
 void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
 {
        int e, clentnum, flags, alpha, glowcolor, glowsize, scale, effects, modelindex;
@@ -878,12 +867,8 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
        eval_t *val;
        trace_t trace;
        model_t *model;
-       entity_frame_t entityframe;
        entity_state_t *s;
 
-       if (client->sendsignon)
-               return;
-
        Mod_CheckLoaded(sv.worldmodel);
 
 // find the client's PVS
@@ -899,7 +884,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
        if (e >= 128)
                e -= 256;
        testorigin[2] -= (float) e;
-       EntityFrame_Clear(&entityframe, testorigin);
+       EntityFrame_Clear(&sv_writeentitiestoclient_entityframe, testorigin);
 
        culled_pvs = 0;
        culled_portal = 0;
@@ -912,7 +897,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
        ent = NEXT_EDICT(sv.edicts);
        for (e = 1;e < sv.num_edicts;e++, ent = NEXT_EDICT(ent))
        {
-               if (ent->free)
+               if (ent->e->free)
                        continue;
                flags = 0;
 
@@ -1162,7 +1147,7 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                if ((effects & EF_LOWPRECISION) && origin[0] >= -32768 && origin[1] >= -32768 && origin[2] >= -32768 && origin[0] <= 32767 && origin[1] <= 32767 && origin[2] <= 32767)
                        flags |= RENDER_LOWPRECISION;
 
-               s = EntityFrame_NewEntity(&entityframe, e);
+               s = EntityFrame_NewEntity(&sv_writeentitiestoclient_entityframe, e);
                // if we run out of space, abort
                if (!s)
                        break;
@@ -1181,8 +1166,8 @@ void SV_WriteEntitiesToClient (client_t *client, edict_t *clent, sizebuf_t *msg)
                s->glowcolor = glowcolor;
                s->flags = flags;
        }
-       entityframe.framenum = ++client->entityframenumber;
-       EntityFrame_Write(&client->entitydatabase, &entityframe, msg);
+       sv_writeentitiestoclient_entityframe.framenum = ++client->entityframenumber;
+       EntityFrame_Write(&client->entitydatabase, &sv_writeentitiestoclient_entityframe, msg);
 
        if (sv_cullentities_stats.integer)
                Con_Printf("client \"%s\" entities: %d total, %d visible, %d culled by: %d pvs %d portal %d trace\n", client->name, totalentities, visibleentities, culled_pvs + culled_portal + culled_trace, culled_pvs, culled_portal, culled_trace);
@@ -1397,17 +1382,14 @@ qboolean SV_SendClientDatagram (client_t *client)
        MSG_WriteByte (&msg, svc_time);
        MSG_WriteFloat (&msg, sv.time);
 
-       if (client->spawned)
-       {
-               // add the client specific data to the datagram
-               SV_WriteClientdataToMessage (client->edict, &msg);
+       // add the client specific data to the datagram
+       SV_WriteClientdataToMessage (client->edict, &msg);
 
-               SV_WriteEntitiesToClient (client, client->edict, &msg);
+       SV_WriteEntitiesToClient (client, client->edict, &msg);
 
-               // copy the server datagram if there is space
-               if (msg.cursize + sv.datagram.cursize < msg.maxsize)
-                       SZ_Write (&msg, sv.datagram.data, sv.datagram.cursize);
-       }
+       // copy the server datagram if there is space
+       if (msg.cursize + sv.datagram.cursize < msg.maxsize)
+               SZ_Write (&msg, sv.datagram.data, sv.datagram.cursize);
 
 // send the datagram
        if (NET_SendUnreliableMessage (client->netconnection, &msg) == -1)
@@ -1426,13 +1408,14 @@ SV_UpdateToReliableMessages
 */
 void SV_UpdateToReliableMessages (void)
 {
-       int                     i, j;
+       int i, j;
        client_t *client;
 
 // check for changes to be sent over the reliable streams
        for (i=0, host_client = svs.clients ; i<svs.maxclients ; i++, host_client++)
        {
-               if (host_client->old_frags != host_client->edict->v->frags)
+               sv_player = host_client->edict;
+               if (host_client->old_frags != sv_player->v->frags)
                {
                        for (j=0, client = svs.clients ; j<svs.maxclients ; j++, client++)
                        {
@@ -1440,18 +1423,18 @@ void SV_UpdateToReliableMessages (void)
                                        continue;
                                MSG_WriteByte (&client->message, svc_updatefrags);
                                MSG_WriteByte (&client->message, i);
-                               MSG_WriteShort (&client->message, host_client->edict->v->frags);
+                               MSG_WriteShort (&client->message, sv_player->v->frags);
                        }
 
-                       host_client->old_frags = host_client->edict->v->frags;
+                       host_client->old_frags = sv_player->v->frags;
                }
        }
 
-       for (j=0, client = svs.clients ; j<svs.maxclients ; j++, client++)
+       for (j=0, host_client = svs.clients ; j<svs.maxclients ; j++, host_client++)
        {
-               if (!client->active)
+               if (!host_client->active)
                        continue;
-               SZ_Write (&client->message, sv.reliable_datagram.data, sv.reliable_datagram.cursize);
+               SZ_Write (&host_client->message, sv.reliable_datagram.data, sv.reliable_datagram.cursize);
        }
 
        SZ_Clear (&sv.reliable_datagram);
@@ -1500,14 +1483,6 @@ void SV_SendClientMessages (void)
                if (!host_client->active)
                        continue;
 
-#ifndef NOROUTINGFIX
-               if (host_client->sendserverinfo)
-               {
-                       host_client->sendserverinfo = false;
-                       SV_SendServerinfo (host_client);
-               }
-#endif
-
                if (host_client->spawned)
                {
                        if (!SV_SendClientDatagram (host_client))
@@ -1533,7 +1508,7 @@ void SV_SendClientMessages (void)
                // changes level
                if (host_client->message.overflowed)
                {
-                       SV_DropClient (true);
+                       SV_DropClient (true); // overflowed
                        host_client->message.overflowed = false;
                        continue;
                }
@@ -1610,31 +1585,31 @@ void SV_CreateBaseline (void)
                svent = EDICT_NUM(entnum);
 
                // LordHavoc: always clear state values, whether the entity is in use or not
-               ClearStateToDefault(&svent->baseline);
+               ClearStateToDefault(&svent->e->baseline);
 
-               if (svent->free)
+               if (svent->e->free)
                        continue;
                if (entnum > svs.maxclients && !svent->v->modelindex)
                        continue;
 
                // create entity baseline
-               VectorCopy (svent->v->origin, svent->baseline.origin);
-               VectorCopy (svent->v->angles, svent->baseline.angles);
-               svent->baseline.frame = svent->v->frame;
-               svent->baseline.skin = svent->v->skin;
+               VectorCopy (svent->v->origin, svent->e->baseline.origin);
+               VectorCopy (svent->v->angles, svent->e->baseline.angles);
+               svent->e->baseline.frame = svent->v->frame;
+               svent->e->baseline.skin = svent->v->skin;
                if (entnum > 0 && entnum <= svs.maxclients)
                {
-                       svent->baseline.colormap = entnum;
-                       svent->baseline.modelindex = SV_ModelIndex("progs/player.mdl");
+                       svent->e->baseline.colormap = entnum;
+                       svent->e->baseline.modelindex = SV_ModelIndex("progs/player.mdl");
                }
                else
                {
-                       svent->baseline.colormap = 0;
-                       svent->baseline.modelindex = svent->v->modelindex;
+                       svent->e->baseline.colormap = 0;
+                       svent->e->baseline.modelindex = svent->v->modelindex;
                }
 
                large = false;
-               if (svent->baseline.modelindex & 0xFF00 || svent->baseline.frame & 0xFF00)
+               if (svent->e->baseline.modelindex & 0xFF00 || svent->e->baseline.frame & 0xFF00)
                        large = true;
 
                // add to the message
@@ -1646,20 +1621,20 @@ void SV_CreateBaseline (void)
 
                if (large)
                {
-                       MSG_WriteShort (&sv.signon, svent->baseline.modelindex);
-                       MSG_WriteShort (&sv.signon, svent->baseline.frame);
+                       MSG_WriteShort (&sv.signon, svent->e->baseline.modelindex);
+                       MSG_WriteShort (&sv.signon, svent->e->baseline.frame);
                }
                else
                {
-                       MSG_WriteByte (&sv.signon, svent->baseline.modelindex);
-                       MSG_WriteByte (&sv.signon, svent->baseline.frame);
+                       MSG_WriteByte (&sv.signon, svent->e->baseline.modelindex);
+                       MSG_WriteByte (&sv.signon, svent->e->baseline.frame);
                }
-               MSG_WriteByte (&sv.signon, svent->baseline.colormap);
-               MSG_WriteByte (&sv.signon, svent->baseline.skin);
+               MSG_WriteByte (&sv.signon, svent->e->baseline.colormap);
+               MSG_WriteByte (&sv.signon, svent->e->baseline.skin);
                for (i=0 ; i<3 ; i++)
                {
-                       MSG_WriteDPCoord(&sv.signon, svent->baseline.origin[i]);
-                       MSG_WriteAngle(&sv.signon, svent->baseline.angles[i]);
+                       MSG_WriteDPCoord(&sv.signon, svent->e->baseline.origin[i]);
+                       MSG_WriteAngle(&sv.signon, svent->e->baseline.angles[i]);
                }
        }
 }
@@ -1718,6 +1693,43 @@ void SV_SaveSpawnparms (void)
        }
 }
 
+void SV_IncreaseEdicts(void)
+{
+       int i;
+       edict_t *ent;
+       int oldmax_edicts = sv.max_edicts;
+       void *oldedictsengineprivate = sv.edictsengineprivate;
+       void *oldedictsfields = sv.edictsfields;
+       void *oldmoved_edicts = sv.moved_edicts;
+
+       // links don't survive the transition, so unlink everything
+       for (i = 0;i < sv.max_edicts;i++)
+               SV_UnlinkEdict (sv.edicts + i);
+       SV_ClearWorld();
+
+       sv.max_edicts   = min(sv.max_edicts + 32, MAX_EDICTS);
+       sv.edictsengineprivate = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_engineprivate_t));
+       sv.edictsfields = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * pr_edict_size);
+       sv.moved_edicts = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_t *));
+
+       memcpy(sv.edictsengineprivate, oldedictsengineprivate, oldmax_edicts * sizeof(edict_engineprivate_t));
+       memcpy(sv.edictsfields, oldedictsfields, oldmax_edicts * pr_edict_size);
+
+       for (i = 0;i < sv.max_edicts;i++)
+       {
+               ent = sv.edicts + i;
+               ent->e = sv.edictsengineprivate + i;
+               ent->v = (void *)((qbyte *)sv.edictsfields + i * pr_edict_size);
+               // link every entity except world
+               if (i > 0)
+                       SV_LinkEdict(ent, false);
+       }
+
+       Mem_Free(oldedictsengineprivate);
+       Mem_Free(oldedictsfields);
+       Mem_Free(oldmoved_edicts);
+}
+
 /*
 ================
 SV_SpawnServer
@@ -1729,8 +1741,9 @@ extern float              scr_centertime_off;
 
 void SV_SpawnServer (const char *server)
 {
-       edict_t         *ent;
-       int                     i;
+       edict_t *ent;
+       int i;
+       qbyte *entities;
 
        // let's not have any servers with no name
        if (hostname.string[0] == 0)
@@ -1768,23 +1781,25 @@ void SV_SpawnServer (const char *server)
        PR_LoadProgs ();
 
 // allocate server memory
-       sv.max_edicts = MAX_EDICTS;
+       // start out with just enough room for clients and a reasonable estimate of entities
+       sv.max_edicts = ((svs.maxclients + 128) + 31) & ~31;
+       sv.max_edicts = min(sv.max_edicts, MAX_EDICTS);
 
        // clear the edict memory pool
        Mem_EmptyPool(sv_edicts_mempool);
        // edict_t structures (hidden from progs)
-       sv.edicts = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_t));
+       sv.edicts = Mem_Alloc(sv_edicts_mempool, MAX_EDICTS * sizeof(edict_t));
+       // engine private structures (hidden from progs)
+       sv.edictsengineprivate = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_engineprivate_t));
        // progs fields, often accessed by server
        sv.edictsfields = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * pr_edict_size);
-       // table of edict pointers, for quicker lookup of edicts
-       sv.edictstable = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_t *));
        // used by PushMove to move back pushed entities
        sv.moved_edicts = Mem_Alloc(sv_edicts_mempool, sv.max_edicts * sizeof(edict_t *));
        for (i = 0;i < sv.max_edicts;i++)
        {
                ent = sv.edicts + i;
+               ent->e = sv.edictsengineprivate + i;
                ent->v = (void *)((qbyte *)sv.edictsfields + i * pr_edict_size);
-               sv.edictstable[i] = ent;
        }
 
        sv.datagram.maxsize = sizeof(sv.datagram_buf);
@@ -1802,10 +1817,7 @@ void SV_SpawnServer (const char *server)
 // leave slots at start for clients only
        sv.num_edicts = svs.maxclients+1;
        for (i=0 ; i<svs.maxclients ; i++)
-       {
-               ent = EDICT_NUM(i+1);
-               svs.clients[i].edict = ent;
-       }
+               svs.clients[i].edict = EDICT_NUM(i+1);
 
        sv.state = ss_loading;
        sv.paused = false;
@@ -1845,7 +1857,7 @@ void SV_SpawnServer (const char *server)
 //
        ent = EDICT_NUM(0);
        memset (ent->v, 0, progs->entityfields * 4);
-       ent->free = false;
+       ent->e->free = false;
        ent->v->model = PR_SetString(sv.modelname);
        ent->v->modelindex = 1;         // world model
        ent->v->solid = SOLID_BSP;
@@ -1861,7 +1873,20 @@ void SV_SpawnServer (const char *server)
 // serverflags are for cross level information (sigils)
        pr_global_struct->serverflags = svs.serverflags;
 
-       ED_LoadFromFile (sv.worldmodel->entities);
+       // load replacement entity file if found
+       entities = NULL;
+       if (sv_entpatch.integer)
+               entities = FS_LoadFile(va("maps/%s.ent", sv.name), true);
+       if (entities)
+       {
+               Con_Printf("Loaded maps/%s.ent\n", sv.name);
+               ED_LoadFromFile (entities);
+               Mem_Free(entities);
+       }
+       else
+               ED_LoadFromFile (sv.worldmodel->entities);
+
+
        // LordHavoc: clear world angles (to fix e3m3.bsp)
        VectorClear(sv.edicts->v->angles);