2017-09-10 15:43:56 +00:00
|
|
|
#include <assert.h>
|
2017-09-11 20:51:47 +00:00
|
|
|
#include <memory>
|
2017-09-10 15:43:56 +00:00
|
|
|
|
2017-09-11 23:49:31 +00:00
|
|
|
#include "../common/tracy_lz4.hpp"
|
2017-09-13 20:56:55 +00:00
|
|
|
#include "../common/TracyProtocol.hpp"
|
2017-09-11 20:51:47 +00:00
|
|
|
#include "../common/TracySocket.hpp"
|
2017-09-12 23:31:55 +00:00
|
|
|
#include "../common/TracySystem.hpp"
|
2017-09-10 15:43:56 +00:00
|
|
|
#include "TracyProfiler.hpp"
|
|
|
|
|
|
|
|
namespace tracy
|
|
|
|
{
|
|
|
|
|
2017-09-10 18:02:40 +00:00
|
|
|
extern const char* PointerCheckA;
|
|
|
|
const char* PointerCheckB = "tracy";
|
|
|
|
|
2017-09-10 18:07:38 +00:00
|
|
|
|
2017-09-10 15:43:56 +00:00
|
|
|
static Profiler* s_instance = nullptr;
|
|
|
|
|
|
|
|
Profiler::Profiler()
|
2017-09-10 18:14:16 +00:00
|
|
|
: m_timeBegin( GetTime() )
|
|
|
|
, m_shutdown( false )
|
2017-09-10 18:08:42 +00:00
|
|
|
, m_id( 0 )
|
2017-09-10 15:43:56 +00:00
|
|
|
{
|
2017-09-10 18:02:40 +00:00
|
|
|
assert( PointerCheckA == PointerCheckB );
|
2017-09-10 15:43:56 +00:00
|
|
|
assert( !s_instance );
|
|
|
|
s_instance = this;
|
|
|
|
|
|
|
|
m_thread = std::thread( [this] { Worker(); } );
|
2017-09-10 15:46:20 +00:00
|
|
|
SetThreadName( m_thread, "Tracy Profiler" );
|
2017-09-10 15:43:56 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
Profiler::~Profiler()
|
|
|
|
{
|
2017-09-13 21:36:40 +00:00
|
|
|
m_shutdown.store( true, std::memory_order_relaxed );
|
|
|
|
m_thread.join();
|
|
|
|
|
2017-09-10 15:43:56 +00:00
|
|
|
assert( s_instance );
|
|
|
|
s_instance = nullptr;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
2017-09-10 18:08:42 +00:00
|
|
|
uint64_t Profiler::GetNewId()
|
|
|
|
{
|
|
|
|
return s_instance->m_id.fetch_add( 1, std::memory_order_relaxed );
|
|
|
|
}
|
|
|
|
|
2017-09-13 23:07:14 +00:00
|
|
|
uint64_t Profiler::ZoneBegin( QueueZoneBegin&& data )
|
2017-09-10 18:09:14 +00:00
|
|
|
{
|
2017-09-13 23:07:14 +00:00
|
|
|
auto id = GetNewId();
|
2017-09-11 22:56:31 +00:00
|
|
|
QueueItem item;
|
|
|
|
item.hdr.type = QueueType::ZoneBegin;
|
2017-09-13 23:07:14 +00:00
|
|
|
item.hdr.id = id;
|
2017-09-10 18:09:14 +00:00
|
|
|
item.zoneBegin = std::move( data );
|
2017-09-10 18:52:10 +00:00
|
|
|
s_instance->m_queue.enqueue( GetToken(), std::move( item ) );
|
2017-09-13 23:07:14 +00:00
|
|
|
return id;
|
2017-09-10 18:09:14 +00:00
|
|
|
}
|
|
|
|
|
2017-09-13 23:07:14 +00:00
|
|
|
void Profiler::ZoneEnd( uint64_t id, QueueZoneEnd&& data )
|
2017-09-10 18:09:14 +00:00
|
|
|
{
|
2017-09-11 22:56:31 +00:00
|
|
|
QueueItem item;
|
|
|
|
item.hdr.type = QueueType::ZoneEnd;
|
2017-09-13 23:07:14 +00:00
|
|
|
item.hdr.id = id;
|
2017-09-10 18:09:14 +00:00
|
|
|
item.zoneEnd = std::move( data );
|
2017-09-10 18:52:10 +00:00
|
|
|
s_instance->m_queue.enqueue( GetToken(), std::move( item ) );
|
|
|
|
}
|
|
|
|
|
|
|
|
Profiler* Profiler::Instance()
|
|
|
|
{
|
|
|
|
return s_instance;
|
2017-09-10 18:09:14 +00:00
|
|
|
}
|
|
|
|
|
2017-09-10 15:43:56 +00:00
|
|
|
void Profiler::Worker()
|
|
|
|
{
|
2017-09-11 22:49:38 +00:00
|
|
|
enum { BulkSize = TargetFrameSize / QueueItemSize };
|
2017-09-11 23:49:31 +00:00
|
|
|
|
2017-09-10 18:23:06 +00:00
|
|
|
moodycamel::ConsumerToken token( m_queue );
|
|
|
|
|
2017-09-11 20:51:47 +00:00
|
|
|
ListenSocket listen;
|
|
|
|
listen.Listen( "8086", 8 );
|
|
|
|
|
2017-09-10 15:43:56 +00:00
|
|
|
for(;;)
|
|
|
|
{
|
2017-09-11 20:51:47 +00:00
|
|
|
std::unique_ptr<Socket> sock;
|
|
|
|
for(;;)
|
2017-09-10 18:23:06 +00:00
|
|
|
{
|
2017-09-11 20:51:47 +00:00
|
|
|
if( m_shutdown.load( std::memory_order_relaxed ) ) return;
|
|
|
|
sock = listen.Accept();
|
|
|
|
if( sock ) break;
|
2017-09-10 18:23:06 +00:00
|
|
|
}
|
2017-09-11 20:51:47 +00:00
|
|
|
|
|
|
|
sock->Send( &m_timeBegin, sizeof( m_timeBegin ) );
|
2017-09-12 00:12:45 +00:00
|
|
|
#ifdef _DEBUG
|
|
|
|
// notify client that lz4 compression is disabled (too slow in debug builds)
|
|
|
|
char val = 0;
|
|
|
|
sock->Send( &val, 1 );
|
|
|
|
#else
|
|
|
|
char val = 1;
|
|
|
|
sock->Send( &val, 1 );
|
|
|
|
#endif
|
2017-09-11 20:51:47 +00:00
|
|
|
|
|
|
|
for(;;)
|
2017-09-10 18:23:06 +00:00
|
|
|
{
|
2017-09-11 20:51:47 +00:00
|
|
|
if( m_shutdown.load( std::memory_order_relaxed ) ) return;
|
|
|
|
|
|
|
|
QueueItem item[BulkSize];
|
|
|
|
const auto sz = m_queue.try_dequeue_bulk( token, item, BulkSize );
|
|
|
|
if( sz > 0 )
|
|
|
|
{
|
2017-09-11 23:14:04 +00:00
|
|
|
char buf[TargetFrameSize];
|
|
|
|
char* ptr = buf;
|
|
|
|
for( int i=0; i<sz; i++ )
|
|
|
|
{
|
2017-09-13 23:05:08 +00:00
|
|
|
const auto dsz = QueueDataSize[item[i].hdr.idx];
|
2017-09-11 23:14:04 +00:00
|
|
|
memcpy( ptr, item+i, dsz );
|
|
|
|
ptr += dsz;
|
|
|
|
}
|
2017-09-12 00:12:45 +00:00
|
|
|
#ifdef _DEBUG
|
|
|
|
if( sock->Send( buf, ptr - buf ) == -1 ) break;
|
|
|
|
#else
|
2017-09-13 20:58:04 +00:00
|
|
|
char lz4[LZ4Size + sizeof( lz4sz_t )];
|
|
|
|
const lz4sz_t lz4sz = LZ4_compress_default( buf, lz4 + sizeof( lz4sz_t ), ptr - buf, LZ4Size );
|
|
|
|
memcpy( lz4, &lz4sz, sizeof( lz4sz ) );
|
|
|
|
if( sock->Send( lz4, lz4sz + sizeof( lz4sz_t ) ) == -1 ) break;
|
2017-09-12 00:12:45 +00:00
|
|
|
#endif
|
2017-09-11 20:51:47 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
std::this_thread::sleep_for( std::chrono::milliseconds( 10 ) );
|
|
|
|
}
|
2017-09-10 18:23:06 +00:00
|
|
|
}
|
2017-09-10 15:43:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
}
|