492e6b59cd
Cierra 0049b. El context de fn::run_app pide ahora GL 4.3 core con forward-compat global, habilitando compute shaders, SSBOs, image load/store y atomic counters — bloques esenciales del graph_renderer GPU del proyecto osint_graph (issues 0049f y 0049h). Cambios: - cpp/framework/app_base.cpp: 4.3 core + forward-compat. Comentario marcando que es backward-compatible con shaders #version 330. - cpp/apps/primitives_gallery/capture.cpp: deja explicitamente 3.3 core porque WSL Mesa no entrega 4.3 offscreen (GLXBadFBConfig); ImGui + ImPlot funcionan igual en 3.3 para los goldens. - primitives_gallery: nuevo demo Gfx > gl_info que muestra Vendor/Renderer/Version/GLSL en runtime + status 4.3 (verde) + limites (MAX_TEXTURE_SIZE, MAX_VERTEX_ATTRIBS, MAX_UNIFORM_BLOCK_SIZE y, si 4.3+, MAX_SHADER_STORAGE_BUFFER_BINDINGS y compute shared mem). Solo glGetString/glGetIntegerv — sin loader extra. - About bumped a 0.4.0 con la nota del nuevo demo y de GL 4.3. - cpp/tests/test_visual.cpp: usa LIBGL_ALWAYS_SOFTWARE=1 al lanzar el capture para alinear el driver con update_goldens.sh; sin esto las diferencias de strings (llvmpipe vs d3d12) hacen que gl_info supere el 1% de tolerancia. - cpp/tests/golden/gl_info.png: nuevo golden. Build verificado en Linux (cmake build OK) + Windows cross-compile (cmake build OK). Las 27 pruebas pasan (incluida test_visual con 42 demos comparadas).
174 lines
5.8 KiB
C++
174 lines
5.8 KiB
C++
// Implementacion de gallery::run_capture — render offscreen + glReadPixels +
|
|
// PNG via stb_image_write. Ver capture.h.
|
|
|
|
#include "capture.h"
|
|
|
|
#include "imgui.h"
|
|
#include "imgui_impl_glfw.h"
|
|
#include "imgui_impl_opengl3.h"
|
|
#include "implot.h"
|
|
#include "implot3d.h"
|
|
#include "core/tokens.h"
|
|
#include "core/icon_font.h"
|
|
#include "core/app_settings.h"
|
|
#include "gfx/gl_loader.h"
|
|
|
|
#include <GLFW/glfw3.h>
|
|
|
|
#define STB_IMAGE_WRITE_IMPLEMENTATION
|
|
#include "stb_image_write.h"
|
|
|
|
#include <cstdio>
|
|
#include <vector>
|
|
|
|
namespace gallery {
|
|
|
|
static void glfw_capture_error(int error, const char* description) {
|
|
std::fprintf(stderr, "GLFW Error %d: %s\n", error, description);
|
|
}
|
|
|
|
// Flip vertical in-place: OpenGL origin = bottom-left, PNG = top-left.
|
|
static void flip_vertical_rgba(unsigned char* px, int w, int h) {
|
|
const int stride = w * 4;
|
|
std::vector<unsigned char> row(stride);
|
|
for (int y = 0; y < h / 2; ++y) {
|
|
unsigned char* a = px + y * stride;
|
|
unsigned char* b = px + (h - 1 - y) * stride;
|
|
std::copy(a, a + stride, row.begin());
|
|
std::copy(b, b + stride, a);
|
|
std::copy(row.begin(), row.end(), b);
|
|
}
|
|
}
|
|
|
|
bool run_capture(const CaptureConfig& cfg, const std::vector<CaptureItem>& items) {
|
|
glfwSetErrorCallback(&glfw_capture_error);
|
|
if (!glfwInit()) {
|
|
std::fprintf(stderr, "capture: glfwInit failed\n");
|
|
return false;
|
|
}
|
|
|
|
// Capture mode usa GL 3.3 deliberadamente: WSL Mesa no entrega contexto
|
|
// 4.3 offscreen (GLXBadFBConfig). Las pruebas visuales no necesitan
|
|
// compute/SSBO — ImGui+ImPlot funciona en 3.3 core. La build interactiva
|
|
// (app_base.cpp) si pide 4.3.
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MAJOR, 3);
|
|
glfwWindowHint(GLFW_CONTEXT_VERSION_MINOR, 3);
|
|
glfwWindowHint(GLFW_OPENGL_PROFILE, GLFW_OPENGL_CORE_PROFILE);
|
|
glfwWindowHint(GLFW_VISIBLE, GLFW_FALSE);
|
|
#ifdef __APPLE__
|
|
glfwWindowHint(GLFW_OPENGL_FORWARD_COMPAT, GL_TRUE);
|
|
#endif
|
|
|
|
GLFWwindow* window = glfwCreateWindow(
|
|
cfg.capture_w, cfg.capture_h, "capture", nullptr, nullptr);
|
|
if (!window) {
|
|
std::fprintf(stderr, "capture: glfwCreateWindow failed (no GL?)\n");
|
|
glfwTerminate();
|
|
return false;
|
|
}
|
|
|
|
glfwMakeContextCurrent(window);
|
|
glfwSwapInterval(0);
|
|
|
|
if (!fn::gfx::gl_loader_init()) {
|
|
std::fprintf(stderr, "capture: gl_loader_init failed\n");
|
|
glfwDestroyWindow(window);
|
|
glfwTerminate();
|
|
return false;
|
|
}
|
|
|
|
IMGUI_CHECKVERSION();
|
|
ImGui::CreateContext();
|
|
ImPlot::CreateContext();
|
|
ImPlot3D::CreateContext();
|
|
|
|
ImGuiIO& io = ImGui::GetIO();
|
|
io.IniFilename = nullptr; // no .ini side effects in capture mode.
|
|
io.DisplaySize = ImVec2((float)cfg.capture_w, (float)cfg.capture_h);
|
|
|
|
fn_ui::settings_load();
|
|
fn_ui::load_fonts_from_settings();
|
|
{
|
|
ImGuiStyle& style = ImGui::GetStyle();
|
|
style.FontSizeBase = fn_ui::settings().font_size_px;
|
|
style._NextFrameFontSizeBase = style.FontSizeBase;
|
|
}
|
|
|
|
fn_tokens::apply_dark_theme();
|
|
|
|
ImGui_ImplGlfw_InitForOpenGL(window, false);
|
|
ImGui_ImplOpenGL3_Init("#version 330");
|
|
|
|
bool ok_all = true;
|
|
std::vector<unsigned char> pixels((size_t)cfg.capture_w * cfg.capture_h * 4u);
|
|
|
|
for (const auto& item : items) {
|
|
// Warmup: rinde varios frames para que ImGui/ImPlot estabilicen layout
|
|
// (el primer frame frecuentemente carece de mediciones de tamaño).
|
|
for (int frame = 0; frame < cfg.warmup_frames + 1; ++frame) {
|
|
glfwPollEvents();
|
|
|
|
ImGui_ImplOpenGL3_NewFrame();
|
|
ImGui_ImplGlfw_NewFrame();
|
|
ImGui::NewFrame();
|
|
|
|
// Ventana fullscreen sobre el viewport con la demo activa,
|
|
// sin sidebar (queremos el render del primitivo lo mas limpio
|
|
// posible para el diff visual).
|
|
const ImGuiViewport* vp = ImGui::GetMainViewport();
|
|
ImGui::SetNextWindowPos(vp->WorkPos);
|
|
ImGui::SetNextWindowSize(vp->WorkSize);
|
|
ImGui::Begin("##capture_root",
|
|
nullptr,
|
|
ImGuiWindowFlags_NoTitleBar | ImGuiWindowFlags_NoResize |
|
|
ImGuiWindowFlags_NoMove | ImGuiWindowFlags_NoCollapse |
|
|
ImGuiWindowFlags_NoBringToFrontOnFocus |
|
|
ImGuiWindowFlags_NoSavedSettings);
|
|
if (item.fn) item.fn();
|
|
ImGui::End();
|
|
|
|
ImGui::Render();
|
|
int dw, dh;
|
|
glfwGetFramebufferSize(window, &dw, &dh);
|
|
glViewport(0, 0, dw, dh);
|
|
glClearColor(fn_tokens::colors::bg.x,
|
|
fn_tokens::colors::bg.y,
|
|
fn_tokens::colors::bg.z, 1.0f);
|
|
glClear(GL_COLOR_BUFFER_BIT);
|
|
ImGui_ImplOpenGL3_RenderDrawData(ImGui::GetDrawData());
|
|
glfwSwapBuffers(window);
|
|
}
|
|
|
|
// Read framebuffer (GL_RGBA / GL_UNSIGNED_BYTE).
|
|
glPixelStorei(GL_PACK_ALIGNMENT, 1);
|
|
glReadPixels(0, 0, cfg.capture_w, cfg.capture_h,
|
|
GL_RGBA, GL_UNSIGNED_BYTE, pixels.data());
|
|
|
|
flip_vertical_rgba(pixels.data(), cfg.capture_w, cfg.capture_h);
|
|
|
|
char path[1024];
|
|
std::snprintf(path, sizeof(path), "%s/%s.png",
|
|
cfg.output_dir.c_str(), item.id.c_str());
|
|
const int rc = stbi_write_png(
|
|
path, cfg.capture_w, cfg.capture_h, 4,
|
|
pixels.data(), cfg.capture_w * 4);
|
|
if (rc == 0) {
|
|
std::fprintf(stderr, "capture: stbi_write_png failed for %s\n", path);
|
|
ok_all = false;
|
|
} else {
|
|
std::fprintf(stdout, "captured: %s\n", path);
|
|
}
|
|
}
|
|
|
|
ImGui_ImplOpenGL3_Shutdown();
|
|
ImGui_ImplGlfw_Shutdown();
|
|
ImPlot3D::DestroyContext();
|
|
ImPlot::DestroyContext();
|
|
ImGui::DestroyContext();
|
|
glfwDestroyWindow(window);
|
|
glfwTerminate();
|
|
return ok_all;
|
|
}
|
|
|
|
} // namespace gallery
|