123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928 |
- //**************************************************************************
- //**
- //** d_net.c : Heretic 2 : Raven Software, Corp.
- //**
- //** $RCSfile: d_net.c,v $
- //** $Revision: 1.16 $
- //** $Date: 96/01/01 03:39:44 $
- //** $Author: bgokey $
- //**
- //** This version has the fixed ticdup code.
- //**
- //**************************************************************************
- #include "h2def.h"
- #include "p_local.h"
- #include <stdlib.h> // for atoi()
- #define NCMD_EXIT 0x80000000
- #define NCMD_RETRANSMIT 0x40000000
- #define NCMD_SETUP 0x20000000
- #define NCMD_KILL 0x10000000 // kill game
- #define NCMD_CHECKSUM 0x0fffffff
- doomcom_t *doomcom;
- doomdata_t *netbuffer; // points inside doomcom
- /*
- ==============================================================================
- NETWORKING
- gametic is the tic about to (or currently being) run
- maketic is the tick that hasn't had control made for it yet
- nettics[] has the maketics for all players
- a gametic cannot be run until nettics[] > gametic for all players
- ==============================================================================
- */
- #define RESENDCOUNT 10
- #define PL_DRONE 0x80 // bit flag in doomdata->player
- ticcmd_t localcmds[BACKUPTICS];
- ticcmd_t netcmds[MAXPLAYERS][BACKUPTICS];
- int nettics[MAXNETNODES];
- boolean nodeingame[MAXNETNODES]; // set false as nodes leave game
- boolean remoteresend[MAXNETNODES]; // set when local needs tics
- int resendto[MAXNETNODES]; // set when remote needs tics
- int resendcount[MAXNETNODES];
- int nodeforplayer[MAXPLAYERS];
- int maketic;
- int lastnettic, skiptics;
- int ticdup;
- int maxsend; // BACKUPTICS/(2*ticdup)-1
- void H2_ProcessEvents (void);
- void G_BuildTiccmd (ticcmd_t *cmd);
- void H2_DoAdvanceDemo (void);
- extern void ST_NetProgress(void);
- extern void ST_NetDone(void);
- boolean reboundpacket;
- doomdata_t reboundstore;
- int NetbufferSize (void)
- {
- return (int)&(((doomdata_t *)0)->cmds[netbuffer->numtics]);
- }
- unsigned NetbufferChecksum (void)
- {
- unsigned c;
- int i,l;
- c = 0x1234567;
- #if defined(NeXT) || defined(NORMALUNIX)
- return 0; // byte order problems
- #endif
- l = (NetbufferSize () - (int)&(((doomdata_t *)0)->retransmitfrom))/4;
- for (i=0 ; i<l ; i++)
- c += ((unsigned *)&netbuffer->retransmitfrom)[i] * (i+1);
- return c & NCMD_CHECKSUM;
- }
- int ExpandTics (int low)
- {
- int delta;
- delta = low - (maketic&0xff);
- if (delta >= -64 && delta <= 64)
- return (maketic&~0xff) + low;
- if (delta > 64)
- return (maketic&~0xff) - 256 + low;
- if (delta < -64)
- return (maketic&~0xff) + 256 + low;
- I_Error ("ExpandTics: strange value %i at maketic %i",low,maketic);
- return 0;
- }
- //============================================================================
- /*
- ==============
- =
- = HSendPacket
- =
- ==============
- */
- void HSendPacket (int node, int flags)
- {
- netbuffer->checksum = NetbufferChecksum () | flags;
- if (!node)
- {
- reboundstore = *netbuffer;
- reboundpacket = true;
- return;
- }
- if (demoplayback)
- return;
- if (!netgame)
- I_Error ("Tried to transmit to another node");
- doomcom->command = CMD_SEND;
- doomcom->remotenode = node;
- doomcom->datalength = NetbufferSize ();
- if (debugfile)
- {
- int i;
- int realretrans;
- if (netbuffer->checksum & NCMD_RETRANSMIT)
- realretrans = ExpandTics (netbuffer->retransmitfrom);
- else
- realretrans = -1;
- fprintf (debugfile,"send (%i + %i, R %i) [%i] "
- ,ExpandTics(netbuffer->starttic),netbuffer->numtics, realretrans, doomcom->datalength);
- for (i=0 ; i<doomcom->datalength ; i++)
- fprintf (debugfile,"%i ",((byte *)netbuffer)[i]);
- fprintf (debugfile,"\n");
- }
- I_NetCmd ();
- }
- //==========================================================================
- //
- // NET_SendFrags
- //
- //==========================================================================
- void NET_SendFrags(player_t *player)
- {
- int i;
- int frags;
- netbuffer->checksum = NetbufferChecksum();
- if (demoplayback)
- {
- return;
- }
- if (!netgame)
- {
- I_Error ("Tried to transmit to another node");
- }
- frags = 0;
- for(i = 0; i < MAXPLAYERS; i++)
- {
- frags += player->frags[i];
- }
- doomcom->command = CMD_FRAG;
- doomcom->remotenode = frags;
- doomcom->datalength = NetbufferSize ();
- I_NetCmd ();
- }
- /*
- ==============
- =
- = HGetPacket
- =
- = Returns false if no packet is waiting
- =
- ==============
- */
- boolean HGetPacket (void)
- {
- if (reboundpacket)
- {
- *netbuffer = reboundstore;
- doomcom->remotenode = 0;
- reboundpacket = false;
- return true;
- }
- if (!netgame)
- return false;
- if (demoplayback)
- return false;
- doomcom->command = CMD_GET;
- I_NetCmd ();
- if (doomcom->remotenode == -1)
- return false;
- if (doomcom->datalength != NetbufferSize ())
- {
- if (debugfile)
- fprintf (debugfile,"bad packet length %i\n",doomcom->datalength);
- return false;
- }
- if (NetbufferChecksum () != (netbuffer->checksum&NCMD_CHECKSUM) )
- {
- if (debugfile)
- fprintf (debugfile,"bad packet checksum\n");
- return false;
- }
- if (debugfile)
- {
- int realretrans;
- int i;
- if (netbuffer->checksum & NCMD_SETUP)
- fprintf (debugfile,"setup packet\n");
- else
- {
- if (netbuffer->checksum & NCMD_RETRANSMIT)
- realretrans = ExpandTics (netbuffer->retransmitfrom);
- else
- realretrans = -1;
- fprintf (debugfile,"get %i = (%i + %i, R %i)[%i] ",doomcom->remotenode,
- ExpandTics(netbuffer->starttic),netbuffer->numtics, realretrans, doomcom->datalength);
- for (i=0 ; i<doomcom->datalength ; i++)
- fprintf (debugfile,"%i ",((byte *)netbuffer)[i]);
- fprintf (debugfile,"\n");
- }
- }
- return true;
- }
- /*
- ===================
- =
- = GetPackets
- =
- ===================
- */
- char exitmsg[80];
- void GetPackets (void)
- {
- int netconsole;
- int netnode;
- ticcmd_t *src, *dest;
- int realend;
- int realstart;
- while (HGetPacket ())
- {
- if (netbuffer->checksum & NCMD_SETUP)
- continue; // extra setup packet
- netconsole = netbuffer->player & ~PL_DRONE;
- netnode = doomcom->remotenode;
- //
- // to save bytes, only the low byte of tic numbers are sent
- // Figure out what the rest of the bytes are
- //
- realstart = ExpandTics (netbuffer->starttic);
- realend = (realstart+netbuffer->numtics);
- //
- // check for exiting the game
- //
- if (netbuffer->checksum & NCMD_EXIT)
- {
- if (!nodeingame[netnode])
- continue;
- nodeingame[netnode] = false;
- playeringame[netconsole] = false;
- strcpy (exitmsg, "PLAYER 1 LEFT THE GAME");
- exitmsg[7] += netconsole;
- P_SetMessage(&players[consoleplayer], exitmsg, true);
- S_StartSound(NULL, SFX_CHAT);
- // players[consoleplayer].message = exitmsg;
- // if (demorecording)
- // G_CheckDemoStatus ();
- continue;
- }
- //
- // check for a remote game kill
- //
- if (netbuffer->checksum & NCMD_KILL)
- I_Error ("Killed by network driver");
- nodeforplayer[netconsole] = netnode;
- //
- // check for retransmit request
- //
- if ( resendcount[netnode] <= 0
- && (netbuffer->checksum & NCMD_RETRANSMIT) )
- {
- resendto[netnode] = ExpandTics(netbuffer->retransmitfrom);
- if (debugfile)
- fprintf (debugfile,"retransmit from %i\n", resendto[netnode]);
- resendcount[netnode] = RESENDCOUNT;
- }
- else
- resendcount[netnode]--;
- //
- // check for out of order / duplicated packet
- //
- if (realend == nettics[netnode])
- continue;
- if (realend < nettics[netnode])
- {
- if (debugfile)
- fprintf (debugfile,"out of order packet (%i + %i)\n" ,realstart,netbuffer->numtics);
- continue;
- }
- //
- // check for a missed packet
- //
- if (realstart > nettics[netnode])
- {
- // stop processing until the other system resends the missed tics
- if (debugfile)
- fprintf (debugfile,"missed tics from %i (%i - %i)\n", netnode, realstart, nettics[netnode]);
- remoteresend[netnode] = true;
- continue;
- }
- //
- // update command store from the packet
- //
- {
- int start;
- remoteresend[netnode] = false;
- start = nettics[netnode] - realstart;
- src = &netbuffer->cmds[start];
- while (nettics[netnode] < realend)
- {
- dest = &netcmds[netconsole][nettics[netnode]%BACKUPTICS];
- nettics[netnode]++;
- *dest = *src;
- src++;
- }
- }
- }
- }
- /*
- =============
- =
- = NetUpdate
- =
- = Builds ticcmds for console player
- = sends out a packet
- =============
- */
- int gametime;
- void NetUpdate (void)
- {
- int nowtime;
- int newtics;
- int i,j;
- int realstart;
- int gameticdiv;
- //
- // check time
- //
- nowtime = I_GetTime ()/ticdup;
- newtics = nowtime - gametime;
- gametime = nowtime;
- if (newtics <= 0) // nothing new to update
- goto listen;
- if (skiptics <= newtics)
- {
- newtics -= skiptics;
- skiptics = 0;
- }
- else
- {
- skiptics -= newtics;
- newtics = 0;
- }
- netbuffer->player = consoleplayer;
- //
- // build new ticcmds for console player
- //
- gameticdiv = gametic/ticdup;
- for (i=0 ; i<newtics ; i++)
- {
- I_StartTic ();
- H2_ProcessEvents ();
- if (maketic - gameticdiv >= BACKUPTICS/2-1)
- break; // can't hold any more
- //printf ("mk:%i ",maketic);
- G_BuildTiccmd (&localcmds[maketic%BACKUPTICS]);
- maketic++;
- }
- if (singletics)
- return; // singletic update is syncronous
- //
- // send the packet to the other nodes
- //
- for (i=0 ; i<doomcom->numnodes ; i++)
- if (nodeingame[i])
- {
- netbuffer->starttic = realstart = resendto[i];
- netbuffer->numtics = maketic - realstart;
- if (netbuffer->numtics > BACKUPTICS)
- I_Error ("NetUpdate: netbuffer->numtics > BACKUPTICS");
- resendto[i] = maketic - doomcom->extratics;
- for (j=0 ; j< netbuffer->numtics ; j++)
- netbuffer->cmds[j] =
- localcmds[(realstart+j)%BACKUPTICS];
- if (remoteresend[i])
- {
- netbuffer->retransmitfrom = nettics[i];
- HSendPacket (i, NCMD_RETRANSMIT);
- }
- else
- {
- netbuffer->retransmitfrom = 0;
- HSendPacket (i, 0);
- }
- }
- //
- // listen for other packets
- //
- listen:
- GetPackets ();
- }
- /*
- =====================
- =
- = CheckAbort
- =
- =====================
- */
- void CheckAbort (void)
- {
- event_t *ev;
- int stoptic;
- stoptic = I_GetTime () + 2;
- while (I_GetTime() < stoptic)
- I_StartTic ();
- I_StartTic ();
- for ( ; eventtail != eventhead
- ; eventtail = (++eventtail)&(MAXEVENTS-1) )
- {
- ev = &events[eventtail];
- if (ev->type == ev_keydown && ev->data1 == KEY_ESCAPE)
- I_Error ("Network game synchronization aborted.");
- }
- }
- /*
- =====================
- =
- = D_ArbitrateNetStart
- =
- =====================
- */
- void D_ArbitrateNetStart (void)
- {
- int i;
- boolean gotinfo[MAXNETNODES];
- boolean gotClass[MAXNETNODES];
- #ifdef __WATCOMC__
- int nextTic;
- extern volatile int ticcount;
- nextTic = ticcount+8;
- #endif
- autostart = true;
- memset (gotClass,0,sizeof(gotClass));
- memset (gotinfo,0,sizeof(gotinfo));
- gotClass[doomcom->consoleplayer] = true;
- do
- {
- i = 0;
- CheckAbort();
- while(HGetPacket())
- { // Check for any incoming packets
- if(netbuffer->checksum&NCMD_SETUP && netbuffer->starttic >= 64)
- {
-
- PlayerClass[netbuffer->player] = netbuffer->starttic&0x3f;
- if(!gotClass[netbuffer->player])
- {
- gotClass[netbuffer->player] = true;
- ST_NetProgress();
- ST_Message("\n");
- }
- if(netbuffer->retransmitfrom)
- { // that node has received info from all other nodes
- gotinfo[netbuffer->player] = true;
- }
- }
- }
- #ifdef __WATCOMC__
- if(ticcount <= nextTic)
- { // only send packets every half second
- continue;
- }
- nextTic = ticcount+8;
- #endif
- // Keep sending out packets containing the console class
- for(i = 0; i < doomcom->numnodes; i++)
- {
- netbuffer->player = doomcom->consoleplayer;
- netbuffer->starttic = PlayerClass[doomcom->consoleplayer]+64;
- netbuffer->retransmitfrom = gotinfo[doomcom->consoleplayer];
- netbuffer->numtics = 0;
- HSendPacket(i, NCMD_SETUP);
- }
- for(i = 0; i < doomcom->numnodes; i++)
- { // Make sure that all nodes have sent class info
- if (!gotClass[i])
- {
- ST_Message(".");
- break;
- }
- }
- if(i < doomcom->numnodes)
- {
- continue;
- }
- else
- { // consoleplayer has received all player classes
- if(gotinfo[doomcom->consoleplayer])
- {
- CheckAbort();
- }
- else
- {
- gotinfo[doomcom->consoleplayer] = true;
- ST_Message("All player classes received, ready to proceed\n");
- ST_NetDone();
- }
- }
- for (i = 0; i < doomcom->numnodes; i++)
- { // Make sure that all nodes are ready to proceed
- if (!gotinfo[i])
- {
- break;
- }
- }
- } while(i < doomcom->numnodes);
- memset (gotinfo,0,sizeof(gotinfo));
- if (doomcom->consoleplayer)
- { // listen for setup info from key player
- // ST_Message ("listening for network start info...\n");
- while (1)
- {
- CheckAbort ();
- if (!HGetPacket ())
- continue;
- if(netbuffer->checksum & NCMD_SETUP && netbuffer->starttic < 64)
- {
- if (netbuffer->player != VERSION)
- I_Error ("Different HEXEN versions cannot play a net game!");
- startskill = netbuffer->retransmitfrom & 15;
- deathmatch = (netbuffer->retransmitfrom & 0xc0) >> 6;
- nomonsters = (netbuffer->retransmitfrom & 0x20) > 0;
- respawnparm = (netbuffer->retransmitfrom & 0x10) > 0;
- startmap = netbuffer->starttic & 0x3f;
- startepisode = 1;
- return;
- }
- }
- }
- else
- { // key player, send the setup info
- // ST_Message ("sending network start info...\n");
- do
- {
- CheckAbort ();
- for (i=0 ; i<doomcom->numnodes ; i++)
- {
- netbuffer->retransmitfrom = startskill;
- if (deathmatch)
- netbuffer->retransmitfrom |= (deathmatch<<6);
- if (nomonsters)
- netbuffer->retransmitfrom |= 0x20;
- if (respawnparm)
- netbuffer->retransmitfrom |= 0x10;
- netbuffer->starttic = startmap&0x3f;
- netbuffer->player = VERSION;
- netbuffer->numtics = 0;
- HSendPacket (i, NCMD_SETUP);
- }
- #if 1
- for(i = 10 ; i && HGetPacket(); --i)
- {
- if((netbuffer->player&0x7f) < MAXNETNODES)
- gotinfo[netbuffer->player&0x7f] = true;
- }
- #else
- while (HGetPacket ())
- {
- gotinfo[netbuffer->player&0x7f] = true;
- }
- #endif
- for (i=1 ; i<doomcom->numnodes ; i++)
- if (!gotinfo[i])
- break;
- } while (i < doomcom->numnodes);
- }
- }
- /*
- ===================
- =
- = D_CheckNetGame
- =
- = Works out player numbers among the net participants
- ===================
- */
- extern int viewangleoffset;
- void D_CheckNetGame (void)
- {
- int i;
- int pClass;
- for (i=0 ; i<MAXNETNODES ; i++)
- {
- nodeingame[i] = false;
- nettics[i] = 0;
- remoteresend[i] = false; // set when local needs tics
- resendto[i] = 0; // which tic to start sending
- }
- // I_InitNetwork sets doomcom and netgame
- I_InitNetwork ();
- if (doomcom->id != DOOMCOM_ID)
- I_Error ("Doomcom buffer invalid!");
- netbuffer = &doomcom->data;
- consoleplayer = displayplayer = doomcom->consoleplayer;
- pClass = PCLASS_FIGHTER;
- if(i = M_CheckParm("-class"))
- {
- pClass = atoi(myargv[i+1]);
- if(pClass > PCLASS_MAGE || pClass < PCLASS_FIGHTER)
- {
- I_Error("Invalid player class: %d\n", pClass);
- }
- ST_Message("\nPlayer Class: %d\n", pClass);
- }
- PlayerClass[consoleplayer] = pClass;
- if (netgame)
- D_ArbitrateNetStart ();
- //ST_Message ("startskill %i deathmatch: %i startmap: %i startepisode: %i\n", startskill, deathmatch, startmap, startepisode);
- // read values out of doomcom
- ticdup = doomcom->ticdup;
- maxsend = BACKUPTICS/(2*ticdup)-1;
- if (maxsend<1)
- maxsend = 1;
- for (i=0 ; i<doomcom->numplayers ; i++)
- playeringame[i] = true;
- for (i=0 ; i<doomcom->numnodes ; i++)
- nodeingame[i] = true;
- //ST_Message ("player %i of %i (%i nodes)\n", consoleplayer+1, doomcom->numplayers, doomcom->numnodes);
- }
- /*
- ==================
- =
- = D_QuitNetGame
- =
- = Called before quitting to leave a net game without hanging the
- = other players
- =
- ==================
- */
- void D_QuitNetGame (void)
- {
- int i, j;
- if (debugfile)
- fclose (debugfile);
- if (!netgame || !usergame || consoleplayer == -1 || demoplayback)
- return;
- // send a bunch of packets for security
- netbuffer->player = consoleplayer;
- netbuffer->numtics = 0;
- for (i=0 ; i<4 ; i++)
- {
- for (j=1 ; j<doomcom->numnodes ; j++)
- if (nodeingame[j])
- HSendPacket (j, NCMD_EXIT);
- I_WaitVBL (1);
- }
- }
- /*
- ===============
- =
- = TryRunTics
- =
- ===============
- */
- int frametics[4], frameon;
- int frameskip[4];
- int oldnettics;
- extern boolean advancedemo;
- void TryRunTics (void)
- {
- int i;
- int lowtic;
- int entertic;
- static int oldentertics;
- int realtics, availabletics;
- int counts;
- int numplaying;
- //
- // get real tics
- //
- entertic = I_GetTime ()/ticdup;
- realtics = entertic - oldentertics;
- oldentertics = entertic;
- //
- // get available tics
- //
- NetUpdate ();
- lowtic = MAXINT;
- numplaying = 0;
- for (i=0 ; i<doomcom->numnodes ; i++)
- if (nodeingame[i])
- {
- numplaying++;
- if (nettics[i] < lowtic)
- lowtic = nettics[i];
- }
- availabletics = lowtic - gametic/ticdup;
- //
- // decide how many tics to run
- //
- if (realtics < availabletics-1)
- counts = realtics+1;
- else if (realtics < availabletics)
- counts = realtics;
- else
- counts = availabletics;
- if (counts < 1)
- counts = 1;
- frameon++;
- if (debugfile)
- fprintf (debugfile,"=======real: %i avail: %i game: %i\n",realtics, availabletics,counts);
- if (!demoplayback)
- {
- //=============================================================================
- //
- // ideally nettics[0] should be 1 - 3 tics above lowtic
- // if we are consistantly slower, speed up time
- //
- for (i=0 ; i<MAXPLAYERS ; i++)
- if (playeringame[i])
- break;
- if (consoleplayer == i)
- { // the key player does not adapt
- }
- else
- {
- if (nettics[0] <= nettics[nodeforplayer[i]])
- {
- gametime--;
- // printf ("-");
- }
- frameskip[frameon&3] = (oldnettics > nettics[nodeforplayer[i]]);
- oldnettics = nettics[0];
- if (frameskip[0] && frameskip[1] && frameskip[2] && frameskip[3])
- {
- skiptics = 1;
- // printf ("+");
- }
- }
- //=============================================================================
- } // demoplayback
- //
- // wait for new tics if needed
- //
- while (lowtic < gametic/ticdup + counts)
- {
- NetUpdate ();
- lowtic = MAXINT;
- for (i=0 ; i<doomcom->numnodes ; i++)
- if (nodeingame[i] && nettics[i] < lowtic)
- lowtic = nettics[i];
- if (lowtic < gametic/ticdup)
- I_Error ("TryRunTics: lowtic < gametic");
- // don't stay in here forever -- give the menu a chance to work
- if (I_GetTime ()/ticdup - entertic >= 20)
- {
- MN_Ticker ();
- return;
- }
- }
- //
- // run the count * ticdup dics
- //
- while (counts--)
- {
- for (i=0 ; i<ticdup ; i++)
- {
- if (gametic/ticdup > lowtic)
- I_Error ("gametic>lowtic");
- if (advancedemo)
- H2_DoAdvanceDemo ();
- MN_Ticker ();
- G_Ticker ();
- gametic++;
- //
- // modify command for duplicated tics
- //
- if (i != ticdup-1)
- {
- ticcmd_t *cmd;
- int buf;
- int j;
- buf = (gametic/ticdup)%BACKUPTICS;
- for (j=0 ; j<MAXPLAYERS ; j++)
- {
- cmd = &netcmds[j][buf];
- cmd->chatchar = 0;
- if (cmd->buttons & BT_SPECIAL)
- cmd->buttons = 0;
- }
- }
- }
- NetUpdate (); // check for new console commands
- }
- }
|