Add initial http server stuff

This commit is contained in:
offtkp 2023-07-09 23:36:27 +03:00
parent eb7b056e28
commit 0949a16e6f
6 changed files with 107 additions and 6 deletions

6
.gitmodules vendored
View file

@ -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

View file

@ -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)

View file

@ -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
};

View file

@ -1,4 +1,9 @@
#include "emulator.hpp"
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
#include <httplib.h>
#endif
#define STB_IMAGE_WRITE_IMPLEMENTATION
#include <stb_image_write.h>
#ifdef _WIN32
#include <windows.h>
@ -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<uint8_t> 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<NCSD> 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<char> buffer(std::istreambuf_iterator<char>(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

1
third_party/httplib vendored Submodule

@ -0,0 +1 @@
Subproject commit be07d2d7a99c0a54b00526f30f175e93c3588f34

1
third_party/stb vendored Submodule

@ -0,0 +1 @@
Subproject commit 5736b15f7ea0ffb08dd38af21067c314d6a3aae9