2021-09-06 06:21:10 +00:00
|
|
|
// Dear ImGui: standalone example application for GLFW + OpenGL 3, using programmable pipeline
|
|
|
|
// (GLFW is a cross-platform general purpose library for handling windows, inputs, OpenGL/Vulkan/Metal graphics context creation, etc.)
|
|
|
|
// If you are new to Dear ImGui, read documentation from the docs/ folder + read the top of imgui.cpp.
|
|
|
|
// Read online: https://github.com/ocornut/imgui/tree/master/docs
|
|
|
|
|
|
|
|
#include "imgui.h"
|
|
|
|
#include "imgui_impl_glfw.h"
|
|
|
|
#include "imgui_impl_opengl3.h"
|
|
|
|
#include <stdio.h>
|
|
|
|
#include <thread>
|
2021-10-18 15:50:50 +00:00
|
|
|
#include <unistd.h>
|
2021-09-06 06:21:10 +00:00
|
|
|
#include "../overlay.h"
|
2022-01-06 09:57:18 +00:00
|
|
|
#include "mangoapp.h"
|
2022-01-05 05:12:21 +00:00
|
|
|
#include <GL/glew.h>
|
2021-09-06 06:21:10 +00:00
|
|
|
#include <GLFW/glfw3.h>
|
|
|
|
|
2021-09-21 12:29:49 +00:00
|
|
|
#define GLFW_EXPOSE_NATIVE_X11
|
|
|
|
#include <GLFW/glfw3native.h>
|
|
|
|
#include <X11/Xatom.h>
|
|
|
|
|
2021-09-06 06:21:10 +00:00
|
|
|
static void glfw_error_callback(int error, const char* description)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Glfw Error %d: %s\n", error, description);
|
|
|
|
}
|
|
|
|
swapchain_stats sw_stats {};
|
2022-01-20 00:24:01 +00:00
|
|
|
overlay_params *params;
|
2021-09-06 06:21:10 +00:00
|
|
|
static ImVec2 window_size;
|
|
|
|
static uint32_t vendorID;
|
|
|
|
static std::string deviceName;
|
2022-01-05 05:12:41 +00:00
|
|
|
static notify_thread notifier;
|
2022-01-06 09:57:18 +00:00
|
|
|
int msgid;
|
|
|
|
bool mangoapp_paused = false;
|
|
|
|
std::mutex mangoapp_m;
|
|
|
|
std::condition_variable mangoapp_cv;
|
2021-12-13 13:38:50 +00:00
|
|
|
static uint8_t raw_msg[1024] = {0};
|
|
|
|
|
2022-01-21 09:35:55 +00:00
|
|
|
void ctrl_thread(){
|
|
|
|
while (1){
|
|
|
|
const struct mangoapp_ctrl_msgid1_v1 *mangoapp_ctrl_v1 = (const struct mangoapp_ctrl_msgid1_v1*) raw_msg;
|
2022-01-22 04:17:55 +00:00
|
|
|
memset(raw_msg, 0, sizeof(raw_msg));
|
2022-01-21 09:35:55 +00:00
|
|
|
size_t msg_size = msgrcv(msgid, (void *) raw_msg, sizeof(raw_msg), 2, 0);
|
|
|
|
switch (mangoapp_ctrl_v1->log_session) {
|
|
|
|
case 1:
|
2022-01-22 04:17:55 +00:00
|
|
|
if (!logger->is_active())
|
|
|
|
logger->start_logging();
|
|
|
|
break;
|
2022-01-21 09:35:55 +00:00
|
|
|
case 2:
|
2022-01-22 04:17:55 +00:00
|
|
|
if (logger->is_active())
|
|
|
|
logger->stop_logging();
|
|
|
|
break;
|
2022-01-21 09:35:55 +00:00
|
|
|
case 3:
|
|
|
|
logger->is_active() ? logger->stop_logging() : logger->start_logging();
|
2022-01-22 04:17:55 +00:00
|
|
|
break;
|
2022-01-21 09:35:55 +00:00
|
|
|
}
|
2022-02-01 01:27:45 +00:00
|
|
|
{
|
|
|
|
std::lock_guard<std::mutex> lk(mangoapp_m);
|
|
|
|
switch (mangoapp_ctrl_v1->no_display){
|
|
|
|
case 1:
|
|
|
|
params->no_display = 1;
|
|
|
|
printf("set no_display 1\n");
|
|
|
|
break;
|
|
|
|
case 2:
|
|
|
|
params->no_display = 0;
|
|
|
|
printf("set no_display 0\n");
|
|
|
|
break;
|
|
|
|
case 3:
|
|
|
|
params->no_display ? params->no_display = 0 : params->no_display = 1;
|
|
|
|
printf("toggle no_display\n");
|
|
|
|
break;
|
|
|
|
}
|
2022-01-21 09:35:55 +00:00
|
|
|
}
|
|
|
|
mangoapp_cv.notify_one();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-12-13 13:38:50 +00:00
|
|
|
void msg_read_thread(){
|
|
|
|
int key = ftok("mangoapp", 65);
|
2022-01-06 09:57:18 +00:00
|
|
|
msgid = msgget(key, 0666 | IPC_CREAT);
|
2021-12-13 13:38:50 +00:00
|
|
|
const struct mangoapp_msg_header *hdr = (const struct mangoapp_msg_header*) raw_msg;
|
|
|
|
const struct mangoapp_msg_v1 *mangoapp_v1 = (const struct mangoapp_msg_v1*) raw_msg;
|
2021-09-06 06:21:10 +00:00
|
|
|
while (1){
|
2021-12-13 13:38:50 +00:00
|
|
|
// make sure that the message recieved is compatible
|
|
|
|
// and that we're not trying to use variables that don't exist (yet)
|
|
|
|
size_t msg_size = msgrcv(msgid, (void *) raw_msg, sizeof(raw_msg), 1, 0);
|
|
|
|
if (hdr->version == 1){
|
|
|
|
if (msg_size > offsetof(struct mangoapp_msg_v1, frametime_ns)){
|
2022-01-20 00:24:01 +00:00
|
|
|
update_hud_info_with_frametime(sw_stats, *params, vendorID, mangoapp_v1->frametime_ns);
|
2021-12-13 13:38:50 +00:00
|
|
|
}
|
2021-09-06 06:21:10 +00:00
|
|
|
} else {
|
2021-12-13 13:38:50 +00:00
|
|
|
printf("Unsupported mangoapp struct version: %i\n", hdr->version);
|
|
|
|
exit(1);
|
2021-09-06 06:21:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-04 06:56:56 +00:00
|
|
|
static const char *GamescopeOverlayProperty = "GAMESCOPE_EXTERNAL_OVERLAY";
|
2021-09-21 12:29:49 +00:00
|
|
|
|
2022-01-06 09:57:18 +00:00
|
|
|
GLFWwindow* init(GLFWwindow* window, const char* glsl_version){
|
|
|
|
window = glfwCreateWindow(1280, 720, "mangoapp overlay window", NULL, NULL);
|
|
|
|
Display *x11_display = glfwGetX11Display();
|
|
|
|
Window x11_window = glfwGetX11Window(window);
|
|
|
|
if (x11_window && x11_display)
|
|
|
|
{
|
|
|
|
// Set atom for gamescope to render as an overlay.
|
|
|
|
Atom overlay_atom = XInternAtom (x11_display, GamescopeOverlayProperty, False);
|
|
|
|
uint32_t value = 1;
|
|
|
|
XChangeProperty(x11_display, x11_window, overlay_atom, XA_ATOM, 32, PropertyNewValue, (unsigned char *)&value, 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
glfwMakeContextCurrent(window);
|
|
|
|
glfwSwapInterval(1); // Enable vsync
|
|
|
|
ImGui::CreateContext();
|
|
|
|
ImGuiIO& io = ImGui::GetIO(); (void)io;
|
|
|
|
ImGui::StyleColorsDark();
|
|
|
|
ImGui_ImplGlfw_InitForOpenGL(window, true);
|
|
|
|
ImGui_ImplOpenGL3_Init(glsl_version);
|
|
|
|
return window;
|
|
|
|
}
|
|
|
|
|
|
|
|
void shutdown(GLFWwindow* window){
|
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
|
|
ImGui_ImplGlfw_Shutdown();
|
|
|
|
ImGui::DestroyContext();
|
|
|
|
glfwDestroyWindow(window);
|
|
|
|
}
|
|
|
|
|
2021-09-06 06:21:10 +00:00
|
|
|
int main(int, char**)
|
|
|
|
{
|
|
|
|
// Setup window
|
|
|
|
glfwSetErrorCallback(glfw_error_callback);
|
|
|
|
if (!glfwInit())
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
const char* glsl_version = "#version 130";
|
|
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
|
|
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 0);
|
|
|
|
|
|
|
|
glfwWindowHint(GLFW_RESIZABLE, 1);
|
|
|
|
glfwWindowHint(GLFW_TRANSPARENT_FRAMEBUFFER, 1);
|
|
|
|
|
|
|
|
// Create window with graphics context
|
2022-01-06 09:57:18 +00:00
|
|
|
GLFWwindow* window = init(window, glsl_version);;
|
2021-09-06 06:21:10 +00:00
|
|
|
// Initialize OpenGL loader
|
2022-01-05 05:12:21 +00:00
|
|
|
|
2021-09-06 06:21:10 +00:00
|
|
|
bool err = glewInit() != GLEW_OK;
|
2022-01-05 05:12:21 +00:00
|
|
|
|
2021-09-06 06:21:10 +00:00
|
|
|
if (err)
|
|
|
|
{
|
|
|
|
fprintf(stderr, "Failed to initialize OpenGL loader!\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Setup Platform/Renderer backends
|
2022-01-20 00:24:01 +00:00
|
|
|
struct device_data *device_data = new struct device_data();
|
|
|
|
device_data->instance = new struct instance_data();
|
|
|
|
device_data->instance->params = {};
|
|
|
|
params = &device_data->instance->params;
|
|
|
|
parse_overlay_config(params, getenv("MANGOHUD_CONFIG"));
|
|
|
|
create_fonts(*params, sw_stats.font1, sw_stats.font_text);
|
|
|
|
HUDElements.convert_colors(*params);
|
|
|
|
init_cpu_stats(*params);
|
|
|
|
notifier.params = params;
|
2022-01-05 05:12:41 +00:00
|
|
|
start_notifier(notifier);
|
2021-09-06 06:21:10 +00:00
|
|
|
deviceName = (char*)glGetString(GL_RENDERER);
|
|
|
|
sw_stats.deviceName = deviceName;
|
|
|
|
if (deviceName.find("Radeon") != std::string::npos
|
|
|
|
|| deviceName.find("AMD") != std::string::npos){
|
|
|
|
vendorID = 0x1002;
|
|
|
|
} else {
|
|
|
|
vendorID = 0x10de;
|
|
|
|
}
|
2022-01-20 00:24:01 +00:00
|
|
|
init_gpu_stats(vendorID, 0, *params);
|
2021-09-06 06:21:10 +00:00
|
|
|
init_system_info();
|
2021-11-18 07:25:30 +00:00
|
|
|
sw_stats.engine = EngineTypes::GAMESCOPE;
|
2021-12-13 13:38:50 +00:00
|
|
|
std::thread(msg_read_thread).detach();
|
2022-01-21 09:35:55 +00:00
|
|
|
std::thread(ctrl_thread).detach();
|
2022-01-15 19:05:32 +00:00
|
|
|
if(!logger) logger = std::make_unique<Logger>(HUDElements.params);
|
2021-09-06 06:21:10 +00:00
|
|
|
// Main loop
|
2022-01-06 09:57:18 +00:00
|
|
|
while (!glfwWindowShouldClose(window)){
|
2022-01-20 00:24:01 +00:00
|
|
|
if (!params->no_display){
|
2022-01-06 09:57:18 +00:00
|
|
|
if (mangoapp_paused){
|
|
|
|
window = init(window, glsl_version);
|
2022-01-20 00:24:01 +00:00
|
|
|
create_fonts(*params, sw_stats.font1, sw_stats.font_text);
|
|
|
|
HUDElements.convert_colors(*params);
|
2022-01-06 09:57:18 +00:00
|
|
|
mangoapp_paused = false;
|
|
|
|
}
|
|
|
|
// Start the Dear ImGui frame
|
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
|
|
ImGui_ImplGlfw_NewFrame();
|
2022-01-07 10:00:49 +00:00
|
|
|
window_size.x = window_size.x * 1.2;
|
2022-01-06 09:57:18 +00:00
|
|
|
ImGui::NewFrame();
|
|
|
|
{
|
2022-01-20 00:24:01 +00:00
|
|
|
check_keybinds(sw_stats, *params, vendorID);
|
|
|
|
position_layer(sw_stats, *params, window_size);
|
|
|
|
render_imgui(sw_stats, *params, window_size, true);
|
|
|
|
if (params->control >= 0) {
|
|
|
|
control_client_check(device_data);
|
|
|
|
process_control_socket(device_data->instance);
|
|
|
|
}
|
2022-01-06 09:57:18 +00:00
|
|
|
}
|
|
|
|
ImGui::PopStyleVar(3);
|
|
|
|
|
|
|
|
// Rendering
|
|
|
|
ImGui::Render();
|
2022-01-07 17:42:06 +00:00
|
|
|
static int display_w, display_h;
|
2022-01-19 04:29:13 +00:00
|
|
|
glfwSetWindowSize(window, window_size.x + 45.f, window_size.y + 325.f);
|
2022-01-06 09:57:18 +00:00
|
|
|
glfwGetFramebufferSize(window, &display_w, &display_h);
|
|
|
|
glEnable(GL_DEPTH_TEST);
|
|
|
|
glEnable(GL_BLEND);
|
|
|
|
glBlendFunc(GL_SRC_ALPHA, GL_ONE_MINUS_SRC_ALPHA);
|
|
|
|
glClearColor(0, 0, 0, 0);
|
|
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
|
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
|
|
|
|
|
|
|
glfwSwapBuffers(window);
|
|
|
|
} else if (!mangoapp_paused) {
|
|
|
|
shutdown(window);
|
|
|
|
mangoapp_paused = true;
|
|
|
|
std::unique_lock<std::mutex> lk(mangoapp_m);
|
2022-01-20 00:24:01 +00:00
|
|
|
mangoapp_cv.wait(lk, []{return !params->no_display;});
|
2021-09-06 06:21:10 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Cleanup
|
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
|
|
ImGui_ImplGlfw_Shutdown();
|
|
|
|
ImGui::DestroyContext();
|
|
|
|
|
|
|
|
glfwDestroyWindow(window);
|
|
|
|
glfwTerminate();
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|