From 0949a16e6f8aa46182cefada7c0a2395206651f7 Mon Sep 17 00:00:00 2001 From: offtkp Date: Sun, 9 Jul 2023 23:36:27 +0300 Subject: [PATCH] Add initial http server stuff --- .gitmodules | 6 +++ CMakeLists.txt | 2 + include/emulator.hpp | 13 +++++++ src/emulator.cpp | 90 +++++++++++++++++++++++++++++++++++++++++--- third_party/httplib | 1 + third_party/stb | 1 + 6 files changed, 107 insertions(+), 6 deletions(-) create mode 160000 third_party/httplib create mode 160000 third_party/stb diff --git a/.gitmodules b/.gitmodules index f8c67366..a2cac3f2 100644 --- a/.gitmodules +++ b/.gitmodules @@ -19,3 +19,9 @@ [submodule "third_party/toml11"] path = third_party/toml11 url = https://github.com/ToruNiina/toml11 +[submodule "cpp-httplib"] + path = third_party/httplib + url = https://github.com/yhirose/cpp-httplib +[submodule "stb"] + path = third_party/stb + url = https://github.com/nothings/stb diff --git a/CMakeLists.txt b/CMakeLists.txt index bea18041..bfbf8ae9 100644 --- a/CMakeLists.txt +++ b/CMakeLists.txt @@ -34,6 +34,8 @@ include_directories(third_party/cryptopp/) include_directories(third_party/cityhash/include) include_directories(third_party/result/include) include_directories(third_party/xxhash/include) +include_directories(third_party/httplib) +include_directories(third_party/stb) add_compile_definitions(NOMINMAX) add_compile_definitions(SDL_MAIN_HANDLED) diff --git a/include/emulator.hpp b/include/emulator.hpp index e1684085..63fe468c 100644 --- a/include/emulator.hpp +++ b/include/emulator.hpp @@ -17,6 +17,8 @@ enum class ROMType { None, ELF, NCSD, CXI }; +enum class HttpAction { None, Screenshot }; + class Emulator { CPU cpu; GPU gpu; @@ -46,6 +48,12 @@ class Emulator { ROMType romType = ROMType::None; bool running = true; +#ifdef PANDA3DS_ENABLE_HTTP_SERVER + std::atomic_bool pendingAction = false; + HttpAction action = HttpAction::None; + std::mutex actionMutex = {}; +#endif + // Keep the handle for the ROM here to reload when necessary and to prevent deleting it // This is currently only used for ELFs, NCSDs use the IOFile API instead std::ifstream loadedELF; @@ -62,10 +70,15 @@ class Emulator { void reset(); void run(); void runFrame(); + void screenshot(const std::string& name); bool loadROM(const std::filesystem::path& path); bool loadNCSD(const std::filesystem::path& path, ROMType type); bool loadELF(const std::filesystem::path& path); bool loadELF(std::ifstream& file); void initGraphicsContext(); + +#ifdef PANDA3DS_ENABLE_HTTP_SERVER + void startHttpServer(); +#endif }; diff --git a/src/emulator.cpp b/src/emulator.cpp index 198c573d..2981505e 100644 --- a/src/emulator.cpp +++ b/src/emulator.cpp @@ -1,4 +1,9 @@ #include "emulator.hpp" +#ifdef PANDA3DS_ENABLE_HTTP_SERVER +#include +#endif +#define STB_IMAGE_WRITE_IMPLEMENTATION +#include #ifdef _WIN32 #include @@ -10,6 +15,10 @@ _declspec(dllexport) DWORD AmdPowerXpressRequestHighPerformance = 1; } #endif +#ifdef PANDA3DS_ENABLE_HTTP_SERVER +constexpr const char* httpServerScreenshotPath = "screenshot.png"; +#endif + Emulator::Emulator() : kernel(cpu, memory, gpu), cpu(memory, kernel), gpu(memory, gl, config), memory(cpu.getTicksRef()) { if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_EVENTS) < 0) { Helpers::panic("Failed to initialize SDL2"); @@ -67,7 +76,7 @@ void Emulator::reset() { // Reloading r13 and r15 needs to happen after everything has been reset // Otherwise resetting the kernel or cpu might nuke them cpu.setReg(13, VirtualAddrs::StackTop); // Set initial SP - + // If a ROM is active and we reset, reload it. This is necessary to set up stack, executable memory, .data/.rodata/.bss all over again if (romType != ROMType::None && romPath.has_value()) { bool success = loadROM(romPath.value()); @@ -80,13 +89,55 @@ void Emulator::reset() { } } +void Emulator::screenshot(const std::string& name) { + std::vector pixels, flippedPixels; + pixels.resize(width * height * 4); + flippedPixels.resize(width * height * 4); + + glReadPixels(0, 0, width, height, GL_BGRA, GL_UNSIGNED_BYTE, pixels.data()); + + // Flip the image vertically + for (int y = 0; y < height; y++) { + memcpy(&flippedPixels[y * width * 4], &pixels[(height - y - 1) * width * 4], width * 4); + // Swap R and B channels + for (int x = 0; x < width; x++) { + std::swap(flippedPixels[y * width * 4 + x * 4 + 0], flippedPixels[y * width * 4 + x * 4 + 2]); + // Set alpha to 0xFF + flippedPixels[y * width * 4 + x * 4 + 3] = 0xFF; + } + } + + stbi_write_png(name.c_str(), width, height, 4, flippedPixels.data(), 0); +} + void Emulator::step() {} void Emulator::render() {} void Emulator::run() { - while (running) { - runFrame(); // Run 1 frame of instructions - gpu.display(); // Display graphics +#ifdef PANDA3DS_ENABLE_HTTP_SERVER + startHttpServer(); +#endif + while (running) { +#ifdef PANDA3DS_ENABLE_HTTP_SERVER + { + std::scoped_lock lock(actionMutex); + if (pendingAction) { + switch (action) { + case HttpAction::Screenshot: { + screenshot(httpServerScreenshotPath); + break; + } + case HttpAction::None: { + break; + } + } + pendingAction = false; + pendingAction.notify_all(); + } + } +#endif + runFrame(); // Run 1 frame of instructions + gpu.display(); // Display graphics ServiceManager& srv = kernel.getServiceManager(); @@ -337,12 +388,12 @@ bool Emulator::loadROM(const std::filesystem::path& path) { romPath = std::nullopt; romType = ROMType::None; } - + return success; } // Used for loading both CXI and NCSD files since they are both so similar and use the same interface -// (We promote CXI files to NCSD internally for ease) +// (We promote CXI files to NCSD internally for ease) bool Emulator::loadNCSD(const std::filesystem::path& path, ROMType type) { romType = type; std::optional opt = (type == ROMType::NCSD) ? memory.loadNCSD(aesEngine, path) : memory.loadCXI(aesEngine, path); @@ -390,3 +441,30 @@ void Emulator::initGraphicsContext() { gl.reset(); // TODO (For when we have multiple backends): Only do this if we are using OpenGL gpu.initGraphicsContext(); } + +#ifdef PANDA3DS_ENABLE_HTTP_SERVER +void Emulator::startHttpServer() { + std::thread http_thread([this]() { + httplib::Server server; + server.Get("/ping", [](const httplib::Request&, httplib::Response& response) { + response.set_content("pong", "text/plain"); + }); + server.Get("/screen", [this](const httplib::Request&, httplib::Response& response) { + { + std::scoped_lock lock(actionMutex); + pendingAction = true; + action = HttpAction::Screenshot; + } + // wait until the screenshot is ready + pendingAction.wait(true); + std::ifstream image(httpServerScreenshotPath, std::ios::binary); + std::vector buffer(std::istreambuf_iterator(image), {}); + response.set_content(buffer.data(), buffer.size(), "image/png"); + }); + // TODO: ability to specify host and port + printf("Starting HTTP server on port 1234\n"); + server.listen("localhost", 1234); + }); + http_thread.detach(); +} +#endif \ No newline at end of file diff --git a/third_party/httplib b/third_party/httplib new file mode 160000 index 00000000..be07d2d7 --- /dev/null +++ b/third_party/httplib @@ -0,0 +1 @@ +Subproject commit be07d2d7a99c0a54b00526f30f175e93c3588f34 diff --git a/third_party/stb b/third_party/stb new file mode 160000 index 00000000..5736b15f --- /dev/null +++ b/third_party/stb @@ -0,0 +1 @@ +Subproject commit 5736b15f7ea0ffb08dd38af21067c314d6a3aae9