1
0

Rewrote ChunkSending queue for significantly improved performance

This commit is contained in:
tycho 2015-05-31 15:28:38 +01:00
parent 19c95ee564
commit bfe52277b4
5 changed files with 103 additions and 190 deletions

@ -1 +1 @@
Subproject commit 94da343b62f0498a5843247f36d6ee00cbeb8f21 Subproject commit 493f2dfa6d39f134e37c4c614cf8d6ffd10c825f

View File

@ -85,6 +85,7 @@ public:
// tolua_begin // tolua_begin
// Position, in absolute block coordinates: // Position, in absolute block coordinates:
Vector3i GetPos(void) const { return Vector3i{m_PosX, m_PosY, m_PosZ}; }
int GetPosX(void) const { return m_PosX; } int GetPosX(void) const { return m_PosX; }
int GetPosY(void) const { return m_PosY; } int GetPosY(void) const { return m_PosY; }
int GetPosZ(void) const { return m_PosZ; } int GetPosZ(void) const { return m_PosZ; }

View File

@ -302,7 +302,7 @@ public:
Returns false if there's no command block at those coords or callback returns true, returns true if found. */ Returns false if there's no command block at those coords or callback returns true, returns true if found. */
bool DoWithCommandBlockAt(int a_BlockX, int a_BlockY, int a_BlockZ, cCommandBlockCallback & a_Callback); // Lua-accessible bool DoWithCommandBlockAt(int a_BlockX, int a_BlockY, int a_BlockZ, cCommandBlockCallback & a_Callback); // Lua-accessible
/** Calls the callback for the mob head block at the specified coords.chu /** Calls the callback for the mob head block at the specified coords.
Returns false if there's no mob head block at those coords or callback returns true, returns true if found. */ Returns false if there's no mob head block at those coords or callback returns true, returns true if found. */
bool DoWithMobHeadAt(int a_BlockX, int a_BlockY, int a_BlockZ, cMobHeadCallback & a_Callback); // Lua-accessible bool DoWithMobHeadAt(int a_BlockX, int a_BlockY, int a_BlockZ, cMobHeadCallback & a_Callback); // Lua-accessible

View File

