mirror of
https://github.com/ZDoom/Raze.git
synced 2024-12-19 00:51:19 +00:00
410 lines
10 KiB
C
410 lines
10 KiB
C
//-------------------------------------------------------------------------
|
|
/*
|
|
Copyright (C) 1997, 2005 - 3D Realms Entertainment
|
|
|
|
This file is part of Shadow Warrior version 1.2
|
|
|
|
Shadow Warrior 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.
|
|
|
|
Original Source: 1997 - Frank Maddin and Jim Norwood
|
|
Prepared for public release: 03/28/2005 - Charlie Wiederhold, 3D Realms
|
|
*/
|
|
//-------------------------------------------------------------------------
|
|
|
|
int32_t g_numSyncBytes = 1;
|
|
char g_szfirstSyncMsg[MAXSYNCBYTES][60];
|
|
int32_t g_foundSyncError = 0;
|
|
|
|
static int32_t crctable[256];
|
|
#define updatecrc(dcrc,xz) (dcrc = (crctable[((dcrc)>>8)^((xz)&255)]^((dcrc)<<8)))
|
|
|
|
void initsynccrc(void)
|
|
{
|
|
int32_t i, j, k, a;
|
|
|
|
for (j=0; j<256; j++) //Calculate CRC table
|
|
{
|
|
k = (j<<8); a = 0;
|
|
for (i=7; i>=0; i--)
|
|
{
|
|
if (((k^a)&0x8000) > 0)
|
|
a = ((a<<1)&65535) ^ 0x1021; //0x1021 = genpoly
|
|
else
|
|
a = ((a<<1)&65535);
|
|
k = ((k<<1)&65535);
|
|
}
|
|
crctable[j] = (a&65535);
|
|
}
|
|
}
|
|
|
|
char Net_PlayerSync(void)
|
|
{
|
|
int16_t i;
|
|
uint16_t crc = 0;
|
|
DukePlayer_t *pp;
|
|
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
pp = g_player[i].ps;
|
|
updatecrc(crc, pp->posx & 255);
|
|
updatecrc(crc, pp->posy & 255);
|
|
updatecrc(crc, pp->posz & 255);
|
|
updatecrc(crc, pp->ang & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char Net_PlayerSync2(void)
|
|
{
|
|
int32_t i;
|
|
int32_t j, nextj;
|
|
uint16_t crc = 0;
|
|
DukePlayer_t *pp;
|
|
spritetype *spr;
|
|
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
pp = g_player[i].ps;
|
|
|
|
updatecrc(crc, pp->horiz & 255);
|
|
updatecrc(crc, sprite[pp->i].extra & 255);
|
|
updatecrc(crc, pp->bobcounter & 255);
|
|
}
|
|
|
|
TRAVERSE_SPRITE_STAT(headspritestat[STAT_PLAYER], j, nextj)
|
|
{
|
|
spr = &sprite[j];
|
|
updatecrc(crc, (spr->x) & 255);
|
|
updatecrc(crc, (spr->y) & 255);
|
|
updatecrc(crc, (spr->z) & 255);
|
|
updatecrc(crc, (spr->ang) & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char Net_ActorSync(void)
|
|
{
|
|
uint16_t crc = 0;
|
|
int32_t j, nextj;
|
|
spritetype *spr;
|
|
|
|
TRAVERSE_SPRITE_STAT(headspritestat[STAT_ACTOR], j, nextj)
|
|
{
|
|
spr = &sprite[j];
|
|
updatecrc(crc, (spr->x) & 255);
|
|
updatecrc(crc, (spr->y) & 255);
|
|
updatecrc(crc, (spr->z) & 255);
|
|
updatecrc(crc, (spr->lotag) & 255);
|
|
updatecrc(crc, (spr->hitag) & 255);
|
|
updatecrc(crc, (spr->ang) & 255);
|
|
}
|
|
|
|
TRAVERSE_SPRITE_STAT(headspritestat[STAT_ZOMBIEACTOR], j, nextj)
|
|
{
|
|
spr = &sprite[j];
|
|
updatecrc(crc, (spr->x) & 255);
|
|
updatecrc(crc, (spr->y) & 255);
|
|
updatecrc(crc, (spr->z) & 255);
|
|
updatecrc(crc, (spr->lotag) & 255);
|
|
updatecrc(crc, (spr->hitag) & 255);
|
|
updatecrc(crc, (spr->ang) & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char Net_WeaponSync(void)
|
|
{
|
|
uint16_t crc = 0;
|
|
int32_t j, nextj;
|
|
spritetype *spr;
|
|
|
|
TRAVERSE_SPRITE_STAT(headspritestat[STAT_PROJECTILE], j, nextj)
|
|
{
|
|
spr = &sprite[j];
|
|
updatecrc(crc, (spr->x) & 255);
|
|
updatecrc(crc, (spr->y) & 255);
|
|
updatecrc(crc, (spr->z) & 255);
|
|
updatecrc(crc, (spr->ang) & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char Net_MapSync(void)
|
|
{
|
|
uint16_t crc = 0;
|
|
int32_t j, nextj;
|
|
spritetype *spr;
|
|
walltype *wal;
|
|
sectortype *sect;
|
|
|
|
TRAVERSE_SPRITE_STAT(headspritestat[STAT_EFFECTOR], j, nextj)
|
|
{
|
|
spr = &sprite[j];
|
|
updatecrc(crc, (spr->x) & 255);
|
|
updatecrc(crc, (spr->y) & 255);
|
|
updatecrc(crc, (spr->z) & 255);
|
|
updatecrc(crc, (spr->ang) & 255);
|
|
updatecrc(crc, (spr->lotag) & 255);
|
|
updatecrc(crc, (spr->hitag) & 255);
|
|
}
|
|
|
|
for (j=numwalls; j>=0; j--)
|
|
{
|
|
wal = &wall[j];
|
|
updatecrc(crc, (wal->x) & 255);
|
|
updatecrc(crc, (wal->y) & 255);
|
|
}
|
|
|
|
for (j=numsectors; j>=0; j--)
|
|
{
|
|
sect = §or[j];
|
|
updatecrc(crc, (sect->floorz) & 255);
|
|
updatecrc(crc, (sect->ceilingz) & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char Net_RandomSync(void)
|
|
{
|
|
uint16_t crc = 0;
|
|
|
|
updatecrc(crc, randomseed & 255);
|
|
updatecrc(crc, (randomseed >> 8) & 255);
|
|
updatecrc(crc, g_globalRandom & 255);
|
|
updatecrc(crc, (g_globalRandom >> 8) & 255);
|
|
|
|
if (g_numSyncBytes == 1)
|
|
{
|
|
updatecrc(crc,Net_PlayerSync() & 255);
|
|
updatecrc(crc,Net_PlayerSync2() & 255);
|
|
updatecrc(crc,Net_WeaponSync() & 255);
|
|
updatecrc(crc,Net_ActorSync() & 255);
|
|
updatecrc(crc,Net_MapSync() & 255);
|
|
}
|
|
|
|
return ((uint8_t) crc & 255);
|
|
}
|
|
|
|
char *SyncNames[] =
|
|
{
|
|
"Net_CheckRandomSync",
|
|
"Net_CheckPlayerSync",
|
|
"Net_CheckPlayerSync2",
|
|
"Net_CheckWeaponSync",
|
|
"Net_CheckActorSync",
|
|
"Net_CheckMapSync",
|
|
NULL
|
|
};
|
|
|
|
static char(*SyncFunc[MAXSYNCBYTES + 1])(void) =
|
|
{
|
|
Net_RandomSync,
|
|
Net_PlayerSync,
|
|
Net_PlayerSync2,
|
|
Net_WeaponSync,
|
|
Net_ActorSync,
|
|
Net_MapSync,
|
|
NULL
|
|
};
|
|
|
|
void Net_GetSyncStat(void)
|
|
{
|
|
int32_t i;
|
|
playerdata_t *pp = &g_player[myconnectindex];
|
|
uint32_t val;
|
|
static uint32_t count;
|
|
|
|
if (numplayers < 2)
|
|
return;
|
|
|
|
for (i = 0; SyncFunc[i]; i++)
|
|
{
|
|
pp->syncval[pp->syncvalhead & (SYNCFIFOSIZ - 1)][i] = (*SyncFunc[i])();
|
|
}
|
|
|
|
val = pp->syncval[pp->syncvalhead & (SYNCFIFOSIZ - 1)][0];
|
|
count += val;
|
|
|
|
pp->syncvalhead++;
|
|
}
|
|
|
|
////////////////////////////////////////////////////////////////////////
|
|
//
|
|
// Sync Message print
|
|
//
|
|
////////////////////////////////////////////////////////////////////////
|
|
|
|
|
|
void Net_DisplaySyncMsg(void)
|
|
{
|
|
int32_t i, j;
|
|
static uint32_t moveCount = 0;
|
|
extern uint32_t g_moveThingsCount;
|
|
|
|
// if (!SyncPrintMode)
|
|
// return;
|
|
|
|
if (numplayers < 2)
|
|
return;
|
|
|
|
for (i = 0; i < g_numSyncBytes; i++)
|
|
{
|
|
// syncstat is NON 0 - out of sync
|
|
if (syncstat[i] != 0)
|
|
{
|
|
if (g_numSyncBytes > 1)
|
|
{
|
|
Bsprintf(tempbuf, "Out Of Sync - %s", SyncNames[i]);
|
|
printext256(4L, 100L + (i * 8), 31, 1, tempbuf, 0);
|
|
}
|
|
|
|
if (!g_foundSyncError && g_szfirstSyncMsg[i][0] == '\0')
|
|
{
|
|
// g_foundSyncError one so test all of them and then never test again
|
|
g_foundSyncError = TRUE;
|
|
|
|
// save off loop count
|
|
moveCount = g_moveThingsCount;
|
|
|
|
for (j = 0; j < g_numSyncBytes; j++)
|
|
{
|
|
if (syncstat[j] != 0 && g_szfirstSyncMsg[j][0] == '\0')
|
|
{
|
|
Bsprintf(tempbuf, "Out Of Sync - %s", SyncNames[j]);
|
|
strcpy(g_szfirstSyncMsg[j], tempbuf);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
// print out the g_szfirstSyncMsg message you got
|
|
for (i = 0; i < g_numSyncBytes; i++)
|
|
{
|
|
if (g_szfirstSyncMsg[i][0] != '\0')
|
|
{
|
|
if (g_numSyncBytes > 1)
|
|
{
|
|
Bsprintf(tempbuf, "FIRST %s", g_szfirstSyncMsg[i]);
|
|
printext256(4L, 44L + (i * 8), 31, 1, tempbuf, 0);
|
|
Bsprintf(tempbuf, "moveCount %d",moveCount);
|
|
printext256(4L, 52L + (i * 8), 31, 1, tempbuf, 0);
|
|
}
|
|
else
|
|
{
|
|
printext256(4L,100L,31,0,"Out Of Sync - Please restart game",0);
|
|
}
|
|
}
|
|
}
|
|
|
|
// if (syncstate != 0)
|
|
// printext256(68L, 92L, 1, 31, "Missed Network packet!", 0);
|
|
}
|
|
|
|
|
|
static inline void Net_AddSyncData(int32_t *j)
|
|
{
|
|
int32_t sb;
|
|
int32_t count = 0;
|
|
|
|
// sync testing
|
|
while (g_player[myconnectindex].syncvalhead != syncvaltail && count++ < 4)
|
|
{
|
|
for (sb = 0; sb < g_numSyncBytes; sb++)
|
|
packbuf[(*j)++] = g_player[myconnectindex].syncval[syncvaltail & (SYNCFIFOSIZ - 1)][sb];
|
|
|
|
syncvaltail++;
|
|
}
|
|
}
|
|
|
|
void Net_GetSyncData(char *packbuf, int32_t packbufleng, int32_t *j, int32_t otherconnectindex)
|
|
{
|
|
int32_t sb, i;
|
|
extern int32_t syncvaltail, syncvaltottail;
|
|
playerdata_t *ppo = &g_player[otherconnectindex];
|
|
char found = 0;
|
|
|
|
// have had problems with this routine crashing when players quit
|
|
// games.
|
|
|
|
// if ready2send is not set then don't try to get sync info
|
|
|
|
if (!ready2send)
|
|
return;
|
|
|
|
// Suspect that its trying to traverse the connect list
|
|
// for a player that does not exist. This tries to take care of that
|
|
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
if (otherconnectindex == i)
|
|
found = 1;
|
|
}
|
|
|
|
if (!found)
|
|
return;
|
|
|
|
// sync testing
|
|
//while ((*j) != packbufleng) // changed this on Kens suggestion
|
|
while ((*j) < packbufleng)
|
|
{
|
|
for (sb = 0; sb < g_numSyncBytes; sb++)
|
|
{
|
|
ppo->syncval[ppo->syncvalhead & (SYNCFIFOSIZ - 1)][sb] = packbuf[(*j)++];
|
|
}
|
|
ppo->syncvalhead++;
|
|
}
|
|
|
|
// update syncstat
|
|
// if any of the syncstat vars is non-0 then there is a problem
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
if (g_player[i].syncvalhead == syncvaltottail)
|
|
return;
|
|
}
|
|
|
|
//for (sb = 0; sb < g_numSyncBytes; sb++)
|
|
// syncstat[sb] = 0;
|
|
|
|
while (1)
|
|
{
|
|
for (i = connectpoint2[connecthead]; i >= 0; i = connectpoint2[i])
|
|
{
|
|
for (sb = 0; sb < g_numSyncBytes; sb++)
|
|
{
|
|
if (g_player[i].syncval[syncvaltottail & (SYNCFIFOSIZ - 1)][sb] != g_player[connecthead].syncval[syncvaltottail & (SYNCFIFOSIZ - 1)][sb])
|
|
{
|
|
syncstat[sb] = 1;
|
|
}
|
|
}
|
|
}
|
|
|
|
syncvaltottail++;
|
|
|
|
TRAVERSE_CONNECT(i)
|
|
{
|
|
if (g_player[i].syncvalhead == syncvaltottail)
|
|
return;
|
|
}
|
|
}
|
|
}
|
|
|
|
|