From 45c016d12eb787b6e34fd36aad39a381ba67d5dd Mon Sep 17 00:00:00 2001 From: wheremyfoodat Date: Mon, 19 Sep 2022 01:33:12 +0300 Subject: [PATCH] [Kernel] Hopefully implement the QueryMemory svc remotely correctly --- include/kernel/kernel.hpp | 2 ++ include/memory.hpp | 39 +++++++++++++++++++++++++++ src/core/kernel/kernel.cpp | 1 + src/core/kernel/memory_management.cpp | 22 +++++++++++++++ src/core/memory.cpp | 28 +++++++++++++++++-- src/core/services/gsp_gpu.cpp | 2 +- src/main.cpp | 1 + 7 files changed, 92 insertions(+), 3 deletions(-) diff --git a/include/kernel/kernel.hpp b/include/kernel/kernel.hpp index f8bdb0ee..1461af74 100644 --- a/include/kernel/kernel.hpp +++ b/include/kernel/kernel.hpp @@ -69,6 +69,8 @@ class Kernel { void createAddressArbiter(); void createEvent(); void controlMemory(); + void mapMemoryBlock(); + void queryMemory(); void getResourceLimit(); void getResourceLimitLimitValues(); void getResourceLimitCurrentValues(); diff --git a/include/memory.hpp b/include/memory.hpp index d7e4cb56..58f6e1ec 100644 --- a/include/memory.hpp +++ b/include/memory.hpp @@ -26,11 +26,49 @@ namespace VirtualAddrs { }; } +// Types for svcQueryMemory +namespace KernelMemoryTypes { + // This makes no sense + enum MemoryState : u32 { + Free = 0, + Reserved = 1, + IO = 2, + Static = 3, + Code = 4, + Private = 5, + Shared = 6, + Continuous = 7, + Aliased = 8, + Alias = 9, + AliasCode = 10, + Locked = 11, + + PERMISSION_R = 1 << 0, + PERMISSION_W = 1 << 1, + PERMISSION_X = 1 << 2 + }; + + // I assume this is referring to a single piece of allocated memory? If it's for pages, it makes no sense. + // If it's for multiple allocations, it also makes no sense + struct MemoryInfo { + u32 baseVaddr; // Base process virtual address. TODO: What even is this + u32 size; // Of what? + u32 perms; // Is this referring to a single page or? + u32 state; + + u32 end() { return baseVaddr + size; } + MemoryInfo(u32 baseVaddr, u32 size, u32 perms, u32 state) : baseVaddr(baseVaddr), size(size) + , perms(perms), state(state) {} + }; +} + class Memory { u8* fcram; // Our dynarmic core uses page tables for reads and writes with 4096 byte pages std::vector readTable, writeTable; + std::vector memoryInfo; + static constexpr u32 pageShift = 12; static constexpr u32 pageSize = 1 << pageShift; static constexpr u32 pageMask = pageSize - 1; @@ -76,4 +114,5 @@ public: } std::string readString(u32 vaddr, u32 maxCharacters); + KernelMemoryTypes::MemoryInfo queryMemory(u32 vaddr); }; \ No newline at end of file diff --git a/src/core/kernel/kernel.cpp b/src/core/kernel/kernel.cpp index 0614e169..6495125c 100644 --- a/src/core/kernel/kernel.cpp +++ b/src/core/kernel/kernel.cpp @@ -5,6 +5,7 @@ void Kernel::serviceSVC(u32 svc) { switch (svc) { case 0x01: controlMemory(); break; + case 0x02: queryMemory(); break; case 0x17: createEvent(); break; case 0x21: createAddressArbiter(); break; case 0x23: svcCloseHandle(); break; diff --git a/src/core/kernel/memory_management.cpp b/src/core/kernel/memory_management.cpp index e7c3b01a..2e430c0b 100644 --- a/src/core/kernel/memory_management.cpp +++ b/src/core/kernel/memory_management.cpp @@ -67,4 +67,26 @@ void Kernel::controlMemory() { } regs[0] = SVCResult::Success; +} + +// Result QueryMemory(MemoryInfo* memInfo, PageInfo* pageInfo, u32 addr) +void Kernel::queryMemory() { + const u32 memInfo = regs[0]; + const u32 pageInfo = regs[1]; + const u32 addr = regs[2]; + + if (addr & 0xfff) { + Helpers::panic("QueryMemory: Address not page aligned\n"); + } + + printf("QueryMemory(mem info pointer = %08X, page info pointer = %08X, addr = %08X)\n", memInfo, pageInfo, addr); + + const auto info = mem.queryMemory(addr); + regs[0] = SVCResult::Success; + + mem.write32(memInfo, info.baseVaddr); // Set memInfo->baseVaddr + mem.write32(memInfo + 4, info.size); // Set memInfo->size + mem.write32(memInfo + 8, info.baseVaddr); // Set memInfo->perms + mem.write32(memInfo + 12, info.state); // Set memInfo->state + mem.write32(pageInfo, 0); // Set pageInfo->flags to 0 } \ No newline at end of file diff --git a/src/core/memory.cpp b/src/core/memory.cpp index 31146cb8..b497831f 100644 --- a/src/core/memory.cpp +++ b/src/core/memory.cpp @@ -2,14 +2,18 @@ #include "config_mem.hpp" #include +using namespace KernelMemoryTypes; + Memory::Memory() { fcram = new uint8_t[FCRAM_SIZE](); readTable.resize(totalPageCount, 0); writeTable.resize(totalPageCount, 0); + memoryInfo.reserve(32); // Pre-allocate some room for memory allocation info to avoid dynamic allocs } void Memory::reset() { // Unallocate all memory + memoryInfo.clear(); usedFCRAMPages.reset(); usedUserMemory = 0_MB; @@ -76,13 +80,17 @@ std::optional Memory::allocateMemory(u32 vaddr, u32 paddr, u32 size, bool l if (w) { writeTable[virtualPage] = uintptr_t(&fcram[physPage * pageSize]); } - + // Mark FCRAM page as allocated and go on usedFCRAMPages[physPage] = true; virtualPage++; physPage++; } - + + // Back up the info for this allocation in our memoryInfo vector + u32 perms = (r ? PERMISSION_R : 0) | (w ? PERMISSION_W : 0) | (x ? PERMISSION_X : 0); + memoryInfo.push_back(std::move(MemoryInfo(vaddr, size, perms, KernelMemoryTypes::Reserved))); + return vaddr; } @@ -218,4 +226,20 @@ std::string Memory::readString(u32 address, u32 maxSize) { string.shrink_to_fit(); return string; +} + +// The way I understand how the kernel's QueryMemory is supposed to work is that you give it a vaddr +// And the kernel looks up the memory allocations it's performed, finds which one it belongs in and returns its info? +// TODO: Verify this +MemoryInfo Memory::queryMemory(u32 vaddr) { + // Check each allocation + for (auto& alloc : memoryInfo) { + // Check if the memory address belongs in this allocation and return the info if so + if (vaddr >= alloc.baseVaddr && vaddr < alloc.end()) { + return alloc; + } + } + + // Otherwise, if this vaddr was never allocated + return MemoryInfo(vaddr, 0, 0, KernelMemoryTypes::Free); } \ No newline at end of file diff --git a/src/core/services/gsp_gpu.cpp b/src/core/services/gsp_gpu.cpp index f6abb814..7aa4e7ce 100644 --- a/src/core/services/gsp_gpu.cpp +++ b/src/core/services/gsp_gpu.cpp @@ -12,7 +12,7 @@ namespace GPUCommands { namespace Result { enum : u32 { Success = 0, - SuccessRegisterIRQ = 0x2A07 + SuccessRegisterIRQ = 0x2A07 // TODO: Is this a reference to the Ricoh 2A07 used in PAL NES systems? }; } diff --git a/src/main.cpp b/src/main.cpp index a188db06..668be366 100644 --- a/src/main.cpp +++ b/src/main.cpp @@ -11,5 +11,6 @@ int main (int argc, char *argv[]) { if (!emu.loadELF(elfPath)) { Helpers::panic("Failed to load ELF file: %s", elfPath.c_str()); } + emu.run(); } \ No newline at end of file