tracy/profiler/src/main.cpp

607 lines
20 KiB
C++
Raw Normal View History

#include <algorithm>
2018-08-17 15:24:50 +00:00
#include <assert.h>
2019-06-17 17:34:48 +00:00
#include <chrono>
2018-07-28 16:50:22 +00:00
#include <inttypes.h>
#include <imgui.h>
2018-08-29 23:31:57 +00:00
#include "imgui_impl_glfw.h"
#include "imgui_impl_opengl3.h"
2018-10-23 17:39:09 +00:00
#include <stdint.h>
#include <stdio.h>
2018-07-15 18:10:34 +00:00
#include <stdlib.h>
2018-08-31 17:38:05 +00:00
#include <string>
2018-10-23 17:39:09 +00:00
#include <unordered_map>
2017-09-15 19:37:28 +00:00
#include <GL/gl3w.h>
#include <GLFW/glfw3.h>
2017-09-15 19:37:28 +00:00
#include <memory>
2017-09-30 12:32:30 +00:00
#include "../nfd/nfd.h"
2017-10-18 21:18:32 +00:00
#include <sys/stat.h>
#include <locale.h>
2017-09-15 19:37:28 +00:00
#ifdef _WIN32
# include <windows.h>
2018-07-15 18:10:34 +00:00
# include <shellapi.h>
#endif
2019-06-02 15:51:58 +00:00
#define STB_IMAGE_IMPLEMENTATION
#define STBI_ONLY_PNG
#include "stb_image.h"
2019-06-17 17:34:48 +00:00
#include "../../server/tracy_flat_hash_map.hpp"
2018-10-23 17:39:09 +00:00
#include "../../server/tracy_pdqsort.h"
#include "../../server/TracyBadVersion.hpp"
2017-09-30 14:58:02 +00:00
#include "../../server/TracyFileRead.hpp"
#include "../../server/TracyImGui.hpp"
2018-08-29 21:22:54 +00:00
#include "../../server/TracyStorage.hpp"
2017-09-15 19:37:28 +00:00
#include "../../server/TracyView.hpp"
2018-07-28 16:07:55 +00:00
#include "../../server/TracyWorker.hpp"
2018-07-29 12:24:24 +00:00
#include "../../server/TracyVersion.hpp"
2018-08-17 21:22:13 +00:00
#include "../../server/IconsFontAwesome5.h"
2018-07-29 12:24:24 +00:00
2018-08-17 19:40:15 +00:00
#include "imgui_freetype.h"
2018-07-27 23:03:26 +00:00
#include "Arimo.hpp"
#include "Cousine.hpp"
#include "FontAwesomeSolid.hpp"
2019-06-02 16:03:56 +00:00
#include "icon.hpp"
2018-07-27 23:03:26 +00:00
2018-04-14 13:12:16 +00:00
static void glfw_error_callback(int error, const char* description)
{
fprintf(stderr, "Error %d: %s\n", error, description);
}
2018-08-17 15:19:06 +00:00
static void OpenWebpage( const char* url )
2018-07-15 18:10:34 +00:00
{
#ifdef _WIN32
ShellExecuteA( nullptr, nullptr, url, nullptr, nullptr, 0 );
2019-04-04 11:58:13 +00:00
#elif defined __APPLE__
char buf[1024];
sprintf( buf, "open %s", url );
system( buf );
2018-07-15 18:10:34 +00:00
#else
char buf[1024];
sprintf( buf, "xdg-open %s", url );
system( buf );
#endif
}
2018-08-17 15:24:50 +00:00
static GLFWwindow* s_glfwWindow = nullptr;
static bool s_customTitle = false;
static void SetWindowTitleCallback( const char* title )
{
assert( s_glfwWindow );
glfwSetWindowTitle( s_glfwWindow, title );
s_customTitle = true;
}
std::vector<std::unordered_map<std::string, uint64_t>::const_iterator> RebuildConnectionHistory( const std::unordered_map<std::string, uint64_t>& connHistMap )
{
std::vector<std::unordered_map<std::string, uint64_t>::const_iterator> ret;
ret.reserve( connHistMap.size() );
for( auto it = connHistMap.begin(); it != connHistMap.end(); ++it )
{
ret.emplace_back( it );
}
tracy::pdqsort_branchless( ret.begin(), ret.end(), []( const auto& lhs, const auto& rhs ) { return lhs->second > rhs->second; } );
return ret;
}
2019-06-17 17:34:48 +00:00
struct ClientData
{
int64_t time;
uint32_t protocolVersion;
std::string procName;
};
int main( int argc, char** argv )
{
2019-06-17 17:34:48 +00:00
tracy::flat_hash_map<uint32_t, ClientData> clients;
std::unique_ptr<tracy::View> view;
2018-04-21 12:31:33 +00:00
int badVer = 0;
if( argc == 2 )
{
auto f = std::unique_ptr<tracy::FileRead>( tracy::FileRead::Open( argv[1] ) );
if( f )
{
view = std::make_unique<tracy::View>( *f );
}
}
else if( argc == 3 && strcmp( argv[1], "-a" ) == 0 )
{
view = std::make_unique<tracy::View>( argv[2] );
}
2018-07-29 12:24:24 +00:00
char title[128];
sprintf( title, "Tracy Profiler %i.%i.%i", tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch );
2018-07-29 12:24:24 +00:00
std::string winPosFile = tracy::GetSavePath( "window.position" );
int x = 200, y = 200, w = 1650, h = 960, maximize = 0;
{
FILE* f = fopen( winPosFile.c_str(), "rb" );
if( f )
{
uint32_t data[5];
fread( data, 1, sizeof( data ), f );
fclose( f );
x = data[0];
y = data[1];
w = data[2];
h = data[3];
maximize = data[4];
}
2019-02-12 00:26:14 +00:00
if( w <= 0 || h <= 0 )
{
x = 200;
y = 200;
w = 1650;
h = 960;
maximize = 0;
}
}
2018-10-23 17:39:09 +00:00
std::string connHistFile = tracy::GetSavePath( "connection.history" );
std::unordered_map<std::string, uint64_t> connHistMap;
std::vector<std::unordered_map<std::string, uint64_t>::const_iterator> connHistVec;
{
FILE* f = fopen( connHistFile.c_str(), "rb" );
if( f )
{
uint64_t sz;
fread( &sz, 1, sizeof( sz ), f );
for( uint64_t i=0; i<sz; i++ )
{
uint64_t ssz, cnt;
fread( &ssz, 1, sizeof( ssz ), f );
assert( ssz < 1024 );
char tmp[1024];
fread( tmp, 1, ssz, f );
fread( &cnt, 1, sizeof( cnt ), f );
connHistMap.emplace( std::string( tmp, tmp+ssz ), cnt );
}
fclose( f );
connHistVec = RebuildConnectionHistory( connHistMap );
2018-10-23 17:39:09 +00:00
}
}
// Setup window
2018-04-14 13:12:16 +00:00
glfwSetErrorCallback(glfw_error_callback);
2018-08-29 23:31:57 +00:00
if( !glfwInit() ) return 1;
2018-08-31 17:40:08 +00:00
glfwWindowHint(GLFW_VISIBLE, 0);
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
2018-04-14 13:12:16 +00:00
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 2);
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
#if __APPLE__
glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
#endif
GLFWwindow* window = glfwCreateWindow( w, h, title, NULL, NULL);
2018-08-29 23:31:57 +00:00
if( !window ) return 1;
2019-06-02 16:03:56 +00:00
{
GLFWimage icon;
icon.pixels = stbi_load_from_memory( (const stbi_uc*)Icon_data, Icon_size, &icon.width, &icon.height, nullptr, 4 );
glfwSetWindowIcon( window, 1, &icon );
free( icon.pixels );
}
glfwSetWindowPos( window, x, y );
2018-08-31 18:05:40 +00:00
#ifdef GLFW_MAXIMIZED
if( maximize ) glfwMaximizeWindow( window );
2018-08-31 18:05:40 +00:00
#endif
2018-08-17 15:24:50 +00:00
s_glfwWindow = window;
glfwMakeContextCurrent(window);
glfwSwapInterval(1); // Enable vsync
gl3wInit();
float dpiScale = 1.f;
#ifdef _WIN32
typedef UINT(*GDFS)(void);
GDFS getDpiForSystem = nullptr;
HMODULE dll = GetModuleHandleW(L"user32.dll");
if (dll != INVALID_HANDLE_VALUE)
getDpiForSystem = (GDFS)GetProcAddress(dll, "GetDpiForSystem");
if (getDpiForSystem)
dpiScale = getDpiForSystem() / 96.f;
#endif
// Setup ImGui binding
2018-08-29 23:31:57 +00:00
IMGUI_CHECKVERSION();
2018-04-14 13:12:16 +00:00
ImGui::CreateContext();
2018-08-29 21:22:54 +00:00
ImGuiIO& io = ImGui::GetIO();
2018-08-31 17:38:05 +00:00
std::string iniFileName = tracy::GetSavePath( "imgui.ini" );
io.IniFilename = iniFileName.c_str();
2018-08-30 00:08:08 +00:00
io.ConfigFlags |= ImGuiConfigFlags_NavEnableKeyboard;
2018-08-29 21:22:54 +00:00
2018-08-29 23:31:57 +00:00
ImGui_ImplGlfw_InitForOpenGL( window, true );
ImGui_ImplOpenGL3_Init( "#version 150" );
static const ImWchar rangesBasic[] = {
2018-07-27 23:03:26 +00:00
0x0020, 0x00FF, // Basic Latin + Latin Supplement
0x03BC, 0x03BC, // micro
2019-03-14 00:14:06 +00:00
0x03C3, 0x03C3, // small sigma
2018-07-27 23:03:26 +00:00
0,
};
static const ImWchar rangesIcons[] = {
2018-08-17 21:22:13 +00:00
ICON_MIN_FA, ICON_MAX_FA,
0
};
ImFontConfig configMerge;
configMerge.MergeMode = true;
io.Fonts->AddFontFromMemoryCompressedTTF( tracy::Arimo_compressed_data, tracy::Arimo_compressed_size, 15.0f * dpiScale, nullptr, rangesBasic );
2018-08-17 16:45:04 +00:00
io.Fonts->AddFontFromMemoryCompressedTTF( tracy::FontAwesomeSolid_compressed_data, tracy::FontAwesomeSolid_compressed_size, 14.0f * dpiScale, &configMerge, rangesIcons );
auto fixedWidth = io.Fonts->AddFontFromMemoryCompressedTTF( tracy::Cousine_compressed_data, tracy::Cousine_compressed_size, 15.0f * dpiScale );
auto bigFont = io.Fonts->AddFontFromMemoryCompressedTTF( tracy::Arimo_compressed_data, tracy::Cousine_compressed_size, 20.0f * dpiScale );
2017-10-18 21:18:32 +00:00
2018-08-17 19:40:15 +00:00
ImGuiFreeType::BuildFontAtlas( io.Fonts, ImGuiFreeType::LightHinting );
2018-01-13 12:59:16 +00:00
ImGui::StyleColorsDark();
2018-01-13 13:08:14 +00:00
auto& style = ImGui::GetStyle();
2018-08-17 17:06:14 +00:00
style.WindowBorderSize = 1.f * dpiScale;
style.FrameBorderSize = 1.f * dpiScale;
style.FrameRounding = 5.f * dpiScale;
2018-08-17 17:06:30 +00:00
style.ScrollbarSize *= dpiScale;
2018-01-13 13:08:14 +00:00
style.Colors[ImGuiCol_ScrollbarBg] = ImVec4( 1, 1, 1, 0.03f );
2018-01-13 12:59:16 +00:00
ImVec4 clear_color = ImColor(114, 144, 154);
2017-09-15 19:37:28 +00:00
char addr[1024] = { "127.0.0.1" };
2018-07-28 16:17:56 +00:00
std::thread loadThread;
2019-06-17 17:34:48 +00:00
tracy::UdpListen* broadcastListen = nullptr;
2018-07-28 16:17:56 +00:00
2018-08-31 17:40:08 +00:00
glfwShowWindow( window );
double time = 0;
// Main loop
while (!glfwWindowShouldClose(window))
{
glfwPollEvents();
if( glfwGetWindowAttrib( window, GLFW_ICONIFIED ) )
{
std::this_thread::sleep_for( std::chrono::milliseconds( 50 ) );
continue;
}
int display_w, display_h;
glfwGetFramebufferSize(window, &display_w, &display_h);
2018-08-29 23:31:57 +00:00
ImGui_ImplOpenGL3_NewFrame();
ImGui_ImplGlfw_NewFrame();
ImGui::NewFrame();
2017-09-15 19:37:28 +00:00
if( !view )
{
2018-08-17 15:24:50 +00:00
if( s_customTitle )
{
s_customTitle = false;
glfwSetWindowTitle( window, title );
}
2019-06-17 17:34:48 +00:00
if( !broadcastListen )
{
broadcastListen = new tracy::UdpListen();
if( !broadcastListen->Listen( 8087 ) )
{
delete broadcastListen;
broadcastListen = nullptr;
}
}
else
{
tracy::IpAddress addr;
int len;
auto msg = broadcastListen->Read( len, addr );
const auto t = std::chrono::duration_cast<std::chrono::milliseconds>( std::chrono::system_clock::now().time_since_epoch() ).count();
if( msg )
{
uint32_t protoVer;
memcpy( &protoVer, msg, sizeof( uint32_t ) );
auto procname = msg + sizeof( uint32_t );
auto it = clients.find( addr.GetNumber() );
if( it == clients.end() )
{
clients.emplace( addr.GetNumber(), ClientData { t, protoVer, procname } );
}
else
{
it->second.time = t;
if( it->second.protocolVersion != protoVer ) it->second.protocolVersion = protoVer;
if( strcmp( it->second.procName.c_str(), procname ) != 0 ) it->second.procName = procname;
}
}
auto it = clients.begin();
while( it != clients.end() )
{
const auto diff = t - it->second.time;
if( diff > 10000 ) // 10s
{
it = clients.erase( it );
}
else
{
++it;
}
}
}
setlocale( LC_NUMERIC, "C" );
2019-03-07 00:18:24 +00:00
style.Colors[ImGuiCol_WindowBg] = ImVec4( 0.129f, 0.137f, 0.11f, 1.f );
ImGui::Begin( "Get started", nullptr, ImGuiWindowFlags_AlwaysAutoResize );
char buf[128];
sprintf( buf, "Tracy Profiler %i.%i.%i", tracy::Version::Major, tracy::Version::Minor, tracy::Version::Patch );
ImGui::PushFont( bigFont );
tracy::TextCentered( buf );
ImGui::PopFont();
ImGui::Spacing();
2019-06-11 00:12:34 +00:00
if( ImGui::Button( ICON_FA_BOOK " Manual" ) )
2018-08-03 23:15:56 +00:00
{
OpenWebpage( "https://bitbucket.org/wolfpld/tracy/downloads/tracy.pdf" );
}
ImGui::SameLine();
2019-06-11 00:12:34 +00:00
if( ImGui::Button( ICON_FA_GLOBE_AMERICAS " Web" ) )
{
OpenWebpage( "https://bitbucket.org/wolfpld/tracy" );
}
ImGui::SameLine();
2019-06-11 00:12:34 +00:00
if( ImGui::Button( ICON_FA_COMMENT " Chat" ) )
{
OpenWebpage( "https://discord.gg/pk78auc" );
}
ImGui::SameLine();
2018-08-17 21:22:13 +00:00
if( ImGui::Button( ICON_FA_VIDEO " Tutorial" ) )
{
ImGui::OpenPopup( "tutorial" );
}
if( ImGui::BeginPopup( "tutorial" ) )
{
if( ImGui::Selectable( ICON_FA_VIDEO " Introduction to the Tracy Profiler" ) )
{
OpenWebpage( "https://www.youtube.com/watch?v=fB5B46lbapc" );
}
if( ImGui::Selectable( ICON_FA_VIDEO " New features in Tracy Profiler v0.3" ) )
{
OpenWebpage( "https://www.youtube.com/watch?v=3SXpDpDh2Uo" );
}
if( ImGui::Selectable( ICON_FA_VIDEO " New features in Tracy Profiler v0.4" ) )
{
OpenWebpage( "https://www.youtube.com/watch?v=eAkgkaO8B9o" );
}
ImGui::EndPopup();
}
ImGui::Separator();
2019-06-16 23:21:55 +00:00
ImGui::TextUnformatted( "Client address" );
bool connectClicked = false;
2019-03-15 00:47:07 +00:00
connectClicked |= ImGui::InputTextWithHint( "###connectaddress", "Enter address", addr, 1024, ImGuiInputTextFlags_EnterReturnsTrue );
if( !connHistVec.empty() )
{
ImGui::SameLine();
if( ImGui::BeginCombo( "##frameCombo", nullptr, ImGuiComboFlags_NoPreview ) )
{
int idxRemove = -1;
const auto sz = std::min<size_t>( 5, connHistVec.size() );
for( size_t i=0; i<sz; i++ )
{
const auto& str = connHistVec[i]->first;
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 )
{
connHistMap.erase( connHistVec[idxRemove] );
connHistVec = RebuildConnectionHistory( connHistMap );
}
ImGui::EndCombo();
}
}
connectClicked |= ImGui::Button( ICON_FA_WIFI " Connect" );
if( connectClicked && *addr && !loadThread.joinable() )
2017-09-15 19:37:28 +00:00
{
2018-10-23 17:39:09 +00:00
std::string addrStr( addr );
2019-01-29 20:53:56 +00:00
auto it = connHistMap.find( addrStr );
2018-10-23 17:39:09 +00:00
if( it != connHistMap.end() )
{
it->second++;
}
else
{
2019-01-29 20:53:56 +00:00
connHistMap.emplace( std::move( addrStr ), 1 );
2018-10-23 17:39:09 +00:00
}
connHistVec = RebuildConnectionHistory( connHistMap );
2018-10-23 17:39:09 +00:00
2018-08-17 15:24:50 +00:00
view = std::make_unique<tracy::View>( addr, fixedWidth, SetWindowTitleCallback );
2017-09-15 19:37:28 +00:00
}
2019-06-16 23:21:55 +00:00
ImGui::SameLine( 0, ImGui::GetFontSize() * 2 );
2018-08-17 21:22:13 +00:00
if( ImGui::Button( ICON_FA_FOLDER_OPEN " Open saved trace" ) && !loadThread.joinable() )
2017-09-30 12:32:30 +00:00
{
nfdchar_t* fn;
auto res = NFD_OpenDialog( "tracy", nullptr, &fn );
if( res == NFD_OKAY )
{
try
2017-09-30 14:58:02 +00:00
{
2018-07-28 16:17:56 +00:00
auto f = std::shared_ptr<tracy::FileRead>( tracy::FileRead::Open( fn ) );
if( f )
2018-04-21 12:31:33 +00:00
{
loadThread = std::thread( [&view, f, &badVer, fixedWidth] {
2018-07-28 16:17:56 +00:00
try
{
2018-08-17 15:24:50 +00:00
view = std::make_unique<tracy::View>( *f, fixedWidth, SetWindowTitleCallback );
2018-07-28 16:17:56 +00:00
}
catch( const tracy::UnsupportedVersion& e )
{
badVer = e.version;
}
} );
2018-04-21 12:31:33 +00:00
}
}
catch( const tracy::NotTracyDump& e )
{
badVer = -1;
2017-09-30 14:58:02 +00:00
}
2017-09-30 12:32:30 +00:00
}
}
2018-04-21 12:31:33 +00:00
2018-07-28 16:17:56 +00:00
if( badVer != 0 )
{
if( loadThread.joinable() ) { loadThread.join(); }
tracy::BadVersion( badVer );
}
2018-04-21 12:31:33 +00:00
ImGui::End();
}
2017-09-15 19:37:28 +00:00
else
{
2019-06-17 17:34:48 +00:00
if( broadcastListen )
{
delete broadcastListen;
broadcastListen = nullptr;
clients.clear();
}
2018-07-28 16:17:56 +00:00
if( loadThread.joinable() ) loadThread.join();
view->NotifyRootWindowSize( display_w, display_h );
2018-04-29 23:16:08 +00:00
if( !view->Draw() )
{
view.reset();
}
}
2018-07-28 16:07:55 +00:00
auto& progress = tracy::Worker::GetLoadProgress();
auto totalProgress = progress.total.load( std::memory_order_relaxed );
if( totalProgress != 0 )
{
ImGui::OpenPopup( "Loading trace..." );
}
2018-07-28 16:48:45 +00:00
if( ImGui::BeginPopupModal( "Loading trace...", nullptr, ImGuiWindowFlags_AlwaysAutoResize ) )
2018-07-28 16:07:55 +00:00
{
2018-08-17 21:29:32 +00:00
tracy::TextCentered( ICON_FA_HOURGLASS_HALF );
time += io.DeltaTime;
2019-03-06 17:12:44 +00:00
tracy::DrawWaitingDots( time );
2018-07-28 16:07:55 +00:00
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;
2019-06-06 21:40:37 +00:00
case tracy::LoadProgress::FrameImages:
ImGui::TextUnformatted( "Frame images..." );
break;
default:
assert( false );
break;
}
2018-07-28 16:07:55 +00:00
ImGui::ProgressBar( float( currProgress ) / totalProgress, ImVec2( 200 * dpiScale, 0 ) );
2018-07-28 16:56:52 +00:00
ImGui::TextUnformatted( "Progress..." );
2018-07-28 16:56:52 +00:00
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 ) );
}
2018-07-28 16:07:55 +00:00
ImGui::EndPopup();
}
// Rendering
2018-08-29 23:31:57 +00:00
ImGui::Render();
glfwMakeContextCurrent(window);
glViewport(0, 0, display_w, display_h);
glClearColor(clear_color.x, clear_color.y, clear_color.z, clear_color.w);
glClear(GL_COLOR_BUFFER_BIT);
2018-08-29 23:31:57 +00:00
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
glfwMakeContextCurrent(window);
glfwSwapBuffers(window);
if( !glfwGetWindowAttrib( window, GLFW_FOCUSED ) )
{
std::this_thread::sleep_for( std::chrono::milliseconds( 50 ) );
}
}
{
2018-10-23 17:39:09 +00:00
FILE* f = fopen( winPosFile.c_str(), "wb" );
if( f )
{
2018-08-31 18:05:40 +00:00
#ifdef GLFW_MAXIMIZED
2018-10-23 17:39:09 +00:00
uint32_t maximized = glfwGetWindowAttrib( window, GLFW_MAXIMIZED );
if( maximized ) glfwRestoreWindow( window );
2018-08-31 18:05:40 +00:00
#else
2018-10-23 17:39:09 +00:00
uint32_t maximized = 0;
2018-08-31 18:05:40 +00:00
#endif
2018-10-23 17:39:09 +00:00
glfwGetWindowPos( window, &x, &y );
glfwGetWindowSize( window, &w, &h );
2018-10-23 17:39:09 +00:00
uint32_t data[5] = { uint32_t( x ), uint32_t( y ), uint32_t( w ), uint32_t( h ), maximized };
fwrite( data, 1, sizeof( data ), f );
fclose( f );
}
}
// Cleanup
2018-08-29 23:31:57 +00:00
ImGui_ImplOpenGL3_Shutdown();
ImGui_ImplGlfw_Shutdown();
2018-04-14 13:12:16 +00:00
ImGui::DestroyContext();
2018-08-29 23:31:57 +00:00
glfwDestroyWindow(window);
glfwTerminate();
2018-10-23 17:39:09 +00:00
{
FILE* f = fopen( connHistFile.c_str(), "wb" );
if( f )
{
uint64_t sz = uint64_t( connHistMap.size() );
fwrite( &sz, 1, sizeof( uint64_t ), f );
for( auto& v : connHistMap )
{
sz = uint64_t( v.first.size() );
fwrite( &sz, 1, sizeof( uint64_t ), f );
fwrite( v.first.c_str(), 1, sz, f );
fwrite( &v.second, 1, sizeof( v.second ), f );
}
fclose( f );
}
}
return 0;
}