tracy/server/TracyView.cpp

167 lines
4.1 KiB
C++
Raw Normal View History

2017-09-12 23:54:22 +00:00
#ifdef _MSC_VER
# include <winsock2.h>
#else
# include <sys/time.h>
#endif
2017-09-12 23:33:50 +00:00
#include <assert.h>
2017-09-13 21:40:28 +00:00
#include "../common/tracy_lz4.hpp"
#include "../common/TracyProtocol.hpp"
2017-09-12 23:33:50 +00:00
#include "../common/TracySystem.hpp"
2017-09-13 21:40:28 +00:00
#include "../common/TracyQueue.hpp"
2017-09-12 23:33:50 +00:00
#include "TracyView.hpp"
namespace tracy
{
static View* s_instance = nullptr;
View::View( const char* addr )
: m_addr( addr )
, m_shutdown( false )
{
assert( s_instance == nullptr );
s_instance = this;
m_thread = std::thread( [this] { Worker(); } );
SetThreadName( m_thread, "Tracy View" );
}
View::~View()
{
m_shutdown.store( true, std::memory_order_relaxed );
m_thread.join();
assert( s_instance != nullptr );
s_instance = nullptr;
2017-09-12 23:33:50 +00:00
}
bool View::ShouldExit()
{
return s_instance->m_shutdown.load( std::memory_order_relaxed );
}
2017-09-12 23:33:50 +00:00
void View::Worker()
{
2017-09-12 23:54:22 +00:00
timeval tv;
tv.tv_sec = 0;
tv.tv_usec = 10000;
2017-09-12 23:33:50 +00:00
for(;;)
{
if( m_shutdown.load( std::memory_order_relaxed ) ) return;
if( !m_sock.Connect( m_addr.c_str(), "8086" ) ) continue;
2017-09-12 23:54:22 +00:00
uint8_t lz4;
if( !m_sock.Read( &m_timeBegin, sizeof( m_timeBegin ), &tv, ShouldExit ) ) goto close;
if( !m_sock.Read( &lz4, sizeof( lz4 ), &tv, ShouldExit ) ) goto close;
2017-09-12 23:54:22 +00:00
for(;;)
{
if( m_shutdown.load( std::memory_order_relaxed ) ) return;
2017-09-13 21:40:28 +00:00
if( lz4 )
{
char buf[TargetFrameSize];
char lz4buf[LZ4Size];
lz4sz_t lz4sz;
if( !m_sock.Read( &lz4sz, sizeof( lz4sz ), &tv, ShouldExit ) ) goto close;
if( !m_sock.Read( lz4buf, lz4sz, &tv, ShouldExit ) ) goto close;
2017-09-13 21:40:28 +00:00
auto sz = LZ4_decompress_safe( lz4buf, buf, lz4sz, TargetFrameSize );
assert( sz >= 0 );
const char* ptr = buf;
const char* end = buf + sz;
while( ptr < end )
{
auto ev = (QueueItem*)ptr;
Process( *ev );
ptr += QueueDataSize[ev->hdr.idx];
2017-09-13 21:40:28 +00:00
}
}
else
{
2017-09-13 23:06:40 +00:00
QueueItem ev;
if( !m_sock.Read( &ev.hdr, sizeof( QueueHeader ), &tv, ShouldExit ) ) goto close;
if( !m_sock.Read( ((char*)&ev) + sizeof( QueueHeader ), QueueDataSize[ev.hdr.idx] - sizeof( QueueHeader ), &tv, ShouldExit ) ) goto close;
2017-09-13 23:06:40 +00:00
Process( ev );
2017-09-13 21:40:28 +00:00
}
2017-09-12 23:54:22 +00:00
}
2017-09-13 00:00:22 +00:00
close:
m_sock.Close();
2017-09-12 23:33:50 +00:00
}
}
2017-09-13 21:40:28 +00:00
void View::Process( const QueueItem& ev )
{
2017-09-14 00:00:13 +00:00
switch( ev.hdr.type )
{
case QueueType::ZoneBegin:
ProcessZoneBegin( ev.hdr.id, ev.zoneBegin );
break;
case QueueType::ZoneEnd:
ProcessZoneEnd( ev.hdr.id, ev.zoneEnd );
break;
default:
assert( false );
break;
}
}
void View::ProcessZoneBegin( uint64_t id, const QueueZoneBegin& ev )
{
auto it = m_pendingEndZone.find( id );
const auto idx = m_data.size();
2017-09-14 00:16:51 +00:00
CheckString( ev.filename );
CheckString( ev.function );
2017-09-14 00:00:13 +00:00
std::unique_lock<std::mutex> lock( m_lock );
if( it == m_pendingEndZone.end() )
{
m_data.emplace_back( Event { ev.time, -1 } );
lock.unlock();
m_openZones.emplace( id, idx );
}
else
{
assert( ev.time <= it->second.time );
m_data.emplace_back( Event { ev.time, it->second.time } );
lock.unlock();
2017-09-13 21:40:28 +00:00
2017-09-14 00:00:13 +00:00
m_pendingEndZone.erase( it );
}
}
void View::ProcessZoneEnd( uint64_t id, const QueueZoneEnd& ev )
{
auto it = m_openZones.find( id );
if( it == m_openZones.end() )
{
m_pendingEndZone.emplace( id, ev );
}
else
{
std::unique_lock<std::mutex> lock( m_lock );
assert( ev.time >= m_data[it->second].start );
m_data[it->second].end = ev.time;
lock.unlock();
m_openZones.erase( it );
}
2017-09-13 21:40:28 +00:00
}
2017-09-14 00:16:51 +00:00
void View::CheckString( uint64_t ptr )
{
if( m_strings.find( ptr ) != m_strings.end() ) return;
if( m_pendingStrings.find( ptr ) != m_pendingStrings.end() ) return;
m_pendingStrings.emplace( ptr );
m_sock.Send( &ptr, sizeof( ptr ) );
}
2017-09-12 23:33:50 +00:00
}