mirror of
				https://github.com/DarkflameUniverse/DarkflameServer.git
				synced 2025-10-22 23:38:13 +00:00 
			
		
		
		
	
		
			
				
	
	
		
			618 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
			
		
		
	
	
			618 lines
		
	
	
		
			20 KiB
		
	
	
	
		
			C++
		
	
	
	
	
	
| /// \file
 | |
| ///
 | |
| /// This file is part of RakNet Copyright 2003 Kevin Jenkins.
 | |
| ///
 | |
| /// Usage of RakNet is subject to the appropriate license agreement.
 | |
| /// Creative Commons Licensees are subject to the
 | |
| /// license found at
 | |
| /// http://creativecommons.org/licenses/by-nc/2.5/
 | |
| /// Single application licensees are subject to the license found at
 | |
| /// http://www.jenkinssoftware.com/SingleApplicationLicense.html
 | |
| /// Custom license users are subject to the terms therein.
 | |
| /// GPL license users are subject to 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.
 | |
| 
 | |
| #include "ConnectionGraph.h"
 | |
| #include "RakPeerInterface.h"
 | |
| #include "MessageIdentifiers.h"
 | |
| #include "BitStream.h"
 | |
| #include "StringCompressor.h"
 | |
| #include "GetTime.h"
 | |
| #include <string.h>
 | |
| #include "RakAssert.h"
 | |
| #include "SHA1.h"
 | |
| 
 | |
| #ifdef _MSC_VER
 | |
| #pragma warning( push )
 | |
| #endif
 | |
| 
 | |
| static const int connectionGraphChannel=31;
 | |
| 
 | |
| ConnectionGraph::SystemAddressAndGroupId::SystemAddressAndGroupId()
 | |
| {
 | |
| 
 | |
| }
 | |
| ConnectionGraph::SystemAddressAndGroupId::~SystemAddressAndGroupId()
 | |
| {
 | |
| 
 | |
| }
 | |
| ConnectionGraph::SystemAddressAndGroupId::SystemAddressAndGroupId(SystemAddress _systemAddress, ConnectionGraphGroupID _groupID)
 | |
| {
 | |
| 	systemAddress=_systemAddress;
 | |
| 	groupId=_groupID;
 | |
| }
 | |
| bool ConnectionGraph::SystemAddressAndGroupId::operator==( const ConnectionGraph::SystemAddressAndGroupId& right ) const
 | |
| {
 | |
| 	return systemAddress==right.systemAddress;
 | |
| }
 | |
| bool ConnectionGraph::SystemAddressAndGroupId::operator!=( const ConnectionGraph::SystemAddressAndGroupId& right ) const
 | |
| {
 | |
| 	return systemAddress!=right.systemAddress;
 | |
| }
 | |
| bool ConnectionGraph::SystemAddressAndGroupId::operator > ( const ConnectionGraph::SystemAddressAndGroupId& right ) const
 | |
| {
 | |
| 	return systemAddress>right.systemAddress;
 | |
| }
 | |
| bool ConnectionGraph::SystemAddressAndGroupId::operator < ( const ConnectionGraph::SystemAddressAndGroupId& right ) const
 | |
| {
 | |
| 	return systemAddress<right.systemAddress;
 | |
| }
 | |
| ConnectionGraph::ConnectionGraph()
 | |
| {
 | |
|     pw=0;
 | |
| 	myGroupId=0;
 | |
| 	autoAddNewConnections=true;
 | |
| //	forceBroadcastTime=0;
 | |
| 	DataStructures::WeightedGraph<ConnectionGraph::SystemAddressAndGroupId, unsigned short, false>::IMPLEMENT_DEFAULT_COMPARISON();
 | |
| 	DataStructures::OrderedList<SystemAddress, SystemAddress>::IMPLEMENT_DEFAULT_COMPARISON();
 | |
| 	DataStructures::OrderedList<ConnectionGraph::SystemAddressAndGroupId, ConnectionGraph::SystemAddressAndGroupId>::IMPLEMENT_DEFAULT_COMPARISON();
 | |
| 	DataStructures::OrderedList<ConnectionGraphGroupID, ConnectionGraphGroupID>::IMPLEMENT_DEFAULT_COMPARISON();
 | |
| 
 | |
| 	subscribedGroups.Insert(0,0, true);
 | |
| }
 | |
