SA-MP/bot/net/netgame.cpp
RD42 d261676570 [bot] Implement/match CNetGame::Packet_Unk12(...)
* Update `CNetGame::UpdateNetwork()`
2024-10-10 23:47:23 +08:00

1249 lines
28 KiB
C++

#include "../main.h"
#include "../../raknet/SocketDataEncryptor.h"
#include "../mathutils.h"
#include "../unnamed_1.h"
#define NETGAME_VERSION 4057
char szGameModeFile[256];
#define PLAYER_STATE_NONE 0
#define PLAYER_STATE_ONFOOT 1
#define PLAYER_STATE_DRIVER 2
#define PLAYER_STATE_PASSENGER 3
#define PLAYER_STATE_WASTED 7
#define PLAYER_STATE_SPAWNED 8
#define PLAYER_STATE_SPECTATING 9
#pragma pack(1)
typedef struct _AIM_SYNC_DATA // size: 31
{
char _gap0[31];
} AIM_SYNC_DATA;
#pragma pack(1)
typedef struct _TRAILER_SYNC_DATA // size: 54
{
char _gap0[54];
} TRAILER_SYNC_DATA;
char unnamed_2[63];
PASSENGER_SYNC_DATA unnamed_5[MAX_PLAYERS];
BOOL bPlayerSlotState[MAX_PLAYERS];
BYTE byteState;
ONFOOT_SYNC_DATA ofSync;
ONFOOT_SYNC_DATA unnamed_3[MAX_PLAYERS];
BYTE bytePlayerState[MAX_PLAYERS];
BOOL bVehicleSlotState[MAX_VEHICLES];
INCAR_SYNC_DATA unnamed_4[MAX_PLAYERS];
BYTE byteMySeatID;
bool bSpawned = false;
void CNetGame::SetPlayerAdded(PLAYERID playerId, BOOL a2)
{
if(playerId < MAX_PLAYERS)
{
bPlayerSlotState[playerId] = a2;
}
}
void CNetGame::SetVehicleAdded(VEHICLEID VehicleID, BOOL a2)
{
if(VehicleID < MAX_VEHICLES)
{
bVehicleSlotState[VehicleID] = a2;
}
}
void CNetGame::SetPlayerState(PLAYERID playerId, BYTE byteState)
{
if(playerId < MAX_PLAYERS)
{
bytePlayerState[playerId] = byteState;
}
}
BYTE CNetGame::GetPlayerState(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return PLAYER_STATE_NONE;
if(bPlayerSlotState[playerId] == FALSE) return PLAYER_STATE_NONE;
return bytePlayerState[playerId];
}
BOOL CNetGame::GetPlayerPos(PLAYERID playerId, PVECTOR Vector)
{
if(playerId >= MAX_PLAYERS || !bPlayerSlotState[playerId])
return FALSE;
switch(bytePlayerState[playerId])
{
case PLAYER_STATE_ONFOOT:
Vector->X = unnamed_3[playerId].vecPos.X;
Vector->Y = unnamed_3[playerId].vecPos.Y;
Vector->Z = unnamed_3[playerId].vecPos.Z;
return TRUE;
case PLAYER_STATE_DRIVER:
Vector->X = unnamed_4[playerId].vecPos.X;
Vector->Y = unnamed_4[playerId].vecPos.Y;
Vector->Z = unnamed_4[playerId].vecPos.Z;
return TRUE;
case PLAYER_STATE_PASSENGER:
Vector->X = unnamed_5[playerId].vecPos.X;
Vector->Y = unnamed_5[playerId].vecPos.Y;
Vector->Z = unnamed_5[playerId].vecPos.Z;
return TRUE;
}
return FALSE;
}
VEHICLEID CNetGame::GetPlayerVehicleID(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return INVALID_VEHICLE_ID;
if(bPlayerSlotState[playerId] == FALSE) return INVALID_VEHICLE_ID;
if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
return unnamed_4[playerId].VehicleID;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
{
return unnamed_5[playerId].VehicleID;
}
else
{
return INVALID_VEHICLE_ID;
}
}
BYTE CNetGame::GetPlayerArmedWeapon(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return 0;
if(bPlayerSlotState[playerId] == FALSE) return 0;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
return unnamed_3[playerId].byteCurrentWeapon;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
return unnamed_4[playerId].byteCurrentWeapon;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
{
return unnamed_5[playerId].byteCurrentWeapon;
}
else
{
return 0;
}
}
BYTE CNetGame::GetPlayerHealth(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return 0;
if(bPlayerSlotState[playerId] == FALSE) return 0;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
return unnamed_3[playerId].byteHealth;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
return unnamed_4[playerId].bytePlayerHealth;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
{
return unnamed_5[playerId].bytePlayerHealth;
}
else
{
return 0;
}
}
BYTE CNetGame::GetPlayerArmour(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return 0;
if(bPlayerSlotState[playerId] == FALSE) return 0;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
return unnamed_3[playerId].byteArmour;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
return unnamed_4[playerId].bytePlayerArmour;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
{
return unnamed_5[playerId].bytePlayerArmour;
}
else
{
return 0;
}
}
BOOL CNetGame::GetPlayerKeys(PLAYERID playerId, WORD *udAnalog, WORD *lrAnalog, WORD *wKeys)
{
if(playerId >= MAX_PLAYERS) return FALSE;
if(bPlayerSlotState[playerId] == FALSE) return FALSE;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
*udAnalog = unnamed_3[playerId].udAnalog;
*lrAnalog = unnamed_3[playerId].lrAnalog;
*wKeys = unnamed_3[playerId].wKeys;
return TRUE;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
*udAnalog = unnamed_4[playerId].udAnalog;
*lrAnalog = unnamed_4[playerId].lrAnalog;
*wKeys = unnamed_4[playerId].wKeys;
return TRUE;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_PASSENGER)
{
*udAnalog = unnamed_5[playerId].udAnalog;
*lrAnalog = unnamed_5[playerId].lrAnalog;
*wKeys = unnamed_5[playerId].wKeys;
return TRUE;
}
else
{
return FALSE;
}
}
float CNetGame::GetPlayerFacingAngle(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return 0.0f;
if(bPlayerSlotState[playerId] == FALSE) return 0.0f;
MATRIX4X4 mat;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
QuaternionToMatrix(&unnamed_3[playerId].quatRotation, &mat);
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f/PI;
// Bound it to [0, 360)
if ( fZAngle < 0.0f )
fZAngle += 360.0f;
else if ( fZAngle >= 360.0f )
fZAngle -= 360.0f;
return fZAngle;
}
else if(bytePlayerState[playerId] == PLAYER_STATE_DRIVER)
{
QuaternionToMatrix(&unnamed_4[playerId].quatRotation, &mat);
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f/PI;
// Bound it to [0, 360)
if ( fZAngle < 0.0f )
fZAngle += 360.0f;
else if ( fZAngle >= 360.0f )
fZAngle -= 360.0f;
return fZAngle;
}
else
{
return 0.0f;
}
}
BYTE CNetGame::GetPlayerSpecialAction(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return SPECIAL_ACTION_NONE;
if(bPlayerSlotState[playerId] == FALSE) return SPECIAL_ACTION_NONE;
if(bytePlayerState[playerId] == PLAYER_STATE_ONFOOT)
{
return unnamed_3[playerId].byteSpecialAction;
}
return SPECIAL_ACTION_NONE;
}
//----------------------------------------------------
// MATCH
BOOL CNetGame::IsPlayerAdded(PLAYERID playerId)
{
if(playerId >= MAX_PLAYERS) return FALSE;
return bPlayerSlotState[playerId] != FALSE;
}
//----------------------------------------------------
BOOL CNetGame::IsVehicleAdded(VEHICLEID VehicleID)
{
if(VehicleID >= MAX_VEHICLES) return FALSE;
return bVehicleSlotState[VehicleID] != FALSE;
}
//----------------------------------------------------
float CNetGame::GetDistanceFromMeToPoint(PVECTOR vecPos)
{
VECTOR vecMyPos;
if(GetMyPos(&vecMyPos))
{
float fX = vecMyPos.X - vecPos->X;
float fY = vecMyPos.Y - vecPos->Y;
float fZ = vecMyPos.Z - vecPos->Z;
return (float)sqrt(fX * fX + fY * fY + fZ * fZ);
}
else
{
return 0.0f;
}
}
//----------------------------------------------------
PVECTOR CNetGame::GetMyPos(PVECTOR Vector)
{
if(byteState == PLAYER_STATE_ONFOOT)
{
Vector->X = ofSync.vecPos.X;
Vector->Y = ofSync.vecPos.Y;
Vector->Z = ofSync.vecPos.Z;
return Vector;
}
else if(byteState == PLAYER_STATE_DRIVER)
{
Vector->X = icSync.vecPos.X;
Vector->Y = icSync.vecPos.Y;
Vector->Z = icSync.vecPos.Z;
return Vector;
}
else
{
return NULL;
}
}
//----------------------------------------------------
void CNetGame::SetMyPos(PVECTOR Vector)
{
if(byteState == PLAYER_STATE_ONFOOT)
{
ofSync.vecPos.X = Vector->X;
ofSync.vecPos.Y = Vector->Y;
ofSync.vecPos.Z = Vector->Z;
}
else if (byteState == PLAYER_STATE_DRIVER)
{
icSync.vecPos.X = Vector->X;
icSync.vecPos.Y = Vector->Y;
icSync.vecPos.Z = Vector->Z;
}
}
//----------------------------------------------------
float CNetGame::GetMyZAngle()
{
MATRIX4X4 mat;
if(byteState == PLAYER_STATE_ONFOOT)
{
QuaternionToMatrix(&ofSync.quatRotation, &mat);
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f / PI;
// Bound it to [0, 360)
if ( fZAngle < 0.0f )
fZAngle += 360.0f;
else if ( fZAngle >= 360.0f )
fZAngle -= 360.0f;
return fZAngle;
}
else if(byteState == PLAYER_STATE_DRIVER)
{
QuaternionToMatrix(&icSync.quatRotation, &mat);
float fZAngle = atan2(-mat.up.X, mat.up.Y) * 180.0f/PI;
// Bound it to [0, 360)
if ( fZAngle < 0.0f )
fZAngle += 360.0f;
else if ( fZAngle >= 360.0f )
fZAngle -= 360.0f;
return fZAngle;
}
return 0.0f;
}
//----------------------------------------------------
// MATCH
void CNetGame::SetMyZAngle(float fAngle)
{
//logprintf("CNetGame::SetMyZAngle(%f)", fAngle);
float fRadians = fAngle * 0.017453292f; // (PI/180.0f)
MATRIX4X4 mat;
memset(&mat,0,sizeof(MATRIX4X4));
mat.right.Z = 0.0f;
mat.up.Z = 0.0f;
mat.at.X = 0.0f;
mat.at.Y = 0.0f;
mat.at.Z = 1.0f;
float fCos = cos(fRadians);
float fSin = sin(fRadians);
mat.right.X = fCos;
mat.right.Y = fSin;
mat.up.X = -fSin;
mat.up.Y = fCos;
MatrixToQuaternion(&mat, &ofSync.quatRotation);
}
//----------------------------------------------------
// TODO: length assert has to be at line 397
// what the fuck is on top of the netgame.cpp that makes GetPacketID is at line 397 ????
BYTE GetPacketID(Packet *p)
{
if (p==0) return 255;
if ((unsigned char)p->data[0] == ID_TIMESTAMP) {
assert(p->length > sizeof(unsigned char) + sizeof(unsigned long));
return (unsigned char) p->data[sizeof(unsigned char) + sizeof(unsigned long)];
}
else {
return (unsigned char) p->data[0];
}
}
//----------------------------------------------------
CNetGame::CNetGame()
{
// nothing
}
//----------------------------------------------------
CNetGame::~CNetGame()
{
m_pRakClient->Disconnect(0);
UnRegisterRPCs(m_pRakClient);
UnRegisterScriptRPCs(m_pRakClient); // Unregister server-side scripting RPCs.
RakNetworkFactory::DestroyRakClientInterface(m_pRakClient);
SAFE_DELETE(m_pGameMode);
SAFE_DELETE(m_pScriptTimers);
SAFE_DELETE(m_pPlayerPool);
SAFE_DELETE(m_pVehiclePool);
}
//----------------------------------------------------
void CNetGame::ShutdownForGameModeRestart()
{
m_byteWorldTime = 12;
m_byteWorldMinute = 0;
m_byteWeather = 10;
m_byteHoldTime = 1;
m_bUseCJWalk = FALSE;
m_fGravity = (float)0.008000000;
m_iDeathDropMoney = 0;
m_iGameState = GAMESTATE_RESTARTING;
// Process the pool one last time
m_pPlayerPool->Process();
ResetVehiclePool();
StopRecordingPlayback();
memset(unnamed_2,0,sizeof(unnamed_2));
memset(unnamed_3,0,sizeof(unnamed_3));
memset(unnamed_4,0,sizeof(unnamed_4));
memset(unnamed_5,0,sizeof(unnamed_5));
memset(&bVehicleSlotState[0],0,sizeof(BOOL)*MAX_VEHICLES);
memset(&ofSync,0,sizeof(ONFOOT_SYNC_DATA));
memset(&bPlayerSlotState[0],0,sizeof(BOOL)*MAX_PLAYERS);
memset(&bytePlayerState[0],0,sizeof(BYTE)*MAX_PLAYERS);
m_bZoneNames = FALSE;
}
//----------------------------------------------------
void CNetGame::Init(PCHAR szHostOrIp, int iPort,
PCHAR szPlayerName, PCHAR szPass,
PCHAR szNpcMode)
{
strcpy(m_szHostName, "San Andreas Multiplayer");
strncpy(m_szHostOrIp, szHostOrIp, sizeof(m_szHostOrIp));
m_iPort = iPort;
m_pGameMode = new CGameMode();
m_pScriptTimers = new CScriptTimers();
// Setup player pool
m_pPlayerPool = new CPlayerPool();
m_pPlayerPool->SetLocalPlayerName(szPlayerName);
m_pVehiclePool = new CVehiclePool();
m_pRakClient = RakNetworkFactory::GetRakClientInterface();
SocketDataEncryptor::SetKey(iPort);
RegisterRPCs(m_pRakClient);
RegisterScriptRPCs(m_pRakClient); // Register server-side scripting RPCs.
sprintf(szGameModeFile, "npcmodes/%s.amx", szNpcMode);
if (!m_pGameMode->Load(szGameModeFile))
{
//logprintf("NPC: I can't load %s so I'm quiting.", szGameModeFile);
exit(1);
}
m_pRakClient->SetPassword(szPass);
m_pRakClient->Connect(szHostOrIp,iPort,0,0,10);
m_iGameState = GAMESTATE_CONNECTING;
//logprintf("NPC(%s): connecting to %s:%d...",szPlayerName,szHostOrIp,iPort);
m_iSpawnsAvailable = 0;
m_byteWorldTime = 12;
m_byteWorldMinute = 0;
m_byteWeather = 10;
m_fGravity = (float)0.008000000;
m_iDeathDropMoney = 0;
m_bLanMode = FALSE;
m_byteHoldTime = 1;
m_bUseCJWalk = FALSE;
int i;
for (i = 0; i < 100; i++) m_dwMapIcon[i] = NULL;
m_byteFriendlyFire = 1;
m_bZoneNames = FALSE;
m_bInstagib = FALSE;
memset(unnamed_2,0,sizeof(unnamed_2));
memset(&ofSync,0,sizeof(ONFOOT_SYNC_DATA));
memset(unnamed_3,0,sizeof(unnamed_3));
memset(unnamed_4,0,sizeof(unnamed_4));
memset(unnamed_5,0,sizeof(unnamed_5));
memset(&bVehicleSlotState[0],0,sizeof(BOOL)*MAX_VEHICLES);
memset(&bPlayerSlotState[0],0,sizeof(BOOL)*MAX_PLAYERS);
memset(&bytePlayerState[0],0,sizeof(BYTE)*MAX_PLAYERS);
field_1DE = 0;
field_1E2 = 0;
field_1F2 = GetTickCount();
byteState = PLAYER_STATE_NONE;
field_1FA = -1;
field_1FE = -1;
srand(0);
}
//----------------------------------------------------
#ifdef WIN32
#pragma comment(lib, "winmm.lib")
float GetElapsedTime()
{
static BOOL bTimerInit = false;
static BOOL bUsingOPF = false;
static LONGLONG nTicksPerSec = 0;
if (!bTimerInit)
{
bTimerInit = true;
LARGE_INTEGER qwTicksPerSec;
bUsingOPF = QueryPerformanceFrequency(&qwTicksPerSec);
if (bUsingOPF) nTicksPerSec = qwTicksPerSec.QuadPart;
}
if (bUsingOPF)
{
LARGE_INTEGER qwTime;
QueryPerformanceCounter(&qwTime);
static LONGLONG llLastTime = qwTime.QuadPart;
double fElapsedTime = (double)(qwTime.QuadPart - llLastTime) / (double) nTicksPerSec;
llLastTime = qwTime.QuadPart;
return (float)fElapsedTime;
} else {
double fTime = timeGetTime() * 0.001;
static double fLastTime = fTime;
double fElapsedTime = (double)(fTime - fLastTime);
fLastTime = fTime;
return (float)fElapsedTime;
}
}
#else
float GetElapsedTime()
{
static timeval lasttv;
timeval tv;
float fRet;
gettimeofday(&tv, NULL);
if (!timerisset(&lasttv)) memcpy(&lasttv, &tv, sizeof(timeval));
fRet = (float)((tv.tv_sec - lasttv.tv_sec) * 1000000) + (tv.tv_usec - lasttv.tv_usec);
fRet /= 1000000.0f;
memcpy(&lasttv,&tv,sizeof(timeval));
return fRet;
}
#endif // WIN32
//----------------------------------------------------
void CNetGame::Process()
{
float fElapsedTime = GetElapsedTime();
UpdateNetwork();
// TODO: CNetGame::Process (W: 00418370 L: 080AD6A4)
/*
if ( this->field_C == 2 )
{
if ( this->field_386 )
sub_80A969E(this->field_386, v2);
if ( this->field_38A )
sub_80B8610(this->field_38A, (signed __int64)(v2 * 1000.0));
if ( byte_810A708 )
{
if ( this->field_1DE )
{
sub_80ADFDE(this);
}
else
{
sub_80AD770(this);
byte_80E16B7 = 1;
}
}
}*/
}
//----------------------------------------------------
// UPDATE NETWORK
//----------------------------------------------------
void CNetGame::UpdateNetwork()
{
Packet* pkt=NULL;
unsigned char packetIdentifier;
while((pkt = m_pRakClient->Receive()))
{
packetIdentifier = GetPacketID(pkt);
switch(packetIdentifier)
{
case ID_UNK_12:
Packet_Unk12(pkt);
break;
case ID_RSA_PUBLIC_KEY_MISMATCH:
Packet_RSAPublicKeyMismatch(pkt);
break;
case ID_CONNECTION_BANNED:
Packet_ConnectionBanned(pkt);
break;
case ID_NO_FREE_INCOMING_CONNECTIONS:
Packet_NoFreeIncomingConnections(pkt);
break;
case ID_DISCONNECTION_NOTIFICATION:
Packet_DisconnectionNotification(pkt);
break;
case ID_CONNECTION_LOST:
Packet_ConnectionLost(pkt);
break;
case ID_INVALID_PASSWORD:
Packet_InvalidPassword(pkt);
break;
case ID_MODIFIED_PACKET:
Packet_ModifiedPacket(pkt);
break;
case ID_CONNECTION_ATTEMPT_FAILED:
Packet_ConnectAttemptFailed(pkt);
break;
case ID_CONNECTION_REQUEST_ACCEPTED:
Packet_ConnectionSucceeded(pkt);
break;
case ID_PLAYER_SYNC:
Packet_PlayerSync(pkt);
break;
case ID_VEHICLE_SYNC:
Packet_VehicleSync(pkt);
break;
case ID_PASSENGER_SYNC:
Packet_PassengerSync(pkt);
break;
case ID_AIM_SYNC:
Packet_AimSync(pkt);
break;
case ID_TRAILER_SYNC:
Packet_TrailerSync(pkt);
break;
}
m_pRakClient->DeallocatePacket(pkt);
}
}
//----------------------------------------------------
// PACKET HANDLERS INTERNAL
//----------------------------------------------------
void CNetGame::Packet_PlayerSync(Packet *p)
{
RakNet::BitStream bsPlayerSync((PCHAR)p->data, p->length, false);
ONFOOT_SYNC_DATA ofSync;
BYTE bytePacketID=0;
PLAYERID playerId=0;
bool bHasLR,bHasUD;
bool bHasSurfingInfo;
bool bUnk;
if(GetGameState() != GAMESTATE_CONNECTED) return;
memset(&ofSync,0,sizeof(ONFOOT_SYNC_DATA));
bsPlayerSync.Read(bytePacketID);
bsPlayerSync.Read(playerId);
// LEFT/RIGHT KEYS
bsPlayerSync.Read(bHasLR);
if(bHasLR) bsPlayerSync.Read(ofSync.lrAnalog);
// UP/DOWN KEYS
bsPlayerSync.Read(bHasUD);
if(bHasUD) bsPlayerSync.Read(ofSync.udAnalog);
// GENERAL KEYS
bsPlayerSync.Read(ofSync.wKeys);
// VECTOR POS
bsPlayerSync.Read((char*)&ofSync.vecPos,sizeof(VECTOR));
// ROTATION
bsPlayerSync.ReadNormQuat(ofSync.quatRotation.W,ofSync.quatRotation.X,ofSync.quatRotation.Y,ofSync.quatRotation.Z);
// HEALTH/ARMOUR (COMPRESSED INTO 1 BYTE)
BYTE byteHealthArmour;
BYTE byteArmTemp=0,byteHlTemp=0;
bsPlayerSync.Read(byteHealthArmour);
byteArmTemp = (byteHealthArmour & 0x0F);
byteHlTemp = (byteHealthArmour >> 4);
if(byteArmTemp == 0xF) ofSync.byteArmour = 100;
else if(byteArmTemp == 0) ofSync.byteArmour = 0;
else ofSync.byteArmour = byteArmTemp * 7;
if(byteHlTemp == 0xF) ofSync.byteHealth = 100;
else if(byteHlTemp == 0) ofSync.byteHealth = 0;
else ofSync.byteHealth = byteHlTemp * 7;
// CURRENT WEAPON
BYTE byteCurrentWeapon=0;
bsPlayerSync.Read(byteCurrentWeapon);
ofSync.byteCurrentWeapon = byteCurrentWeapon;
// Special Action
bsPlayerSync.Read(ofSync.byteSpecialAction);
// READ MOVESPEED VECTORS
bsPlayerSync.ReadVector(ofSync.vecMoveSpeed.X,ofSync.vecMoveSpeed.Y,ofSync.vecMoveSpeed.Z);
bsPlayerSync.Read(bHasSurfingInfo);
if(bHasSurfingInfo) {
bsPlayerSync.Read(ofSync.wSurfInfo);
bsPlayerSync.Read((char*)&ofSync.vecSurfOffsets,sizeof(VECTOR));
} else {
ofSync.wSurfInfo = 0xFFFF;
}
bsPlayerSync.Read(bUnk);
if(bUnk) bsPlayerSync.Read(ofSync.field_40);
else ofSync.field_40 = 0;
if(playerId < MAX_PLAYERS)
{
memcpy(&unnamed_3[playerId],&ofSync,sizeof(ONFOOT_SYNC_DATA));
bytePlayerState[playerId] = PLAYER_STATE_ONFOOT;
}
}
//----------------------------------------------------
void CNetGame::Packet_AimSync(Packet *p)
{
RakNet::BitStream bsAimSync((PCHAR)p->data, p->length, false);
AIM_SYNC_DATA aimSync;
BYTE bytePacketID=0;
BYTE bytePlayerID=0;
if(GetGameState() != GAMESTATE_CONNECTED) return;
bsAimSync.Read(bytePacketID);
bsAimSync.Read(bytePlayerID);
bsAimSync.Read((PCHAR)&aimSync,sizeof(AIM_SYNC_DATA));
}
//----------------------------------------------------
void CNetGame::Packet_VehicleSync(Packet *p)
{
RakNet::BitStream bsSync((PCHAR)p->data, p->length, false);
BYTE bytePacketID=0;
PLAYERID playerId=0;
INCAR_SYNC_DATA icSync;
bool bSiren,bLandingGear;
bool bHydra,bTrailer;
if(GetGameState() != GAMESTATE_CONNECTED) return;
memset(&icSync,0,sizeof(INCAR_SYNC_DATA));
bsSync.Read(bytePacketID);
bsSync.Read(playerId);
bsSync.Read(icSync.VehicleID);
// KEYS
bsSync.Read(icSync.lrAnalog);
bsSync.Read(icSync.udAnalog);
bsSync.Read(icSync.wKeys);
// ROLL / DIRECTION / POSITION / MOVE SPEED
bsSync.ReadNormQuat(icSync.quatRotation.W,icSync.quatRotation.X,icSync.quatRotation.Y,icSync.quatRotation.Z);
bsSync.Read((char*)&icSync.vecPos,sizeof(VECTOR));
bsSync.ReadVector(icSync.vecMoveSpeed.X,icSync.vecMoveSpeed.Y,icSync.vecMoveSpeed.Z);
// VEHICLE HEALTH
WORD wTempVehicleHealth;
bsSync.Read(wTempVehicleHealth);
icSync.fCarHealth = (float)wTempVehicleHealth;
// HEALTH/ARMOUR (COMPRESSED INTO 1 BYTE)
BYTE byteHealthArmour;
BYTE byteArmTemp=0,byteHlTemp=0;
bsSync.Read(byteHealthArmour);
byteArmTemp = (byteHealthArmour & 0x0F);
byteHlTemp = (byteHealthArmour >> 4);
if(byteArmTemp == 0xF) icSync.bytePlayerArmour = 100;
else if(byteArmTemp == 0) icSync.bytePlayerArmour = 0;
else icSync.bytePlayerArmour = byteArmTemp * 7;
if(byteHlTemp == 0xF) icSync.bytePlayerHealth = 100;
else if(byteHlTemp == 0) icSync.bytePlayerHealth = 0;
else icSync.bytePlayerHealth = byteHlTemp * 7;
// CURRENT WEAPON
BYTE byteCurrentWeapon=0;
bsSync.Read(byteCurrentWeapon);
icSync.byteCurrentWeapon = byteCurrentWeapon;
// SIREN
bsSync.Read(bSiren);
if(bSiren) icSync.byteSirenOn = 1;
// LANDING GEAR
bsSync.Read(bLandingGear);
if(bLandingGear) icSync.byteLandingGearState = 1;
// HYDRA SPECIAL
bsSync.Read(bHydra);
if(bHydra) bsSync.Read(icSync.dwHydraThrustAngle);
// TRAILER ID
bsSync.Read(bTrailer);
if(bTrailer) bsSync.Read(icSync.TrailerID);
if(playerId < MAX_PLAYERS)
{
memcpy(&unnamed_4[playerId],&icSync,sizeof(INCAR_SYNC_DATA));
bytePlayerState[playerId] = PLAYER_STATE_DRIVER;
}
}
//----------------------------------------------------
void CNetGame::Packet_PassengerSync(Packet *p)
{
RakNet::BitStream bsPassengerSync((PCHAR)p->data, p->length, false);
BYTE bytePacketID=0;
PLAYERID playerId=0;
PASSENGER_SYNC_DATA psSync;
if(GetGameState() != GAMESTATE_CONNECTED) return;
bsPassengerSync.Read(bytePacketID);
bsPassengerSync.Read(playerId);
bsPassengerSync.Read((PCHAR)&psSync,sizeof(PASSENGER_SYNC_DATA));
if (playerId < MAX_PLAYERS)
{
memcpy(&unnamed_5[playerId],&psSync,sizeof(PASSENGER_SYNC_DATA));
bytePlayerState[playerId] = PLAYER_STATE_PASSENGER;
}
}
//----------------------------------------------------
void CNetGame::Packet_TrailerSync(Packet *p)
{
RakNet::BitStream bsTrailerSync((PCHAR)p->data, p->length, false);
if(GetGameState() != GAMESTATE_CONNECTED) return;
BYTE bytePacketID=0;
BYTE bytePlayerID=0;
TRAILER_SYNC_DATA trSync;
bsTrailerSync.Read(bytePacketID);
bsTrailerSync.Read(bytePlayerID);
bsTrailerSync.Read((PCHAR)&trSync, sizeof(TRAILER_SYNC_DATA));
}
//----------------------------------------------------
void CNetGame::Packet_RSAPublicKeyMismatch(Packet* packet)
{
//logprintf("NPC: Failed to initialize encryption.");
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_ConnectionBanned(Packet* packet)
{
//logprintf("NPC: You're banned from the server.");
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_ConnectionRequestAccepted(Packet* packet)
{
//logprintf("NPC: Server has accepted the connection.");
}
//----------------------------------------------------
void CNetGame::Packet_NoFreeIncomingConnections(Packet* packet)
{
//logprintf("NPC: The server is full.");
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_DisconnectionNotification(Packet* packet)
{
//logprintf("NPC: Disconnected.");
m_pRakClient->Disconnect(0);
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_ConnectionLost(Packet* packet)
{
//logprintf("NPC: Lost connection to the server.");
m_pRakClient->Disconnect(0);
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_InvalidPassword(Packet* packet)
{
//logprintf("NPC: Wrong server password.");
m_pRakClient->Disconnect(0);
exit(1);
}
//----------------------------------------------------
void CNetGame::Packet_ModifiedPacket(Packet* packet)
{
}
//----------------------------------------------------
// RST
void CNetGame::Packet_ConnectAttemptFailed(Packet* packet)
{
//logprintf("NPC: Connection attempt failed.");
exit(1);
}
//----------------------------------------------------
// Connection Success
void CNetGame::Packet_ConnectionSucceeded(Packet *p)
{
m_iGameState = GAMESTATE_AWAIT_JOIN;
RakNet::BitStream bsReturnParams((PCHAR)p->data, p->length, true);
BYTE bytePacketID=0;
unsigned int binaryAddr=0;
unsigned short port=0;
unsigned short playerId=0;
unsigned int uiChallenge=0;
bsReturnParams.Read(bytePacketID);
bsReturnParams.Read(binaryAddr);
bsReturnParams.Read(port);
bsReturnParams.Read(playerId);
bsReturnParams.Read(uiChallenge);
uiChallenge ^= NETGAME_VERSION;
//logprintf("NPC: Connection Succeeded");
int iVersion = NETGAME_VERSION;
BYTE byteMod = 1;
BYTE byteNameLen = (BYTE)strlen(m_pPlayerPool->GetLocalPlayerName());
RakNet::BitStream bsSend;
bsSend.Write(iVersion);
bsSend.Write(byteMod);
bsSend.Write(byteNameLen);
bsSend.Write(m_pPlayerPool->GetLocalPlayerName(),byteNameLen);
bsSend.Write(uiChallenge);
m_pRakClient->RPC(RPC_NPCJoin,&bsSend,HIGH_PRIORITY,RELIABLE,0,FALSE);
}
//----------------------------------------------------
void CNetGame::Packet_Unk12(Packet *p)
{
RakNet::BitStream bsRecv((PCHAR)p->data, p->length, false);
char szIdent[4];
strcpy(szIdent, "NPC");
BYTE byteIdentLen = strlen(szIdent)+1;
RakNet::BitStream bsSend;
bsSend.Write((BYTE)ID_UNK_12);
bsSend.Write(byteIdentLen);
bsSend.Write(szIdent, byteIdentLen);
m_pRakClient->Send(&bsSend,SYSTEM_PRIORITY,RELIABLE,0);
}
//----------------------------------------------------
void CNetGame::ResetVehiclePool()
{
if(m_pVehiclePool) {
delete m_pVehiclePool;
}
m_pVehiclePool = new CVehiclePool();
}
//----------------------------------------------------
void CNetGame::ResetPlayerPool()
{
if(m_pPlayerPool) {
delete m_pPlayerPool;
}
m_pPlayerPool = new CPlayerPool();
}
//----------------------------------------------------
void CNetGame::SendChat(char *szText)
{
BYTE byteTextLen = strlen(szText);
RakNet::BitStream bsSend;
bsSend.Write(byteTextLen);
bsSend.Write(szText,byteTextLen);
GetRakClient()->RPC(RPC_Chat,&bsSend,HIGH_PRIORITY,RELIABLE,0,false);
}
void CNetGame::SendCommand(char *szCommand)
{
RakNet::BitStream bsParams;
int iStrlen = strlen(szCommand);
bsParams.Write(iStrlen);
bsParams.Write(szCommand, iStrlen);
GetRakClient()->RPC(RPC_ServerCommand,&bsParams,HIGH_PRIORITY,RELIABLE,0,false);
}
DWORD *GLOBAL_7;
ONFOOT_SYNC_DATA *GLOBAL_8;
INCAR_SYNC_DATA *GLOBAL_6;
DWORD GLOBAL_5;
void CNetGame::FUNC_5(FILE *a1, DWORD a2)
{
DWORD i=0;
if(0x43 * (a2 / 0x43) != a2)
{
//logprintf("NPC: Total failure. File doesn't have correct data alignment for driver recording.");
exit(1);
}
GLOBAL_5 = a2 / 0x43;
if(GLOBAL_6)
free(GLOBAL_6);
if(GLOBAL_7)
free(GLOBAL_7);
GLOBAL_6 = (INCAR_SYNC_DATA *)calloc(1, sizeof(INCAR_SYNC_DATA) * GLOBAL_5);
GLOBAL_7 = (DWORD *)calloc(1, sizeof(DWORD) * GLOBAL_5);
while(!feof(a1))
{
fread(&GLOBAL_7[i], 1, sizeof(DWORD), a1);
fread(&GLOBAL_6[i], 1, sizeof(INCAR_SYNC_DATA), a1);
i++;
}
}
void CNetGame::FUNC_6(FILE *a1, DWORD a2)
{
DWORD i=0;
if(0x48 * (a2 / 0x48) != a2)
{
//logprintf("NPC: Total failure. File doesn't have correct data alignment for onfoot recording.");
exit(1);
}
GLOBAL_5 = a2 / 0x48;
if(GLOBAL_8)
free(GLOBAL_8);
if(GLOBAL_7)
free(GLOBAL_7);
GLOBAL_8 = (ONFOOT_SYNC_DATA *)calloc(1, sizeof(ONFOOT_SYNC_DATA) * GLOBAL_5);
GLOBAL_7 = (DWORD *)calloc(1, sizeof(DWORD) * GLOBAL_5);
while(!feof(a1))
{
fread(&GLOBAL_7[i], 1, sizeof(DWORD), a1);
fread(&GLOBAL_8[i], 1, sizeof(ONFOOT_SYNC_DATA), a1);
i++;
}
}
void CNetGame::StartRecordingPlayback(int iPlaybackType, char *szRecordName)
{
DWORD v5 = 0;
if(field_1E2)
{
fclose(field_1E2);
field_1E2 = NULL;
}
char s[MAX_PATH];
sprintf(s, "./npcmodes/recordings/%s.rec", szRecordName);
field_1E2 = fopen(s, "rb");
if(!field_1E2)
{
//logprintf("NPC: Total failure. Can't open recording playback file %s.", s);
exit(1);
}
fseek(field_1E2, 0, SEEK_END);
v5 = ftell(field_1E2);
rewind(field_1E2);
if(v5 == 0)
{
//logprintf("NPC: Total failure. %s is a 0 length file.", s);
exit(1);
}
DWORD v4 = 0;
DWORD v3 = 0;
fread(&v4, 1, sizeof(DWORD), field_1E2);
fread(&v3, 1, sizeof(DWORD), field_1E2);
if(v4 != 1000)
{
//logprintf("NPC: %s is not the correct recording version for this bot.", s);
//logprintf("NPC: Trying to upgrade %s...", s);
fclose(field_1E2);
if(!UpgradeRecordFile(s, v5, iPlaybackType))
{
//logprintf("NPC: Fatal Error. Could not upgrade file. I'm out of options so I'm quiting.");
exit(1);
}
//logprintf("NPC: File %s was upgraded. Please restart the bot.", s);
exit(1);
}
if(iPlaybackType == PLAYER_RECORDING_TYPE_DRIVER)
{
FUNC_5(field_1E2, v5 - 8);
}
else if(iPlaybackType == PLAYER_RECORDING_TYPE_ONFOOT)
{
FUNC_6(field_1E2, v5 - 8);
}
else
{
//logprintf("NPC: Total failure. Unknown recording type specified.");
exit(1);
}
fclose(field_1E2);
field_1E2 = 0;
field_1DA = iPlaybackType;
field_1DE = 1;
field_1EA = GetTickCount();
field_1E6 = 0;
}
void CNetGame::StopRecordingPlayback()
{
field_1DE = 0;
field_1DA = 0;
}
void CNetGame::PauseRecordingPlayback()
{
if(field_1DE == 1 || field_1DE == 2)
field_1FA = 11;
}
void CNetGame::ResumeRecordingPlayback()
{
field_1FE = 9;
field_1DE = 1;
if(field_1FA == -1)
field_1EA += GetTickCount() - field_1F6 + 100;
else
field_1FA = -1;
}