2017-09-12 23:54:22 +00:00
|
|
|
#ifdef _MSC_VER
|
|
|
|
# include <winsock2.h>
|
|
|
|
#else
|
|
|
|
# include <sys/time.h>
|
|
|
|
#endif
|
|
|
|
|
2017-09-14 19:28:40 +00:00
|
|
|
#include <algorithm>
|
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-15 00:30:22 +00:00
|
|
|
#include "../imgui/imgui.h"
|
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 )
|
2017-09-15 00:29:48 +00:00
|
|
|
, m_mbps( 64 )
|
2017-09-12 23:33:50 +00:00
|
|
|
{
|
|
|
|
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();
|
2017-09-13 21:36:40 +00:00
|
|
|
|
|
|
|
assert( s_instance != nullptr );
|
|
|
|
s_instance = nullptr;
|
2017-09-12 23:33:50 +00:00
|
|
|
}
|
|
|
|
|
2017-09-13 00:08:35 +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;
|
2017-09-14 00:15:04 +00:00
|
|
|
if( !m_sock.Connect( m_addr.c_str(), "8086" ) ) continue;
|
2017-09-12 23:54:22 +00:00
|
|
|
|
2017-09-15 19:09:19 +00:00
|
|
|
std::chrono::time_point<std::chrono::high_resolution_clock> t0;
|
|
|
|
|
2017-09-12 23:54:22 +00:00
|
|
|
uint8_t lz4;
|
2017-09-15 20:45:03 +00:00
|
|
|
uint64_t bytes = 0;
|
2017-09-15 22:40:51 +00:00
|
|
|
uint64_t timeStart;
|
2017-09-13 00:08:35 +00:00
|
|
|
|
2017-09-15 22:40:51 +00:00
|
|
|
if( !m_sock.Read( &timeStart, sizeof( timeStart ), &tv, ShouldExit ) ) goto close;
|
2017-09-14 00:15:04 +00:00
|
|
|
if( !m_sock.Read( &lz4, sizeof( lz4 ), &tv, ShouldExit ) ) goto close;
|
2017-09-12 23:54:22 +00:00
|
|
|
|
2017-09-15 22:40:51 +00:00
|
|
|
m_frames.push_back( timeStart );
|
|
|
|
|
2017-09-15 19:09:19 +00:00
|
|
|
t0 = std::chrono::high_resolution_clock::now();
|
2017-09-15 00:29:48 +00:00
|
|
|
|
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;
|
2017-09-14 00:15:04 +00:00
|
|
|
if( !m_sock.Read( &lz4sz, sizeof( lz4sz ), &tv, ShouldExit ) ) goto close;
|
|
|
|
if( !m_sock.Read( lz4buf, lz4sz, &tv, ShouldExit ) ) goto close;
|
2017-09-15 00:29:48 +00:00
|
|
|
bytes += sizeof( lz4sz ) + lz4sz;
|
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;
|
2017-09-14 17:44:49 +00:00
|
|
|
DispatchProcess( *ev, ptr );
|
2017-09-13 21:40:28 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-09-13 23:06:40 +00:00
|
|
|
QueueItem ev;
|
2017-09-14 00:15:04 +00:00
|
|
|
if( !m_sock.Read( &ev.hdr, sizeof( QueueHeader ), &tv, ShouldExit ) ) goto close;
|
2017-09-14 17:44:49 +00:00
|
|
|
const auto payload = QueueDataSize[ev.hdr.idx] - sizeof( QueueHeader );
|
|
|
|
if( payload > 0 )
|
|
|
|
{
|
|
|
|
if( !m_sock.Read( ((char*)&ev) + sizeof( QueueHeader ), payload, &tv, ShouldExit ) ) goto close;
|
|
|
|
}
|
2017-09-15 00:29:48 +00:00
|
|
|
bytes += sizeof( QueueHeader ) + payload; // ignores string transfer
|
2017-09-14 17:44:49 +00:00
|
|
|
DispatchProcess( ev );
|
2017-09-13 21:40:28 +00:00
|
|
|
}
|
2017-09-15 00:29:48 +00:00
|
|
|
|
|
|
|
auto t1 = std::chrono::high_resolution_clock::now();
|
|
|
|
auto td = std::chrono::duration_cast<std::chrono::milliseconds>( t1 - t0 ).count();
|
|
|
|
enum { MbpsUpdateTime = 200 };
|
|
|
|
if( td > MbpsUpdateTime )
|
|
|
|
{
|
2017-09-15 18:31:59 +00:00
|
|
|
std::lock_guard<std::mutex> lock( m_mbpslock );
|
2017-09-15 00:29:48 +00:00
|
|
|
m_mbps.erase( m_mbps.begin() );
|
|
|
|
m_mbps.emplace_back( 8.f * MbpsUpdateTime * bytes / ( td * 1000 * 1000 ) );
|
|
|
|
t0 = t1;
|
|
|
|
bytes = 0;
|
|
|
|
}
|
2017-09-12 23:54:22 +00:00
|
|
|
}
|
|
|
|
|
2017-09-13 00:00:22 +00:00
|
|
|
close:
|
2017-09-14 00:15:04 +00:00
|
|
|
m_sock.Close();
|
2017-09-12 23:33:50 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-09-14 17:44:49 +00:00
|
|
|
void View::DispatchProcess( const QueueItem& ev )
|
|
|
|
{
|
|
|
|
if( ev.hdr.type == QueueType::StringData )
|
|
|
|
{
|
|
|
|
timeval tv;
|
|
|
|
tv.tv_sec = 0;
|
|
|
|
tv.tv_usec = 10000;
|
|
|
|
|
|
|
|
char buf[TargetFrameSize];
|
|
|
|
uint16_t sz;
|
|
|
|
m_sock.Read( &sz, sizeof( sz ), &tv, ShouldExit );
|
|
|
|
m_sock.Read( buf, sz, &tv, ShouldExit );
|
|
|
|
AddString( ev.hdr.id, std::string( buf, buf+sz ) );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Process( ev );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void View::DispatchProcess( const QueueItem& ev, const char*& ptr )
|
|
|
|
{
|
|
|
|
ptr += QueueDataSize[ev.hdr.idx];
|
|
|
|
if( ev.hdr.type == QueueType::StringData )
|
|
|
|
{
|
|
|
|
uint16_t sz;
|
|
|
|
memcpy( &sz, ptr, sizeof( sz ) );
|
|
|
|
ptr += sizeof( sz );
|
|
|
|
AddString( ev.hdr.id, std::string( ptr, ptr+sz ) );
|
|
|
|
ptr += sz;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Process( ev );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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;
|
2017-09-15 22:30:27 +00:00
|
|
|
case QueueType::FrameMark:
|
2017-09-15 22:40:51 +00:00
|
|
|
ProcessFrameMark( ev.hdr.id );
|
2017-09-15 22:30:27 +00:00
|
|
|
break;
|
2017-09-14 00:00:13 +00:00
|
|
|
default:
|
|
|
|
assert( false );
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
void View::ProcessZoneBegin( uint64_t id, const QueueZoneBegin& ev )
|
|
|
|
{
|
|
|
|
auto it = m_pendingEndZone.find( id );
|
2017-09-15 17:56:55 +00:00
|
|
|
auto zone = m_slab.Alloc<Event>();
|
2017-09-14 00:16:51 +00:00
|
|
|
CheckString( ev.filename );
|
|
|
|
CheckString( ev.function );
|
2017-09-15 17:56:55 +00:00
|
|
|
zone->start = ev.time;
|
2017-09-14 00:00:13 +00:00
|
|
|
std::unique_lock<std::mutex> lock( m_lock );
|
|
|
|
if( it == m_pendingEndZone.end() )
|
|
|
|
{
|
2017-09-15 17:56:55 +00:00
|
|
|
zone->end = -1;
|
|
|
|
NewZone( zone );
|
2017-09-14 00:00:13 +00:00
|
|
|
lock.unlock();
|
2017-09-15 17:56:55 +00:00
|
|
|
m_openZones.emplace( id, zone );
|
2017-09-14 00:00:13 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
assert( ev.time <= it->second.time );
|
2017-09-15 17:56:55 +00:00
|
|
|
zone->end = it->second.time;
|
|
|
|
NewZone( zone );
|
2017-09-14 00:00:13 +00:00
|
|
|
lock.unlock();
|
|
|
|
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
|
|
|
|
{
|
2017-09-15 17:56:55 +00:00
|
|
|
auto zone = it->second;
|
2017-09-14 00:00:13 +00:00
|
|
|
std::unique_lock<std::mutex> lock( m_lock );
|
2017-09-15 17:56:55 +00:00
|
|
|
assert( ev.time >= zone->start );
|
|
|
|
zone->end = ev.time;
|
|
|
|
UpdateZone( zone );
|
2017-09-14 00:00:13 +00:00
|
|
|
lock.unlock();
|
|
|
|
m_openZones.erase( it );
|
|
|
|
}
|
2017-09-13 21:40:28 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 22:40:51 +00:00
|
|
|
void View::ProcessFrameMark( uint64_t id )
|
|
|
|
{
|
|
|
|
assert( !m_frames.empty() );
|
|
|
|
const auto lastframe = m_frames.back();
|
|
|
|
if( lastframe < id )
|
|
|
|
{
|
|
|
|
std::unique_lock<std::mutex> lock( m_lock );
|
|
|
|
m_frames.push_back( id );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
auto it = std::lower_bound( m_frames.begin(), m_frames.end(), id );
|
|
|
|
std::unique_lock<std::mutex> lock( m_lock );
|
|
|
|
m_frames.insert( it, id );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
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-14 17:43:40 +00:00
|
|
|
void View::AddString( uint64_t ptr, std::string&& str )
|
|
|
|
{
|
|
|
|
assert( m_strings.find( ptr ) == m_strings.end( ptr ) );
|
|
|
|
auto it = m_pendingStrings.find( ptr );
|
|
|
|
assert( it != m_pendingStrings.end() );
|
|
|
|
m_pendingStrings.erase( it );
|
|
|
|
std::lock_guard<std::mutex> lock( m_lock );
|
|
|
|
m_strings.emplace( ptr, std::move( str ) );
|
|
|
|
}
|
|
|
|
|
2017-09-15 17:56:55 +00:00
|
|
|
void View::NewZone( Event* zone )
|
2017-09-14 19:05:01 +00:00
|
|
|
{
|
2017-09-14 19:28:40 +00:00
|
|
|
if( !m_timeline.empty() )
|
|
|
|
{
|
2017-09-15 17:56:55 +00:00
|
|
|
const auto lastend = m_timeline.back()->end;
|
|
|
|
if( lastend != -1 && lastend < zone->start )
|
2017-09-14 19:28:40 +00:00
|
|
|
{
|
2017-09-15 18:17:39 +00:00
|
|
|
m_timeline.push_back( zone );
|
2017-09-14 19:28:40 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-09-15 18:17:39 +00:00
|
|
|
m_timeline.push_back( zone );
|
2017-09-14 19:28:40 +00:00
|
|
|
}
|
2017-09-14 19:05:01 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 17:56:55 +00:00
|
|
|
void View::UpdateZone( Event* zone )
|
2017-09-14 19:05:01 +00:00
|
|
|
{
|
2017-09-15 17:56:55 +00:00
|
|
|
assert( zone->end != -1 );
|
2017-09-14 19:05:01 +00:00
|
|
|
}
|
|
|
|
|
2017-09-15 00:30:22 +00:00
|
|
|
void View::Draw()
|
|
|
|
{
|
|
|
|
s_instance->DrawImpl();
|
|
|
|
}
|
|
|
|
|
|
|
|
void View::DrawImpl()
|
|
|
|
{
|
|
|
|
// Connection window
|
2017-09-15 22:57:50 +00:00
|
|
|
ImGui::Begin( m_addr.c_str() );
|
2017-09-15 00:30:22 +00:00
|
|
|
{
|
2017-09-15 18:31:59 +00:00
|
|
|
std::lock_guard<std::mutex> lock( m_mbpslock );
|
2017-09-15 00:30:22 +00:00
|
|
|
const auto mbps = m_mbps.back();
|
|
|
|
char buf[64];
|
|
|
|
if( mbps < 0.1f )
|
|
|
|
{
|
|
|
|
sprintf( buf, "%.2f Kbps", mbps * 1000.f );
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sprintf( buf, "%.2f Mbps", mbps );
|
|
|
|
}
|
|
|
|
ImGui::PlotLines( buf, m_mbps.data(), m_mbps.size(), 0, nullptr, 0 );
|
|
|
|
}
|
2017-09-15 18:31:59 +00:00
|
|
|
|
|
|
|
std::lock_guard<std::mutex> lock( m_lock );
|
2017-09-15 22:57:50 +00:00
|
|
|
{
|
|
|
|
const auto sz = m_frames.size();
|
|
|
|
if( sz > 1 )
|
|
|
|
{
|
|
|
|
const auto dt = m_frames[sz-1] - m_frames[sz-2];
|
|
|
|
const auto dtm = dt / 1000000.f;
|
|
|
|
const auto fps = 1000.f / dtm;
|
|
|
|
ImGui::Text( "FPS: %.1f Frame time: %.2f ms", fps, dtm );
|
|
|
|
}
|
|
|
|
}
|
|
|
|
ImGui::End();
|
2017-09-15 00:30:22 +00:00
|
|
|
}
|
|
|
|
|
2017-09-12 23:33:50 +00:00
|
|
|
}
|