| 
 | |
| ConnectionGraph::~ConnectionGraph()
 | |
| {
 | |
| 	if (pw)
 | |
| 		delete [] pw;
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::SetPassword(const char *password)
 | |
| {
 | |
| 	if (pw)
 | |
| 	{
 | |
| 		delete [] pw;
 | |
| 		pw=0;
 | |
| 	}
 | |
| 
 | |
| 	if (password && password[0])
 | |
| 	{
 | |
| 		assert(strlen(password)<256);
 | |
| 		pw=(char*) rakMalloc( strlen(password)+1 );
 | |
| 		strcpy(pw, password);
 | |
| 	}
 | |
| }
 | |
| DataStructures::WeightedGraph<ConnectionGraph::SystemAddressAndGroupId, unsigned short, false> *ConnectionGraph::GetGraph(void)
 | |
| {
 | |
| 	return &graph;
 | |
| }
 | |
| void ConnectionGraph::SetAutoAddNewConnections(bool autoAdd)
 | |
| {
 | |
| 	autoAddNewConnections=autoAdd;
 | |
| }
 | |
| void ConnectionGraph::OnShutdown(RakPeerInterface *peer)
 | |
| {
 | |
| 	(void) peer;
 | |
| 	graph.Clear();
 | |
| 	participantList.Clear();
 | |
| //	forceBroadcastTime=0;
 | |
| }
 | |
| void ConnectionGraph::Update(RakPeerInterface *peer)
 | |
| {
 | |
| 	(void) peer;
 | |
| 
 | |
| //	RakNetTime time = RakNet::GetTime();
 | |
| 
 | |
| 	// If the time is past the next weight update time, then refresh all pings of all connected participants and send these out if substantially different.
 | |
| //	if (forceBroadcastTime && time > forceBroadcastTime)
 | |
| //	{
 | |
| //		DataStructures::OrderedList<SystemAddress,SystemAddress> none;
 | |
| 	//	BroadcastGraphUpdate(none, peer);
 | |
| //		forceBroadcastTime=0;
 | |
| //	}
 | |
| }
 | |
| PluginReceiveResult ConnectionGraph::OnReceive(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	switch (packet->data[0]) 
 | |
| 	{
 | |
| 	case ID_NEW_INCOMING_CONNECTION:
 | |
| 		OnNewIncomingConnection(peer, packet);
 | |
| 		return RR_CONTINUE_PROCESSING;
 | |
| 	case ID_CONNECTION_REQUEST_ACCEPTED:
 | |
| 		OnConnectionRequestAccepted(peer, packet);
 | |
| 		return RR_CONTINUE_PROCESSING;
 | |
| 	case ID_CONNECTION_GRAPH_REQUEST:
 | |
| 		OnConnectionGraphRequest(peer, packet);
 | |
| 		return RR_STOP_PROCESSING_AND_DEALLOCATE;
 | |
| 	case ID_CONNECTION_GRAPH_REPLY:
 | |
| 		OnConnectionGraphReply(peer, packet);
 | |
| 		return RR_CONTINUE_PROCESSING;
 | |
| 	case ID_CONNECTION_GRAPH_UPDATE:
 | |
| 		OnConnectionGraphUpdate(peer, packet);
 | |
| 		return RR_STOP_PROCESSING_AND_DEALLOCATE;
 | |
| 	case ID_CONNECTION_GRAPH_NEW_CONNECTION:
 | |
| 		OnNewConnection(peer, packet);
 | |
| 		return RR_STOP_PROCESSING_AND_DEALLOCATE;
 | |
| 		// Remove connection lost
 | |
| 	case ID_CONNECTION_GRAPH_CONNECTION_LOST:
 | |
| 	case ID_CONNECTION_GRAPH_DISCONNECTION_NOTIFICATION:
 | |
| 		if (OnConnectionLost(peer, packet, packet->data[0]))
 | |
| 		{
 | |
| 			if (packet->data[0]==ID_CONNECTION_GRAPH_CONNECTION_LOST)
 | |
| 				packet->data[0]=ID_REMOTE_CONNECTION_LOST;
 | |
| 			else
 | |
| 				packet->data[0]=ID_REMOTE_DISCONNECTION_NOTIFICATION;
 | |
| 			return RR_CONTINUE_PROCESSING; // Return this packet to the user
 | |
| 		}		
 | |
| 		return RR_STOP_PROCESSING_AND_DEALLOCATE;
 | |
| 		// Local connection lost
 | |
| 	case ID_CONNECTION_LOST:
 | |
| 	case ID_DISCONNECTION_NOTIFICATION:
 | |
| 		{
 | |
| 			unsigned char packetId;
 | |
| 			// Change to remote connection lost and relay the message
 | |
| 			if (packet->data[0]==ID_CONNECTION_LOST)
 | |
| 				packetId=ID_CONNECTION_GRAPH_CONNECTION_LOST;
 | |
| 			else
 | |
| 				packetId=ID_CONNECTION_GRAPH_DISCONNECTION_NOTIFICATION;
 | |
| 			HandleDroppedConnection(peer, packet->systemAddress, packetId);
 | |
| 		}
 | |
| 	}
 | |
| 
 | |
| 	return RR_CONTINUE_PROCESSING;
 | |
| }
 | |
| void ConnectionGraph::OnCloseConnection(RakPeerInterface *peer, SystemAddress systemAddress)
 | |
| {
 | |
| 	HandleDroppedConnection(peer, systemAddress, ID_CONNECTION_GRAPH_DISCONNECTION_NOTIFICATION);
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::HandleDroppedConnection(RakPeerInterface *peer, SystemAddress systemAddress, unsigned char packetId)
 | |
| {
 | |
| 	assert(peer);
 | |
| 	RemoveParticipant(systemAddress);
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	RemoveAndRelayConnection(ignoreList, packetId, systemAddress, peer->GetExternalID(systemAddress), peer);
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::OnNewIncomingConnection(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	if (autoAddNewConnections==false)
 | |
| 		return;
 | |
| 
 | |
| 	// 0 is the default groupId
 | |
| 	AddNewConnection(peer, packet->systemAddress, 0);
 | |
| }
 | |
| void ConnectionGraph::OnConnectionRequestAccepted(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	if (autoAddNewConnections==false)
 | |
| 		return;
 | |
| 
 | |
| 	RequestConnectionGraph(peer, packet->systemAddress);
 | |
| 
 | |
| 	// 0 is the default groupId
 | |
| 	AddNewConnection(peer, packet->systemAddress, 0);
 | |
| }
 | |
| void ConnectionGraph::SetGroupId(ConnectionGraphGroupID groupId)
 | |
| {
 | |
| 	myGroupId=groupId;
 | |
| }
 | |
| void ConnectionGraph::AddNewConnection(RakPeerInterface *peer, SystemAddress systemAddress, ConnectionGraphGroupID groupId)
 | |
| {
 | |
| 	if (autoAddNewConnections==false && subscribedGroups.HasData(groupId)==false)
 | |
| 		return;
 | |
| 
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	
 | |
| 	SystemAddressAndGroupId first, second;
 | |
| 	first.systemAddress=systemAddress;
 | |
| 	first.groupId=groupId;
 | |
| 	second.systemAddress=peer->GetExternalID(systemAddress);
 | |
| 	second.groupId=myGroupId;
 | |
| 
 | |
| 	AddAndRelayConnection(ignoreList, first, second, (unsigned short)peer->GetAveragePing(systemAddress), peer);
 | |
| }
 | |
| void ConnectionGraph::SubscribeToGroup(ConnectionGraphGroupID groupId)
 | |
| {
 | |
| 	subscribedGroups.Insert(groupId, groupId, true);
 | |
| }
 | |
| void ConnectionGraph::UnsubscribeFromGroup(ConnectionGraphGroupID groupId)
 | |
| {
 | |
| 	subscribedGroups.Remove(groupId);
 | |
| }
 | |
| void ConnectionGraph::RequestConnectionGraph(RakPeerInterface *peer, SystemAddress systemAddress)
 | |
| {
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write((MessageID)ID_CONNECTION_GRAPH_REQUEST);
 | |
| 	stringCompressor->EncodeString(pw,256,&outBitstream);
 | |
| 	peer->Send(&outBitstream, LOW_PRIORITY, RELIABLE_ORDERED, connectionGraphChannel, systemAddress, false);
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 	printf("ID_CONNECTION_GRAPH_REQUEST from %i to %i\n", peer->GetInternalID().port, systemAddress.port);
 | |
| #endif
 | |
| }
 | |
| void ConnectionGraph::OnConnectionGraphRequest(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	char password[256];
 | |
| 	RakNet::BitStream inBitstream(packet->data, packet->length, false);
 | |
| 	inBitstream.IgnoreBits(8);
 | |
| 	stringCompressor->DecodeString(password,256,&inBitstream);
 | |
| 	if (pw && pw[0] && strcmp(pw, password)!=0)
 | |
| 		return;
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 	printf("ID_CONNECTION_GRAPH_REPLY ");
 | |
| #endif
 | |
| 
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write((MessageID)ID_CONNECTION_GRAPH_REPLY);
 | |
| 	stringCompressor->EncodeString(pw,256,&outBitstream);
 | |
| 	SerializeWeightedGraph(&outBitstream, graph);
 | |
| 	peer->Send(&outBitstream, LOW_PRIORITY, RELIABLE_ORDERED, connectionGraphChannel, packet->systemAddress, false);
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 	printf("from %i to %i\n", peer->GetInternalID().port, packet->systemAddress.port);
 | |
| #endif
 | |
| 
 | |
| 	// Add packet->systemAddress to the participant list if it is not already there
 | |
| 	AddParticipant(packet->systemAddress);
 | |
| }
 | |
| void ConnectionGraph::OnConnectionGraphReply(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	unsigned char password[256];
 | |
| 	RakNet::BitStream inBitstream(packet->data, packet->length, false);
 | |
| 	inBitstream.IgnoreBits(8);
 | |
| 	stringCompressor->DecodeString((char*)password,256,&inBitstream);
 | |
| 	if (pw && pw[0] && strcmp(pw, (const char*)password)!=0)
 | |
| 		return;
 | |
| 
 | |
| 	// Serialize the weighted graph and send it to them
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write((MessageID)ID_CONNECTION_GRAPH_UPDATE);
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 	printf("ID_CONNECTION_GRAPH_UPDATE ");
 | |
| #endif
 | |
| 
 | |
| 	// Send our current graph to the sender
 | |
| 	SerializeWeightedGraph(&outBitstream, graph);
 | |
| 
 | |
| 
 | |
| 	// Write the systems that have processed this graph so we don't resend to these systems
 | |
| 	outBitstream.Write((unsigned short) 1);
 | |
| 	outBitstream.Write(peer->GetExternalID(packet->systemAddress));
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 	printf("from %i to %i\n", peer->GetInternalID().port, packet->systemAddress.port);
 | |
| #endif
 | |
| 
 | |
| 	peer->Send(&outBitstream, LOW_PRIORITY, RELIABLE_ORDERED, connectionGraphChannel, packet->systemAddress, false);
 | |
| 
 | |
| 	// Add packet->systemAddress to the participant list if it is not already there
 | |
| 	AddParticipant(packet->systemAddress);
 | |
| 
 | |
| 	if (DeserializeWeightedGraph(&inBitstream, peer)==false)
 | |
| 		return;
 | |
| 
 | |
| 	// Forward the updated graph to all current participants
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	ignoreList.Insert(packet->systemAddress,packet->systemAddress, true);
 | |
| 	BroadcastGraphUpdate(ignoreList, peer);
 | |
| }
 | |
| void ConnectionGraph::OnConnectionGraphUpdate(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	// Only accept from participants
 | |
| 	if (participantList.HasData(packet->systemAddress)==false)
 | |
| 		return;
 | |
| 
 | |
| 	RakNet::BitStream inBitstream(packet->data, packet->length, false);
 | |
| 	inBitstream.IgnoreBits(8);
 | |
| 	
 | |
| 	if (DeserializeWeightedGraph(&inBitstream, peer)==false)
 | |
| 		return;
 | |
| 
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	DeserializeIgnoreList(ignoreList, &inBitstream);
 | |
| 
 | |
| 	// Forward the updated graph to all participants.
 | |
| 	ignoreList.Insert(packet->systemAddress,packet->systemAddress, false);
 | |
| 	BroadcastGraphUpdate(ignoreList, peer);
 | |
| }
 | |
| void ConnectionGraph::OnNewConnection(RakPeerInterface *peer, Packet *packet)
 | |
| {
 | |
| 	// Only accept from participants
 | |
| 	if (participantList.HasData(packet->systemAddress)==false)
 | |
| 		return;
 | |
| 
 | |
| 	SystemAddressAndGroupId node1, node2;
 | |
| 	unsigned short ping;
 | |
| 	RakNet::BitStream inBitstream(packet->data, packet->length, false);
 | |
| 	inBitstream.IgnoreBits(8);
 | |
| 	inBitstream.Read(node1.systemAddress);
 | |
| 	inBitstream.Read(node1.groupId);
 | |
| 	inBitstream.Read(node2.systemAddress);
 | |
| 	inBitstream.Read(node2.groupId);
 | |
| 	if (inBitstream.Read(ping)==false)
 | |
| 		return;
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	DeserializeIgnoreList(ignoreList, &inBitstream);
 | |
| 	ignoreList.Insert(packet->systemAddress,packet->systemAddress, false);
 | |
| 	AddAndRelayConnection(ignoreList, node1, node2, ping, peer);	
 | |
| }
 | |
| bool ConnectionGraph::OnConnectionLost(RakPeerInterface *peer, Packet *packet, unsigned char packetId)
 | |
| {
 | |
| 	// Only accept from participants
 | |
| 	if (participantList.HasData(packet->systemAddress)==false)
 | |
| 		return false;
 | |
| 
 | |
| 	SystemAddress node1, node2;
 | |
| 	RakNet::BitStream inBitstream(packet->data, packet->length, false);
 | |
| 	inBitstream.IgnoreBits(8);
 | |
| 	// This is correct - group IDs are not written for removal, only addition.
 | |
| 	inBitstream.Read(node1);
 | |
| 	if (inBitstream.Read(node2)==false)
 | |
| 		return false;
 | |
| 	DataStructures::OrderedList<SystemAddress,SystemAddress> ignoreList;
 | |
| 	DeserializeIgnoreList(ignoreList, &inBitstream);
 | |
| 	ignoreList.Insert(packet->systemAddress, packet->systemAddress, false);
 | |
| 	
 | |
| 	return RemoveAndRelayConnection(ignoreList, packetId, node1, node2, peer);
 | |
| }
 | |
| bool ConnectionGraph::DeserializeIgnoreList(DataStructures::OrderedList<SystemAddress,SystemAddress> &ignoreList, RakNet::BitStream *inBitstream )
 | |
| {
 | |
| 	unsigned short count;
 | |
| 	SystemAddress temp;
 | |
| 	unsigned i;
 | |
| 	inBitstream->Read(count);
 | |
| 	for (i=0; i < count; i++)
 | |
| 	{
 | |
| 		if (inBitstream->Read(temp)==false)
 | |
| 		{
 | |
| 			assert(0);
 | |
| 			return false;
 | |
| 		}
 | |
| 		ignoreList.Insert(temp,temp, true);
 | |
| 	}
 | |
| 	return true;
 | |
| }
 | |
| void ConnectionGraph::SerializeWeightedGraph(RakNet::BitStream *out, const DataStructures::WeightedGraph<ConnectionGraph::SystemAddressAndGroupId, unsigned short, false> &g) const
 | |
| {
 | |
| 	unsigned nodeIndex, connectionIndex;
 | |
| 	BitSize_t countOffset, oldOffset;
 | |
| 	unsigned short count;
 | |
| 	SystemAddressAndGroupId node1, node2;
 | |
| 	unsigned short weight;
 | |
| 	out->WriteCompressed(g.GetNodeCount());
 | |
| 	for (nodeIndex=0; nodeIndex < g.GetNodeCount(); nodeIndex++)
 | |
| 	{
 | |
| 		// Write the node
 | |
| 		node1=g.GetNodeAtIndex(nodeIndex);
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 		printf("[%i] ", node1.systemAddress.port);
 | |
| #endif
 | |
| 		out->Write(node1.systemAddress);
 | |
| 		out->Write(node1.groupId);
 | |
| 
 | |
| 		// Write the adjacency list count
 | |
| 		count=(unsigned short)g.GetConnectionCount(nodeIndex);
 | |
| 		out->AlignWriteToByteBoundary();
 | |
| 		countOffset=out->GetWriteOffset();
 | |
| 		out->Write(count);
 | |
| 		count=0;
 | |
| 		for (connectionIndex=0; connectionIndex < g.GetConnectionCount(nodeIndex); connectionIndex++)
 | |
| 		{
 | |
| 			g.GetConnectionAtIndex(nodeIndex, connectionIndex, node2, weight);
 | |
| 			// For efficiencies' sake, only serialize the upper half of the connection pairs
 | |
| 			if (node2 > node1)
 | |
| 			{
 | |
| 				count++;
 | |
| 				out->Write(node2.systemAddress);
 | |
| 				out->Write(node2.groupId);
 | |
| 				out->Write(weight);
 | |
| 
 | |
| #ifdef _CONNECTION_GRAPH_DEBUG_PRINT
 | |
| 				printf("(%i) ", node2.systemAddress.port);
 | |
| #endif
 | |
| 			}
 | |
| 		}
 | |
| 
 | |
| 		// Go back and change how many elements were written
 | |
| 		oldOffset=out->GetWriteOffset();
 | |
| 		out->SetWriteOffset(countOffset);
 | |
| 		out->Write(count);
 | |
| 		out->SetWriteOffset(oldOffset);
 | |
| 	}
 | |
| }
 | |
| bool ConnectionGraph::DeserializeWeightedGraph(RakNet::BitStream *inBitstream, RakPeerInterface *peer)
 | |
| {
 | |
| 	unsigned nodeCount, nodeIndex, connectionIndex;
 | |
| 	unsigned short connectionCount;
 | |
| 	SystemAddressAndGroupId node, connection;
 | |
| 	bool anyConnectionsNew=false;
 | |
| 	unsigned short weight;
 | |
| 	inBitstream->ReadCompressed(nodeCount);
 | |
| 	for (nodeIndex=0; nodeIndex < nodeCount; nodeIndex++)
 | |
| 	{
 | |
| 		inBitstream->Read(node.systemAddress);
 | |
| 		inBitstream->Read(node.groupId);
 | |
| 
 | |
| 		inBitstream->AlignReadToByteBoundary();
 | |
| 		if (inBitstream->Read(connectionCount)==false)
 | |
| 		{
 | |
| 			assert(0);
 | |
| 			return false;
 | |
| 		}
 | |
| 		for (connectionIndex=0; connectionIndex < connectionCount; connectionIndex++)
 | |
| 		{
 | |
| 			inBitstream->Read(connection.systemAddress);
 | |
| 			inBitstream->Read(connection.groupId);
 | |
| 			if (inBitstream->Read(weight)==false)
 | |
| 			{
 | |
| 				assert(0);
 | |
| 				return false;
 | |
| 			}
 | |
| 			if (subscribedGroups.HasData(connection.groupId)==false ||
 | |
| 				subscribedGroups.HasData(node.groupId)==false)
 | |
| 				continue;
 | |
| 			RakAssert(node.systemAddress!=UNASSIGNED_SYSTEM_ADDRESS);
 | |
| 			RakAssert(connection.systemAddress!=UNASSIGNED_SYSTEM_ADDRESS);
 | |
| 			if (IsNewRemoteConnection(node,connection,peer))
 | |
| 				NotifyUserOfRemoteConnection(node,connection,weight,peer);
 | |
| 
 | |
| 			if (graph.HasConnection(node,connection)==false)
 | |
| 				anyConnectionsNew=true;
 | |
| 
 | |
| 			graph.AddConnection(node,connection,weight);
 | |
| 		}
 | |
| 	}
 | |
| 	return anyConnectionsNew;
 | |
| }
 | |
| void ConnectionGraph::RemoveParticipant(SystemAddress systemAddress)
 | |
| {
 | |
| 	unsigned index;
 | |
| 	bool objectExists;
 | |
| 	index=participantList.GetIndexFromKey(systemAddress, &objectExists);
 | |
| 	if (objectExists)
 | |
| 		participantList.RemoveAtIndex(index);
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::AddParticipant(SystemAddress systemAddress)
 | |
| {
 | |
| 	participantList.Insert(systemAddress,systemAddress, false);
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::AddAndRelayConnection(DataStructures::OrderedList<SystemAddress,SystemAddress> &ignoreList, const SystemAddressAndGroupId &conn1, const SystemAddressAndGroupId &conn2, unsigned short ping, RakPeerInterface *peer)
 | |
| {
 | |
| 	if (graph.HasConnection(conn1,conn2))
 | |
| 		return;
 | |
| 
 | |
| 	if (ping==65535)
 | |
| 		ping=0;
 | |
| 	assert(conn1.systemAddress!=UNASSIGNED_SYSTEM_ADDRESS);
 | |
| 	assert(conn2.systemAddress!=UNASSIGNED_SYSTEM_ADDRESS);
 | |
| 
 | |
| 	if (IsNewRemoteConnection(conn1,conn2,peer))
 | |
| 	{
 | |
| 		NotifyUserOfRemoteConnection(conn1,conn2,ping,peer);
 | |
| 
 | |
| 		// What was this return here for?
 | |
| 	//	return;
 | |
| 	}
 | |
| 
 | |
| 	graph.AddConnection(conn1,conn2,ping);
 | |
| 
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write((MessageID)ID_CONNECTION_GRAPH_NEW_CONNECTION);
 | |
| 	outBitstream.Write(conn1.systemAddress);
 | |
| 	outBitstream.Write(conn1.groupId);
 | |
| 	outBitstream.Write(conn2.systemAddress);
 | |
| 	outBitstream.Write(conn2.groupId);
 | |
| 	outBitstream.Write(ping);
 | |
| 	ignoreList.Insert(conn2.systemAddress,conn2.systemAddress, false);
 | |
| 	ignoreList.Insert(conn1.systemAddress,conn1.systemAddress, false);
 | |
| 	SerializeIgnoreListAndBroadcast(&outBitstream, ignoreList, peer);
 | |
| }
 | |
| bool ConnectionGraph::RemoveAndRelayConnection(DataStructures::OrderedList<SystemAddress,SystemAddress> &ignoreList, unsigned char packetId, const SystemAddress node1, const SystemAddress node2, RakPeerInterface *peer)
 | |
| {
 | |
| 	SystemAddressAndGroupId n1, n2;
 | |
| 	n1.systemAddress=node1;
 | |
| 	n2.systemAddress=node2;
 | |
| 	if (graph.HasConnection(n1,n2)==false)
 | |
| 		return false;
 | |
| 	graph.RemoveConnection(n1,n2);
 | |
| 
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write(packetId);
 | |
| 	outBitstream.Write(node1);
 | |
| 	outBitstream.Write(node2);
 | |
| 
 | |
| 	ignoreList.Insert(node1,node1, false);
 | |
| 	ignoreList.Insert(node2,node2, false);
 | |
| 	SerializeIgnoreListAndBroadcast(&outBitstream, ignoreList, peer);
 | |
| 
 | |
| 	return true;
 | |
| }
 | |
| 
 | |
| void ConnectionGraph::BroadcastGraphUpdate(DataStructures::OrderedList<SystemAddress,SystemAddress> &ignoreList, RakPeerInterface *peer)
 | |
| {
 | |
| 	RakNet::BitStream outBitstream;
 | |
| 	outBitstream.Write((MessageID)ID_CONNECTION_GRAPH_UPDATE);
 | |
| 	SerializeWeightedGraph(&outBitstream, graph);
 | |
| 	SerializeIgnoreListAndBroadcast(&outBitstream, ignoreList, peer);
 | |
| }
 | |
| void ConnectionGraph::SerializeIgnoreListAndBroadcast(RakNet::BitStream *outBitstream, DataStructures::OrderedList<SystemAddress,SystemAddress> &ignoreList, RakPeerInterface *peer)
 | |
| {
 | |
| 	DataStructures::List<SystemAddress> sendList;
 | |
| 	unsigned i;
 | |
| 	for (i=0; i < participantList.Size(); i++)
 | |
| 	{
 | |
| 		if (ignoreList.HasData(participantList[i])==false)
 | |
| 			sendList.Insert(participantList[i]);
 | |
| 	}
 | |
| 	if (sendList.Size()==0)
 | |
| 		return;
 | |
| 
 | |
| 	SystemAddress self = peer->GetExternalID(sendList[0]);
 | |
| 	ignoreList.Insert(self,self, false);
 | |
| 	outBitstream->Write((unsigned short) (ignoreList.Size()+sendList.Size()));
 | |
| 	for (i=0; i < ignoreList.Size(); i++)
 | |
| 		outBitstream->Write(ignoreList[i]);
 | |
| 	for (i=0; i < sendList.Size(); i++)
 | |
| 		outBitstream->Write(sendList[i]);
 | |
| 
 | |
| 	for (i=0; i < sendList.Size(); i++)
 | |
| 	{
 | |
| 		peer->Send(outBitstream, LOW_PRIORITY, RELIABLE_ORDERED, connectionGraphChannel, sendList[i], false);
 | |
| 	}
 | |
| }
 | |
| bool ConnectionGraph::IsNewRemoteConnection(const SystemAddressAndGroupId &conn1, const SystemAddressAndGroupId &conn2,RakPeerInterface *peer)
 | |
| {
 | |
| 	if (graph.HasConnection(conn1,conn2)==false &&
 | |
| 		subscribedGroups.HasData(conn1.groupId) &&
 | |
| 		subscribedGroups.HasData(conn2.groupId) &&
 | |
| 		(peer->IsConnected(conn1.systemAddress)==false || peer->IsConnected(conn2.systemAddress)==false))
 | |
| 	{
 | |
| 		SystemAddress externalId1, externalId2;
 | |
| 		externalId1=peer->GetExternalID(conn1.systemAddress);
 | |
| 		externalId2=peer->GetExternalID(conn2.systemAddress);
 | |
| 		return (externalId1!=conn1.systemAddress && externalId1!=conn2.systemAddress &&
 | |
| 			externalId2!=conn1.systemAddress && externalId2!=conn2.systemAddress);
 | |
| 	}
 | |
| 	return false;
 | |
| }
 | |
| void ConnectionGraph::NotifyUserOfRemoteConnection(const SystemAddressAndGroupId &conn1, const SystemAddressAndGroupId &conn2,unsigned short ping, RakPeerInterface *peer)
 | |
| {
 | |
| 	// Create a packet to tell the user of this event
 | |
| 	static const int length=sizeof(MessageID) + (sizeof(SystemAddress) + sizeof(ConnectionGraphGroupID)) * 2 + sizeof(unsigned short);
 | |
| 	Packet *p = peer->AllocatePacket(length);
 | |
| 	RakNet::BitStream b(p->data, length, false);
 | |
| 	p->bitSize=p->length*8;
 | |
| 	b.SetWriteOffset(0);
 | |
| 	b.Write((MessageID)ID_REMOTE_NEW_INCOMING_CONNECTION);
 | |
| 	b.Write(conn1.systemAddress);
 | |
| 	b.Write(conn1.groupId);
 | |
| 	b.Write(conn2.systemAddress);
 | |
| 	b.Write(conn2.groupId);
 | |
| 	b.Write(ping);
 | |
| 	if (peer->IsConnected(conn2.systemAddress)==false)
 | |
| 		p->systemAddress=conn2.systemAddress;
 | |
| 	else
 | |
| 		p->systemAddress=conn1.systemAddress;
 | |
| 	peer->PushBackPacket(p, false);
 | |
| }
 | |
| 
 | |
| #ifdef _MSC_VER
 | |
| #pragma warning( pop )
 | |
| #endif
 | 
