You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
262 lines
6.3 KiB
262 lines
6.3 KiB
#include "diablo.h" |
|
#include "../3rdParty/Storm/Source/storm.h" |
|
|
|
DEVILUTION_BEGIN_NAMESPACE |
|
|
|
char byte_679704; // weak |
|
int gdwMsgLenTbl[MAX_PLRS]; |
|
#ifdef __cplusplus |
|
static CCritSect sgMemCrit; |
|
#endif |
|
int gdwDeltaBytesSec; // weak |
|
char nthread_should_run; // weak |
|
DWORD gdwTurnsInTransit; // weak |
|
char *glpMsgTbl[MAX_PLRS]; |
|
unsigned int glpNThreadId; |
|
char sgbSyncCountdown; // weak |
|
int turn_upper_bit; // weak |
|
char byte_679758; // weak |
|
char sgbPacketCountdown; // weak |
|
char sgbThreadIsRunning; // weak |
|
DWORD gdwLargestMsgSize; // weak |
|
DWORD gdwNormalMsgSize; // weak |
|
int last_tick; // weak |
|
|
|
/* data */ |
|
static HANDLE sghThread = INVALID_HANDLE_VALUE; |
|
|
|
void nthread_terminate_game(const char *pszFcn) |
|
{ |
|
DWORD sErr; |
|
|
|
sErr = SErrGetLastError(); |
|
if (sErr == STORM_ERROR_INVALID_PLAYER) { |
|
return; |
|
} else if (sErr == STORM_ERROR_GAME_TERMINATED) { |
|
gbGameDestroyed = TRUE; |
|
} else if (sErr == STORM_ERROR_NOT_IN_GAME) { |
|
gbGameDestroyed = TRUE; |
|
} else { |
|
app_fatal("%s:\n%s", pszFcn, TraceLastError()); |
|
} |
|
} |
|
|
|
DWORD nthread_send_and_recv_turn(DWORD cur_turn, int turn_delta) |
|
{ |
|
DWORD new_cur_turn; |
|
int turn_tmp; |
|
int turn; |
|
int curTurnsInTransit; |
|
|
|
new_cur_turn = cur_turn; |
|
if (!SNetGetTurnsInTransit(&curTurnsInTransit)) { |
|
nthread_terminate_game("SNetGetTurnsInTransit"); |
|
return 0; |
|
} |
|
while (curTurnsInTransit < gdwTurnsInTransit) { |
|
curTurnsInTransit++; |
|
|
|
turn_tmp = turn_upper_bit | new_cur_turn & 0x7FFFFFFF; |
|
turn_upper_bit = 0; |
|
turn = turn_tmp; |
|
|
|
if (!SNetSendTurn((char *)&turn, sizeof(turn))) { |
|
nthread_terminate_game("SNetSendTurn"); |
|
return 0; |
|
} |
|
|
|
new_cur_turn += turn_delta; |
|
if (new_cur_turn >= 0x7FFFFFFF) |
|
new_cur_turn &= 0xFFFF; |
|
} |
|
return new_cur_turn; |
|
} |
|
|
|
|
|
BOOL nthread_recv_turns(BOOL *pfSendAsync) |
|
{ |
|
*pfSendAsync = FALSE; |
|
sgbPacketCountdown--; |
|
if (sgbPacketCountdown) { |
|
last_tick += 50; |
|
return TRUE; |
|
} |
|
sgbSyncCountdown--; |
|
sgbPacketCountdown = byte_679704; |
|
if (sgbSyncCountdown != 0) { |
|
|
|
*pfSendAsync = TRUE; |
|
last_tick += 50; |
|
return TRUE; |
|
} |
|
if (!SNetReceiveTurns(0, MAX_PLRS, (char **)glpMsgTbl, (unsigned int *)gdwMsgLenTbl, (LPDWORD)player_state)) { |
|
if (SErrGetLastError() != STORM_ERROR_NO_MESSAGES_WAITING) |
|
nthread_terminate_game("SNetReceiveTurns"); |
|
byte_679758 = 0; |
|
sgbSyncCountdown = 1; |
|
sgbPacketCountdown = 1; |
|
return 0; |
|
} |
|
else{ |
|
|
|
if (!byte_679758) { |
|
byte_679758 = 1; |
|
last_tick = GetTickCount(); |
|
} |
|
sgbSyncCountdown = 4; |
|
multi_msg_countdown(); |
|
*pfSendAsync = TRUE; |
|
last_tick += 50; |
|
return TRUE; |
|
} |
|
} |
|
// 679704: using guessed type char byte_679704; |
|
// 679750: using guessed type char sgbSyncCountdown; |
|
// 679758: using guessed type char byte_679758; |
|
// 679759: using guessed type char sgbPacketCountdown; |
|
// 679764: using guessed type int last_tick; |
|
|
|
void nthread_set_turn_upper_bit() |
|
{ |
|
turn_upper_bit = 0x80000000; |
|
} |
|
|
|
void nthread_start(BOOL set_turn_upper_bit) |
|
{ |
|
char *err; // eax |
|
unsigned int largestMsgSize; // esi |
|
unsigned int normalMsgSize; // eax |
|
char *err2; // eax |
|
_SNETCAPS caps; // [esp+8h] [ebp-24h] |
|
|
|
last_tick = GetTickCount(); |
|
sgbPacketCountdown = 1; |
|
sgbSyncCountdown = 1; |
|
byte_679758 = 1; |
|
if (set_turn_upper_bit) |
|
nthread_set_turn_upper_bit(); |
|
else |
|
turn_upper_bit = 0; |
|
caps.size = 36; |
|
if (!SNetGetProviderCaps(&caps)) { |
|
err = TraceLastError(); |
|
app_fatal("SNetGetProviderCaps:\n%s", err); |
|
} |
|
gdwTurnsInTransit = caps.defaultturnsintransit; |
|
if (!caps.defaultturnsintransit) |
|
gdwTurnsInTransit = 1; |
|
if (caps.defaultturnssec <= 0x14u && caps.defaultturnssec) |
|
byte_679704 = 0x14u / caps.defaultturnssec; |
|
else |
|
byte_679704 = 1; |
|
largestMsgSize = 512; |
|
if (caps.maxmessagesize < 0x200u) |
|
largestMsgSize = caps.maxmessagesize; |
|
gdwDeltaBytesSec = (unsigned int)caps.bytessec >> 2; |
|
gdwLargestMsgSize = largestMsgSize; |
|
gdwNormalMsgSize = caps.bytessec * (unsigned int)(unsigned char)byte_679704 / 0x14; |
|
gdwNormalMsgSize *= 3; |
|
gdwNormalMsgSize = gdwNormalMsgSize >> 2; |
|
if (caps.maxplayers > 4u) |
|
caps.maxplayers = 4; |
|
gdwNormalMsgSize /= caps.maxplayers; |
|
while (gdwNormalMsgSize < 0x80){ |
|
gdwNormalMsgSize *= 2; |
|
byte_679704 *= 2; |
|
} |
|
if (gdwNormalMsgSize > largestMsgSize) |
|
gdwNormalMsgSize = largestMsgSize; |
|
if ((unsigned char)gbMaxPlayers > 1u) { |
|
sgbThreadIsRunning = 0; |
|
#ifdef __cplusplus |
|
sgMemCrit.Enter(); |
|
#endif |
|
nthread_should_run = 1; |
|
sghThread = (HANDLE)_beginthreadex(NULL, 0, nthread_handler, NULL, 0, &glpNThreadId); |
|
if (sghThread == INVALID_HANDLE_VALUE) { |
|
err2 = TraceLastError(); |
|
app_fatal("nthread2:\n%s", err2); |
|
} |
|
SetThreadPriority(sghThread, THREAD_PRIORITY_HIGHEST); |
|
} |
|
} |
|
|
|
unsigned int __stdcall nthread_handler(void *a1) |
|
{ |
|
int delta; |
|
int received; |
|
|
|
if (nthread_should_run) { |
|
while (1) { |
|
#ifdef __cplusplus |
|
sgMemCrit.Enter(); |
|
#endif |
|
if (!nthread_should_run) |
|
break; |
|
nthread_send_and_recv_turn(0, 0); |
|
if (nthread_recv_turns(&received)) |
|
delta = last_tick - GetTickCount(); |
|
else |
|
delta = 50; |
|
#ifdef __cplusplus |
|
sgMemCrit.Leave(); |
|
#endif |
|
if (delta > 0) |
|
Sleep(delta); |
|
if (!nthread_should_run) |
|
return 0; |
|
} |
|
#ifdef __cplusplus |
|
sgMemCrit.Leave(); |
|
#endif |
|
} |
|
return 0; |
|
} |
|
|
|
void nthread_cleanup() |
|
{ |
|
nthread_should_run = 0; |
|
gdwTurnsInTransit = 0; |
|
gdwNormalMsgSize = 0; |
|
gdwLargestMsgSize = 0; |
|
if (sghThread != INVALID_HANDLE_VALUE && glpNThreadId != GetCurrentThreadId()) { |
|
#ifdef __cplusplus |
|
if (!sgbThreadIsRunning) |
|
sgMemCrit.Leave(); |
|
#endif |
|
if (WaitForSingleObject(sghThread, 0xFFFFFFFF) == -1) { |
|
app_fatal("nthread3:\n(%s)", TraceLastError()); |
|
} |
|
CloseHandle(sghThread); |
|
sghThread = INVALID_HANDLE_VALUE; |
|
} |
|
} |
|
|
|
void nthread_ignore_mutex(BOOL bStart) |
|
{ |
|
if (sghThread != INVALID_HANDLE_VALUE) { |
|
#ifdef __cplusplus |
|
if (bStart) |
|
sgMemCrit.Leave(); |
|
else |
|
sgMemCrit.Enter(); |
|
#endif |
|
sgbThreadIsRunning = bStart; |
|
} |
|
} |
|
|
|
BOOL nthread_has_500ms_passed(BOOL unused) |
|
{ |
|
DWORD currentTickCount; |
|
int ticksElapsed; |
|
|
|
currentTickCount = GetTickCount(); |
|
ticksElapsed = currentTickCount - last_tick; |
|
if (gbMaxPlayers == 1 && ticksElapsed > 500) { |
|
last_tick = currentTickCount; |
|
ticksElapsed = 0; |
|
} |
|
return ticksElapsed >= 0; |
|
} |
|
|
|
DEVILUTION_END_NAMESPACE
|
|
|