#include #include #include #include #include #include #include #include #include #include #include #include #include #include #include #ifdef _WIN32 # include #endif #define STB_IMAGE_IMPLEMENTATION #define STBI_ONLY_PNG #include "stb_image.h" #define STB_IMAGE_RESIZE_IMPLEMENTATION #include "stb_image_resize.h" #include "../../public/common/TracyProtocol.hpp" #include "../../public/common/TracyVersion.hpp" #include "../../server/tracy_pdqsort.h" #include "../../server/tracy_robin_hood.h" #include "../../server/TracyBadVersion.hpp" #include "../../server/TracyFileHeader.hpp" #include "../../server/TracyFileRead.hpp" #include "../../server/TracyFileselector.hpp" #include "../../server/TracyImGui.hpp" #include "../../server/TracyMouse.hpp" #include "../../server/TracyPrint.hpp" #include "../../server/TracyProtoHistory.hpp" #include "../../server/TracyStorage.hpp" #include "../../server/TracyTexture.hpp" #include "../../server/TracyView.hpp" #include "../../server/TracyWeb.hpp" #include "../../server/TracyWorker.hpp" #include "../../server/IconsFontAwesome6.h" #include "icon.hpp" #include "zigzag01.hpp" #include "zigzag02.hpp" #include "zigzag04.hpp" #include "zigzag08.hpp" #include "zigzag16.hpp" #include "zigzag32.hpp" #include "Backend.hpp" #include "ConnectionHistory.hpp" #include "Filters.hpp" #include "Fonts.hpp" #include "HttpRequest.hpp" #include "IsElevated.hpp" #include "ImGuiContext.hpp" #include "ResolvService.hpp" #include "RunQueue.hpp" struct ClientData { int64_t time; uint32_t protocolVersion; int32_t activeTime; uint16_t port; uint64_t pid; std::string procName; std::string address; }; enum class ViewShutdown { False, True, Join }; static tracy::unordered_flat_map clients; static std::unique_ptr view; static tracy::BadVersionState badVer; static uint16_t port = 8086; static const char* connectTo = nullptr; static char title[128]; static std::thread loadThread, updateThread, updateNotesThread; static std::unique_ptr broadcastListen; static std::mutex resolvLock; static tracy::unordered_flat_map resolvMap; static ResolvService resolv( port ); static char addr[1024] = { "127.0.0.1" }; static ConnectionHistory* connHist; static std::atomic viewShutdown { ViewShutdown::False }; static double animTime = 0; static float dpiScale = 1.f; static Filters* filt; static RunQueue mainThreadTasks; static uint32_t updateVersion = 0; static bool showReleaseNotes = false; static std::string releaseNotes; static uint8_t* iconPx; static int iconX, iconY; static void* iconTex; static int iconTexSz; static uint8_t* zigzagPx[6]; static int zigzagX[6], zigzagY[6]; void* zigzagTex; static Backend* bptr; static bool s_customTitle = false; static bool s_isElevated = false; static void SetWindowTitleCallback( const char* title ) { char tmp[1024]; sprintf( tmp, "%s - Tracy Profiler %i.%i.%i", title, tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch ); bptr->SetTitle( tmp ); s_customTitle = true; } static void AttentionCallback() { bptr->Attention(); } static void DrawContents(); static void RunOnMainThread( const std::function& cb, bool forceDelay = false ) { mainThreadTasks.Queue( cb, forceDelay ); } static void SetupDPIScale( float scale, ImFont*& cb_fixedWidth, ImFont*& cb_bigFont, ImFont*& cb_smallFont ) { LoadFonts( scale, cb_fixedWidth, cb_bigFont, cb_smallFont ); auto& style = ImGui::GetStyle(); style = ImGuiStyle(); ImGui::StyleColorsDark(); style.WindowBorderSize = 1.f * scale; style.FrameBorderSize = 1.f * scale; style.FrameRounding = 5.f; style.Colors[ImGuiCol_ScrollbarBg] = ImVec4( 1, 1, 1, 0.03f ); style.Colors[ImGuiCol_Header] = ImVec4(0.26f, 0.59f, 0.98f, 0.25f); style.Colors[ImGuiCol_HeaderHovered] = ImVec4(0.26f, 0.59f, 0.98f, 0.35f); style.Colors[ImGuiCol_HeaderActive] = ImVec4(0.26f, 0.59f, 0.98f, 0.45f); style.ScaleAllSizes( scale ); const auto ty = int( 80 * scale ); iconTexSz = ty; auto scaleIcon = new uint8_t[4*ty*ty]; stbir_resize_uint8( iconPx, iconX, iconY, 0, scaleIcon, ty, ty, 0, 4 ); tracy::UpdateTextureRGBA( iconTex, scaleIcon, ty, ty ); delete[] scaleIcon; } static void SetupScaleCallback( float scale, ImFont*& cb_fixedWidth, ImFont*& cb_bigFont, ImFont*& cb_smallFont ) { RunOnMainThread( [scale, &cb_fixedWidth, &cb_bigFont, &cb_smallFont] { SetupDPIScale( scale * dpiScale, cb_fixedWidth, cb_bigFont, cb_smallFont ); }, true ); } int main( int argc, char** argv ) { sprintf( title, "Tracy Profiler %i.%i.%i", tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch ); std::unique_ptr initFileOpen; #ifdef __EMSCRIPTEN__ initFileOpen = std::unique_ptr( tracy::FileRead::Open( "embed.tracy" ) ); #endif if( argc == 2 ) { if( strcmp( argv[1], "--help" ) == 0 ) { printf( "%s\n\n", title ); printf( "Usage:\n\n" ); printf( " Open trace file stored on disk:\n" ); printf( " %s file.tracy\n\n", argv[0] ); printf( " Connect to a running client:\n" ); printf( " %s -a address [-p port]\n", argv[0] ); exit( 0 ); } initFileOpen = std::unique_ptr( tracy::FileRead::Open( argv[1] ) ); if( !initFileOpen ) { fprintf( stderr, "Cannot open trace file: %s\n", argv[1] ); exit( 1 ); } } else { while( argc >= 3 ) { if( strcmp( argv[1], "-a" ) == 0 ) { connectTo = argv[2]; } else if( strcmp( argv[1], "-p" ) == 0 ) { port = (uint16_t)atoi( argv[2] ); } else { fprintf( stderr, "Bad parameter: %s", argv[1] ); exit( 1 ); } argc -= 2; argv += 2; } } ConnectionHistory connHistory; Filters filters; connHist = &connHistory; filt = &filters; #ifndef __EMSCRIPTEN__ updateThread = std::thread( [] { HttpRequest( "nereid.pl", "/tracy/version", 8099, [] ( int size, char* data ) { if( size == 4 ) { uint32_t ver; memcpy( &ver, data, 4 ); RunOnMainThread( [ver] { updateVersion = ver; tracy::s_wasActive = true; } ); } delete[] data; } ); } ); #endif auto iconThread = std::thread( [] { iconPx = stbi_load_from_memory( (const stbi_uc*)Icon_data, Icon_size, &iconX, &iconY, nullptr, 4 ); zigzagPx[0] = stbi_load_from_memory( (const stbi_uc*)ZigZag32_data, ZigZag32_size, &zigzagX[0], &zigzagY[0], nullptr, 4 ); zigzagPx[1] = stbi_load_from_memory( (const stbi_uc*)ZigZag16_data, ZigZag16_size, &zigzagX[1], &zigzagY[1], nullptr, 4 ); zigzagPx[2] = stbi_load_from_memory( (const stbi_uc*)ZigZag08_data, ZigZag08_size, &zigzagX[2], &zigzagY[2], nullptr, 4 ); zigzagPx[3] = stbi_load_from_memory( (const stbi_uc*)ZigZag04_data, ZigZag04_size, &zigzagX[3], &zigzagY[3], nullptr, 4 ); zigzagPx[4] = stbi_load_from_memory( (const stbi_uc*)ZigZag02_data, ZigZag02_size, &zigzagX[4], &zigzagY[4], nullptr, 4 ); zigzagPx[5] = stbi_load_from_memory( (const stbi_uc*)ZigZag01_data, ZigZag01_size, &zigzagX[5], &zigzagY[5], nullptr, 4 ); } ); ImGuiTracyContext imguiContext; Backend backend( title, DrawContents, &mainThreadTasks ); tracy::InitTexture(); iconTex = tracy::MakeTexture(); zigzagTex = tracy::MakeTexture( true ); iconThread.join(); backend.SetIcon( iconPx, iconX, iconY ); bptr = &backend; dpiScale = backend.GetDpiScale(); const auto envDpiScale = getenv( "TRACY_DPI_SCALE" ); if( envDpiScale ) { const auto cnv = atof( envDpiScale ); if( cnv != 0 ) dpiScale = cnv; } SetupDPIScale( dpiScale, s_fixedWidth, s_bigFont, s_smallFont ); tracy::UpdateTextureRGBAMips( zigzagTex, (void**)zigzagPx, zigzagX, zigzagY, 6 ); for( auto& v : zigzagPx ) free( v ); if( initFileOpen ) { view = std::make_unique( RunOnMainThread, *initFileOpen, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); initFileOpen.reset(); } else if( connectTo ) { view = std::make_unique( RunOnMainThread, connectTo, port, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } tracy::Fileselector::Init(); s_isElevated = IsElevated(); backend.Show(); backend.Run(); if( loadThread.joinable() ) loadThread.join(); if( updateThread.joinable() ) updateThread.join(); if( updateNotesThread.joinable() ) updateNotesThread.join(); view.reset(); tracy::FreeTexture( zigzagTex, RunOnMainThread ); tracy::FreeTexture( iconTex, RunOnMainThread ); free( iconPx ); tracy::Fileselector::Shutdown(); return 0; } static void DrawContents() { static bool reconnect = false; static std::string reconnectAddr; static uint16_t reconnectPort; static bool showFilter = false; #ifndef __EMSCRIPTEN__ if( !view ) { const auto time = std::chrono::duration_cast( std::chrono::system_clock::now().time_since_epoch() ).count(); if( !broadcastListen ) { broadcastListen = std::make_unique(); if( !broadcastListen->Listen( port ) ) { broadcastListen.reset(); } } else { tracy::IpAddress addr; size_t len; for(;;) { auto msg = broadcastListen->Read( len, addr, 0 ); if( !msg ) break; if( len > sizeof( tracy::BroadcastMessage ) ) continue; uint16_t broadcastVersion; memcpy( &broadcastVersion, msg, sizeof( uint16_t ) ); if( broadcastVersion <= tracy::BroadcastVersion ) { uint32_t protoVer; char procname[tracy::WelcomeMessageProgramNameSize]; int32_t activeTime; uint16_t listenPort; uint64_t pid; switch( broadcastVersion ) { case 3: { tracy::BroadcastMessage bm; memcpy( &bm, msg, len ); protoVer = bm.protocolVersion; strcpy( procname, bm.programName ); activeTime = bm.activeTime; listenPort = bm.listenPort; pid = bm.pid; break; } case 2: { if( len > sizeof( tracy::BroadcastMessage_v2 ) ) continue; tracy::BroadcastMessage_v2 bm; memcpy( &bm, msg, len ); protoVer = bm.protocolVersion; strcpy( procname, bm.programName ); activeTime = bm.activeTime; listenPort = bm.listenPort; pid = 0; break; } case 1: { if( len > sizeof( tracy::BroadcastMessage_v1 ) ) continue; tracy::BroadcastMessage_v1 bm; memcpy( &bm, msg, len ); protoVer = bm.protocolVersion; strcpy( procname, bm.programName ); activeTime = bm.activeTime; listenPort = bm.listenPort; pid = 0; break; } case 0: { if( len > sizeof( tracy::BroadcastMessage_v0 ) ) continue; tracy::BroadcastMessage_v0 bm; memcpy( &bm, msg, len ); protoVer = bm.protocolVersion; strcpy( procname, bm.programName ); activeTime = bm.activeTime; listenPort = 8086; pid = 0; break; } default: assert( false ); break; } auto address = addr.GetText(); const auto ipNumerical = addr.GetNumber(); const auto clientId = uint64_t( ipNumerical ) | ( uint64_t( listenPort ) << 32 ); auto it = clients.find( clientId ); if( activeTime >= 0 ) { if( it == clients.end() ) { std::string ip( address ); resolvLock.lock(); if( resolvMap.find( ip ) == resolvMap.end() ) { resolvMap.emplace( ip, ip ); resolv.Query( ipNumerical, [ip] ( std::string&& name ) { std::lock_guard lock( resolvLock ); auto it = resolvMap.find( ip ); assert( it != resolvMap.end() ); std::swap( it->second, name ); } ); } resolvLock.unlock(); clients.emplace( clientId, ClientData { time, protoVer, activeTime, listenPort, pid, procname, std::move( ip ) } ); } else { it->second.time = time; it->second.activeTime = activeTime; it->second.port = listenPort; it->second.pid = pid; it->second.protocolVersion = protoVer; if( strcmp( it->second.procName.c_str(), procname ) != 0 ) it->second.procName = procname; } } else if( it != clients.end() ) { clients.erase( it ); } } } auto it = clients.begin(); while( it != clients.end() ) { const auto diff = time - it->second.time; if( diff > 4000 ) // 4s { it = clients.erase( it ); } else { ++it; } } } } else if( !clients.empty() ) { clients.clear(); } #endif int display_w, display_h; bptr->NewFrame( display_w, display_h ); static int activeFrames = 3; if( tracy::WasActive() || !clients.empty() || ( view && view->WasActive() ) ) { activeFrames = 3; } else { auto ctx = ImGui::GetCurrentContext(); if( ctx->NavWindowingTarget || ( ctx->DimBgRatio != 0 && ctx->DimBgRatio != 1 ) ) { activeFrames = 3; } else { auto& inputQueue = ctx->InputEventsQueue; if( !inputQueue.empty() ) { for( auto& v : inputQueue ) { if( v.Type != ImGuiInputEventType_MouseViewport ) { activeFrames = 3; break; } } } } } if( activeFrames == 0 ) { std::this_thread::sleep_for( std::chrono::milliseconds( 16 ) ); return; } activeFrames--; ImGui::NewFrame(); tracy::MouseFrame(); setlocale( LC_NUMERIC, "C" ); if( !view ) { if( s_customTitle ) { s_customTitle = false; bptr->SetTitle( title ); } auto& style = ImGui::GetStyle(); style.Colors[ImGuiCol_WindowBg] = ImVec4( 0.129f, 0.137f, 0.11f, 1.f ); ImGui::Begin( "Get started", nullptr, ImGuiWindowFlags_AlwaysAutoResize | ImGuiWindowFlags_NoCollapse ); char buf[128]; sprintf( buf, "Tracy Profiler %i.%i.%i", tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch ); ImGui::PushFont( s_bigFont ); tracy::TextCentered( buf ); ImGui::PopFont(); ImGui::SameLine( ImGui::GetWindowContentRegionMax().x - ImGui::CalcTextSize( ICON_FA_WRENCH ).x - ImGui::GetStyle().FramePadding.x * 2 ); if( ImGui::Button( ICON_FA_WRENCH ) ) { ImGui::OpenPopup( "About Tracy" ); } bool keepOpenAbout = true; if( ImGui::BeginPopupModal( "About Tracy", &keepOpenAbout, ImGuiWindowFlags_AlwaysAutoResize ) ) { tracy::ImageCentered( iconTex, ImVec2( iconTexSz, iconTexSz ) ); ImGui::Spacing(); ImGui::PushFont( s_bigFont ); tracy::TextCentered( buf ); ImGui::PopFont(); ImGui::Spacing(); ImGui::TextUnformatted( "A real time, nanosecond resolution, remote telemetry, hybrid\nframe and sampling profiler for games and other applications." ); ImGui::Spacing(); ImGui::TextUnformatted( "Created by Bartosz Taudul" ); ImGui::SameLine(); tracy::TextDisabledUnformatted( "" ); tracy::TextDisabledUnformatted( "Additional authors listed in AUTHORS file and in git history." ); ImGui::Separator(); ImGui::PushFont( s_smallFont ); tracy::TextFocused( "Protocol version", tracy::RealToString( tracy::ProtocolVersion ) ); ImGui::SameLine(); ImGui::SeparatorEx( ImGuiSeparatorFlags_Vertical ); ImGui::SameLine(); tracy::TextFocused( "Broadcast version", tracy::RealToString( tracy::BroadcastVersion ) ); ImGui::SameLine(); ImGui::SeparatorEx( ImGuiSeparatorFlags_Vertical ); ImGui::SameLine(); tracy::TextFocused( "Build date", __DATE__ ", " __TIME__ ); ImGui::PopFont(); ImGui::EndPopup(); } ImGui::Spacing(); if( ImGui::Button( ICON_FA_BOOK " Manual" ) ) { tracy::OpenWebpage( "https://github.com/wolfpld/tracy/releases" ); } ImGui::SameLine(); if( ImGui::Button( ICON_FA_EARTH_AMERICAS " Web" ) ) { ImGui::OpenPopup( "web" ); } if( ImGui::BeginPopup( "web" ) ) { if( ImGui::Selectable( ICON_FA_HOUSE_CHIMNEY " Tracy Profiler home page" ) ) { tracy::OpenWebpage( "https://github.com/wolfpld/tracy" ); } ImGui::Separator(); if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.8" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=30wpRpHTTag" ); } if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.7" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=_hU7vw00MZ4" ); } if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.6" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=uJkrFgriuOo" ); } if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.5" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=P6E7qLMmzTQ" ); } if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.4" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=eAkgkaO8B9o" ); } if( ImGui::Selectable( ICON_FA_VIDEO " New features in v0.3" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=3SXpDpDh2Uo" ); } if( ImGui::Selectable( ICON_FA_VIDEO " Overview of v0.2" ) ) { tracy::OpenWebpage( "https://www.youtube.com/watch?v=fB5B46lbapc" ); } ImGui::EndPopup(); } ImGui::SameLine(); if( ImGui::Button( ICON_FA_COMMENT " Chat" ) ) { tracy::OpenWebpage( "https://discord.gg/pk78auc" ); } ImGui::SameLine(); if( ImGui::Button( ICON_FA_HEART " Sponsor" ) ) { tracy::OpenWebpage( "https://github.com/sponsors/wolfpld/" ); } if( updateVersion > tracy::FileVersion( tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch ) ) { ImGui::Separator(); ImGui::TextColored( ImVec4( 1, 1, 0, 1 ), ICON_FA_EXCLAMATION " Update to %i.%i.%i is available!", ( updateVersion >> 16 ) & 0xFF, ( updateVersion >> 8 ) & 0xFF, updateVersion & 0xFF ); ImGui::SameLine(); if( ImGui::SmallButton( ICON_FA_GIFT " Get it!" ) ) { showReleaseNotes = true; if( !updateNotesThread.joinable() ) { updateNotesThread = std::thread( [] { HttpRequest( "nereid.pl", "/tracy/notes", 8099, [] ( int size, char* data ) { std::string notes( data, data+size ); delete[] data; RunOnMainThread( [notes = std::move( notes )] () mutable { releaseNotes = std::move( notes ); tracy::s_wasActive = true; } ); } ); } ); } } } if( s_isElevated ) { ImGui::Separator(); ImGui::TextColored( ImVec4( 1, 0.25f, 0.25f, 1 ), ICON_FA_TRIANGLE_EXCLAMATION " Profiler has elevated privileges! " ICON_FA_TRIANGLE_EXCLAMATION ); ImGui::PushFont( s_smallFont ); ImGui::TextColored( ImVec4( 1, 0.25f, 0.25f, 1 ), "You are running the profiler interface with admin privileges. This is" ); ImGui::TextColored( ImVec4( 1, 0.25f, 0.25f, 1 ), "most likely a mistake, as there is no reason to do so. Instead, you" ); ImGui::TextColored( ImVec4( 1, 0.25f, 0.25f, 1 ), "probably wanted to run the client (the application you are profiling)" ); ImGui::TextColored( ImVec4( 1, 0.25f, 0.25f, 1 ), "with elevated privileges." ); ImGui::PopFont(); } ImGui::Separator(); ImGui::TextUnformatted( "Client address" ); bool connectClicked = false; connectClicked |= ImGui::InputTextWithHint( "###connectaddress", "Enter address", addr, 1024, ImGuiInputTextFlags_EnterReturnsTrue ); if( !connHist->empty() ) { ImGui::SameLine(); if( ImGui::BeginCombo( "##frameCombo", nullptr, ImGuiComboFlags_NoPreview ) ) { int idxRemove = -1; const auto sz = std::min( 5, connHist->size() ); for( size_t i=0; iName( i ); if( ImGui::Selectable( str.c_str() ) ) { memcpy( addr, str.c_str(), str.size() + 1 ); } if( ImGui::IsItemHovered() && ImGui::IsKeyPressed( ImGui::GetKeyIndex( ImGuiKey_Delete ), false ) ) { idxRemove = (int)i; } } if( idxRemove >= 0 ) { connHist->Erase( idxRemove ); } ImGui::EndCombo(); } } connectClicked |= ImGui::Button( ICON_FA_WIFI " Connect" ); if( connectClicked && *addr && !loadThread.joinable() ) { connHist->Count( addr ); const auto addrLen = strlen( addr ); auto ptr = addr + addrLen - 1; while( ptr > addr && *ptr != ':' ) ptr--; if( *ptr == ':' ) { std::string addrPart = std::string( addr, ptr ); uint16_t portPart = (uint16_t)atoi( ptr+1 ); view = std::make_unique( RunOnMainThread, addrPart.c_str(), portPart, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } else { view = std::make_unique( RunOnMainThread, addr, port, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } } ImGui::SameLine( 0, ImGui::GetTextLineHeight() * 2 ); #ifndef TRACY_NO_FILESELECTOR if( ImGui::Button( ICON_FA_FOLDER_OPEN " Open saved trace" ) && !loadThread.joinable() ) { tracy::Fileselector::OpenFile( "tracy", "Tracy Profiler trace file", []( const char* fn ) { try { auto f = std::shared_ptr( tracy::FileRead::Open( fn ) ); if( f ) { loadThread = std::thread( [f] { try { view = std::make_unique( RunOnMainThread, *f, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } catch( const tracy::UnsupportedVersion& e ) { badVer.state = tracy::BadVersionState::UnsupportedVersion; badVer.version = e.version; } catch( const tracy::LegacyVersion& e ) { badVer.state = tracy::BadVersionState::LegacyVersion; badVer.version = e.version; } } ); } } catch( const tracy::NotTracyDump& ) { badVer.state = tracy::BadVersionState::BadFile; } catch( const tracy::FileReadError& ) { badVer.state = tracy::BadVersionState::ReadError; } } ); } if( badVer.state != tracy::BadVersionState::Ok ) { if( loadThread.joinable() ) { loadThread.join(); } tracy::BadVersion( badVer, s_bigFont ); } #endif if( !clients.empty() ) { ImGui::Separator(); ImGui::TextUnformatted( "Discovered clients:" ); ImGui::SameLine(); tracy::SmallToggleButton( ICON_FA_FILTER " Filter", showFilter ); if( filt->IsActive() ) { ImGui::SameLine(); tracy::TextColoredUnformatted( 0xFF00FFFF, ICON_FA_TRIANGLE_EXCLAMATION ); tracy::TooltipIfHovered( "Filters are active" ); if( showFilter ) { ImGui::SameLine(); if( ImGui::SmallButton( ICON_FA_DELETE_LEFT " Clear" ) ) { filt->Clear(); } } } if( showFilter ) { const auto w = ImGui::GetTextLineHeight() * 12; ImGui::Separator(); filt->Draw( w ); } ImGui::Separator(); static bool widthSet = false; ImGui::Columns( 3 ); if( !widthSet ) { widthSet = true; const auto w = ImGui::GetWindowWidth(); ImGui::SetColumnWidth( 0, w * 0.35f ); ImGui::SetColumnWidth( 1, w * 0.175f ); ImGui::SetColumnWidth( 2, w * 0.425f ); } const auto time = std::chrono::duration_cast( std::chrono::system_clock::now().time_since_epoch() ).count(); int idx = 0; int passed = 0; std::lock_guard lock( resolvLock ); for( auto& v : clients ) { const bool badProto = v.second.protocolVersion != tracy::ProtocolVersion; bool sel = false; const auto& name = resolvMap.find( v.second.address ); assert( name != resolvMap.end() ); if( filt->FailAddr( name->second.c_str() ) && filt->FailAddr( v.second.address.c_str() ) ) continue; if( filt->FailPort( v.second.port ) ) continue; if( filt->FailProg( v.second.procName.c_str() ) ) continue; ImGuiSelectableFlags flags = ImGuiSelectableFlags_SpanAllColumns; if( badProto ) flags |= ImGuiSelectableFlags_Disabled; ImGui::PushID( idx++ ); const bool selected = ImGui::Selectable( name->second.c_str(), &sel, flags ); ImGui::PopID(); if( ImGui::IsItemHovered( ImGuiHoveredFlags_AllowWhenDisabled ) ) { char portstr[32]; sprintf( portstr, "%" PRIu16, v.second.port ); ImGui::BeginTooltip(); if( badProto ) { tracy::TextColoredUnformatted( 0xFF0000FF, "Incompatible protocol!" ); ImGui::SameLine(); auto ph = tracy::ProtocolHistory; ImGui::TextDisabled( "(used: %i, required: %i)", v.second.protocolVersion, tracy::ProtocolVersion ); while( ph->protocol && ph->protocol != v.second.protocolVersion ) ph++; if( ph->protocol ) { if( ph->maxVer ) { ImGui::TextDisabled( "Compatible Tracy versions: %i.%i.%i to %i.%i.%i", ph->minVer >> 16, ( ph->minVer >> 8 ) & 0xFF, ph->minVer & 0xFF, ph->maxVer >> 16, ( ph->maxVer >> 8 ) & 0xFF, ph->maxVer & 0xFF ); } else { ImGui::TextDisabled( "Compatible Tracy version: %i.%i.%i", ph->minVer >> 16, ( ph->minVer >> 8 ) & 0xFF, ph->minVer & 0xFF ); } } ImGui::Separator(); } tracy::TextFocused( "IP:", v.second.address.c_str() ); tracy::TextFocused( "Port:", portstr ); if( v.second.pid != 0 ) { tracy::TextFocused( "PID:", tracy::RealToString( v.second.pid ) ); } ImGui::EndTooltip(); } if( v.second.port != port ) { ImGui::SameLine(); ImGui::TextDisabled( ":%" PRIu16, v.second.port ); } if( selected && !loadThread.joinable() ) { view = std::make_unique( RunOnMainThread, v.second.address.c_str(), v.second.port, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } ImGui::NextColumn(); const auto acttime = ( v.second.activeTime + ( time - v.second.time ) / 1000 ) * 1000000000ll; if( badProto ) { tracy::TextDisabledUnformatted( tracy::TimeToString( acttime ) ); } else { ImGui::TextUnformatted( tracy::TimeToString( acttime ) ); } ImGui::NextColumn(); if( badProto ) { tracy::TextDisabledUnformatted( v.second.procName.c_str() ); } else { ImGui::TextUnformatted( v.second.procName.c_str() ); } ImGui::NextColumn(); passed++; } ImGui::EndColumns(); if( passed == 0 ) { ImGui::TextUnformatted( "All clients are filtered." ); } } ImGui::End(); if( showReleaseNotes ) { assert( updateNotesThread.joinable() ); ImGui::SetNextWindowSize( ImVec2( 600 * dpiScale, 400 * dpiScale ), ImGuiCond_FirstUseEver ); ImGui::Begin( "Update available!", &showReleaseNotes ); if( ImGui::Button( ICON_FA_DOWNLOAD " Download" ) ) { tracy::OpenWebpage( "https://github.com/wolfpld/tracy/releases" ); } ImGui::BeginChild( "###notes", ImVec2( 0, 0 ), true ); if( releaseNotes.empty() ) { static float rnTime = 0; rnTime += ImGui::GetIO().DeltaTime; tracy::TextCentered( "Fetching release notes..." ); tracy::DrawWaitingDots( rnTime ); } else { ImGui::PushFont( s_fixedWidth ); ImGui::TextUnformatted( releaseNotes.c_str() ); ImGui::PopFont(); } ImGui::EndChild(); ImGui::End(); } } else { if( showReleaseNotes ) showReleaseNotes = false; if( broadcastListen ) { broadcastListen.reset(); clients.clear(); } if( loadThread.joinable() ) loadThread.join(); view->NotifyRootWindowSize( display_w, display_h ); if( !view->Draw() ) { viewShutdown.store( ViewShutdown::True, std::memory_order_relaxed ); reconnect = view->ReconnectRequested(); if( reconnect ) { reconnectAddr = view->GetAddress(); reconnectPort = view->GetPort(); } loadThread = std::thread( [view = std::move( view )] () mutable { view.reset(); viewShutdown.store( ViewShutdown::Join, std::memory_order_relaxed ); } ); } } auto& progress = tracy::Worker::GetLoadProgress(); auto totalProgress = progress.total.load( std::memory_order_relaxed ); if( totalProgress != 0 ) { ImGui::OpenPopup( "Loading trace..." ); } if( ImGui::BeginPopupModal( "Loading trace...", nullptr, ImGuiWindowFlags_AlwaysAutoResize ) ) { ImGui::PushFont( s_bigFont ); tracy::TextCentered( ICON_FA_HOURGLASS_HALF ); ImGui::PopFont(); animTime += ImGui::GetIO().DeltaTime; tracy::DrawWaitingDots( animTime ); auto currProgress = progress.progress.load( std::memory_order_relaxed ); if( totalProgress == 0 ) { ImGui::CloseCurrentPopup(); totalProgress = currProgress; } switch( currProgress ) { case tracy::LoadProgress::Initialization: ImGui::TextUnformatted( "Initialization..." ); break; case tracy::LoadProgress::Locks: ImGui::TextUnformatted( "Locks..." ); break; case tracy::LoadProgress::Messages: ImGui::TextUnformatted( "Messages..." ); break; case tracy::LoadProgress::Zones: ImGui::TextUnformatted( "CPU zones..." ); break; case tracy::LoadProgress::GpuZones: ImGui::TextUnformatted( "GPU zones..." ); break; case tracy::LoadProgress::Plots: ImGui::TextUnformatted( "Plots..." ); break; case tracy::LoadProgress::Memory: ImGui::TextUnformatted( "Memory..." ); break; case tracy::LoadProgress::CallStacks: ImGui::TextUnformatted( "Call stacks..." ); break; case tracy::LoadProgress::FrameImages: ImGui::TextUnformatted( "Frame images..." ); break; case tracy::LoadProgress::ContextSwitches: ImGui::TextUnformatted( "Context switches..." ); break; case tracy::LoadProgress::ContextSwitchesPerCpu: ImGui::TextUnformatted( "CPU context switches..." ); break; default: assert( false ); break; } ImGui::ProgressBar( float( currProgress ) / totalProgress, ImVec2( 200 * dpiScale, 0 ) ); ImGui::TextUnformatted( "Progress..." ); auto subTotal = progress.subTotal.load( std::memory_order_relaxed ); auto subProgress = progress.subProgress.load( std::memory_order_relaxed ); if( subTotal == 0 ) { ImGui::ProgressBar( 1.f, ImVec2( 200 * dpiScale, 0 ) ); } else { ImGui::ProgressBar( float( subProgress ) / subTotal, ImVec2( 200 * dpiScale, 0 ) ); } ImGui::EndPopup(); } switch( viewShutdown.load( std::memory_order_relaxed ) ) { case ViewShutdown::True: ImGui::OpenPopup( "Capture cleanup..." ); break; case ViewShutdown::Join: loadThread.join(); viewShutdown.store( ViewShutdown::False, std::memory_order_relaxed ); if( reconnect ) { view = std::make_unique( RunOnMainThread, reconnectAddr.c_str(), reconnectPort, s_fixedWidth, s_smallFont, s_bigFont, SetWindowTitleCallback, SetupScaleCallback, AttentionCallback ); } break; default: break; } if( ImGui::BeginPopupModal( "Capture cleanup...", nullptr, ImGuiWindowFlags_AlwaysAutoResize ) ) { if( viewShutdown.load( std::memory_order_relaxed ) != ViewShutdown::True ) ImGui::CloseCurrentPopup(); ImGui::PushFont( s_bigFont ); tracy::TextCentered( ICON_FA_BROOM ); ImGui::PopFont(); animTime += ImGui::GetIO().DeltaTime; tracy::DrawWaitingDots( animTime ); ImGui::TextUnformatted( "Please wait, cleanup is in progress" ); ImGui::EndPopup(); } if( tracy::Fileselector::HasFailed() ) ImGui::OpenPopup( "File selector is not available" ); if( ImGui::BeginPopupModal( "File selector is not available", nullptr, ImGuiWindowFlags_AlwaysAutoResize ) ) { ImGui::TextUnformatted( "File selector cannot be displayed." ); ImGui::TextUnformatted( "Check nfd library implementation for details." ); ImGui::Separator(); if( ImGui::Button( "Ok" ) ) ImGui::CloseCurrentPopup(); ImGui::EndPopup(); } bptr->EndFrame(); }