mirror of
https://github.com/DarkflameUniverse/DarkflameServer.git
synced 2025-10-24 00:08:07 +00:00
562 lines
16 KiB
C++
562 lines
16 KiB
C++
#include "FileListTransfer.h"
|
|
#include "DS_HuffmanEncodingTree.h"
|
|
#include "FileListTransferCBInterface.h"
|
|
#include "StringCompressor.h"
|
|
#include "FileList.h"
|
|
#include "DS_Queue.h"
|
|
#include "MessageIdentifiers.h"
|
|
#include "RakNetTypes.h"
|
|
#include "RakPeerInterface.h"
|
|
#include <assert.h>
|
|
#ifdef _PS3
|
|
#include <alloca.h>
|
|
#endif
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( push )
|
|
#endif
|
|
|
|
struct FileListTransfer::FileListReceiver
|
|
{
|
|
FileListTransferCBInterface *downloadHandler;
|
|
SystemAddress allowedSender;
|
|
HuffmanEncodingTree tree;
|
|
unsigned short setID;
|
|
unsigned setCount;
|
|
unsigned setTotalCompressedTransmissionLength;
|
|
unsigned setTotalFinalLength;
|
|
bool gotSetHeader;
|
|
bool deleteDownloadHandler;
|
|
bool isCompressed;
|
|
};
|
|
|
|
FileListTransfer::FileListTransfer()
|
|
{
|
|
rakPeer=0;
|
|
setId=0;
|
|
callback=0;
|
|
DataStructures::Map<unsigned short, FileListReceiver*>::IMPLEMENT_DEFAULT_COMPARISON();
|
|
}
|
|
FileListTransfer::~FileListTransfer()
|
|
{
|
|
Clear();
|
|
}
|
|
unsigned short FileListTransfer::SetupReceive(FileListTransferCBInterface *handler, bool deleteHandler, SystemAddress allowedSender)
|
|
{
|
|
if (rakPeer->IsConnected(allowedSender)==false)
|
|
return (unsigned short)-1;
|
|
FileListReceiver *receiver;
|
|
|
|
if (fileListReceivers.Has(setId))
|
|
{
|
|
receiver=fileListReceivers.Get(setId);
|
|
receiver->downloadHandler->OnDereference();
|
|
if (receiver->deleteDownloadHandler)
|
|
delete receiver->downloadHandler;
|
|
delete receiver;
|
|
fileListReceivers.Delete(setId);
|
|
}
|
|
|
|
unsigned short oldId;
|
|
receiver = new FileListReceiver;
|
|
receiver->downloadHandler=handler;
|
|
receiver->allowedSender=allowedSender;
|
|
receiver->gotSetHeader=false;
|
|
receiver->deleteDownloadHandler=deleteHandler;
|
|
fileListReceivers.Set(setId, receiver);
|
|
oldId=setId;
|
|
if (++setId==(unsigned short)-1)
|
|
setId=0;
|
|
return oldId;
|
|
}
|
|
|
|
void FileListTransfer::Send(FileList *fileList, RakPeerInterface *rakPeer, SystemAddress recipient, unsigned short setID, PacketPriority priority, char orderingChannel, bool compressData)
|
|
{
|
|
RakNet::BitStream outBitstream, encodedData;
|
|
HuffmanEncodingTree tree;
|
|
unsigned int frequencyTable[ 256 ];
|
|
unsigned int i,j;
|
|
unsigned totalCompressedLength, totalLength;
|
|
DataStructures::Queue<FileListNode> compressedFiles;
|
|
FileListNode n;
|
|
|
|
if (callback)
|
|
fileList->SetCallback(callback);
|
|
|
|
totalCompressedLength=totalLength=0;
|
|
if (compressData)
|
|
{
|
|
memset(frequencyTable,0,256*sizeof(unsigned int));
|
|
|
|
for (i=0; i < fileList->fileList.Size(); i++)
|
|
{
|
|
const FileListNode &fileListNode = fileList->fileList[i];
|
|
for (j=0; j < fileListNode.dataLengthBytes; j++)
|
|
{
|
|
++frequencyTable[(unsigned char)(fileListNode.data[j])];
|
|
}
|
|
}
|
|
|
|
tree.GenerateFromFrequencyTable(frequencyTable);
|
|
|
|
// Compress all the files, so we know the total compressed size to be sent
|
|
|
|
for (i=0; i < fileList->fileList.Size(); i++)
|
|
{
|
|
encodedData.Reset();
|
|
// Why send compressed chunks if we are not sending the whole file?
|
|
assert(fileList->fileList[i].fileLengthBytes==fileList->fileList[i].fileLengthBytes);
|
|
tree.EncodeArray((unsigned char*)fileList->fileList[i].data, (size_t) fileList->fileList[i].dataLengthBytes, &encodedData);
|
|
n.dataLengthBytes=encodedData.GetNumberOfBitsUsed(); // Temporarily actually storing bits
|
|
totalCompressedLength+=(unsigned int) BITS_TO_BYTES(n.dataLengthBytes);
|
|
totalLength+=fileList->fileList[i].fileLengthBytes;
|
|
n.data = (char*) rakMalloc( (size_t) BITS_TO_BYTES(n.dataLengthBytes) );
|
|
memcpy(n.data, encodedData.GetData(), (size_t) BITS_TO_BYTES(n.dataLengthBytes));
|
|
compressedFiles.Push(n);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i=0; i < fileList->fileList.Size(); i++)
|
|
{
|
|
const FileListNode &fileListNode = fileList->fileList[i];
|
|
totalLength+=fileListNode.fileLengthBytes;
|
|
}
|
|
}
|
|
|
|
// Write the chunk header, which contains the frequency table, the total number of files, and the total number of bytes
|
|
bool anythingToWrite;
|
|
outBitstream.Write((MessageID)ID_FILE_LIST_TRANSFER_HEADER);
|
|
outBitstream.Write(setID);
|
|
anythingToWrite=fileList->fileList.Size()>0;
|
|
outBitstream.Write(anythingToWrite);
|
|
if (anythingToWrite)
|
|
{
|
|
if (compressData)
|
|
{
|
|
outBitstream.Write(true);
|
|
for (i=0; i < 256; i++)
|
|
outBitstream.WriteCompressed(frequencyTable[i]);
|
|
outBitstream.WriteCompressed(fileList->fileList.Size()); // SetCount
|
|
outBitstream.WriteCompressed(totalLength);
|
|
outBitstream.WriteCompressed(totalCompressedLength);
|
|
}
|
|
else
|
|
{
|
|
outBitstream.Write(false);
|
|
outBitstream.WriteCompressed(fileList->fileList.Size());
|
|
outBitstream.WriteCompressed(totalLength);
|
|
}
|
|
|
|
rakPeer->Send(&outBitstream, priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
|
|
// Next part arrives at FileListTransfer::DecodeFile
|
|
if (compressData)
|
|
{
|
|
// Send each possibly compressed file
|
|
for (i=0; i < compressedFiles.Size(); i++)
|
|
{
|
|
outBitstream.Reset();
|
|
outBitstream.Write((MessageID)ID_FILE_LIST_TRANSFER_FILE);
|
|
outBitstream.Write(fileList->fileList[i].context);
|
|
outBitstream.Write(setID);
|
|
outBitstream.WriteCompressed(i);
|
|
outBitstream.WriteCompressed(fileList->fileList[i].dataLengthBytes); // Original length
|
|
outBitstream.WriteCompressed(compressedFiles[i].dataLengthBytes); // Compressed bitlength (yes, not bytes)
|
|
stringCompressor->EncodeString(fileList->fileList[i].filename, 512, &outBitstream);
|
|
outBitstream.WriteBits((const unsigned char*)compressedFiles[i].data, compressedFiles[i].dataLengthBytes);
|
|
// rakFree(compressedFiles[i].data);
|
|
|
|
// rakPeer->Send(&outBitstream, priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
|
|
// This is much more efficient than the commented code, because it doesn't require writing the file to the bitstream just to pass to Send
|
|
char *dataBlocks[2];
|
|
int lengths[2];
|
|
dataBlocks[0]=(char*) outBitstream.GetData();
|
|
lengths[0]=outBitstream.GetNumberOfBytesUsed();
|
|
dataBlocks[1]=compressedFiles[i].data;
|
|
lengths[1]=BITS_TO_BYTES(compressedFiles[i].dataLengthBytes);
|
|
rakPeer->SendList(dataBlocks,lengths,2,priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
rakFree(compressedFiles[i].data);
|
|
}
|
|
}
|
|
else
|
|
{
|
|
for (i=0; i < fileList->fileList.Size(); i++)
|
|
{
|
|
outBitstream.Reset();
|
|
outBitstream.Write((MessageID)ID_FILE_LIST_TRANSFER_FILE);
|
|
outBitstream.Write(fileList->fileList[i].context);
|
|
outBitstream.Write(setID);
|
|
outBitstream.WriteCompressed(i);
|
|
outBitstream.WriteCompressed(fileList->fileList[i].dataLengthBytes); // Original length in bytes
|
|
stringCompressor->EncodeString(fileList->fileList[i].filename, 512, &outBitstream);
|
|
outBitstream.AlignWriteToByteBoundary();
|
|
// outBitstream.Write((const char*) fileList->fileList[i].data, fileList->fileList[i].dataLengthBytes);
|
|
|
|
// rakPeer->Send(&outBitstream, priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
|
|
// This is much more efficient than the commented code, because it doesn't require writing the file to the bitstream just to pass to Send
|
|
char *dataBlocks[2];
|
|
int lengths[2];
|
|
dataBlocks[0]=(char*) outBitstream.GetData();
|
|
lengths[0]=outBitstream.GetNumberOfBytesUsed();
|
|
dataBlocks[1]=fileList->fileList[i].data;
|
|
lengths[1]=fileList->fileList[i].dataLengthBytes;
|
|
rakPeer->SendList(dataBlocks,lengths,2,priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
}
|
|
|
|
}
|
|
}
|
|
else
|
|
rakPeer->Send(&outBitstream, priority, RELIABLE_ORDERED, orderingChannel, recipient, false);
|
|
}
|
|
|
|
bool FileListTransfer::DecodeSetHeader(Packet *packet)
|
|
{
|
|
unsigned i;
|
|
unsigned int frequencyTable[ 256 ];
|
|
bool anythingToWrite=false;
|
|
unsigned short setID;
|
|
RakNet::BitStream inBitStream(packet->data, packet->length, false);
|
|
inBitStream.IgnoreBits(8);
|
|
inBitStream.Read(setID);
|
|
FileListReceiver *fileListReceiver;
|
|
if (fileListReceivers.Has(setID)==false)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return false;
|
|
}
|
|
fileListReceiver=fileListReceivers.Get(setID);
|
|
if (fileListReceiver->allowedSender!=packet->systemAddress)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
assert(fileListReceiver->gotSetHeader==false);
|
|
#endif
|
|
|
|
inBitStream.Read(anythingToWrite);
|
|
if (anythingToWrite)
|
|
{
|
|
inBitStream.Read(fileListReceiver->isCompressed);
|
|
if (fileListReceiver->isCompressed)
|
|
{
|
|
for (i=0; i < 256; i++)
|
|
inBitStream.ReadCompressed(frequencyTable[i]);
|
|
fileListReceiver->tree.GenerateFromFrequencyTable(frequencyTable);
|
|
inBitStream.ReadCompressed(fileListReceiver->setCount);
|
|
inBitStream.ReadCompressed(fileListReceiver->setTotalFinalLength);
|
|
if (inBitStream.ReadCompressed(fileListReceiver->setTotalCompressedTransmissionLength))
|
|
{
|
|
fileListReceiver->gotSetHeader=true;
|
|
return true;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
inBitStream.ReadCompressed(fileListReceiver->setCount);
|
|
if (inBitStream.ReadCompressed(fileListReceiver->setTotalFinalLength))
|
|
{
|
|
fileListReceiver->setTotalCompressedTransmissionLength=fileListReceiver->setTotalFinalLength;
|
|
fileListReceiver->gotSetHeader=true;
|
|
return true;
|
|
}
|
|
}
|
|
}
|
|
else
|
|
{
|
|
FileListTransferCBInterface::OnFileStruct s;
|
|
memset(&s,0,sizeof(FileListTransferCBInterface::OnFileStruct));
|
|
s.setID=setID;
|
|
fileListReceiver->downloadHandler->OnFile(&s);
|
|
return true;
|
|
}
|
|
|
|
return false;
|
|
}
|
|
|
|
bool FileListTransfer::DecodeFile(Packet *packet, bool fullFile)
|
|
{
|
|
FileListTransferCBInterface::OnFileStruct onFileStruct;
|
|
BitSize_t bitLength=0;
|
|
RakNet::BitStream inBitStream(packet->data, packet->length, false);
|
|
inBitStream.IgnoreBits(8);
|
|
|
|
unsigned int partCount=0;
|
|
unsigned int partTotal=0;
|
|
unsigned int partLength=0;
|
|
onFileStruct.fileData=0;
|
|
if (fullFile==false)
|
|
{
|
|
// Disable endian swapping on reading this, as it's generated locally in ReliabilityLayer.cpp
|
|
inBitStream.ReadBits( (unsigned char* ) &partCount, BYTES_TO_BITS(sizeof(partCount)), true );
|
|
inBitStream.ReadBits( (unsigned char* ) &partTotal, BYTES_TO_BITS(sizeof(partTotal)), true );
|
|
inBitStream.ReadBits( (unsigned char* ) &partLength, BYTES_TO_BITS(sizeof(partLength)), true );
|
|
inBitStream.IgnoreBits(8);
|
|
// The header is appended to every chunk, which we continue to read after this statement block
|
|
}
|
|
inBitStream.Read(onFileStruct.context);
|
|
inBitStream.Read(onFileStruct.setID);
|
|
FileListReceiver *fileListReceiver;
|
|
if (fileListReceivers.Has(onFileStruct.setID)==false)
|
|
{
|
|
return false;
|
|
}
|
|
fileListReceiver=fileListReceivers.Get(onFileStruct.setID);
|
|
if (fileListReceiver->allowedSender!=packet->systemAddress)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
#ifdef _DEBUG
|
|
assert(fileListReceiver->gotSetHeader==true);
|
|
#endif
|
|
|
|
inBitStream.ReadCompressed(onFileStruct.fileIndex);
|
|
inBitStream.ReadCompressed(onFileStruct.finalDataLength);
|
|
if (fileListReceiver->isCompressed)
|
|
{
|
|
inBitStream.ReadCompressed(bitLength);
|
|
onFileStruct.compressedTransmissionLength=(unsigned int) BITS_TO_BYTES(bitLength);
|
|
}
|
|
else
|
|
{
|
|
// Read header uncompressed so the data is byte aligned, for speed
|
|
onFileStruct.compressedTransmissionLength=(unsigned int) onFileStruct.finalDataLength;
|
|
}
|
|
|
|
if (stringCompressor->DecodeString(onFileStruct.fileName, 512, &inBitStream)==false)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return false;
|
|
}
|
|
|
|
if (fullFile)
|
|
{
|
|
// Support SendLists
|
|
inBitStream.AlignReadToByteBoundary();
|
|
|
|
onFileStruct.fileData = (char*) rakMalloc( (size_t) onFileStruct.finalDataLength );
|
|
|
|
if (fileListReceiver->isCompressed)
|
|
{
|
|
BitSize_t len=fileListReceiver->tree.DecodeArray(&inBitStream, bitLength, (size_t) onFileStruct.finalDataLength, (unsigned char*) onFileStruct.fileData);
|
|
if (len!=onFileStruct.finalDataLength)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
rakFree(onFileStruct.fileData);
|
|
return false;
|
|
}
|
|
}
|
|
else
|
|
{
|
|
inBitStream.Read((char*)onFileStruct.fileData, onFileStruct.finalDataLength);
|
|
}
|
|
}
|
|
|
|
|
|
onFileStruct.setCount=fileListReceiver->setCount;
|
|
onFileStruct.setTotalCompressedTransmissionLength=fileListReceiver->setTotalCompressedTransmissionLength;
|
|
onFileStruct.setTotalFinalLength=fileListReceiver->setTotalFinalLength;
|
|
|
|
// User callback for this file.
|
|
if (fullFile)
|
|
{
|
|
if (fileListReceiver->downloadHandler->OnFile(&onFileStruct))
|
|
rakFree(onFileStruct.fileData);
|
|
|
|
// If this set is done, free the memory for it.
|
|
if (fileListReceiver->setCount==onFileStruct.fileIndex+1)
|
|
{
|
|
if (fileListReceiver->downloadHandler->OnDownloadComplete()==false)
|
|
{
|
|
fileListReceiver->downloadHandler->OnDereference();
|
|
if (fileListReceiver->deleteDownloadHandler)
|
|
delete fileListReceiver->downloadHandler;
|
|
fileListReceivers.Delete(onFileStruct.setID);
|
|
delete fileListReceiver;
|
|
}
|
|
}
|
|
|
|
}
|
|
else
|
|
{
|
|
inBitStream.AlignReadToByteBoundary();
|
|
|
|
bool usedAlloca=false;
|
|
char *firstDataChunk;
|
|
unsigned int unreadBits = inBitStream.GetNumberOfUnreadBits();
|
|
unsigned int unreadBytes = BITS_TO_BYTES(unreadBits);
|
|
if (unreadBytes>0)
|
|
{
|
|
#if !defined(_XBOX360)
|
|
if (unreadBits < MAX_ALLOCA_STACK_ALLOCATION)
|
|
{
|
|
firstDataChunk = ( char* ) alloca( unreadBytes );
|
|
usedAlloca=true;
|
|
}
|
|
else
|
|
#endif
|
|
firstDataChunk = (char*) rakMalloc( unreadBytes );
|
|
|
|
// Read partLength bytes, reported to OnFileProgress
|
|
|
|
if (fileListReceiver->isCompressed)
|
|
fileListReceiver->tree.DecodeArray(&inBitStream, unreadBits, partLength, (unsigned char*) firstDataChunk);
|
|
else
|
|
inBitStream.Read((char*)firstDataChunk, unreadBytes );
|
|
|
|
}
|
|
else
|
|
firstDataChunk=0;
|
|
|
|
fileListReceiver->downloadHandler->OnFileProgress(&onFileStruct, partCount, partTotal, unreadBytes, firstDataChunk);
|
|
|
|
if (usedAlloca==false)
|
|
delete [] firstDataChunk;
|
|
}
|
|
|
|
return true;
|
|
}
|
|
PluginReceiveResult FileListTransfer::OnReceive(RakPeerInterface *peer, Packet *packet)
|
|
{
|
|
(void) peer;
|
|
switch (packet->data[0])
|
|
{
|
|
case ID_CONNECTION_LOST:
|
|
case ID_DISCONNECTION_NOTIFICATION:
|
|
RemoveReceiver(packet->systemAddress);
|
|
break;
|
|
case ID_FILE_LIST_TRANSFER_HEADER:
|
|
DecodeSetHeader(packet);
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
case ID_FILE_LIST_TRANSFER_FILE:
|
|
DecodeFile(packet, true);
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
case ID_DOWNLOAD_PROGRESS:
|
|
if (packet->length>sizeof(MessageID)+sizeof(unsigned int)*3 &&
|
|
packet->data[sizeof(MessageID)+sizeof(unsigned int)*3]==ID_FILE_LIST_TRANSFER_FILE)
|
|
{
|
|
DecodeFile(packet, false);
|
|
return RR_STOP_PROCESSING_AND_DEALLOCATE;
|
|
}
|
|
break;
|
|
}
|
|
|
|
return RR_CONTINUE_PROCESSING;
|
|
}
|
|
void FileListTransfer::OnShutdown(RakPeerInterface *peer)
|
|
{
|
|
(void) peer;
|
|
Clear();
|
|
}
|
|
void FileListTransfer::Clear(void)
|
|
{
|
|
unsigned i;
|
|
for (i=0; i < fileListReceivers.Size(); i++)
|
|
{
|
|
fileListReceivers[i]->downloadHandler->OnDereference();
|
|
if (fileListReceivers[i]->deleteDownloadHandler)
|
|
{
|
|
delete fileListReceivers[i]->downloadHandler;
|
|
}
|
|
delete fileListReceivers[i];
|
|
}
|
|
fileListReceivers.Clear();
|
|
}
|
|
void FileListTransfer::OnCloseConnection(RakPeerInterface *peer, SystemAddress systemAddress)
|
|
{
|
|
(void) peer;
|
|
RemoveReceiver(systemAddress);
|
|
}
|
|
void FileListTransfer::CancelReceive(unsigned short setId)
|
|
{
|
|
if (fileListReceivers.Has(setId)==false)
|
|
{
|
|
#ifdef _DEBUG
|
|
assert(0);
|
|
#endif
|
|
return;
|
|
}
|
|
FileListReceiver *fileListReceiver=fileListReceivers.Get(setId);
|
|
fileListReceiver->downloadHandler->OnDereference();
|
|
if (fileListReceiver->deleteDownloadHandler)
|
|
delete fileListReceiver->downloadHandler;
|
|
delete fileListReceiver;
|
|
fileListReceivers.Delete(setId);
|
|
}
|
|
void FileListTransfer::RemoveReceiver(SystemAddress systemAddress)
|
|
{
|
|
unsigned i;
|
|
i=0;
|
|
while (i < fileListReceivers.Size())
|
|
{
|
|
if (fileListReceivers[i]->allowedSender==systemAddress)
|
|
{
|
|
fileListReceivers[i]->downloadHandler->OnDereference();
|
|
if (fileListReceivers[i]->deleteDownloadHandler)
|
|
{
|
|
delete fileListReceivers[i]->downloadHandler;
|
|
}
|
|
delete fileListReceivers[i];
|
|
fileListReceivers.RemoveAtIndex(i);
|
|
}
|
|
else
|
|
i++;
|
|
}
|
|
}
|
|
bool FileListTransfer::IsHandlerActive(unsigned short setId)
|
|
{
|
|
return fileListReceivers.Has(setId);
|
|
}
|
|
void FileListTransfer::SetCallback(FileListProgress *cb)
|
|
{
|
|
callback=cb;
|
|
}
|
|
FileListProgress *FileListTransfer::GetCallback(void) const
|
|
{
|
|
return callback;
|
|
}
|
|
void FileListTransfer::OnAttach(RakPeerInterface *peer)
|
|
{
|
|
rakPeer=peer;
|
|
}
|
|
void FileListTransfer::Update(RakPeerInterface *peer)
|
|
{
|
|
(void) peer;
|
|
|
|
unsigned i;
|
|
i=0;
|
|
while (i < fileListReceivers.Size())
|
|
{
|
|
if (fileListReceivers[i]->downloadHandler->Update()==false)
|
|
{
|
|
fileListReceivers[i]->downloadHandler->OnDereference();
|
|
if (fileListReceivers[i]->deleteDownloadHandler)
|
|
delete fileListReceivers[i]->downloadHandler;
|
|
delete fileListReceivers[i];
|
|
fileListReceivers.RemoveAtIndex(i);
|
|
}
|
|
else
|
|
i++;
|
|
}
|
|
}
|
|
|
|
#ifdef _MSC_VER
|
|
#pragma warning( pop )
|
|
#endif
|