mirror of
https://github.com/wheremyfoodat/Panda3DS.git
synced 2025-04-06 22:25:41 +12:00
Initial implementation of ports and sessions
This commit is contained in:
parent
aa643c44bb
commit
0e66af8dae
7 changed files with 100 additions and 17 deletions
|
@ -48,7 +48,7 @@ set(SOURCE_FILES src/main.cpp src/emulator.cpp src/core/CPU/cpu_dynarmic.cpp src
|
||||||
|
|
||||||
)
|
)
|
||||||
set(KERNEL_SOURCE_FILES src/core/kernel/kernel.cpp src/core/kernel/resource_limits.cpp
|
set(KERNEL_SOURCE_FILES src/core/kernel/kernel.cpp src/core/kernel/resource_limits.cpp
|
||||||
src/core/kernel/memory_management.cpp
|
src/core/kernel/memory_management.cpp src/core/kernel/ports.cpp
|
||||||
)
|
)
|
||||||
|
|
||||||
set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/opengl.hpp include/termcolor.hpp
|
set(HEADER_FILES include/emulator.hpp include/helpers.hpp include/opengl.hpp include/termcolor.hpp
|
||||||
|
|
|
@ -130,7 +130,6 @@ public:
|
||||||
void runFrame() {
|
void runFrame() {
|
||||||
env.ticksLeft = 268111856 / 60;
|
env.ticksLeft = 268111856 / 60;
|
||||||
const auto exitReason = jit->Run();
|
const auto exitReason = jit->Run();
|
||||||
for (u32 i = 0; i < 4; i++)printf("r%d: %08X\n", i, getReg(i));
|
|
||||||
Helpers::panic("Exit reason: %d\nPC: %08X", (u32)exitReason, getReg(15));
|
Helpers::panic("Exit reason: %d\nPC: %08X", (u32)exitReason, getReg(15));
|
||||||
}
|
}
|
||||||
};
|
};
|
|
@ -14,6 +14,7 @@ class Kernel {
|
||||||
// The handle number for the next kernel object to be created
|
// The handle number for the next kernel object to be created
|
||||||
u32 handleCounter;
|
u32 handleCounter;
|
||||||
std::vector<KernelObject> objects;
|
std::vector<KernelObject> objects;
|
||||||
|
std::vector<Handle> portHandles;
|
||||||
|
|
||||||
u32 currentProcess;
|
u32 currentProcess;
|
||||||
|
|
||||||
|
@ -47,6 +48,10 @@ class Kernel {
|
||||||
}
|
}
|
||||||
|
|
||||||
Handle makeProcess();
|
Handle makeProcess();
|
||||||
|
Handle makePort(const char* name);
|
||||||
|
std::optional<Handle> getPortHandle(const char* name);
|
||||||
|
void deleteObjectData(KernelObject& object);
|
||||||
|
|
||||||
KernelObject* getProcessFromPID(Handle handle);
|
KernelObject* getProcessFromPID(Handle handle);
|
||||||
s32 getCurrentResourceValue(const KernelObject* limit, u32 resourceName);
|
s32 getCurrentResourceValue(const KernelObject* limit, u32 resourceName);
|
||||||
u32 getMaxForResource(const KernelObject* limit, u32 resourceName);
|
u32 getMaxForResource(const KernelObject* limit, u32 resourceName);
|
||||||
|
@ -64,6 +69,7 @@ class Kernel {
|
||||||
public:
|
public:
|
||||||
Kernel(std::array<u32, 16>& regs, Memory& mem) : regs(regs), mem(mem), handleCounter(0) {
|
Kernel(std::array<u32, 16>& regs, Memory& mem) : regs(regs), mem(mem), handleCounter(0) {
|
||||||
objects.reserve(512); // Make room for a few objects to avoid further memory allocs later
|
objects.reserve(512); // Make room for a few objects to avoid further memory allocs later
|
||||||
|
portHandles.reserve(32);
|
||||||
}
|
}
|
||||||
void serviceSVC(u32 svc);
|
void serviceSVC(u32 svc);
|
||||||
void reset();
|
void reset();
|
||||||
|
|
|
@ -5,9 +5,11 @@ namespace SVCResult {
|
||||||
enum : u32 {
|
enum : u32 {
|
||||||
Success = 0,
|
Success = 0,
|
||||||
Failure = 0xFFFFFFFF,
|
Failure = 0xFFFFFFFF,
|
||||||
|
ObjectNotFound = 0xD88007FA,
|
||||||
// Different calls return a different value
|
// Different calls return a different value
|
||||||
BadHandle = 0xD8E007F7,
|
BadHandle = 0xD8E007F7,
|
||||||
BadHandleAlt = 0xD9001BF7
|
BadHandleAlt = 0xD9001BF7,
|
||||||
|
PortNameTooLong = 0xE0E0181E
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -19,7 +21,7 @@ namespace KernelHandles {
|
||||||
}
|
}
|
||||||
|
|
||||||
enum class KernelObjectType : u8 {
|
enum class KernelObjectType : u8 {
|
||||||
ResourceLimit, Process
|
Port, Process, ResourceLimit, Session, Dummy
|
||||||
};
|
};
|
||||||
|
|
||||||
enum class ResourceLimitCategory : int {
|
enum class ResourceLimitCategory : int {
|
||||||
|
@ -55,10 +57,24 @@ struct ProcessData {
|
||||||
ResourceLimits limits;
|
ResourceLimits limits;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
struct PortData {
|
||||||
|
static constexpr u32 maxNameLen = 11;
|
||||||
|
|
||||||
|
char name[maxNameLen + 1] = {};
|
||||||
|
bool isPublic = false; // Setting name=NULL creates a private port not accessible from svcConnectToPort.
|
||||||
|
};
|
||||||
|
|
||||||
|
struct SessionData {
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
static const char* kernelObjectTypeToString(KernelObjectType t) {
|
static const char* kernelObjectTypeToString(KernelObjectType t) {
|
||||||
switch (t) {
|
switch (t) {
|
||||||
|
case KernelObjectType::Port: return "port";
|
||||||
case KernelObjectType::Process: return "process";
|
case KernelObjectType::Process: return "process";
|
||||||
case KernelObjectType::ResourceLimit: return "resource limit";
|
case KernelObjectType::ResourceLimit: return "resource limit";
|
||||||
|
case KernelObjectType::Session: return "session";
|
||||||
|
case KernelObjectType::Dummy: return "dummy";
|
||||||
default: return "unknown";
|
default: return "unknown";
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -39,18 +39,32 @@ KernelObject* Kernel::getProcessFromPID(Handle handle) {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
void Kernel::deleteObjectData(KernelObject& object) {
|
||||||
|
// Resource limit and dummy objects do not allocate heap data, so we don't delete anything
|
||||||
|
if (object.type == KernelObjectType::ResourceLimit || object.type == KernelObjectType::Dummy) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (object.data != nullptr) {
|
||||||
|
delete object.data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
void Kernel::reset() {
|
void Kernel::reset() {
|
||||||
handleCounter = 0;
|
handleCounter = 0;
|
||||||
|
|
||||||
for (auto& object : objects) {
|
for (auto& object : objects) {
|
||||||
if (object.data != nullptr) {
|
deleteObjectData(object);
|
||||||
delete object.data;
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
objects.clear();
|
objects.clear();
|
||||||
|
portHandles.clear();
|
||||||
|
|
||||||
// Make a main process object
|
// Allocate handle #0 to a dummy object and make a main process object
|
||||||
|
makeObject(KernelObjectType::Dummy);
|
||||||
currentProcess = makeProcess();
|
currentProcess = makeProcess();
|
||||||
|
|
||||||
|
// Create global service manager port
|
||||||
|
makePort("srv:");
|
||||||
}
|
}
|
||||||
|
|
||||||
// Result CreateAddressArbiter(Handle* arbiter)
|
// Result CreateAddressArbiter(Handle* arbiter)
|
||||||
|
@ -65,14 +79,6 @@ void Kernel::svcCloseHandle() {
|
||||||
regs[0] = SVCResult::Success;
|
regs[0] = SVCResult::Success;
|
||||||
}
|
}
|
||||||
|
|
||||||
void Kernel::connectToPort() {
|
|
||||||
const u32 handlePointer = regs[0];
|
|
||||||
const char* port = static_cast<const char*>(mem.getReadPointer(regs[1]));
|
|
||||||
|
|
||||||
printf("ConnectToPort(handle pointer = %08X, port = \"%s\")\n", handlePointer, port);
|
|
||||||
Helpers::panic("Unimplemented IPC");
|
|
||||||
}
|
|
||||||
|
|
||||||
std::string Kernel::getProcessName(u32 pid) {
|
std::string Kernel::getProcessName(u32 pid) {
|
||||||
if (pid == KernelHandles::CurrentProcess) {
|
if (pid == KernelHandles::CurrentProcess) {
|
||||||
return "current";
|
return "current";
|
||||||
|
|
57
src/core/kernel/ports.cpp
Normal file
57
src/core/kernel/ports.cpp
Normal file
|
@ -0,0 +1,57 @@
|
||||||
|
#include "kernel.hpp"
|
||||||
|
#include <cstring>
|
||||||
|
|
||||||
|
Handle Kernel::makePort(const char* name) {
|
||||||
|
Handle ret = makeObject(KernelObjectType::Port);
|
||||||
|
portHandles.push_back(ret); // Push the port handle to our cache of port handles
|
||||||
|
|
||||||
|
objects[ret].data = new PortData();
|
||||||
|
const auto data = static_cast<PortData*>(objects[ret].data);
|
||||||
|
|
||||||
|
// If the name is empty (ie the first char is the null terminator) then the port is private
|
||||||
|
data->isPublic = name[0] != '\0';
|
||||||
|
std::strncpy(data->name, name, PortData::maxNameLen);
|
||||||
|
|
||||||
|
// printf("Created %s port \"%s\" with handle %d\n", data->isPublic ? "public" : "private", data->name, ret);
|
||||||
|
return ret;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Get the handle of a port based on its name
|
||||||
|
// If there's no such port, return nullopt
|
||||||
|
std::optional<Handle> Kernel::getPortHandle(const char* name) {
|
||||||
|
for (auto handle : portHandles) {
|
||||||
|
const auto data = static_cast<PortData*>(objects[handle].data);
|
||||||
|
if (std::strncmp(name, data->name, PortData::maxNameLen) == 0) {
|
||||||
|
return handle;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return std::nullopt;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Result ConnectToPort(Handle* out, const char* portName)
|
||||||
|
void Kernel::connectToPort() {
|
||||||
|
const u32 handlePointer = regs[0];
|
||||||
|
const char* port = static_cast<const char*>(mem.getReadPointer(regs[1]));
|
||||||
|
printf("ConnectToPort(handle pointer = %08X, port = \"%s\")\n", handlePointer, port);
|
||||||
|
|
||||||
|
// TODO: This is unsafe
|
||||||
|
if (std::strlen(port) > PortData::maxNameLen) {
|
||||||
|
Helpers::panic("ConnectToPort: Port name too long\n");
|
||||||
|
regs[0] = SVCResult::PortNameTooLong;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const auto portHandle = getPortHandle(port);
|
||||||
|
if (!portHandle.has_value()) {
|
||||||
|
Helpers::panic("ConnectToPort: Port doesn't exist\n");
|
||||||
|
regs[0] = SVCResult::ObjectNotFound;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// TODO: Actually create session
|
||||||
|
Handle sessionHandle = makeObject(KernelObjectType::Session);
|
||||||
|
|
||||||
|
regs[0] = SVCResult::Success;
|
||||||
|
regs[1] = sessionHandle;
|
||||||
|
}
|
|
@ -2,7 +2,6 @@
|
||||||
#include "config_mem.hpp"
|
#include "config_mem.hpp"
|
||||||
#include <cassert>
|
#include <cassert>
|
||||||
|
|
||||||
|
|
||||||
Memory::Memory() {
|
Memory::Memory() {
|
||||||
fcram = new uint8_t[FCRAM_SIZE]();
|
fcram = new uint8_t[FCRAM_SIZE]();
|
||||||
readTable.resize(totalPageCount, 0);
|
readTable.resize(totalPageCount, 0);
|
||||||
|
|
Loading…
Add table
Reference in a new issue