Files
FlaxEngine/Source/Engine/Networking/NetworkManager.cpp
2021-03-11 20:30:03 +01:00

176 lines
4.6 KiB
C++

// Copyright (c) 2012-2021 Wojciech Figat. All rights reserved.
#include "NetworkManager.h"
#include "NetworkMessage.h"
#include "NetworkConfig.h"
#include "NetworkConnection.h"
#include "INetworkDriver.h"
#include "Drivers/ENetDriver.h"
#include "Engine/Core/Log.h"
#include "Engine/Core/Collections/Array.h"
#include "Engine/Core/Math/Math.h"
#include "Engine/Platform/CPUInfo.h"
namespace
{
NetworkConfig Config;
INetworkDriver* NetworkDriver = nullptr;
uint8* MessageBuffer = nullptr;
Array<uint32, HeapAllocation> MessagePool;
}
bool NetworkManager::Initialize(const NetworkConfig& config)
{
Config = config;
ASSERT(NetworkDriver == nullptr);
ASSERT(Config.NetworkDriverType != NetworkTransportType::Undefined);
ASSERT(Config.ConnectionsLimit > 0);
ASSERT(Config.MessageSize > 32); // TODO: Adjust this, not sure what the lowest limit should be.
ASSERT(Config.MessagePoolSize > 128);
// Setup messages
CreateMessageBuffers();
MessagePool.Clear();
// Warmup message pool
for (uint32 messageId = Config.MessagePoolSize; messageId > 0; messageId --)
MessagePool.Push(messageId);
// Setup network driver
NetworkDriver = New<ENetDriver>();
NetworkDriver->Initialize(Config);
LOG(Info, "NetworkManager initialized using driver = {0}", static_cast<int>(Config.NetworkDriverType));
return false;
}
void NetworkManager::Shutdown()
{
Delete(NetworkDriver);
DisposeMessageBuffers();
LOG(Info, "NetworkManager shutdown");
}
bool NetworkManager::Listen()
{
LOG(Info, "NetworkManager starting to listen on address = any:{0}", Config.Port);
ASSERT(NetworkDriver != nullptr);
return NetworkDriver->Listen();
}
void NetworkManager::Connect()
{
// TODO: Support multiple hosts
LOG(Info, "Connecting to 127.0.0.1:{0}...", Config.Port); // TODO: Proper IP address
ASSERT(NetworkDriver != nullptr);
// TODO: Assert address/endpoint
NetworkDriver->Connect();
}
void NetworkManager::Disconnect()
{
LOG(Info, "Disconnecting...");
ASSERT(NetworkDriver != nullptr);
NetworkDriver->Disconnect();
}
void NetworkManager::Disconnect(const NetworkConnection& connection)
{
LOG(Info, "Disconnecting connection with id = {0}...", connection.ConnectionId);
ASSERT(NetworkDriver != nullptr);
NetworkDriver->Disconnect(connection);
}
bool NetworkManager::PopEvent(NetworkEvent& eventPtr)
{
ASSERT(NetworkDriver != nullptr);
return NetworkDriver->PopEvent(&eventPtr);
}
NetworkMessage NetworkManager::CreateMessage()
{
ASSERT(MessagePool.Count() > 0);
const uint32 messageId = MessagePool.Pop();
uint8* messageBuffer = GetMessageBuffer(messageId);
return NetworkMessage(messageBuffer, messageId, Config.MessageSize, 0, 0);
}
void NetworkManager::RecycleMessage(const NetworkMessage& message)
{
ASSERT(message.IsValid());
#ifdef BUILD_DEBUG
ASSERT(MessagePool.Contains(message.MessageId) == false);
#endif
// Return the message id
MessagePool.Push(message.MessageId);
}
NetworkMessage NetworkManager::BeginSendMessage()
{
ASSERT(NetworkDriver != nullptr);
return CreateMessage();
}
void NetworkManager::AbortSendMessage(const NetworkMessage& message)
{
ASSERT(NetworkDriver != nullptr);
ASSERT(message.IsValid());
RecycleMessage(message);
}
bool NetworkManager::EndSendMessage(const NetworkChannelType channelType, const NetworkMessage& message, const Array<NetworkConnection> targets)
{
ASSERT(NetworkDriver != nullptr);
ASSERT(message.IsValid());
NetworkDriver->SendMessage(channelType, message, targets);
RecycleMessage(message);
return false;
}
void NetworkManager::CreateMessageBuffers()
{
ASSERT(MessageBuffer == nullptr);
const uint32 pageSize = Platform::GetCPUInfo().PageSize;
// Calculate total size in bytes
const uint64 totalSize = static_cast<uint64>(Config.MessagePoolSize) * Config.MessageSize;
// Calculate the amount of pages that we need
const uint32 numPages = totalSize > pageSize ? Math::CeilToInt(totalSize / static_cast<float>(pageSize)) : 1;
MessageBuffer = static_cast<uint8*>(Platform::AllocatePages(numPages, pageSize));
Platform::MemorySet(MessageBuffer, 0, numPages * pageSize);
}
void NetworkManager::DisposeMessageBuffers()
{
ASSERT(MessageBuffer != nullptr);
Platform::FreePages(MessageBuffer);
MessageBuffer = nullptr;
}
uint8* NetworkManager::GetMessageBuffer(const uint32 messageId)
{
// Calculate and return the buffer slice using previously calculated slice.
return MessageBuffer + Config.MessageSize * messageId;
}