mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 06:05:40 +12:00
Add initial http server stuff
This commit is contained in:
parent
eb7b056e28
commit
0949a16e6f
6 changed files with 107 additions and 6 deletions
6
.gitmodules
vendored
6
.gitmodules
vendored
|
@ -19,3 +19,9 @@
|
||||||
[submodule "third_party/toml11"]
|
[submodule "third_party/toml11"]
|
||||||
path = third_party/toml11
|
path = third_party/toml11
|
||||||
url = https://github.com/ToruNiina/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
|
||||||
|
|
|
@ -34,6 +34,8 @@ include_directories(third_party/cryptopp/)
|
||||||
include_directories(third_party/cityhash/include)
|
include_directories(third_party/cityhash/include)
|
||||||
include_directories(third_party/result/include)
|
include_directories(third_party/result/include)
|
||||||
include_directories(third_party/xxhash/include)
|
include_directories(third_party/xxhash/include)
|
||||||
|
include_directories(third_party/httplib)
|
||||||
|
include_directories(third_party/stb)
|
||||||
|
|
||||||
add_compile_definitions(NOMINMAX)
|
add_compile_definitions(NOMINMAX)
|
||||||
add_compile_definitions(SDL_MAIN_HANDLED)
|
add_compile_definitions(SDL_MAIN_HANDLED)
|
||||||
|
|
|
@ -17,6 +17,8 @@
|
||||||
|
|
||||||
enum class ROMType { None, ELF, NCSD, CXI };
|
enum class ROMType { None, ELF, NCSD, CXI };
|
||||||
|
|
||||||
|
enum class HttpAction { None, Screenshot };
|
||||||
|
|
||||||
class Emulator {
|
class Emulator {
|
||||||
CPU cpu;
|
CPU cpu;
|
||||||
GPU gpu;
|
GPU gpu;
|
||||||
|
@ -46,6 +48,12 @@ class Emulator {
|
||||||
ROMType romType = ROMType::None;
|
ROMType romType = ROMType::None;
|
||||||
bool running = true;
|
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
|
// 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
|
// This is currently only used for ELFs, NCSDs use the IOFile API instead
|
||||||
std::ifstream loadedELF;
|
std::ifstream loadedELF;
|
||||||
|
@ -62,10 +70,15 @@ class Emulator {
|
||||||
void reset();
|
void reset();
|
||||||
void run();
|
void run();
|
||||||
void runFrame();
|
void runFrame();
|
||||||
|
void screenshot(const std::string& name);
|
||||||
|
|
||||||
bool loadROM(const std::filesystem::path& path);
|
bool loadROM(const std::filesystem::path& path);
|
||||||
bool loadNCSD(const std::filesystem::path& path, ROMType type);
|
bool loadNCSD(const std::filesystem::path& path, ROMType type);
|
||||||
bool loadELF(const std::filesystem::path& path);
|
bool loadELF(const std::filesystem::path& path);
|
||||||
bool loadELF(std::ifstream& file);
|
bool loadELF(std::ifstream& file);
|
||||||
void initGraphicsContext();
|
void initGraphicsContext();
|
||||||
|
|
||||||
|
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
|
||||||
|
void startHttpServer();
|
||||||
|
#endif
|
||||||
};
|
};
|
||||||
|
|
|
@ -1,4 +1,9 @@
|
||||||
#include "emulator.hpp"
|
#include "emulator.hpp"
|
||||||
|
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
|
||||||
|
#include <httplib.h>
|
||||||
|
#endif
|
||||||
|
#define STB_IMAGE_WRITE_IMPLEMENTATION
|
||||||
|
#include <stb_image_write.h>
|
||||||
|
|
||||||
#ifdef _WIN32
|
#ifdef _WIN32
|
||||||
#include <windows.h>
|
#include <windows.h>
|
||||||
|
@ -10,6 +15,10 @@ _declspec(dllexport) DWORD AmdPowerXpressRequestHighPerformance = 1;
|
||||||
}
|
}
|
||||||
#endif
|
#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()) {
|
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) {
|
if (SDL_Init(SDL_INIT_VIDEO | SDL_INIT_EVENTS) < 0) {
|
||||||
Helpers::panic("Failed to initialize SDL2");
|
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
|
// Reloading r13 and r15 needs to happen after everything has been reset
|
||||||
// Otherwise resetting the kernel or cpu might nuke them
|
// Otherwise resetting the kernel or cpu might nuke them
|
||||||
cpu.setReg(13, VirtualAddrs::StackTop); // Set initial SP
|
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 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()) {
|
if (romType != ROMType::None && romPath.has_value()) {
|
||||||
bool success = loadROM(romPath.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::step() {}
|
||||||
void Emulator::render() {}
|
void Emulator::render() {}
|
||||||
|
|
||||||
void Emulator::run() {
|
void Emulator::run() {
|
||||||
while (running) {
|
#ifdef PANDA3DS_ENABLE_HTTP_SERVER
|
||||||
runFrame(); // Run 1 frame of instructions
|
startHttpServer();
|
||||||
gpu.display(); // Display graphics
|
#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();
|
ServiceManager& srv = kernel.getServiceManager();
|
||||||
|
|
||||||
|
@ -337,12 +388,12 @@ bool Emulator::loadROM(const std::filesystem::path& path) {
|
||||||
romPath = std::nullopt;
|
romPath = std::nullopt;
|
||||||
romType = ROMType::None;
|
romType = ROMType::None;
|
||||||
}
|
}
|
||||||
|
|
||||||
return success;
|
return success;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Used for loading both CXI and NCSD files since they are both so similar and use the same interface
|
// 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) {
|
bool Emulator::loadNCSD(const std::filesystem::path& path, ROMType type) {
|
||||||
romType = type;
|
romType = type;
|
||||||
std::optional<NCSD> opt = (type == ROMType::NCSD) ? memory.loadNCSD(aesEngine, path) : memory.loadCXI(aesEngine, path);
|
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
|
gl.reset(); // TODO (For when we have multiple backends): Only do this if we are using OpenGL
|
||||||
gpu.initGraphicsContext();
|
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
1
third_party/httplib
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit be07d2d7a99c0a54b00526f30f175e93c3588f34
|
1
third_party/stb
vendored
Submodule
1
third_party/stb
vendored
Submodule
|
@ -0,0 +1 @@
|
||||||
|
Subproject commit 5736b15f7ea0ffb08dd38af21067c314d6a3aae9
|
Loading…
Add table
Reference in a new issue