@ -18,6 +18,7 @@
//////////////////////////////////////////////////////////////////////////////// ////////////////////////////////////////////////////////////////////////////////
// cNotifyChunkSender: // cNotifyChunkSender:
@ -93,41 +94,25 @@ void cChunkSender::QueueSendChunkTo(int a_ChunkX, int a_ChunkZ, eChunkPriority a
{ {
ASSERT(a_Client != nullptr); ASSERT(a_Client != nullptr);
{ {
sSendChunk Chunk(a_ChunkX, a_ChunkZ, a_Client); cChunkCoords Chunk{a_ChunkX, a_ChunkZ};
cCSLock Lock(m_CS); cCSLock Lock(m_CS);
if ( auto iter = m_ChunkInfo.find(Chunk);
std::find(m_SendChunksLowPriority.begin(), m_SendChunksLowPriority.end(), Chunk) != m_SendChunksLowPriority.end() || if (iter != m_ChunkInfo.end())
std::find(m_SendChunksMediumPriority.begin(), m_SendChunksMediumPriority.end(), Chunk) != m_SendChunksMediumPriority.end() ||
std::find(m_SendChunksHighPriority.begin(), m_SendChunksHighPriority.end(), Chunk) != m_SendChunksHighPriority.end()
)
{ {
// Already queued, bail out auto & info = iter->second;
return; if (info.m_Priority > a_Priority)
{
m_SendChunks.push(sChunkQueue{a_Priority, Chunk});
info.m_Priority = a_Priority;
}
info.m_Clients.insert(a_Client);
} }
else
switch (a_Priority)
{ {
case E_CHUNK_PRIORITY_LOW: m_SendChunks.push(sChunkQueue{a_Priority, Chunk});
{ auto info = sSendChunk{Chunk, a_Priority};
m_SendChunksLowPriority.push_back(Chunk); info.m_Clients.insert(a_Client);
break; m_ChunkInfo.emplace(Chunk, info);
}
case E_CHUNK_PRIORITY_MEDIUM:
{
m_SendChunksMediumPriority.push_back(Chunk);
break;
}
case E_CHUNK_PRIORITY_HIGH:
{
m_SendChunksHighPriority.push_back(Chunk);
break;
}
case PRIORITY_BROADCAST:
{
m_SendChunksBroadcastPriority.push_back(Chunk);
break;
}
} }
} }
m_evtQueue.Set(); m_evtQueue.Set();
@ -141,43 +126,25 @@ void cChunkSender::QueueSendChunkTo(int a_ChunkX, int a_ChunkZ, eChunkPriority a
void cChunkSender::QueueSendChunkTo(int a_ChunkX, int a_ChunkZ, eChunkPriority a_Priority, std::list<cClientHandle *> a_Clients) void cChunkSender::QueueSendChunkTo(int a_ChunkX, int a_ChunkZ, eChunkPriority a_Priority, std::list<cClientHandle *> a_Clients)
{ {
{ {
cChunkCoords Chunk{a_ChunkX, a_ChunkZ};
cCSLock Lock(m_CS); cCSLock Lock(m_CS);
for (auto client : a_Clients) auto iter = m_ChunkInfo.find(Chunk);
if (iter != m_ChunkInfo.end())
{ {
sSendChunk Chunk(a_ChunkX, a_ChunkZ, client); auto & info = iter->second;
if ( if (info.m_Priority > a_Priority)
std::find(m_SendChunksLowPriority.begin(), m_SendChunksLowPriority.end(), Chunk) != m_SendChunksLowPriority.end() ||
std::find(m_SendChunksMediumPriority.begin(), m_SendChunksMediumPriority.end(), Chunk) != m_SendChunksMediumPriority.end() ||
std::find(m_SendChunksHighPriority.begin(), m_SendChunksHighPriority.end(), Chunk) != m_SendChunksHighPriority.end()
)
{ {
// Already queued, bail out m_SendChunks.push(sChunkQueue{a_Priority, Chunk});
continue; info.m_Priority = a_Priority;
}
switch (a_Priority)
{
case E_CHUNK_PRIORITY_LOW:
{
m_SendChunksLowPriority.push_back(Chunk);
break;
}
case E_CHUNK_PRIORITY_MEDIUM:
{
m_SendChunksMediumPriority.push_back(Chunk);
break;
}
case E_CHUNK_PRIORITY_HIGH:
{
m_SendChunksHighPriority.push_back(Chunk);
break;
}
case PRIORITY_BROADCAST:
{
m_SendChunksBroadcastPriority.push_back(Chunk);
break;
}
} }
info.m_Clients.insert(a_Clients.begin(), a_Clients.end());
}
else
{
m_SendChunks.push(sChunkQueue{a_Priority, Chunk});
auto info = sSendChunk{Chunk, a_Priority};
info.m_Clients.insert(a_Clients.begin(), a_Clients.end());
m_ChunkInfo.emplace(Chunk, info);
} }
} }
m_evtQueue.Set(); m_evtQueue.Set();
@ -191,33 +158,12 @@ void cChunkSender::RemoveClient(cClientHandle * a_Client)
{ {
{ {
cCSLock Lock(m_CS); cCSLock Lock(m_CS);
for (sSendChunkList::iterator itr = m_SendChunksLowPriority.begin(); itr != m_SendChunksLowPriority.end();) for (auto && pair : m_ChunkInfo)
{ {
if (itr->m_Client == a_Client) auto && clients = pair.second.m_Clients;
{ clients.erase(a_Client); // nop for sets that do not contain a_Client
itr = m_SendChunksLowPriority.erase(itr); }
continue;
}
++itr;
} // for itr - m_SendChunksLowPriority[]
for (sSendChunkList::iterator itr = m_SendChunksMediumPriority.begin(); itr != m_SendChunksMediumPriority.end();)
{
if (itr->m_Client == a_Client)
{
itr = m_SendChunksMediumPriority.erase(itr);
continue;
}
++itr;
} // for itr - m_SendChunksMediumPriority[]
for (sSendChunkList::iterator itr = m_SendChunksHighPriority.begin(); itr != m_SendChunksHighPriority.end();)
{
if (itr->m_Client == a_Client)
{
itr = m_SendChunksHighPriority.erase(itr);
continue;
}
++itr;
} // for itr - m_SendChunksHighPriority[]
m_RemoveCount++; m_RemoveCount++;
} }
m_evtQueue.Set(); m_evtQueue.Set();
@ -233,7 +179,7 @@ void cChunkSender::Execute(void)
while (!m_ShouldTerminate) while (!m_ShouldTerminate)
{ {
cCSLock Lock(m_CS); cCSLock Lock(m_CS);
while (m_SendChunksBroadcastPriority.empty() && m_SendChunksLowPriority.empty() && m_SendChunksMediumPriority.empty() && m_SendChunksHighPriority.empty()) do
{ {
int RemoveCount = m_RemoveCount; int RemoveCount = m_RemoveCount;
m_RemoveCount = 0; m_RemoveCount = 0;
@ -247,52 +193,24 @@ void cChunkSender::Execute(void)
{ {
return; return;
} }
} // while (empty) } while (m_SendChunks.empty());
if (!m_SendChunksHighPriority.empty()) // Take one from the queue:
auto Chunk = m_SendChunks.top().m_Chunk;
m_SendChunks.pop();
auto itr = m_ChunkInfo.find(Chunk);
if (itr == m_ChunkInfo.end())
{ {
// Take one from the queue: continue;
sSendChunk Chunk(m_SendChunksHighPriority.front());
m_SendChunksHighPriority.pop_front();
Lock.Unlock();
SendChunk(Chunk.m_ChunkX, Chunk.m_ChunkZ, Chunk.m_Client);
} }
else if (!m_SendChunksBroadcastPriority.empty())
{
// Take one from the queue:
sSendChunk Chunk(m_SendChunksBroadcastPriority.front());
m_SendChunksBroadcastPriority.pop_front();
Lock.Unlock();
SendChunk(Chunk.m_ChunkX, Chunk.m_ChunkZ, Chunk.m_Client); std::unordered_set<cClientHandle *> clients;
} std::swap(itr->second.m_Clients, clients);
else if (!m_SendChunksMediumPriority.empty()) m_ChunkInfo.erase(itr);
{
// Take one from the queue:
sSendChunk Chunk(m_SendChunksMediumPriority.front());
m_SendChunksMediumPriority.pop_front();
Lock.Unlock();
SendChunk(Chunk.m_ChunkX, Chunk.m_ChunkZ, Chunk.m_Client);
}
else
{
// Take one from the queue:
sSendChunk Chunk(m_SendChunksLowPriority.front());
m_SendChunksLowPriority.pop_front();
Lock.Unlock();
SendChunk(Chunk.m_ChunkX, Chunk.m_ChunkZ, Chunk.m_Client);
}
Lock.Lock();
int RemoveCount = m_RemoveCount;
m_RemoveCount = 0;
Lock.Unlock(); Lock.Unlock();
for (int i = 0; i < RemoveCount; i++)
{ SendChunk(Chunk.m_ChunkX, Chunk.m_ChunkZ, clients);
m_evtRemoved.Set(); // Notify that the removed clients are safe to be deleted
}
} // while (!mShouldTerminate) } // while (!mShouldTerminate)
} }
@ -300,13 +218,22 @@ void cChunkSender::Execute(void)
void cChunkSender::SendChunk(int a_ChunkX, int a_ChunkZ, cClientHandle * a_Client) void cChunkSender::SendChunk(int a_ChunkX, int a_ChunkZ, std::unordered_set<cClientHandle *> a_Clients)
{ {
// Ask the client if it still wants the chunk: // Ask the client if it still wants the chunk:
if (!a_Client->WantsSendChunk(a_ChunkX, a_ChunkZ)) for(auto itr = a_Clients.begin(); itr != a_Clients.end();)
{ {
return; if(!(*itr)->WantsSendChunk(a_ChunkX, a_ChunkZ))
{
auto toremove = itr;
itr++;
a_Clients.erase(toremove);
}
else
{
itr++;
}
} }
// If the chunk has no clients, no need to packetize it: // If the chunk has no clients, no need to packetize it:
@ -335,14 +262,18 @@ void cChunkSender::SendChunk(int a_ChunkX, int a_ChunkZ, cClientHandle * a_Clien
} }
cChunkDataSerializer Data(m_BlockTypes, m_BlockMetas, m_BlockLight, m_BlockSkyLight, m_BiomeMap); cChunkDataSerializer Data(m_BlockTypes, m_BlockMetas, m_BlockLight, m_BlockSkyLight, m_BiomeMap);
// Send: for (auto client : a_Clients)
a_Client->SendChunkData(a_ChunkX, a_ChunkZ, Data);
// Send block-entity packets:
for (sBlockCoords::iterator itr = m_BlockEntities.begin(); itr != m_BlockEntities.end(); ++itr)
{ {
m_World.SendBlockEntity(itr->m_BlockX, itr->m_BlockY, itr->m_BlockZ, *a_Client); // Send:
} // for itr - m_Packets[] client->SendChunkData(a_ChunkX, a_ChunkZ, Data);
// Send block-entity packets:
for (auto Pos : m_BlockEntities)
{
m_World.SendBlockEntity(Pos.x, Pos.y, Pos.z, *client);
} // for itr - m_Packets[]
}
m_BlockEntities.clear(); m_BlockEntities.clear();
// TODO: Send entity spawn packets // TODO: Send entity spawn packets
@ -354,7 +285,7 @@ void cChunkSender::SendChunk(int a_ChunkX, int a_ChunkZ, cClientHandle * a_Clien
void cChunkSender::BlockEntity(cBlockEntity * a_Entity) void cChunkSender::BlockEntity(cBlockEntity * a_Entity)
{ {
m_BlockEntities.push_back(sBlockCoord(a_Entity->GetPosX(), a_Entity->GetPosY(), a_Entity->GetPosZ())); m_BlockEntities.push_back(a_Entity->GetPos());
} }

View File

@ -29,6 +29,9 @@ Note that it may be called by world's BroadcastToChunk() if the client is still
#include "ChunkDef.h" #include "ChunkDef.h"
#include "ChunkDataCallback.h" #include "ChunkDataCallback.h"
#include <unordered_set>
#include <unordered_map>
@ -59,9 +62,10 @@ public:
enum eChunkPriority enum eChunkPriority
{ {
E_CHUNK_PRIORITY_HIGH = 0, E_CHUNK_PRIORITY_HIGH = 0,
E_CHUNK_PRIORITY_MEDIUM = 1, PRIORITY_BROADCAST,
E_CHUNK_PRIORITY_LOW = 2, E_CHUNK_PRIORITY_MEDIUM,
PRIORITY_BROADCAST E_CHUNK_PRIORITY_LOW,
}; };
bool Start(); bool Start();
@ -77,61 +81,39 @@ public:
protected: protected:
struct sChunkQueue
{
eChunkPriority m_Priority;
cChunkCoords m_Chunk;
bool operator <(const sChunkQueue & a_Other) const { return this->m_Priority < a_Other.m_Priority; }
};
/// Used for sending chunks to specific clients /// Used for sending chunks to specific clients
struct sSendChunk struct sSendChunk
{ {
int m_ChunkX; cChunkCoords m_Chunk;
int m_ChunkZ; std::unordered_set<cClientHandle *> m_Clients;
cClientHandle * m_Client; eChunkPriority m_Priority;
sSendChunk(cChunkCoords a_Chunk, eChunkPriority a_Priority) :
sSendChunk(int a_ChunkX, int a_ChunkZ, cClientHandle * a_Client) : m_Chunk(a_Chunk),
m_ChunkX(a_ChunkX), m_Priority(a_Priority)
m_ChunkZ(a_ChunkZ),
m_Client(a_Client)
{ {
} }
};
bool operator ==(const sSendChunk & a_Other)
{
return (
(a_Other.m_ChunkX == m_ChunkX) &&
(a_Other.m_ChunkZ == m_ChunkZ) &&
(a_Other.m_Client == m_Client)
);
}
} ;
typedef std::list<sSendChunk> sSendChunkList;
struct sBlockCoord
{
int m_BlockX;
int m_BlockY;
int m_BlockZ;
sBlockCoord(int a_BlockX, int a_BlockY, int a_BlockZ) :
m_BlockX(a_BlockX),
m_BlockY(a_BlockY),
m_BlockZ(a_BlockZ)
{
}
} ;
typedef std::vector<sBlockCoord> sBlockCoords;
cWorld & m_World; cWorld & m_World;
cCriticalSection m_CS; cCriticalSection m_CS;
sSendChunkList m_SendChunksLowPriority; std::priority_queue<sChunkQueue> m_SendChunks;
sSendChunkList m_SendChunksMediumPriority; std::unordered_map<cChunkCoords, sSendChunk, cChunkCoordsHash> m_ChunkInfo;
sSendChunkList m_SendChunksBroadcastPriority;
sSendChunkList m_SendChunksHighPriority;
cEvent m_evtQueue; // Set when anything is added to m_ChunksReady cEvent m_evtQueue; // Set when anything is added to m_ChunksReady
cEvent m_evtRemoved; // Set when removed clients are safe to be deleted cEvent m_evtRemoved; // Set when removed clients are safe to be deleted
int m_RemoveCount; // Number of threads waiting for a client removal (m_evtRemoved needs to be set this many times) int m_RemoveCount; // Number of threads waiting for a client removal (m_evtRemoved needs to be set this many times)
// Data about the chunk that is being sent: // Data about the chunk that is being sent:
// NOTE that m_BlockData[] is inherited from the cChunkDataCollector // NOTE that m_BlockData[] is inherited from the cChunkDataCollector
unsigned char m_BiomeMap[cChunkDef::Width * cChunkDef::Width]; unsigned char m_BiomeMap[cChunkDef::Width * cChunkDef::Width];
sBlockCoords m_BlockEntities; // Coords of the block entities to send std::vector<Vector3i> m_BlockEntities; // Coords of the block entities to send
// TODO: sEntityIDs m_Entities; // Entity-IDs of the entities to send // TODO: sEntityIDs m_Entities; // Entity-IDs of the entities to send
// cIsThread override: // cIsThread override:
@ -144,9 +126,8 @@ protected:
virtual void BlockEntity (cBlockEntity * a_Entity) override; virtual void BlockEntity (cBlockEntity * a_Entity) override;
/// Sends the specified chunk to a_Client, or to all chunk clients if a_Client == nullptr /// Sends the specified chunk to a_Client, or to all chunk clients if a_Client == nullptr
void SendChunk(int a_ChunkX, int a_ChunkZ, cClientHandle * a_Client); void SendChunk(int a_ChunkX, int a_ChunkZ, std::unordered_set<cClientHandle *> a_Clients);
} ; } ;