mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-10-24 08:18:10 +00:00
330 lines
10 KiB
C++
330 lines
10 KiB
C++
#include "Router.h"
|
|
#include "BitStream.h"
|
|
#include "RakPeerInterface.h"
|
|
#include "MessageIdentifiers.h"
|
|
#include "RakAssert.h"
|
|
|
|
//#define _DO_PRINTF
|
|
|
|
#ifdef _DO_PRINTF
|
|
#include <stdio.h>
|
|
#endif
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( push )
|
|
#endif
|
|
|
|
Router::Router()
|
|
{
|
|
graph=0;
|
|
restrictByType=false;
|
|
rakPeer=0;
|
|
DataStructures::OrderedList<unsigned char,unsigned char>::IMPLEMENT_DEFAULT_COMPARISON();
|
|
}
|
|
Router::~Router()
|
|
{
|
|
}
|
|
void Router::SetRestrictRoutingByType(bool restrict__)
|
|
{
|
|
restrictByType=restrict__;
|
|
}
|
|
void Router::AddAllowedType(unsigned char messageId)
|
|
{
|
|
if (allowedTypes.HasData(messageId)==false)
|
|
allowedTypes.Insert(messageId,messageId, true);
|
|
}
|
|
void Router::RemoveAllowedType(unsigned char messageId)
|
|
{
|
|
if (allowedTypes.HasData(messageId)==true)
|
|
allowedTypes.Remove(messageId);
|
|
}
|
|
void Router::SetConnectionGraph(DataStructures::WeightedGraph<ConnectionGraph::SystemAddressAndGroupId, unsigned short, false> *connectionGraph)
|
|
{
|
|
graph=connectionGraph;
|
|
}
|
|
bool Router::Send( const char *data, BitSize_t bitLength, PacketPriority priority, PacketReliability reliability, char orderingChannel, SystemAddress systemAddress )
|
|
{
|
|
if (systemAddress!=UNASSIGNED_SYSTEM_ADDRESS)
|
|
{
|
|
RakAssert(data);
|
|
RakAssert(bitLength);
|
|
// Prevent recursion in case a routing call itself calls the router
|
|
if (bitLength>=8 && data[0]==ID_ROUTE_AND_MULTICAST)
|
|
return false;
|
|
|
|
SystemAddressList systemAddressList;
|
|
systemAddressList.AddSystem(systemAddress);
|
|
return Send((char*)data, bitLength, priority, reliability, orderingChannel, &systemAddressList);
|
|
}
|
|
return false;
|
|
}
|
|
bool Router::Send( char *data, BitSize_t bitLength, PacketPriority priority, PacketReliability reliability, char orderingChannel, SystemAddressList *recipients )
|
|
{
|
|
RakAssert(data);
|
|
RakAssert(bitLength);
|
|
if (recipients->GetList()->Size()==0)
|
|
return false;
|
|
if (bitLength==0)
|
|
return false;
|
|
DataStructures::Tree<ConnectionGraph::SystemAddressAndGroupId> tree;
|
|
SystemAddress root;
|
|
root = rakPeer->GetExternalID(rakPeer->GetSystemAddressFromIndex(0));
|
|
if (root==UNASSIGNED_SYSTEM_ADDRESS)
|
|
return false;
|
|
DataStructures::List<ConnectionGraph::SystemAddressAndGroupId> recipientList;
|
|
unsigned i;
|
|
for (i=0; i < recipients->Size(); i++)
|
|
recipientList.Insert(ConnectionGraph::SystemAddressAndGroupId(recipients->GetList()->operator [](i),0));
|
|
if (graph->GetSpanningTree(tree, &recipientList, ConnectionGraph::SystemAddressAndGroupId(root,0), 65535)==false)
|
|
return false;
|
|
|
|
RakNet::BitStream out;
|
|
|
|
// Write timestamp first, if the user had a timestamp
|
|
if (data[0]==ID_TIMESTAMP && bitLength >= BYTES_TO_BITS(sizeof(MessageID)+sizeof(RakNetTime)))
|
|
{
|
|
out.Write(data, sizeof(MessageID)+sizeof(RakNetTime));
|
|
data+=sizeof(MessageID)+sizeof(RakNetTime);
|
|
bitLength-=BYTES_TO_BITS(sizeof(MessageID)+sizeof(RakNetTime));
|
|
}
|
|
|
|
SendTree(priority, reliability, orderingChannel, &tree, data, bitLength, &out, recipients);
|
|
return true;
|
|
}
|
|
void Router::SendTree(PacketPriority priority, PacketReliability reliability, char orderingChannel, DataStructures::Tree<ConnectionGraph::SystemAddressAndGroupId> *tree, const char *data, BitSize_t bitLength, RakNet::BitStream *out, SystemAddressList *recipients)
|
|
{
|
|
BitSize_t outputOffset;
|
|
|
|
// Write routing identifer
|
|
out->Write((MessageID)ID_ROUTE_AND_MULTICAST);
|
|
|
|
// Write the send parameters
|
|
out->WriteCompressed((unsigned char)priority);
|
|
out->WriteCompressed((unsigned char)reliability);
|
|
out->WriteCompressed((unsigned char)orderingChannel);
|
|
|
|
// Write the user payload length
|
|
out->Write((unsigned int)bitLength);
|
|
// out->PrintBits();
|
|
// payload->PrintBits();
|
|
|
|
out->AlignWriteToByteBoundary();
|
|
// payload->AlignReadToByteBoundary();
|
|
// out->Write(payload, payload->GetNumberOfUnreadBits());
|
|
// out->PrintBits();
|
|
if ((bitLength % 8)==0)
|
|
out->Write(data, BITS_TO_BYTES(bitLength));
|
|
else
|
|
out->WriteBits((const unsigned char*)data, bitLength, false);
|
|
|
|
// Save where to start writing per-system data
|
|
outputOffset=out->GetWriteOffset();
|
|
|
|
// Write every child of the root of the tree (SystemAddress, isRecipient, branch)
|
|
unsigned i;
|
|
for (i=0; i < tree->children.Size(); i++)
|
|
{
|
|
// Start writing at the per-system data byte
|
|
out->SetWriteOffset(outputOffset);
|
|
|
|
// Write our external IP to designate the sender
|
|
out->Write(rakPeer->GetExternalID(tree->children[i]->data.systemAddress));
|
|
|
|
// Serialize the tree
|
|
SerializePreorder(tree->children[i], out, recipients);
|
|
|
|
// Send to the first hop
|
|
#ifdef _DO_PRINTF
|
|
printf("%i sending to %i\n", rakPeer->GetExternalID(tree->children[i]->data.systemAddress).port, tree->children[i]->data.systemAddress.port);
|
|
#endif
|
|
rakPeer->Send(out, priority, reliability, orderingChannel, tree->children[i]->data.systemAddress, false);
|
|
}
|
|
}
|
|
PluginReceiveResult Router::OnReceive(RakPeerInterface *peer, Packet *packet)
|
|
{
|
|
(void) peer;
|
|
|
|
if (packet->data[0]==ID_ROUTE_AND_MULTICAST ||
|
|
(packet->length>5 && packet->data[0]==ID_TIMESTAMP && packet->data[5]==ID_ROUTE_AND_MULTICAST))
|
|
{
|
|
#ifdef _DO_PRINTF
|
|
printf("%i got routed message from %i\n", peer->GetExternalID(packet->systemAddress).port, packet->systemAddress.port);
|
|
#endif
|
|
RakNetTime timestamp;
|
|
PacketPriority priority;
|
|
PacketReliability reliability;
|
|
unsigned char orderingChannel;
|
|
SystemAddress originalSender;
|
|
RakNet::BitStream out;
|
|
BitSize_t outStartingOffset;
|
|
unsigned int payloadBitLength;
|
|
unsigned payloadWriteByteOffset;
|
|
RakNet::BitStream incomingBitstream(packet->data, packet->length, false);
|
|
incomingBitstream.IgnoreBits(8);
|
|
|
|
if (packet->data[0]==ID_TIMESTAMP)
|
|
{
|
|
incomingBitstream.Read(timestamp);
|
|
out.Write((MessageID)ID_TIMESTAMP);
|
|
out.Write(timestamp);
|
|
incomingBitstream.IgnoreBits(8);
|
|
}
|
|
|
|
// Read the send parameters
|
|
unsigned char c;
|
|
incomingBitstream.ReadCompressed(c);
|
|
priority=(PacketPriority)c;
|
|
incomingBitstream.ReadCompressed(c);
|
|
reliability=(PacketReliability)c;
|
|
incomingBitstream.ReadCompressed(orderingChannel);
|
|
incomingBitstream.Read(payloadBitLength);
|
|
|
|
out.Write((MessageID)ID_ROUTE_AND_MULTICAST);
|
|
out.WriteCompressed((unsigned char)priority);
|
|
out.WriteCompressed((unsigned char)reliability);
|
|
out.WriteCompressed((unsigned char)orderingChannel);
|
|
out.Write(payloadBitLength);
|
|
out.AlignWriteToByteBoundary();
|
|
incomingBitstream.AlignReadToByteBoundary();
|
|
payloadWriteByteOffset=(unsigned int) BITS_TO_BYTES(out.GetWriteOffset());
|
|
out.Write(&incomingBitstream, payloadBitLength); // This write also does a read on incomingBitStream
|
|
|
|
if (restrictByType)
|
|
{
|
|
RakNet::BitStream t(out.GetData()+payloadWriteByteOffset, sizeof(unsigned char), false);
|
|
MessageID messageID;
|
|
t.Read(messageID);
|
|
if (allowedTypes.HasData(messageID)==false)
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE; // Don't route restricted types
|
|
}
|
|
|
|
incomingBitstream.Read(originalSender);
|
|
out.Write(originalSender);
|
|
outStartingOffset=out.GetWriteOffset();
|
|
|
|
// Deserialize the root
|
|
bool hasData=false;
|
|
SystemAddress recipient;
|
|
unsigned short numberOfChildren;
|
|
incomingBitstream.Read(hasData);
|
|
incomingBitstream.Read(recipient); // This should be our own address
|
|
if (incomingBitstream.ReadCompressed(numberOfChildren)==false)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
}
|
|
|
|
unsigned childIndex;
|
|
bool childHasData=false;
|
|
SystemAddress childRecipient;
|
|
unsigned short childNumberOfChildren;
|
|
SystemAddress immediateRecipient;
|
|
immediateRecipient=UNASSIGNED_SYSTEM_ADDRESS;
|
|
int pendingNodeCount=0;
|
|
|
|
for (childIndex=0; childIndex < numberOfChildren; childIndex++)
|
|
{
|
|
while (pendingNodeCount!=-1)
|
|
{
|
|
// Copy out the serialized subtree for this child
|
|
incomingBitstream.Read(childHasData);
|
|
incomingBitstream.Read(childRecipient);
|
|
if (!incomingBitstream.ReadCompressed(childNumberOfChildren))
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
if (immediateRecipient==UNASSIGNED_SYSTEM_ADDRESS)
|
|
{
|
|
immediateRecipient=childRecipient;
|
|
}
|
|
|
|
pendingNodeCount+=childNumberOfChildren-1;
|
|
|
|
out.Write(childHasData);
|
|
out.Write(childRecipient);
|
|
out.WriteCompressed(childNumberOfChildren);
|
|
}
|
|
|
|
#ifdef _DO_PRINTF
|
|
printf("%i routing to %i\n", peer->GetExternalID(packet->systemAddress).port, immediateRecipient.port);
|
|
#endif
|
|
|
|
// Send what we got so far
|
|
rakPeer->Send(&out, priority, reliability, orderingChannel, immediateRecipient, false);
|
|
|
|
// Restart writing the per recipient data
|
|
out.SetWriteOffset(outStartingOffset);
|
|
|
|
// Reread the top level node
|
|
immediateRecipient=UNASSIGNED_SYSTEM_ADDRESS;
|
|
|
|
pendingNodeCount=0;
|
|
|
|
}
|
|
|
|
// Write the user payload to the packet struct if this is a destination and change the sender and return true
|
|
if (hasData)
|
|
{
|
|
#ifdef _DO_PRINTF
|
|
printf("%i returning payload to user\n", peer->GetExternalID(packet->systemAddress).port);
|
|
#endif
|
|
|
|
if (packet->data[0]==ID_TIMESTAMP )
|
|
{
|
|
memcpy( packet->data + sizeof(RakNetTime)+sizeof(unsigned char), out.GetData()+payloadWriteByteOffset, BITS_TO_BYTES(payloadBitLength) );
|
|
packet->bitSize=BYTES_TO_BITS(sizeof(RakNetTime)+sizeof(unsigned char))+payloadBitLength;
|
|
}
|
|
else
|
|
{
|
|
memcpy( packet->data, out.GetData()+payloadWriteByteOffset, BITS_TO_BYTES(payloadBitLength) );
|
|
packet->bitSize=payloadBitLength;
|
|
}
|
|
packet->length=(unsigned int) BITS_TO_BYTES(packet->bitSize);
|
|
packet->systemIndex=65535;
|
|
packet->systemAddress=originalSender;
|
|
|
|
return RR_CONTINUE_PROCESSING;
|
|
}
|
|
|
|
// Absorb
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
}
|
|
|
|
return RR_CONTINUE_PROCESSING;
|
|
}
|
|
void Router::OnAttach(RakPeerInterface *peer)
|
|
{
|
|
rakPeer=peer;
|
|
peer->SetRouterInterface(this);
|
|
}
|
|
void Router::OnDetach(RakPeerInterface *peer)
|
|
{
|
|
peer->RemoveRouterInterface(this);
|
|
}
|
|
void Router::OnShutdown(RakPeerInterface *peer)
|
|
{
|
|
(void) peer;
|
|
}
|
|
void Router::Update(RakPeerInterface *peer)
|
|
{
|
|
(void) peer;
|
|
}
|
|
void Router::OnCloseConnection(RakPeerInterface *peer, SystemAddress systemAddress)
|
|
{
|
|
(void) peer;
|
|
(void) systemAddress;
|
|
}
|
|
void Router::SerializePreorder(DataStructures::Tree<ConnectionGraph::SystemAddressAndGroupId> *tree, RakNet::BitStream *out, SystemAddressList *recipients) const
|
|
{
|
|
unsigned i;
|
|
out->Write((bool) (recipients->GetList()->GetIndexOf(tree->data.systemAddress)!=MAX_UNSIGNED_LONG));
|
|
out->Write(tree->data.systemAddress);
|
|
out->WriteCompressed((unsigned short) tree->children.Size());
|
|
for (i=0; i < tree->children.Size(); i++)
|
|
SerializePreorder(tree->children[i], out, recipients);
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( pop )
|
|
#endif
|