#ifdef _MSC_VER # include #else # include #endif #include #include "../common/tracy_lz4.hpp" #include "../common/TracyProtocol.hpp" #include "../common/TracySystem.hpp" #include "../common/TracyQueue.hpp" #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; } bool View::ShouldExit() { return s_instance->m_shutdown.load( std::memory_order_relaxed ); } void View::Worker() { timeval tv; tv.tv_sec = 0; tv.tv_usec = 10000; for(;;) { if( m_shutdown.load( std::memory_order_relaxed ) ) return; if( !m_sock.Connect( m_addr.c_str(), "8086" ) ) continue; 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; for(;;) { if( m_shutdown.load( std::memory_order_relaxed ) ) return; 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; 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]; } } else { 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; Process( ev ); } } close: m_sock.Close(); } } void View::Process( const QueueItem& ev ) { 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(); CheckString( ev.filename ); CheckString( ev.function ); std::unique_lock 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(); 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 lock( m_lock ); assert( ev.time >= m_data[it->second].start ); m_data[it->second].end = ev.time; lock.unlock(); m_openZones.erase( it ); } } 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 ) ); } }