HW/Memmap: Refactor Memory to class, move to Core::System.

This commit is contained in:
Admiral H. Curtiss 2022-12-02 20:07:30 +01:00
parent 7cd9a78ebf
commit 839db591d9
No known key found for this signature in database
GPG Key ID: F051B4C4044F33FB
83 changed files with 2222 additions and 1361 deletions

View File

@ -55,6 +55,7 @@ namespace fs = std::filesystem;
#include "Core/PowerPC/PPCAnalyst.h" #include "Core/PowerPC/PPCAnalyst.h"
#include "Core/PowerPC/PPCSymbolDB.h" #include "Core/PowerPC/PPCSymbolDB.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
#include "DiscIO/Enums.h" #include "DiscIO/Enums.h"
#include "DiscIO/GameModDescriptor.h" #include "DiscIO/GameModDescriptor.h"
@ -348,7 +349,10 @@ bool CBoot::DVDRead(const DiscIO::VolumeDisc& disc, u64 dvd_offset, u32 output_a
std::vector<u8> buffer(length); std::vector<u8> buffer(length);
if (!disc.Read(dvd_offset, length, buffer.data(), partition)) if (!disc.Read(dvd_offset, length, buffer.data(), partition))
return false; return false;
Memory::CopyToEmu(output_address, buffer.data(), length);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(output_address, buffer.data(), length);
return true; return true;
} }
@ -357,7 +361,11 @@ bool CBoot::DVDReadDiscID(const DiscIO::VolumeDisc& disc, u32 output_address)
std::array<u8, 0x20> buffer; std::array<u8, 0x20> buffer;
if (!disc.Read(0, buffer.size(), buffer.data(), DiscIO::PARTITION_NONE)) if (!disc.Read(0, buffer.size(), buffer.data(), DiscIO::PARTITION_NONE))
return false; return false;
Memory::CopyToEmu(output_address, buffer.data(), buffer.size());
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(output_address, buffer.data(), buffer.size());
// Transition out of the DiscIdNotRead state (which the drive should be in at this point, // Transition out of the DiscIdNotRead state (which the drive should be in at this point,
// on the assumption that this is only used for the first read) // on the assumption that this is only used for the first read)
DVDInterface::SetDriveState(DVDInterface::DriveState::ReadyNoReadsMade); DVDInterface::SetDriveState(DVDInterface::DriveState::ReadyNoReadsMade);
@ -455,8 +463,10 @@ bool CBoot::Load_BS2(const std::string& boot_rom_filename)
// copying the initial boot code to 0x81200000 is a hack. // copying the initial boot code to 0x81200000 is a hack.
// For now, HLE the first few instructions and start at 0x81200150 // For now, HLE the first few instructions and start at 0x81200150
// to work around this. // to work around this.
Memory::CopyToEmu(0x01200000, data.data() + 0x100, 0x700); auto& system = Core::System::GetInstance();
Memory::CopyToEmu(0x01300000, data.data() + 0x820, 0x1AFE00); auto& memory = system.GetMemory();
memory.CopyToEmu(0x01200000, data.data() + 0x100, 0x700);
memory.CopyToEmu(0x01300000, data.data() + 0x820, 0x1AFE00);
PowerPC::ppcState.gpr[3] = 0xfff0001f; PowerPC::ppcState.gpr[3] = 0xfff0001f;
PowerPC::ppcState.gpr[4] = 0x00002030; PowerPC::ppcState.gpr[4] = 0x00002030;
@ -491,9 +501,11 @@ static void CopyDefaultExceptionHandlers()
0x00000900, 0x00000C00, 0x00000D00, 0x00000F00, 0x00000900, 0x00000C00, 0x00000D00, 0x00000F00,
0x00001300, 0x00001400, 0x00001700}; 0x00001300, 0x00001400, 0x00001700};
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
constexpr u32 RFI_INSTRUCTION = 0x4C000064; constexpr u32 RFI_INSTRUCTION = 0x4C000064;
for (const u32 address : EXCEPTION_HANDLER_ADDRESSES) for (const u32 address : EXCEPTION_HANDLER_ADDRESSES)
Memory::Write_U32(RFI_INSTRUCTION, address); memory.Write_U32(RFI_INSTRUCTION, address);
} }
// Third boot step after BootManager and Core. See Call schedule in BootManager.cpp // Third boot step after BootManager and Core. See Call schedule in BootManager.cpp

View File

@ -32,6 +32,7 @@
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
#include "DiscIO/Enums.h" #include "DiscIO/Enums.h"
#include "DiscIO/RiivolutionPatcher.h" #include "DiscIO/RiivolutionPatcher.h"
@ -212,11 +213,14 @@ bool CBoot::RunApploader(bool is_wii, const DiscIO::VolumeDisc& volume,
void CBoot::SetupGCMemory() void CBoot::SetupGCMemory()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Booted from bootrom. 0xE5207C22 = booted from jtag // Booted from bootrom. 0xE5207C22 = booted from jtag
PowerPC::HostWrite_U32(0x0D15EA5E, 0x80000020); PowerPC::HostWrite_U32(0x0D15EA5E, 0x80000020);
// Physical Memory Size (24MB on retail) // Physical Memory Size (24MB on retail)
PowerPC::HostWrite_U32(Memory::GetRamSizeReal(), 0x80000028); PowerPC::HostWrite_U32(memory.GetRamSizeReal(), 0x80000028);
// Console type - DevKit (retail ID == 0x00000003) see YAGCD 4.2.1.1.2 // Console type - DevKit (retail ID == 0x00000003) see YAGCD 4.2.1.1.2
// TODO: determine why some games fail when using a retail ID. // TODO: determine why some games fail when using a retail ID.
@ -271,10 +275,12 @@ bool CBoot::EmulatedBS2_GC(const DiscIO::VolumeDisc& volume,
DVDReadDiscID(volume, 0x00000000); DVDReadDiscID(volume, 0x00000000);
bool streaming = Memory::Read_U8(0x80000008); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
bool streaming = memory.Read_U8(0x80000008);
if (streaming) if (streaming)
{ {
u8 streaming_size = Memory::Read_U8(0x80000009); u8 streaming_size = memory.Read_U8(0x80000009);
// If the streaming buffer size is 0, then BS2 uses a default size of 10 instead. // If the streaming buffer size is 0, then BS2 uses a default size of 10 instead.
// No known game uses a size other than the default. // No known game uses a size other than the default.
if (streaming_size == 0) if (streaming_size == 0)
@ -412,8 +418,11 @@ bool CBoot::SetupWiiMemory(IOS::HLE::IOSC::ConsoleType console_type)
return false; return false;
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Write the 256 byte setting.txt to memory. // Write the 256 byte setting.txt to memory.
Memory::CopyToEmu(0x3800, gen.GetBytes().data(), gen.GetBytes().size()); memory.CopyToEmu(0x3800, gen.GetBytes().data(), gen.GetBytes().size());
INFO_LOG_FMT(BOOT, "Setup Wii Memory..."); INFO_LOG_FMT(BOOT, "Setup Wii Memory...");
@ -427,28 +436,28 @@ bool CBoot::SetupWiiMemory(IOS::HLE::IOSC::ConsoleType console_type)
0x80000060 Copyright code 0x80000060 Copyright code
*/ */
Memory::Write_U32(0x0D15EA5E, 0x00000020); // Another magic word memory.Write_U32(0x0D15EA5E, 0x00000020); // Another magic word
Memory::Write_U32(0x00000001, 0x00000024); // Unknown memory.Write_U32(0x00000001, 0x00000024); // Unknown
Memory::Write_U32(Memory::GetRamSizeReal(), 0x00000028); // MEM1 size 24MB memory.Write_U32(memory.GetRamSizeReal(), 0x00000028); // MEM1 size 24MB
const Core::ConsoleType board_model = console_type == IOS::HLE::IOSC::ConsoleType::RVT ? const Core::ConsoleType board_model = console_type == IOS::HLE::IOSC::ConsoleType::RVT ?
Core::ConsoleType::NDEV2_1 : Core::ConsoleType::NDEV2_1 :
Core::ConsoleType::RVL_Retail3; Core::ConsoleType::RVL_Retail3;
Memory::Write_U32(static_cast<u32>(board_model), 0x0000002c); // Board Model memory.Write_U32(static_cast<u32>(board_model), 0x0000002c); // Board Model
Memory::Write_U32(0x00000000, 0x00000030); // Init memory.Write_U32(0x00000000, 0x00000030); // Init
Memory::Write_U32(0x817FEC60, 0x00000034); // Init memory.Write_U32(0x817FEC60, 0x00000034); // Init
// 38, 3C should get start, size of FST through apploader // 38, 3C should get start, size of FST through apploader
Memory::Write_U32(0x8008f7b8, 0x000000e4); // Thread Init memory.Write_U32(0x8008f7b8, 0x000000e4); // Thread Init
Memory::Write_U32(Memory::GetRamSizeReal(), 0x000000f0); // "Simulated memory size" (debug mode?) memory.Write_U32(memory.GetRamSizeReal(), 0x000000f0); // "Simulated memory size" (debug mode?)
Memory::Write_U32(0x8179b500, 0x000000f4); // __start memory.Write_U32(0x8179b500, 0x000000f4); // __start
Memory::Write_U32(0x0e7be2c0, 0x000000f8); // Bus speed memory.Write_U32(0x0e7be2c0, 0x000000f8); // Bus speed
Memory::Write_U32(0x2B73A840, 0x000000fc); // CPU speed memory.Write_U32(0x2B73A840, 0x000000fc); // CPU speed
Memory::Write_U16(0x0000, 0x000030e6); // Console type memory.Write_U16(0x0000, 0x000030e6); // Console type
Memory::Write_U32(0x00000000, 0x000030c0); // EXI memory.Write_U32(0x00000000, 0x000030c0); // EXI
Memory::Write_U32(0x00000000, 0x000030c4); // EXI memory.Write_U32(0x00000000, 0x000030c4); // EXI
Memory::Write_U32(0x00000000, 0x000030dc); // Time memory.Write_U32(0x00000000, 0x000030dc); // Time
Memory::Write_U32(0xffffffff, 0x000030d8); // Unknown, set by any official NAND title memory.Write_U32(0xffffffff, 0x000030d8); // Unknown, set by any official NAND title
Memory::Write_U16(0x8201, 0x000030e6); // Dev console / debug capable memory.Write_U16(0x8201, 0x000030e6); // Dev console / debug capable
Memory::Write_U32(0x00000000, 0x000030f0); // Apploader memory.Write_U32(0x00000000, 0x000030f0); // Apploader
// During the boot process, 0x315c is first set to 0xdeadbeef by IOS // During the boot process, 0x315c is first set to 0xdeadbeef by IOS
// in the boot_ppc syscall. The value is then partly overwritten by SDK titles. // in the boot_ppc syscall. The value is then partly overwritten by SDK titles.
@ -457,19 +466,19 @@ bool CBoot::SetupWiiMemory(IOS::HLE::IOSC::ConsoleType console_type)
// //
// 0x0113 appears to mean v1.13, which is the latest version. // 0x0113 appears to mean v1.13, which is the latest version.
// It is fine to always use the latest value as apploaders work with all versions. // It is fine to always use the latest value as apploaders work with all versions.
Memory::Write_U16(0x0113, 0x0000315e); memory.Write_U16(0x0113, 0x0000315e);
Memory::Write_U8(0x80, 0x0000315c); // OSInit memory.Write_U8(0x80, 0x0000315c); // OSInit
Memory::Write_U16(0x0000, 0x000030e0); // PADInit memory.Write_U16(0x0000, 0x000030e0); // PADInit
Memory::Write_U32(0x80000000, 0x00003184); // GameID Address memory.Write_U32(0x80000000, 0x00003184); // GameID Address
// Fake the VI Init of the IPL // Fake the VI Init of the IPL
Memory::Write_U32(DiscIO::IsNTSC(SConfig::GetInstance().m_region) ? 0 : 1, 0x000000CC); memory.Write_U32(DiscIO::IsNTSC(SConfig::GetInstance().m_region) ? 0 : 1, 0x000000CC);
// Clear exception handler. Why? Don't we begin with only zeros? // Clear exception handler. Why? Don't we begin with only zeros?
for (int i = 0x3000; i <= 0x3038; i += 4) for (int i = 0x3000; i <= 0x3038; i += 4)
{ {
Memory::Write_U32(0x00000000, i); memory.Write_U32(0x00000000, i);
} }
return true; return true;
@ -513,6 +522,9 @@ bool CBoot::EmulatedBS2_Wii(const DiscIO::VolumeDisc& volume,
state->discstate = 0x01; state->discstate = 0x01;
}); });
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// The system menu clears the RTC flags. // The system menu clears the RTC flags.
// However, the system menu also updates the disc cache when this happens; see // However, the system menu also updates the disc cache when this happens; see
// https://wiibrew.org/wiki/MX23L4005#DI and // https://wiibrew.org/wiki/MX23L4005#DI and
@ -526,8 +538,8 @@ bool CBoot::EmulatedBS2_Wii(const DiscIO::VolumeDisc& volume,
// When launching the disc game, it copies the partition type and offset to 0x3194 // When launching the disc game, it copies the partition type and offset to 0x3194
// and 0x3198 respectively. // and 0x3198 respectively.
const DiscIO::Partition data_partition = volume.GetGamePartition(); const DiscIO::Partition data_partition = volume.GetGamePartition();
Memory::Write_U32(0, 0x3194); memory.Write_U32(0, 0x3194);
Memory::Write_U32(static_cast<u32>(data_partition.offset >> 2), 0x3198); memory.Write_U32(static_cast<u32>(data_partition.offset >> 2), 0x3198);
const s32 ios_override = Config::Get(Config::MAIN_OVERRIDE_BOOT_IOS); const s32 ios_override = Config::Get(Config::MAIN_OVERRIDE_BOOT_IOS);
const u64 ios = ios_override >= 0 ? Titles::IOS(static_cast<u32>(ios_override)) : tmd.GetIOSId(); const u64 ios = ios_override >= 0 ? Titles::IOS(static_cast<u32>(ios_override)) : tmd.GetIOSId();
@ -554,9 +566,9 @@ bool CBoot::EmulatedBS2_Wii(const DiscIO::VolumeDisc& volume,
SetupHID(/*is_wii*/ true); SetupHID(/*is_wii*/ true);
SetupBAT(/*is_wii*/ true); SetupBAT(/*is_wii*/ true);
Memory::Write_U32(0x4c000064, 0x00000300); // Write default DSI Handler: rfi memory.Write_U32(0x4c000064, 0x00000300); // Write default DSI Handler: rfi
Memory::Write_U32(0x4c000064, 0x00000800); // Write default FPU Handler: rfi memory.Write_U32(0x4c000064, 0x00000800); // Write default FPU Handler: rfi
Memory::Write_U32(0x4c000064, 0x00000C00); // Write default Syscall Handler: rfi memory.Write_U32(0x4c000064, 0x00000C00); // Write default Syscall Handler: rfi
PowerPC::ppcState.gpr[1] = 0x816ffff0; // StackPointer PowerPC::ppcState.gpr[1] = 0x816ffff0; // StackPointer

View File

@ -12,6 +12,7 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/Boot/AncastTypes.h" #include "Core/Boot/AncastTypes.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
DolReader::DolReader(std::vector<u8> buffer) : BootExecutableReader(std::move(buffer)) DolReader::DolReader(std::vector<u8> buffer) : BootExecutableReader(std::move(buffer))
{ {
@ -117,25 +118,32 @@ bool DolReader::LoadIntoMemory(bool only_in_mem1) const
if (m_is_ancast) if (m_is_ancast)
return LoadAncastIntoMemory(); return LoadAncastIntoMemory();
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// load all text (code) sections // load all text (code) sections
for (size_t i = 0; i < m_text_sections.size(); ++i) for (size_t i = 0; i < m_text_sections.size(); ++i)
{
if (!m_text_sections[i].empty() && if (!m_text_sections[i].empty() &&
!(only_in_mem1 && !(only_in_mem1 &&
m_dolheader.textAddress[i] + m_text_sections[i].size() >= Memory::GetRamSizeReal())) m_dolheader.textAddress[i] + m_text_sections[i].size() >= memory.GetRamSizeReal()))
{ {
Memory::CopyToEmu(m_dolheader.textAddress[i], m_text_sections[i].data(), memory.CopyToEmu(m_dolheader.textAddress[i], m_text_sections[i].data(),
m_text_sections[i].size()); m_text_sections[i].size());
} }
}
// load all data sections // load all data sections
for (size_t i = 0; i < m_data_sections.size(); ++i) for (size_t i = 0; i < m_data_sections.size(); ++i)
{
if (!m_data_sections[i].empty() && if (!m_data_sections[i].empty() &&
!(only_in_mem1 && !(only_in_mem1 &&
m_dolheader.dataAddress[i] + m_data_sections[i].size() >= Memory::GetRamSizeReal())) m_dolheader.dataAddress[i] + m_data_sections[i].size() >= memory.GetRamSizeReal()))
{ {
Memory::CopyToEmu(m_dolheader.dataAddress[i], m_data_sections[i].data(), memory.CopyToEmu(m_dolheader.dataAddress[i], m_data_sections[i].data(),
m_data_sections[i].size()); m_data_sections[i].size());
} }
}
return true; return true;
} }
@ -219,11 +227,14 @@ bool DolReader::LoadAncastIntoMemory() const
body_size)) body_size))
return false; return false;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Copy the Ancast header to the emu // Copy the Ancast header to the emu
Memory::CopyToEmu(section_address, header, sizeof(EspressoAncastHeader)); memory.CopyToEmu(section_address, header, sizeof(EspressoAncastHeader));
// Copy the decrypted body to the emu // Copy the decrypted body to the emu
Memory::CopyToEmu(section_address + sizeof(EspressoAncastHeader), decrypted.data(), body_size); memory.CopyToEmu(section_address + sizeof(EspressoAncastHeader), decrypted.data(), body_size);
return true; return true;
} }

View File

@ -14,6 +14,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/PowerPC/PPCSymbolDB.h" #include "Core/PowerPC/PPCSymbolDB.h"
#include "Core/System.h"
static void bswap(u32& w) static void bswap(u32& w)
{ {
@ -135,6 +136,9 @@ bool ElfReader::LoadIntoMemory(bool only_in_mem1) const
INFO_LOG_FMT(BOOT, "{} segments:", header->e_phnum); INFO_LOG_FMT(BOOT, "{} segments:", header->e_phnum);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Copy segments into ram. // Copy segments into ram.
for (int i = 0; i < header->e_phnum; i++) for (int i = 0; i < header->e_phnum; i++)
{ {
@ -150,12 +154,12 @@ bool ElfReader::LoadIntoMemory(bool only_in_mem1) const
u32 srcSize = p->p_filesz; u32 srcSize = p->p_filesz;
u32 dstSize = p->p_memsz; u32 dstSize = p->p_memsz;
if (only_in_mem1 && p->p_vaddr >= Memory::GetRamSizeReal()) if (only_in_mem1 && p->p_vaddr >= memory.GetRamSizeReal())
continue; continue;
Memory::CopyToEmu(writeAddr, src, srcSize); memory.CopyToEmu(writeAddr, src, srcSize);
if (srcSize < dstSize) if (srcSize < dstSize)
Memory::Memset(writeAddr + srcSize, 0, dstSize - srcSize); // zero out bss memory.Memset(writeAddr + srcSize, 0, dstSize - srcSize); // zero out bss
INFO_LOG_FMT(BOOT, "Loadable Segment Copied to {:08x}, size {:08x}", writeAddr, p->p_memsz); INFO_LOG_FMT(BOOT, "Loadable Segment Copied to {:08x}, size {:08x}", writeAddr, p->p_memsz);
} }

View File

@ -13,6 +13,7 @@
#include "Common/MsgHandler.h" #include "Common/MsgHandler.h"
#include "Core/Config/MainSettings.h" #include "Core/Config/MainSettings.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
constexpr u32 FILE_ID = 0x0d01f1f0; constexpr u32 FILE_ID = 0x0d01f1f0;
constexpr u32 VERSION_NUMBER = 5; constexpr u32 VERSION_NUMBER = 5;
@ -161,8 +162,10 @@ bool FifoDataFile::Save(const std::string& filename)
header.flags = m_Flags; header.flags = m_Flags;
header.mem1_size = Memory::GetRamSizeReal(); auto& system = Core::System::GetInstance();
header.mem2_size = Memory::GetExRamSizeReal(); auto& memory = system.GetMemory();
header.mem1_size = memory.GetRamSizeReal();
header.mem2_size = memory.GetExRamSizeReal();
file.Seek(0, File::SeekOrigin::Begin); file.Seek(0, File::SeekOrigin::Begin);
file.WriteBytes(&header, sizeof(FileHeader)); file.WriteBytes(&header, sizeof(FileHeader));
@ -267,14 +270,15 @@ std::unique_ptr<FifoDataFile> FifoDataFile::Load(const std::string& filename, bo
// It should be noted, however, that Dolphin *will still crash* from the nullptr being returned // It should be noted, however, that Dolphin *will still crash* from the nullptr being returned
// in a non-flagsOnly context, so if this code becomes necessary, it should be moved above the // in a non-flagsOnly context, so if this code becomes necessary, it should be moved above the
// prior conditional. // prior conditional.
if (header.mem1_size != Memory::GetRamSizeReal() || auto& system = Core::System::GetInstance();
header.mem2_size != Memory::GetExRamSizeReal()) auto& memory = system.GetMemory();
if (header.mem1_size != memory.GetRamSizeReal() || header.mem2_size != memory.GetExRamSizeReal())
{ {
CriticalAlertFmtT("Emulated memory size mismatch!\n" CriticalAlertFmtT("Emulated memory size mismatch!\n"
"Current: MEM1 {0:08X} ({1} MiB), MEM2 {2:08X} ({3} MiB)\n" "Current: MEM1 {0:08X} ({1} MiB), MEM2 {2:08X} ({3} MiB)\n"
"DFF: MEM1 {4:08X} ({5} MiB), MEM2 {6:08X} ({7} MiB)", "DFF: MEM1 {4:08X} ({5} MiB), MEM2 {6:08X} ({7} MiB)",
Memory::GetRamSizeReal(), Memory::GetRamSizeReal() / 0x100000, memory.GetRamSizeReal(), memory.GetRamSizeReal() / 0x100000,
Memory::GetExRamSizeReal(), Memory::GetExRamSizeReal() / 0x100000, memory.GetExRamSizeReal(), memory.GetExRamSizeReal() / 0x100000,
header.mem1_size, header.mem1_size / 0x100000, header.mem2_size, header.mem1_size, header.mem1_size / 0x100000, header.mem2_size,
header.mem2_size / 0x100000); header.mem2_size / 0x100000);
return nullptr; return nullptr;

View File

@ -494,12 +494,14 @@ void FifoPlayer::WriteAllMemoryUpdates()
void FifoPlayer::WriteMemory(const MemoryUpdate& memUpdate) void FifoPlayer::WriteMemory(const MemoryUpdate& memUpdate)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* mem = nullptr; u8* mem = nullptr;
if (memUpdate.address & 0x10000000) if (memUpdate.address & 0x10000000)
mem = &Memory::m_pEXRAM[memUpdate.address & Memory::GetExRamMask()]; mem = &memory.GetEXRAM()[memUpdate.address & memory.GetExRamMask()];
else else
mem = &Memory::m_pRAM[memUpdate.address & Memory::GetRamMask()]; mem = &memory.GetRAM()[memUpdate.address & memory.GetRamMask()];
std::copy(memUpdate.data.begin(), memUpdate.data.end(), mem); std::copy(memUpdate.data.begin(), memUpdate.data.end(), mem);
} }

View File

@ -12,6 +12,7 @@
#include "Core/ConfigManager.h" #include "Core/ConfigManager.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/OpcodeDecoding.h" #include "VideoCommon/OpcodeDecoding.h"
#include "VideoCommon/XFStructs.h" #include "VideoCommon/XFStructs.h"
@ -229,8 +230,10 @@ void FifoRecorder::StartRecording(s32 numFrames, CallbackFunc finishedCb)
// - Global variables suck // - Global variables suck
// - Multithreading with the above two sucks // - Multithreading with the above two sucks
// //
m_Ram.resize(Memory::GetRamSize()); auto& system = Core::System::GetInstance();
m_ExRam.resize(Memory::GetExRamSize()); auto& memory = system.GetMemory();
m_Ram.resize(memory.GetRamSize());
m_ExRam.resize(memory.GetExRamSize());
std::fill(m_Ram.begin(), m_Ram.end(), 0); std::fill(m_Ram.begin(), m_Ram.end(), 0);
std::fill(m_ExRam.begin(), m_ExRam.end(), 0); std::fill(m_ExRam.begin(), m_ExRam.end(), 0);
@ -310,17 +313,20 @@ void FifoRecorder::WriteGPCommand(const u8* data, u32 size)
void FifoRecorder::UseMemory(u32 address, u32 size, MemoryUpdate::Type type, bool dynamicUpdate) void FifoRecorder::UseMemory(u32 address, u32 size, MemoryUpdate::Type type, bool dynamicUpdate)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* curData; u8* curData;
u8* newData; u8* newData;
if (address & 0x10000000) if (address & 0x10000000)
{ {
curData = &m_ExRam[address & Memory::GetExRamMask()]; curData = &m_ExRam[address & memory.GetExRamMask()];
newData = &Memory::m_pEXRAM[address & Memory::GetExRamMask()]; newData = &memory.GetEXRAM()[address & memory.GetExRamMask()];
} }
else else
{ {
curData = &m_Ram[address & Memory::GetRamMask()]; curData = &m_Ram[address & memory.GetRamMask()];
newData = &Memory::m_pRAM[address & Memory::GetRamMask()]; newData = &memory.GetRAM()[address & memory.GetRamMask()];
} }
if (!dynamicUpdate && memcmp(curData, newData, size) != 0) if (!dynamicUpdate && memcmp(curData, newData, size) != 0)

View File

@ -19,6 +19,7 @@
#include "Core/IOS/ES/ES.h" #include "Core/IOS/ES/ES.h"
#include "Core/PowerPC/PPCSymbolDB.h" #include "Core/PowerPC/PPCSymbolDB.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
namespace HLE namespace HLE
{ {
@ -90,7 +91,9 @@ void PatchFixedFunctions()
if (!Config::Get(Config::MAIN_ENABLE_CHEATS)) if (!Config::Get(Config::MAIN_ENABLE_CHEATS))
{ {
Patch(0x80001800, "HBReload"); Patch(0x80001800, "HBReload");
Memory::CopyToEmu(0x00001804, "STUBHAXX", 8); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(0x00001804, "STUBHAXX", 8);
} }
// Not part of the binary itself, but either we or Gecko OS might insert // Not part of the binary itself, but either we or Gecko OS might insert

View File

@ -10,6 +10,7 @@
#include "Core/HW/DSP.h" #include "Core/HW/DSP.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/System.h"
namespace AddressSpace namespace AddressSpace
{ {
@ -92,6 +93,9 @@ struct EffectiveAddressSpaceAccessors : Accessors
bool Matches(u32 haystack_start, const u8* needle_start, std::size_t needle_size) const bool Matches(u32 haystack_start, const u8* needle_start, std::size_t needle_size) const
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 page_base = haystack_start & 0xfffff000; u32 page_base = haystack_start & 0xfffff000;
u32 offset = haystack_start & 0x0000fff; u32 offset = haystack_start & 0x0000fff;
do do
@ -114,7 +118,7 @@ struct EffectiveAddressSpaceAccessors : Accessors
return false; return false;
} }
u8* page_ptr = Memory::GetPointer(*page_physical_address); u8* page_ptr = memory.GetPointer(*page_physical_address);
if (page_ptr == nullptr) if (page_ptr == nullptr)
{ {
return false; return false;
@ -417,9 +421,12 @@ Accessors* GetAccessors(Type address_space)
void Init() void Init()
{ {
s_mem1_address_space_accessors = {&Memory::m_pRAM, Memory::GetRamSizeReal()}; auto& system = Core::System::GetInstance();
s_mem2_address_space_accessors = {&Memory::m_pEXRAM, Memory::GetExRamSizeReal()}; auto& memory = system.GetMemory();
s_fake_address_space_accessors = {&Memory::m_pFakeVMEM, Memory::GetFakeVMemSize()};
s_mem1_address_space_accessors = {&memory.GetRAM(), memory.GetRamSizeReal()};
s_mem2_address_space_accessors = {&memory.GetEXRAM(), memory.GetExRamSizeReal()};
s_fake_address_space_accessors = {&memory.GetFakeVMEM(), memory.GetFakeVMemSize()};
s_physical_address_space_accessors_gcn = {{0x00000000, &s_mem1_address_space_accessors}}; s_physical_address_space_accessors_gcn = {{0x00000000, &s_mem1_address_space_accessors}};
s_physical_address_space_accessors_wii = {{0x00000000, &s_mem1_address_space_accessors}, s_physical_address_space_accessors_wii = {{0x00000000, &s_mem1_address_space_accessors},
{0x10000000, &s_mem2_address_space_accessors}}; {0x10000000, &s_mem2_address_space_accessors}};

View File

@ -205,16 +205,18 @@ void Init(bool hle)
void Reinit(bool hle) void Reinit(bool hle)
{ {
auto& state = Core::System::GetInstance().GetDSPState().GetData(); auto& system = Core::System::GetInstance();
auto& state = system.GetDSPState().GetData();
state.dsp_emulator = CreateDSPEmulator(hle); state.dsp_emulator = CreateDSPEmulator(hle);
state.is_lle = state.dsp_emulator->IsLLE(); state.is_lle = state.dsp_emulator->IsLLE();
if (SConfig::GetInstance().bWii) if (SConfig::GetInstance().bWii)
{ {
auto& memory = system.GetMemory();
state.aram.wii_mode = true; state.aram.wii_mode = true;
state.aram.size = Memory::GetExRamSizeReal(); state.aram.size = memory.GetExRamSizeReal();
state.aram.mask = Memory::GetExRamMask(); state.aram.mask = memory.GetExRamMask();
state.aram.ptr = Memory::m_pEXRAM; state.aram.ptr = memory.GetEXRAM();
} }
else else
{ {
@ -527,7 +529,8 @@ void UpdateAudioDMA()
// Read audio at g_audioDMA.current_source_address in RAM and push onto an // Read audio at g_audioDMA.current_source_address in RAM and push onto an
// external audio fifo in the emulator, to be mixed with the disc // external audio fifo in the emulator, to be mixed with the disc
// streaming output. // streaming output.
void* address = Memory::GetPointer(state.audio_dma.current_source_address); auto& memory = system.GetMemory();
void* address = memory.GetPointer(state.audio_dma.current_source_address);
AudioCommon::SendAIBuffer(system, reinterpret_cast<short*>(address), 8); AudioCommon::SendAIBuffer(system, reinterpret_cast<short*>(address), 8);
if (state.audio_dma.remaining_blocks_count != 0) if (state.audio_dma.remaining_blocks_count != 0)
@ -554,6 +557,7 @@ static void Do_ARAM_DMA()
{ {
auto& system = Core::System::GetInstance(); auto& system = Core::System::GetInstance();
auto& core_timing = system.GetCoreTiming(); auto& core_timing = system.GetCoreTiming();
auto& memory = system.GetMemory();
auto& state = system.GetDSPState().GetData(); auto& state = system.GetDSPState().GetData();
state.dsp_control.DMAState = 1; state.dsp_control.DMAState = 1;
@ -581,17 +585,17 @@ static void Do_ARAM_DMA()
// See below in the write section for more information // See below in the write section for more information
if ((state.aram_info.Hex & 0xf) == 3) if ((state.aram_info.Hex & 0xf) == 3)
{ {
Memory::Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask], memory.Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask],
state.aram_dma.MMAddr); state.aram_dma.MMAddr);
} }
else if ((state.aram_info.Hex & 0xf) == 4) else if ((state.aram_info.Hex & 0xf) == 4)
{ {
Memory::Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask], memory.Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask],
state.aram_dma.MMAddr); state.aram_dma.MMAddr);
} }
else else
{ {
Memory::Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask], memory.Write_U64_Swap(*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask],
state.aram_dma.MMAddr); state.aram_dma.MMAddr);
} }
@ -604,7 +608,7 @@ static void Do_ARAM_DMA()
{ {
while (state.aram_dma.Cnt.count) while (state.aram_dma.Cnt.count)
{ {
Memory::Write_U64(HSP::Read(state.aram_dma.ARAddr), state.aram_dma.MMAddr); memory.Write_U64(HSP::Read(state.aram_dma.ARAddr), state.aram_dma.MMAddr);
state.aram_dma.MMAddr += 8; state.aram_dma.MMAddr += 8;
state.aram_dma.ARAddr += 8; state.aram_dma.ARAddr += 8;
state.aram_dma.Cnt.count -= 8; state.aram_dma.Cnt.count -= 8;
@ -628,22 +632,22 @@ static void Do_ARAM_DMA()
if ((state.aram_info.Hex & 0xf) == 3) if ((state.aram_info.Hex & 0xf) == 3)
{ {
*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] = *(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] =
Common::swap64(Memory::Read_U64(state.aram_dma.MMAddr)); Common::swap64(memory.Read_U64(state.aram_dma.MMAddr));
} }
else if ((state.aram_info.Hex & 0xf) == 4) else if ((state.aram_info.Hex & 0xf) == 4)
{ {
if (state.aram_dma.ARAddr < 0x400000) if (state.aram_dma.ARAddr < 0x400000)
{ {
*(u64*)&state.aram.ptr[(state.aram_dma.ARAddr + 0x400000) & state.aram.mask] = *(u64*)&state.aram.ptr[(state.aram_dma.ARAddr + 0x400000) & state.aram.mask] =
Common::swap64(Memory::Read_U64(state.aram_dma.MMAddr)); Common::swap64(memory.Read_U64(state.aram_dma.MMAddr));
} }
*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] = *(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] =
Common::swap64(Memory::Read_U64(state.aram_dma.MMAddr)); Common::swap64(memory.Read_U64(state.aram_dma.MMAddr));
} }
else else
{ {
*(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] = *(u64*)&state.aram.ptr[state.aram_dma.ARAddr & state.aram.mask] =
Common::swap64(Memory::Read_U64(state.aram_dma.MMAddr)); Common::swap64(memory.Read_U64(state.aram_dma.MMAddr));
} }
state.aram_dma.MMAddr += 8; state.aram_dma.MMAddr += 8;
@ -655,7 +659,7 @@ static void Do_ARAM_DMA()
{ {
while (state.aram_dma.Cnt.count) while (state.aram_dma.Cnt.count)
{ {
HSP::Write(state.aram_dma.ARAddr, Memory::Read_U64(state.aram_dma.MMAddr)); HSP::Write(state.aram_dma.ARAddr, memory.Read_U64(state.aram_dma.MMAddr));
state.aram_dma.MMAddr += 8; state.aram_dma.MMAddr += 8;
state.aram_dma.ARAddr += 8; state.aram_dma.ARAddr += 8;
@ -670,14 +674,20 @@ static void Do_ARAM_DMA()
// (LM) It just means that DSP reads via '0xffdd' on Wii can end up in EXRAM or main RAM // (LM) It just means that DSP reads via '0xffdd' on Wii can end up in EXRAM or main RAM
u8 ReadARAM(u32 address) u8 ReadARAM(u32 address)
{ {
auto& state = Core::System::GetInstance().GetDSPState().GetData(); auto& system = Core::System::GetInstance();
auto& state = system.GetDSPState().GetData();
if (state.aram.wii_mode) if (state.aram.wii_mode)
{ {
if (address & 0x10000000) if (address & 0x10000000)
{
return state.aram.ptr[address & state.aram.mask]; return state.aram.ptr[address & state.aram.mask];
}
else else
return Memory::Read_U8(address & Memory::GetRamMask()); {
auto& memory = system.GetMemory();
return memory.Read_U8(address & memory.GetRamMask());
}
} }
else else
{ {

View File

@ -19,6 +19,7 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/HW/DSPHLE/UCodes/UCodes.h" #include "Core/HW/DSPHLE/UCodes/UCodes.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace DSP::HLE namespace DSP::HLE
{ {
@ -142,8 +143,10 @@ protected:
template <int Millis, size_t BufCount> template <int Millis, size_t BufCount>
void InitMixingBuffers(u32 init_addr, const std::array<BufferDesc, BufCount>& buffers) void InitMixingBuffers(u32 init_addr, const std::array<BufferDesc, BufCount>& buffers)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::array<u16, 3 * BufCount> init_array; std::array<u16, 3 * BufCount> init_array;
Memory::CopyFromEmuSwapped(init_array.data(), init_addr, sizeof(init_array)); memory.CopyFromEmuSwapped(init_array.data(), init_addr, sizeof(init_array));
for (size_t i = 0; i < BufCount; ++i) for (size_t i = 0; i < BufCount; ++i)
{ {
const BufferDesc& buf = buffers[i]; const BufferDesc& buf = buffers[i];

View File

@ -22,6 +22,7 @@
#include "Core/HW/DSPHLE/UCodes/AX.h" #include "Core/HW/DSPHLE/UCodes/AX.h"
#include "Core/HW/DSPHLE/UCodes/AXStructs.h" #include "Core/HW/DSPHLE/UCodes/AXStructs.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace DSP::HLE namespace DSP::HLE
{ {
@ -101,10 +102,13 @@ bool HasLpf(u32 crc)
// Read a PB from MRAM/ARAM // Read a PB from MRAM/ARAM
void ReadPB(u32 addr, PB_TYPE& pb, u32 crc) void ReadPB(u32 addr, PB_TYPE& pb, u32 crc)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (HasLpf(crc)) if (HasLpf(crc))
{ {
u16* dst = (u16*)&pb; u16* dst = (u16*)&pb;
Memory::CopyFromEmuSwapped<u16>(dst, addr, sizeof(pb)); memory.CopyFromEmuSwapped<u16>(dst, addr, sizeof(pb));
} }
else else
{ {
@ -116,19 +120,22 @@ void ReadPB(u32 addr, PB_TYPE& pb, u32 crc)
constexpr size_t lpf_off = offsetof(AXPB, lpf); constexpr size_t lpf_off = offsetof(AXPB, lpf);
constexpr size_t lc_off = offsetof(AXPB, loop_counter); constexpr size_t lc_off = offsetof(AXPB, loop_counter);
Memory::CopyFromEmuSwapped<u16>((u16*)dst, addr, lpf_off); memory.CopyFromEmuSwapped<u16>((u16*)dst, addr, lpf_off);
memset(dst + lpf_off, 0, lc_off - lpf_off); memset(dst + lpf_off, 0, lc_off - lpf_off);
Memory::CopyFromEmuSwapped<u16>((u16*)(dst + lc_off), addr + lpf_off, sizeof(pb) - lc_off); memory.CopyFromEmuSwapped<u16>((u16*)(dst + lc_off), addr + lpf_off, sizeof(pb) - lc_off);
} }
} }
// Write a PB back to MRAM/ARAM // Write a PB back to MRAM/ARAM
void WritePB(u32 addr, const PB_TYPE& pb, u32 crc) void WritePB(u32 addr, const PB_TYPE& pb, u32 crc)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (HasLpf(crc)) if (HasLpf(crc))
{ {
const u16* src = (const u16*)&pb; const u16* src = (const u16*)&pb;
Memory::CopyToEmuSwapped<u16>(addr, src, sizeof(pb)); memory.CopyToEmuSwapped<u16>(addr, src, sizeof(pb));
} }
else else
{ {
@ -140,8 +147,8 @@ void WritePB(u32 addr, const PB_TYPE& pb, u32 crc)
constexpr size_t lpf_off = offsetof(AXPB, lpf); constexpr size_t lpf_off = offsetof(AXPB, lpf);
constexpr size_t lc_off = offsetof(AXPB, loop_counter); constexpr size_t lc_off = offsetof(AXPB, loop_counter);
Memory::CopyToEmuSwapped<u16>(addr, (const u16*)src, lpf_off); memory.CopyToEmuSwapped<u16>(addr, (const u16*)src, lpf_off);
Memory::CopyToEmuSwapped<u16>(addr + lpf_off, (const u16*)(src + lc_off), sizeof(pb) - lc_off); memory.CopyToEmuSwapped<u16>(addr + lpf_off, (const u16*)(src + lc_off), sizeof(pb) - lc_off);
} }
} }

View File

@ -30,6 +30,7 @@
#include "Core/HW/DSPHLE/UCodes/ROM.h" #include "Core/HW/DSPHLE/UCodes/ROM.h"
#include "Core/HW/DSPHLE/UCodes/Zelda.h" #include "Core/HW/DSPHLE/UCodes/Zelda.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace DSP::HLE namespace DSP::HLE
{ {
@ -40,28 +41,37 @@ constexpr bool ExramRead(u32 address)
u8 HLEMemory_Read_U8(u32 address) u8 HLEMemory_Read_U8(u32 address)
{ {
if (ExramRead(address)) auto& system = Core::System::GetInstance();
return Memory::m_pEXRAM[address & Memory::GetExRamMask()]; auto& memory = system.GetMemory();
return Memory::m_pRAM[address & Memory::GetRamMask()]; if (ExramRead(address))
return memory.GetEXRAM()[address & memory.GetExRamMask()];
return memory.GetRAM()[address & memory.GetRamMask()];
} }
void HLEMemory_Write_U8(u32 address, u8 value) void HLEMemory_Write_U8(u32 address, u8 value)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (ExramRead(address)) if (ExramRead(address))
Memory::m_pEXRAM[address & Memory::GetExRamMask()] = value; memory.GetEXRAM()[address & memory.GetExRamMask()] = value;
else else
Memory::m_pRAM[address & Memory::GetRamMask()] = value; memory.GetRAM()[address & memory.GetRamMask()] = value;
} }
u16 HLEMemory_Read_U16LE(u32 address) u16 HLEMemory_Read_U16LE(u32 address)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u16 value; u16 value;
if (ExramRead(address)) if (ExramRead(address))
std::memcpy(&value, &Memory::m_pEXRAM[address & Memory::GetExRamMask()], sizeof(u16)); std::memcpy(&value, &memory.GetEXRAM()[address & memory.GetExRamMask()], sizeof(u16));
else else
std::memcpy(&value, &Memory::m_pRAM[address & Memory::GetRamMask()], sizeof(u16)); std::memcpy(&value, &memory.GetRAM()[address & memory.GetRamMask()], sizeof(u16));
return value; return value;
} }
@ -73,10 +83,13 @@ u16 HLEMemory_Read_U16(u32 address)
void HLEMemory_Write_U16LE(u32 address, u16 value) void HLEMemory_Write_U16LE(u32 address, u16 value)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (ExramRead(address)) if (ExramRead(address))
std::memcpy(&Memory::m_pEXRAM[address & Memory::GetExRamMask()], &value, sizeof(u16)); std::memcpy(&memory.GetEXRAM()[address & memory.GetExRamMask()], &value, sizeof(u16));
else else
std::memcpy(&Memory::m_pRAM[address & Memory::GetRamMask()], &value, sizeof(u16)); std::memcpy(&memory.GetRAM()[address & memory.GetRamMask()], &value, sizeof(u16));
} }
void HLEMemory_Write_U16(u32 address, u16 value) void HLEMemory_Write_U16(u32 address, u16 value)
@ -86,12 +99,15 @@ void HLEMemory_Write_U16(u32 address, u16 value)
u32 HLEMemory_Read_U32LE(u32 address) u32 HLEMemory_Read_U32LE(u32 address)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 value; u32 value;
if (ExramRead(address)) if (ExramRead(address))
std::memcpy(&value, &Memory::m_pEXRAM[address & Memory::GetExRamMask()], sizeof(u32)); std::memcpy(&value, &memory.GetEXRAM()[address & memory.GetExRamMask()], sizeof(u32));
else else
std::memcpy(&value, &Memory::m_pRAM[address & Memory::GetRamMask()], sizeof(u32)); std::memcpy(&value, &memory.GetRAM()[address & memory.GetRamMask()], sizeof(u32));
return value; return value;
} }
@ -103,10 +119,13 @@ u32 HLEMemory_Read_U32(u32 address)
void HLEMemory_Write_U32LE(u32 address, u32 value) void HLEMemory_Write_U32LE(u32 address, u32 value)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (ExramRead(address)) if (ExramRead(address))
std::memcpy(&Memory::m_pEXRAM[address & Memory::GetExRamMask()], &value, sizeof(u32)); std::memcpy(&memory.GetEXRAM()[address & memory.GetExRamMask()], &value, sizeof(u32));
else else
std::memcpy(&Memory::m_pRAM[address & Memory::GetRamMask()], &value, sizeof(u32)); std::memcpy(&memory.GetRAM()[address & memory.GetRamMask()], &value, sizeof(u32));
} }
void HLEMemory_Write_U32(u32 address, u32 value) void HLEMemory_Write_U32(u32 address, u32 value)
@ -116,10 +135,13 @@ void HLEMemory_Write_U32(u32 address, u32 value)
void* HLEMemory_Get_Pointer(u32 address) void* HLEMemory_Get_Pointer(u32 address)
{ {
if (ExramRead(address)) auto& system = Core::System::GetInstance();
return &Memory::m_pEXRAM[address & Memory::GetExRamMask()]; auto& memory = system.GetMemory();
return &Memory::m_pRAM[address & Memory::GetRamMask()]; if (ExramRead(address))
return &memory.GetEXRAM()[address & memory.GetExRamMask()];
return &memory.GetRAM()[address & memory.GetRamMask()];
} }
UCodeInterface::UCodeInterface(DSPHLE* dsphle, u32 crc) UCodeInterface::UCodeInterface(DSPHLE* dsphle, u32 crc)
@ -188,7 +210,9 @@ void UCodeInterface::PrepareBootUCode(u32 mail)
if (Config::Get(Config::MAIN_DUMP_UCODE)) if (Config::Get(Config::MAIN_DUMP_UCODE))
{ {
DSP::DumpDSPCode(Memory::GetPointer(m_next_ucode.iram_mram_addr), m_next_ucode.iram_size, auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
DSP::DumpDSPCode(memory.GetPointer(m_next_ucode.iram_mram_addr), m_next_ucode.iram_size,
ector_crc); ector_crc);
} }

View File

@ -18,6 +18,7 @@
#include "Core/HW/DSPLLE/DSPSymbols.h" #include "Core/HW/DSPLLE/DSPSymbols.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/Host.h" #include "Core/Host.h"
#include "Core/System.h"
#include "VideoCommon/OnScreenDisplay.h" #include "VideoCommon/OnScreenDisplay.h"
// The user of the DSPCore library must supply a few functions so that the // The user of the DSPCore library must supply a few functions so that the
@ -39,12 +40,16 @@ void WriteHostMemory(u8 value, u32 addr)
void DMAToDSP(u16* dst, u32 addr, u32 size) void DMAToDSP(u16* dst, u32 addr, u32 size)
{ {
Memory::CopyFromEmuSwapped(dst, addr, size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmuSwapped(dst, addr, size);
} }
void DMAFromDSP(const u16* src, u32 addr, u32 size) void DMAFromDSP(const u16* src, u32 addr, u32 size)
{ {
Memory::CopyToEmuSwapped(addr, src, size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmuSwapped(addr, src, size);
} }
void OSD_AddMessage(std::string str, u32 ms) void OSD_AddMessage(std::string str, u32 ms)
@ -70,7 +75,9 @@ void InterruptRequest()
void CodeLoaded(DSPCore& dsp, u32 addr, size_t size) void CodeLoaded(DSPCore& dsp, u32 addr, size_t size)
{ {
CodeLoaded(dsp, Memory::GetPointer(addr), size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
CodeLoaded(dsp, memory.GetPointer(addr), size);
} }
void CodeLoaded(DSPCore& dsp, const u8* ptr, size_t size) void CodeLoaded(DSPCore& dsp, const u8* ptr, size_t size)

View File

@ -896,15 +896,17 @@ void ExecuteCommand(ReplyType reply_type)
{ {
// Used by both GC and Wii // Used by both GC and Wii
case DICommand::Inquiry: case DICommand::Inquiry:
{
// (shuffle2) Taken from my Wii // (shuffle2) Taken from my Wii
Memory::Write_U32(0x00000002, state.DIMAR); // Revision level, device code auto& memory = system.GetMemory();
Memory::Write_U32(0x20060526, state.DIMAR + 4); // Release date memory.Write_U32(0x00000002, state.DIMAR); // Revision level, device code
Memory::Write_U32(0x41000000, state.DIMAR + 8); // Version memory.Write_U32(0x20060526, state.DIMAR + 4); // Release date
memory.Write_U32(0x41000000, state.DIMAR + 8); // Version
INFO_LOG_FMT(DVDINTERFACE, "DVDLowInquiry (Buffer {:#010x}, {:#x})", state.DIMAR, INFO_LOG_FMT(DVDINTERFACE, "DVDLowInquiry (Buffer {:#010x}, {:#x})", state.DIMAR,
state.DILENGTH); state.DILENGTH);
break; break;
}
// GC-only patched drive firmware command, used by libogc // GC-only patched drive firmware command, used by libogc
case DICommand::Unknown55: case DICommand::Unknown55:
INFO_LOG_FMT(DVDINTERFACE, "SetExtension"); INFO_LOG_FMT(DVDINTERFACE, "SetExtension");
@ -1024,17 +1026,20 @@ void ExecuteCommand(ReplyType reply_type)
break; break;
// Wii-exclusive // Wii-exclusive
case DICommand::ReadBCA: case DICommand::ReadBCA:
{
WARN_LOG_FMT(DVDINTERFACE, "DVDLowReadDiskBca - supplying dummy data to appease NSMBW"); WARN_LOG_FMT(DVDINTERFACE, "DVDLowReadDiskBca - supplying dummy data to appease NSMBW");
DolphinAnalytics::Instance().ReportGameQuirk(GameQuirk::USES_DVD_LOW_READ_DISK_BCA); DolphinAnalytics::Instance().ReportGameQuirk(GameQuirk::USES_DVD_LOW_READ_DISK_BCA);
// NSMBW checks that the first 0x33 bytes of the BCA are 0, then it expects a 1. // NSMBW checks that the first 0x33 bytes of the BCA are 0, then it expects a 1.
// Most (all?) other games have 0x34 0's at the start of the BCA, but don't actually // Most (all?) other games have 0x34 0's at the start of the BCA, but don't actually
// read it. NSMBW doesn't care about the other 12 bytes (which contain manufacturing data?) // read it. NSMBW doesn't care about the other 12 bytes (which contain manufacturing data?)
auto& memory = system.GetMemory();
// TODO: Read the .bca file that cleanrip generates, if it exists // TODO: Read the .bca file that cleanrip generates, if it exists
// Memory::CopyToEmu(output_address, bca_data, 0x40); // memory.CopyToEmu(output_address, bca_data, 0x40);
Memory::Memset(state.DIMAR, 0, 0x40); memory.Memset(state.DIMAR, 0, 0x40);
Memory::Write_U8(1, state.DIMAR + 0x33); memory.Write_U8(1, state.DIMAR + 0x33);
break; break;
}
// Wii-exclusive // Wii-exclusive
case DICommand::RequestDiscStatus: case DICommand::RequestDiscStatus:
ERROR_LOG_FMT(DVDINTERFACE, "DVDLowRequestDiscStatus"); ERROR_LOG_FMT(DVDINTERFACE, "DVDLowRequestDiscStatus");

View File

@ -391,7 +391,10 @@ static void FinishRead(Core::System& system, u64 id, s64 cycles_late)
else else
{ {
if (request.copy_to_ram) if (request.copy_to_ram)
Memory::CopyToEmu(request.output_address, buffer.data(), request.length); {
auto& memory = system.GetMemory();
memory.CopyToEmu(request.output_address, buffer.data(), request.length);
}
interrupt = DVDInterface::DIInterruptType::TCINT; interrupt = DVDInterface::DIInterruptType::TCINT;
} }

View File

@ -15,6 +15,7 @@
#include "Core/HW/EXI/EXI_DeviceMemoryCard.h" #include "Core/HW/EXI/EXI_DeviceMemoryCard.h"
#include "Core/HW/EXI/EXI_DeviceMic.h" #include "Core/HW/EXI/EXI_DeviceMic.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace ExpansionInterface namespace ExpansionInterface
{ {
@ -47,20 +48,24 @@ void IEXIDevice::ImmReadWrite(u32& data, u32 size)
void IEXIDevice::DMAWrite(u32 address, u32 size) void IEXIDevice::DMAWrite(u32 address, u32 size)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
while (size--) while (size--)
{ {
u8 byte = Memory::Read_U8(address++); u8 byte = memory.Read_U8(address++);
TransferByte(byte); TransferByte(byte);
} }
} }
void IEXIDevice::DMARead(u32 address, u32 size) void IEXIDevice::DMARead(u32 address, u32 size)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
while (size--) while (size--)
{ {
u8 byte = 0; u8 byte = 0;
TransferByte(byte); TransferByte(byte);
Memory::Write_U8(byte, address++); memory.Write_U8(byte, address++);
} }
} }

View File

@ -19,6 +19,7 @@
#include "Core/HW/EXI/EXI.h" #include "Core/HW/EXI/EXI.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
namespace ExpansionInterface namespace ExpansionInterface
{ {
@ -233,7 +234,9 @@ void CEXIETHERNET::DMAWrite(u32 addr, u32 size)
if (transfer.region == transfer.MX && transfer.direction == transfer.WRITE && if (transfer.region == transfer.MX && transfer.direction == transfer.WRITE &&
transfer.address == BBA_WRTXFIFOD) transfer.address == BBA_WRTXFIFOD)
{ {
DirectFIFOWrite(Memory::GetPointer(addr), size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
DirectFIFOWrite(memory.GetPointer(addr), size);
} }
else else
{ {
@ -246,7 +249,9 @@ void CEXIETHERNET::DMAWrite(u32 addr, u32 size)
void CEXIETHERNET::DMARead(u32 addr, u32 size) void CEXIETHERNET::DMARead(u32 addr, u32 size)
{ {
DEBUG_LOG_FMT(SP1, "DMA read: {:08x} {:x}", addr, size); DEBUG_LOG_FMT(SP1, "DMA read: {:08x} {:x}", addr, size);
Memory::CopyToEmu(addr, &mBbaMem[transfer.address], size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(addr, &mBbaMem[transfer.address], size);
transfer.address += size; transfer.address += size;
} }

View File

@ -523,7 +523,9 @@ void CEXIMemoryCard::DoState(PointerWrap& p)
// read all at once instead of single byte at a time as done by IEXIDevice::DMARead // read all at once instead of single byte at a time as done by IEXIDevice::DMARead
void CEXIMemoryCard::DMARead(u32 addr, u32 size) void CEXIMemoryCard::DMARead(u32 addr, u32 size)
{ {
m_memory_card->Read(m_address, size, Memory::GetPointer(addr)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
m_memory_card->Read(m_address, size, memory.GetPointer(addr));
if ((m_address + size) % Memcard::BLOCK_SIZE == 0) if ((m_address + size) % Memcard::BLOCK_SIZE == 0)
{ {
@ -531,7 +533,7 @@ void CEXIMemoryCard::DMARead(u32 addr, u32 size)
} }
// Schedule transfer complete later based on read speed // Schedule transfer complete later based on read speed
Core::System::GetInstance().GetCoreTiming().ScheduleEvent( system.GetCoreTiming().ScheduleEvent(
size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_READ), size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_READ),
s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot)); s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot));
} }
@ -540,7 +542,9 @@ void CEXIMemoryCard::DMARead(u32 addr, u32 size)
// write all at once instead of single byte at a time as done by IEXIDevice::DMAWrite // write all at once instead of single byte at a time as done by IEXIDevice::DMAWrite
void CEXIMemoryCard::DMAWrite(u32 addr, u32 size) void CEXIMemoryCard::DMAWrite(u32 addr, u32 size)
{ {
m_memory_card->Write(m_address, size, Memory::GetPointer(addr)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
m_memory_card->Write(m_address, size, memory.GetPointer(addr));
if (((m_address + size) % Memcard::BLOCK_SIZE) == 0) if (((m_address + size) % Memcard::BLOCK_SIZE) == 0)
{ {
@ -548,7 +552,7 @@ void CEXIMemoryCard::DMAWrite(u32 addr, u32 size)
} }
// Schedule transfer complete later based on write speed // Schedule transfer complete later based on write speed
Core::System::GetInstance().GetCoreTiming().ScheduleEvent( system.GetCoreTiming().ScheduleEvent(
size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_WRITE), size * (SystemTimers::GetTicksPerSecond() / MC_TRANSFER_RATE_WRITE),
s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot)); s_et_transfer_complete[m_card_slot], static_cast<u64>(m_card_slot));
} }

View File

@ -83,9 +83,12 @@ void ResetGatherPipe()
void UpdateGatherPipe() void UpdateGatherPipe()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
size_t pipe_count = GetGatherPipeCount(); size_t pipe_count = GetGatherPipeCount();
size_t processed; size_t processed;
u8* cur_mem = Memory::GetPointer(ProcessorInterface::Fifo_CPUWritePointer); u8* cur_mem = memory.GetPointer(ProcessorInterface::Fifo_CPUWritePointer);
for (processed = 0; pipe_count >= GATHER_PIPE_SIZE; processed += GATHER_PIPE_SIZE) for (processed = 0; pipe_count >= GATHER_PIPE_SIZE; processed += GATHER_PIPE_SIZE)
{ {
// copy the GatherPipe // copy the GatherPipe
@ -96,7 +99,7 @@ void UpdateGatherPipe()
if (ProcessorInterface::Fifo_CPUWritePointer == ProcessorInterface::Fifo_CPUEnd) if (ProcessorInterface::Fifo_CPUWritePointer == ProcessorInterface::Fifo_CPUEnd)
{ {
ProcessorInterface::Fifo_CPUWritePointer = ProcessorInterface::Fifo_CPUBase; ProcessorInterface::Fifo_CPUWritePointer = ProcessorInterface::Fifo_CPUBase;
cur_mem = Memory::GetPointer(ProcessorInterface::Fifo_CPUWritePointer); cur_mem = memory.GetPointer(ProcessorInterface::Fifo_CPUWritePointer);
} }
else else
{ {
@ -104,7 +107,6 @@ void UpdateGatherPipe()
ProcessorInterface::Fifo_CPUWritePointer += GATHER_PIPE_SIZE; ProcessorInterface::Fifo_CPUWritePointer += GATHER_PIPE_SIZE;
} }
auto& system = Core::System::GetInstance();
system.GetCommandProcessor().GatherPipeBursted(system); system.GetCommandProcessor().GatherPipeBursted(system);
} }

View File

@ -33,7 +33,8 @@ namespace HW
{ {
void Init(const Sram* override_sram) void Init(const Sram* override_sram)
{ {
Core::System::GetInstance().GetCoreTiming().Init(); auto& system = Core::System::GetInstance();
system.GetCoreTiming().Init();
SystemTimers::PreInit(); SystemTimers::PreInit();
State::Init(); State::Init();
@ -45,7 +46,7 @@ void Init(const Sram* override_sram)
ProcessorInterface::Init(); ProcessorInterface::Init();
ExpansionInterface::Init(override_sram); // Needs to be initialized before Memory ExpansionInterface::Init(override_sram); // Needs to be initialized before Memory
HSP::Init(); HSP::Init();
Memory::Init(); // Needs to be initialized before AddressSpace system.GetMemory().Init(); // Needs to be initialized before AddressSpace
AddressSpace::Init(); AddressSpace::Init();
MemoryInterface::Init(); MemoryInterface::Init();
DSP::Init(Config::Get(Config::MAIN_DSP_HLE)); DSP::Init(Config::Get(Config::MAIN_DSP_HLE));
@ -63,6 +64,8 @@ void Init(const Sram* override_sram)
void Shutdown() void Shutdown()
{ {
auto& system = Core::System::GetInstance();
// IOS should always be shut down regardless of bWii because it can be running in GC mode (MIOS). // IOS should always be shut down regardless of bWii because it can be running in GC mode (MIOS).
IOS::HLE::Shutdown(); // Depends on Memory IOS::HLE::Shutdown(); // Depends on Memory
IOS::Shutdown(); IOS::Shutdown();
@ -73,19 +76,20 @@ void Shutdown()
DSP::Shutdown(); DSP::Shutdown();
MemoryInterface::Shutdown(); MemoryInterface::Shutdown();
AddressSpace::Shutdown(); AddressSpace::Shutdown();
Memory::Shutdown(); system.GetMemory().Shutdown();
HSP::Shutdown(); HSP::Shutdown();
ExpansionInterface::Shutdown(); ExpansionInterface::Shutdown();
SerialInterface::Shutdown(); SerialInterface::Shutdown();
AudioInterface::Shutdown(); AudioInterface::Shutdown();
State::Shutdown(); State::Shutdown();
Core::System::GetInstance().GetCoreTiming().Shutdown(); system.GetCoreTiming().Shutdown();
} }
void DoState(PointerWrap& p) void DoState(PointerWrap& p)
{ {
Memory::DoState(p); auto& system = Core::System::GetInstance();
system.GetMemory().DoState(p);
p.DoMarker("Memory"); p.DoMarker("Memory");
MemoryInterface::DoState(p); MemoryInterface::DoState(p);
p.DoMarker("MemoryInterface"); p.DoMarker("MemoryInterface");

View File

@ -41,190 +41,35 @@
namespace Memory namespace Memory
{ {
// ================================= MemoryManager::MemoryManager() = default;
// Init() declarations MemoryManager::~MemoryManager() = default;
// ----------------
// Store the MemArena here
u8* physical_base = nullptr;
u8* logical_base = nullptr;
u8* physical_page_mappings_base = nullptr;
u8* logical_page_mappings_base = nullptr;
static bool is_fastmem_arena_initialized = false;
// The MemArena class void MemoryManager::InitMMIO(bool is_wii)
static Common::MemArena g_arena;
// ==============
// STATE_TO_SAVE
static bool m_IsInitialized = false; // Save the Init(), Shutdown() state
// END STATE_TO_SAVE
u8* m_pRAM;
u8* m_pL1Cache;
u8* m_pEXRAM;
u8* m_pFakeVMEM;
// s_ram_size is the amount allocated by the emulator, whereas s_ram_size_real
// is what will be reported in lowmem, and thus used by emulated software.
// Note: Writing to lowmem is done by IPL. If using retail IPL, it will
// always be set to 24MB.
static u32 s_ram_size_real;
static u32 s_ram_size;
static u32 s_ram_mask;
static u32 s_fakevmem_size;
static u32 s_fakevmem_mask;
static u32 s_L1_cache_size;
static u32 s_L1_cache_mask;
// s_exram_size is the amount allocated by the emulator, whereas s_exram_size_real
// is what gets used by emulated software. If using retail IOS, it will
// always be set to 64MB.
static u32 s_exram_size_real;
static u32 s_exram_size;
static u32 s_exram_mask;
u32 GetRamSizeReal()
{ {
return s_ram_size_real; m_mmio_mapping = std::make_unique<MMIO::Mapping>();
}
u32 GetRamSize()
{
return s_ram_size;
}
u32 GetRamMask()
{
return s_ram_mask;
}
u32 GetFakeVMemSize()
{
return s_fakevmem_size;
}
u32 GetFakeVMemMask()
{
return s_fakevmem_mask;
}
u32 GetL1CacheSize()
{
return s_L1_cache_size;
}
u32 GetL1CacheMask()
{
return s_L1_cache_mask;
}
u32 GetExRamSizeReal()
{
return s_exram_size_real;
}
u32 GetExRamSize()
{
return s_exram_size;
}
u32 GetExRamMask()
{
return s_exram_mask;
}
// MMIO mapping object.
std::unique_ptr<MMIO::Mapping> mmio_mapping;
static void InitMMIO(bool is_wii)
{
mmio_mapping = std::make_unique<MMIO::Mapping>();
auto& system = Core::System::GetInstance(); auto& system = Core::System::GetInstance();
system.GetCommandProcessor().RegisterMMIO(system, mmio_mapping.get(), 0x0C000000); system.GetCommandProcessor().RegisterMMIO(system, m_mmio_mapping.get(), 0x0C000000);
PixelEngine::RegisterMMIO(mmio_mapping.get(), 0x0C001000); PixelEngine::RegisterMMIO(m_mmio_mapping.get(), 0x0C001000);
VideoInterface::RegisterMMIO(mmio_mapping.get(), 0x0C002000); VideoInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C002000);
ProcessorInterface::RegisterMMIO(mmio_mapping.get(), 0x0C003000); ProcessorInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C003000);
MemoryInterface::RegisterMMIO(mmio_mapping.get(), 0x0C004000); MemoryInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C004000);
DSP::RegisterMMIO(mmio_mapping.get(), 0x0C005000); DSP::RegisterMMIO(m_mmio_mapping.get(), 0x0C005000);
DVDInterface::RegisterMMIO(mmio_mapping.get(), 0x0C006000, false); DVDInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C006000, false);
SerialInterface::RegisterMMIO(mmio_mapping.get(), 0x0C006400); SerialInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C006400);
ExpansionInterface::RegisterMMIO(mmio_mapping.get(), 0x0C006800); ExpansionInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C006800);
AudioInterface::RegisterMMIO(mmio_mapping.get(), 0x0C006C00); AudioInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0C006C00);
if (is_wii) if (is_wii)
{ {
IOS::RegisterMMIO(mmio_mapping.get(), 0x0D000000); IOS::RegisterMMIO(m_mmio_mapping.get(), 0x0D000000);
DVDInterface::RegisterMMIO(mmio_mapping.get(), 0x0D006000, true); DVDInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0D006000, true);
SerialInterface::RegisterMMIO(mmio_mapping.get(), 0x0D006400); SerialInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0D006400);
ExpansionInterface::RegisterMMIO(mmio_mapping.get(), 0x0D006800); ExpansionInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0D006800);
AudioInterface::RegisterMMIO(mmio_mapping.get(), 0x0D006C00); AudioInterface::RegisterMMIO(m_mmio_mapping.get(), 0x0D006C00);
} }
} }
bool IsInitialized() void MemoryManager::Init()
{
return m_IsInitialized;
}
struct PhysicalMemoryRegion
{
u8** out_pointer;
u32 physical_address;
u32 size;
enum : u32
{
ALWAYS = 0,
FAKE_VMEM = 1,
WII_ONLY = 2,
} flags;
u32 shm_position;
bool active;
};
struct LogicalMemoryView
{
void* mapped_pointer;
u32 mapped_size;
};
// Dolphin allocates memory to represent four regions:
// - 32MB RAM (actually 24MB on hardware), available on GameCube and Wii
// - 64MB "EXRAM", RAM only available on Wii
// - 32MB FakeVMem, allocated in GameCube mode when MMU support is turned off.
// This is used to approximate the behavior of a common library which pages
// memory to and from the DSP's dedicated RAM. The DSP's RAM (ARAM) isn't
// directly addressable on GameCube.
// - 256KB Locked L1, to represent cache lines allocated out of the L1 data
// cache in Locked L1 mode. Dolphin does not emulate this hardware feature
// accurately; it just pretends there is extra memory at 0xE0000000.
//
// The 4GB starting at physical_base represents access from the CPU
// with address translation turned off. (This is only used by the CPU;
// other devices, like the GPU, use other rules, approximated by
// Memory::GetPointer.) This memory is laid out as follows:
// [0x00000000, 0x02000000) - 32MB RAM
// [0x02000000, 0x08000000) - Mirrors of 32MB RAM (not handled here)
// [0x08000000, 0x0C000000) - EFB "mapping" (not handled here)
// [0x0C000000, 0x0E000000) - MMIO etc. (not handled here)
// [0x10000000, 0x14000000) - 64MB RAM (Wii-only; slightly slower)
// [0x7E000000, 0x80000000) - FakeVMEM
// [0xE0000000, 0xE0040000) - 256KB locked L1
//
// The 4GB starting at logical_base represents access from the CPU
// with address translation turned on. This mapping is computed based
// on the BAT registers.
//
// Each of these 4GB regions is followed by 4GB of empty space so overflows
// in address computation in the JIT don't access the wrong memory.
//
// The neighboring mirrors of RAM ([0x02000000, 0x08000000), etc.) exist because
// the bus masks off the bits in question for RAM accesses; using them is a
// terrible idea because the CPU cache won't handle them correctly, but a
// few buggy games (notably Rogue Squadron 2) use them by accident. They
// aren't backed by memory mappings because they are used very rarely.
//
// Dolphin doesn't emulate the difference between cached and uncached access.
//
// TODO: The actual size of RAM is 24MB; the other 8MB shouldn't be backed by actual memory.
// TODO: Do we want to handle the mirrors of the GC RAM?
static std::array<PhysicalMemoryRegion, 4> s_physical_regions;
static std::vector<LogicalMemoryView> logical_mapped_entries;
static std::array<void*, PowerPC::BAT_PAGE_COUNT> s_physical_page_mappings;
static std::array<void*, PowerPC::BAT_PAGE_COUNT> s_logical_page_mappings;
void Init()
{ {
const auto get_mem1_size = [] { const auto get_mem1_size = [] {
if (Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE)) if (Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE))
@ -236,25 +81,25 @@ void Init()
return Config::Get(Config::MAIN_MEM2_SIZE); return Config::Get(Config::MAIN_MEM2_SIZE);
return Memory::MEM2_SIZE_RETAIL; return Memory::MEM2_SIZE_RETAIL;
}; };
s_ram_size_real = get_mem1_size(); m_ram_size_real = get_mem1_size();
s_ram_size = MathUtil::NextPowerOf2(GetRamSizeReal()); m_ram_size = MathUtil::NextPowerOf2(GetRamSizeReal());
s_ram_mask = GetRamSize() - 1; m_ram_mask = GetRamSize() - 1;
s_fakevmem_size = 0x02000000; m_fakevmem_size = 0x02000000;
s_fakevmem_mask = GetFakeVMemSize() - 1; m_fakevmem_mask = GetFakeVMemSize() - 1;
s_L1_cache_size = 0x00040000; m_l1_cache_size = 0x00040000;
s_L1_cache_mask = GetL1CacheSize() - 1; m_l1_cache_mask = GetL1CacheSize() - 1;
s_exram_size_real = get_mem2_size(); m_exram_size_real = get_mem2_size();
s_exram_size = MathUtil::NextPowerOf2(GetExRamSizeReal()); m_exram_size = MathUtil::NextPowerOf2(GetExRamSizeReal());
s_exram_mask = GetExRamSize() - 1; m_exram_mask = GetExRamSize() - 1;
s_physical_regions[0] = PhysicalMemoryRegion{ m_physical_regions[0] = PhysicalMemoryRegion{
&m_pRAM, 0x00000000, GetRamSize(), PhysicalMemoryRegion::ALWAYS, 0, false}; &m_ram, 0x00000000, GetRamSize(), PhysicalMemoryRegion::ALWAYS, 0, false};
s_physical_regions[1] = PhysicalMemoryRegion{ m_physical_regions[1] = PhysicalMemoryRegion{
&m_pL1Cache, 0xE0000000, GetL1CacheSize(), PhysicalMemoryRegion::ALWAYS, 0, false}; &m_l1_cache, 0xE0000000, GetL1CacheSize(), PhysicalMemoryRegion::ALWAYS, 0, false};
s_physical_regions[2] = PhysicalMemoryRegion{ m_physical_regions[2] = PhysicalMemoryRegion{
&m_pFakeVMEM, 0x7E000000, GetFakeVMemSize(), PhysicalMemoryRegion::FAKE_VMEM, 0, false}; &m_fake_vmem, 0x7E000000, GetFakeVMemSize(), PhysicalMemoryRegion::FAKE_VMEM, 0, false};
s_physical_regions[3] = PhysicalMemoryRegion{ m_physical_regions[3] = PhysicalMemoryRegion{
&m_pEXRAM, 0x10000000, GetExRamSize(), PhysicalMemoryRegion::WII_ONLY, 0, false}; &m_exram, 0x10000000, GetExRamSize(), PhysicalMemoryRegion::WII_ONLY, 0, false};
const bool wii = SConfig::GetInstance().bWii; const bool wii = SConfig::GetInstance().bWii;
const bool mmu = Core::System::GetInstance().IsMMUMode(); const bool mmu = Core::System::GetInstance().IsMMUMode();
@ -268,7 +113,7 @@ void Init()
#endif #endif
u32 mem_size = 0; u32 mem_size = 0;
for (PhysicalMemoryRegion& region : s_physical_regions) for (PhysicalMemoryRegion& region : m_physical_regions)
{ {
if (!wii && (region.flags & PhysicalMemoryRegion::WII_ONLY)) if (!wii && (region.flags & PhysicalMemoryRegion::WII_ONLY))
continue; continue;
@ -279,17 +124,17 @@ void Init()
region.active = true; region.active = true;
mem_size += region.size; mem_size += region.size;
} }
g_arena.GrabSHMSegment(mem_size); m_arena.GrabSHMSegment(mem_size);
s_physical_page_mappings.fill(nullptr); m_physical_page_mappings.fill(nullptr);
// Create an anonymous view of the physical memory // Create an anonymous view of the physical memory
for (const PhysicalMemoryRegion& region : s_physical_regions) for (const PhysicalMemoryRegion& region : m_physical_regions)
{ {
if (!region.active) if (!region.active)
continue; continue;
*region.out_pointer = (u8*)g_arena.CreateView(region.shm_position, region.size); *region.out_pointer = (u8*)m_arena.CreateView(region.shm_position, region.size);
if (!*region.out_pointer) if (!*region.out_pointer)
{ {
@ -302,43 +147,43 @@ void Init()
for (u32 i = 0; i < region.size; i += PowerPC::BAT_PAGE_SIZE) for (u32 i = 0; i < region.size; i += PowerPC::BAT_PAGE_SIZE)
{ {
const size_t index = (i + region.physical_address) >> PowerPC::BAT_INDEX_SHIFT; const size_t index = (i + region.physical_address) >> PowerPC::BAT_INDEX_SHIFT;
s_physical_page_mappings[index] = *region.out_pointer + i; m_physical_page_mappings[index] = *region.out_pointer + i;
} }
} }
physical_page_mappings_base = reinterpret_cast<u8*>(s_physical_page_mappings.data()); m_physical_page_mappings_base = reinterpret_cast<u8*>(m_physical_page_mappings.data());
logical_page_mappings_base = reinterpret_cast<u8*>(s_logical_page_mappings.data()); m_logical_page_mappings_base = reinterpret_cast<u8*>(m_logical_page_mappings.data());
InitMMIO(wii); InitMMIO(wii);
Clear(); Clear();
INFO_LOG_FMT(MEMMAP, "Memory system initialized. RAM at {}", fmt::ptr(m_pRAM)); INFO_LOG_FMT(MEMMAP, "Memory system initialized. RAM at {}", fmt::ptr(m_ram));
m_IsInitialized = true; m_is_initialized = true;
} }
bool InitFastmemArena() bool MemoryManager::InitFastmemArena()
{ {
#if _ARCH_32 #if _ARCH_32
const size_t memory_size = 0x31000000; const size_t memory_size = 0x31000000;
#else #else
const size_t memory_size = 0x400000000; const size_t memory_size = 0x400000000;
#endif #endif
physical_base = g_arena.ReserveMemoryRegion(memory_size); m_physical_base = m_arena.ReserveMemoryRegion(memory_size);
if (!physical_base) if (!m_physical_base)
{ {
PanicAlertFmt("Memory::InitFastmemArena(): Failed finding a memory base."); PanicAlertFmt("Memory::InitFastmemArena(): Failed finding a memory base.");
return false; return false;
} }
for (const PhysicalMemoryRegion& region : s_physical_regions) for (const PhysicalMemoryRegion& region : m_physical_regions)
{ {
if (!region.active) if (!region.active)
continue; continue;
u8* base = physical_base + region.physical_address; u8* base = m_physical_base + region.physical_address;
u8* view = (u8*)g_arena.MapInMemoryRegion(region.shm_position, region.size, base); u8* view = (u8*)m_arena.MapInMemoryRegion(region.shm_position, region.size, base);
if (base != view) if (base != view)
{ {
@ -350,22 +195,22 @@ bool InitFastmemArena()
} }
#ifndef _ARCH_32 #ifndef _ARCH_32
logical_base = physical_base + 0x200000000; m_logical_base = m_physical_base + 0x200000000;
#endif #endif
is_fastmem_arena_initialized = true; m_is_fastmem_arena_initialized = true;
return true; return true;
} }
void UpdateLogicalMemory(const PowerPC::BatTable& dbat_table) void MemoryManager::UpdateLogicalMemory(const PowerPC::BatTable& dbat_table)
{ {
for (auto& entry : logical_mapped_entries) for (auto& entry : m_logical_mapped_entries)
{ {
g_arena.UnmapFromMemoryRegion(entry.mapped_pointer, entry.mapped_size); m_arena.UnmapFromMemoryRegion(entry.mapped_pointer, entry.mapped_size);
} }
logical_mapped_entries.clear(); m_logical_mapped_entries.clear();
s_logical_page_mappings.fill(nullptr); m_logical_page_mappings.fill(nullptr);
for (u32 i = 0; i < dbat_table.size(); ++i) for (u32 i = 0; i < dbat_table.size(); ++i)
{ {
@ -375,7 +220,7 @@ void UpdateLogicalMemory(const PowerPC::BatTable& dbat_table)
// TODO: Merge adjacent mappings to make this faster. // TODO: Merge adjacent mappings to make this faster.
u32 logical_size = PowerPC::BAT_PAGE_SIZE; u32 logical_size = PowerPC::BAT_PAGE_SIZE;
u32 translated_address = dbat_table[i] & PowerPC::BAT_RESULT_MASK; u32 translated_address = dbat_table[i] & PowerPC::BAT_RESULT_MASK;
for (const auto& physical_region : s_physical_regions) for (const auto& physical_region : m_physical_regions)
{ {
if (!physical_region.active) if (!physical_region.active)
continue; continue;
@ -388,13 +233,13 @@ void UpdateLogicalMemory(const PowerPC::BatTable& dbat_table)
{ {
// Found an overlapping region; map it. // Found an overlapping region; map it.
if (is_fastmem_arena_initialized) if (m_is_fastmem_arena_initialized)
{ {
u32 position = physical_region.shm_position + intersection_start - mapping_address; u32 position = physical_region.shm_position + intersection_start - mapping_address;
u8* base = logical_base + logical_address + intersection_start - translated_address; u8* base = m_logical_base + logical_address + intersection_start - translated_address;
u32 mapped_size = intersection_end - intersection_start; u32 mapped_size = intersection_end - intersection_start;
void* mapped_pointer = g_arena.MapInMemoryRegion(position, mapped_size, base); void* mapped_pointer = m_arena.MapInMemoryRegion(position, mapped_size, base);
if (!mapped_pointer) if (!mapped_pointer)
{ {
PanicAlertFmt( PanicAlertFmt(
@ -403,10 +248,10 @@ void UpdateLogicalMemory(const PowerPC::BatTable& dbat_table)
intersection_start, mapped_size, logical_address); intersection_start, mapped_size, logical_address);
exit(0); exit(0);
} }
logical_mapped_entries.push_back({mapped_pointer, mapped_size}); m_logical_mapped_entries.push_back({mapped_pointer, mapped_size});
} }
s_logical_page_mappings[i] = m_logical_page_mappings[i] =
*physical_region.out_pointer + intersection_start - mapping_address; *physical_region.out_pointer + intersection_start - mapping_address;
} }
} }
@ -414,13 +259,13 @@ void UpdateLogicalMemory(const PowerPC::BatTable& dbat_table)
} }
} }
void DoState(PointerWrap& p) void MemoryManager::DoState(PointerWrap& p)
{ {
const u32 current_ram_size = GetRamSize(); const u32 current_ram_size = GetRamSize();
const u32 current_l1_cache_size = GetL1CacheSize(); const u32 current_l1_cache_size = GetL1CacheSize();
const bool current_have_fake_vmem = !!m_pFakeVMEM; const bool current_have_fake_vmem = !!m_fake_vmem;
const u32 current_fake_vmem_size = current_have_fake_vmem ? GetFakeVMemSize() : 0; const u32 current_fake_vmem_size = current_have_fake_vmem ? GetFakeVMemSize() : 0;
const bool current_have_exram = !!m_pEXRAM; const bool current_have_exram = !!m_exram;
const u32 current_exram_size = current_have_exram ? GetExRamSize() : 0; const u32 current_exram_size = current_have_exram ? GetExRamSize() : 0;
u32 state_ram_size = current_ram_size; u32 state_ram_size = current_ram_size;
@ -452,76 +297,76 @@ void DoState(PointerWrap& p)
return; return;
} }
p.DoArray(m_pRAM, current_ram_size); p.DoArray(m_ram, current_ram_size);
p.DoArray(m_pL1Cache, current_l1_cache_size); p.DoArray(m_l1_cache, current_l1_cache_size);
p.DoMarker("Memory RAM"); p.DoMarker("Memory RAM");
if (current_have_fake_vmem) if (current_have_fake_vmem)
p.DoArray(m_pFakeVMEM, current_fake_vmem_size); p.DoArray(m_fake_vmem, current_fake_vmem_size);
p.DoMarker("Memory FakeVMEM"); p.DoMarker("Memory FakeVMEM");
if (current_have_exram) if (current_have_exram)
p.DoArray(m_pEXRAM, current_exram_size); p.DoArray(m_exram, current_exram_size);
p.DoMarker("Memory EXRAM"); p.DoMarker("Memory EXRAM");
} }
void Shutdown() void MemoryManager::Shutdown()
{ {
ShutdownFastmemArena(); ShutdownFastmemArena();
m_IsInitialized = false; m_is_initialized = false;
for (const PhysicalMemoryRegion& region : s_physical_regions) for (const PhysicalMemoryRegion& region : m_physical_regions)
{ {
if (!region.active) if (!region.active)
continue; continue;
g_arena.ReleaseView(*region.out_pointer, region.size); m_arena.ReleaseView(*region.out_pointer, region.size);
*region.out_pointer = nullptr; *region.out_pointer = nullptr;
} }
g_arena.ReleaseSHMSegment(); m_arena.ReleaseSHMSegment();
mmio_mapping.reset(); m_mmio_mapping.reset();
INFO_LOG_FMT(MEMMAP, "Memory system shut down."); INFO_LOG_FMT(MEMMAP, "Memory system shut down.");
} }
void ShutdownFastmemArena() void MemoryManager::ShutdownFastmemArena()
{ {
if (!is_fastmem_arena_initialized) if (!m_is_fastmem_arena_initialized)
return; return;
for (const PhysicalMemoryRegion& region : s_physical_regions) for (const PhysicalMemoryRegion& region : m_physical_regions)
{ {
if (!region.active) if (!region.active)
continue; continue;
u8* base = physical_base + region.physical_address; u8* base = m_physical_base + region.physical_address;
g_arena.UnmapFromMemoryRegion(base, region.size); m_arena.UnmapFromMemoryRegion(base, region.size);
} }
for (auto& entry : logical_mapped_entries) for (auto& entry : m_logical_mapped_entries)
{ {
g_arena.UnmapFromMemoryRegion(entry.mapped_pointer, entry.mapped_size); m_arena.UnmapFromMemoryRegion(entry.mapped_pointer, entry.mapped_size);
} }
logical_mapped_entries.clear(); m_logical_mapped_entries.clear();
g_arena.ReleaseMemoryRegion(); m_arena.ReleaseMemoryRegion();
physical_base = nullptr; m_physical_base = nullptr;
logical_base = nullptr; m_logical_base = nullptr;
is_fastmem_arena_initialized = false; m_is_fastmem_arena_initialized = false;
} }
void Clear() void MemoryManager::Clear()
{ {
if (m_pRAM) if (m_ram)
memset(m_pRAM, 0, GetRamSize()); memset(m_ram, 0, GetRamSize());
if (m_pL1Cache) if (m_l1_cache)
memset(m_pL1Cache, 0, GetL1CacheSize()); memset(m_l1_cache, 0, GetL1CacheSize());
if (m_pFakeVMEM) if (m_fake_vmem)
memset(m_pFakeVMEM, 0, GetFakeVMemSize()); memset(m_fake_vmem, 0, GetFakeVMemSize());
if (m_pEXRAM) if (m_exram)
memset(m_pEXRAM, 0, GetExRamSize()); memset(m_exram, 0, GetExRamSize());
} }
u8* GetPointerForRange(u32 address, size_t size) u8* MemoryManager::GetPointerForRange(u32 address, size_t size) const
{ {
// Make sure we don't have a range spanning 2 separate banks // Make sure we don't have a range spanning 2 separate banks
if (size >= GetExRamSizeReal()) if (size >= GetExRamSizeReal())
@ -541,7 +386,7 @@ u8* GetPointerForRange(u32 address, size_t size)
return pointer; return pointer;
} }
void CopyFromEmu(void* data, u32 address, size_t size) void MemoryManager::CopyFromEmu(void* data, u32 address, size_t size) const
{ {
if (size == 0) if (size == 0)
return; return;
@ -555,7 +400,7 @@ void CopyFromEmu(void* data, u32 address, size_t size)
memcpy(data, pointer, size); memcpy(data, pointer, size);
} }
void CopyToEmu(u32 address, const void* data, size_t size) void MemoryManager::CopyToEmu(u32 address, const void* data, size_t size)
{ {
if (size == 0) if (size == 0)
return; return;
@ -569,7 +414,7 @@ void CopyToEmu(u32 address, const void* data, size_t size)
memcpy(pointer, data, size); memcpy(pointer, data, size);
} }
void Memset(u32 address, u8 value, size_t size) void MemoryManager::Memset(u32 address, u8 value, size_t size)
{ {
if (size == 0) if (size == 0)
return; return;
@ -583,7 +428,7 @@ void Memset(u32 address, u8 value, size_t size)
memset(pointer, value, size); memset(pointer, value, size);
} }
std::string GetString(u32 em_address, size_t size) std::string MemoryManager::GetString(u32 em_address, size_t size)
{ {
const char* ptr = reinterpret_cast<const char*>(GetPointer(em_address)); const char* ptr = reinterpret_cast<const char*>(GetPointer(em_address));
if (ptr == nullptr) if (ptr == nullptr)
@ -600,81 +445,81 @@ std::string GetString(u32 em_address, size_t size)
} }
} }
u8* GetPointer(u32 address) u8* MemoryManager::GetPointer(u32 address) const
{ {
// TODO: Should we be masking off more bits here? Can all devices access // TODO: Should we be masking off more bits here? Can all devices access
// EXRAM? // EXRAM?
address &= 0x3FFFFFFF; address &= 0x3FFFFFFF;
if (address < GetRamSizeReal()) if (address < GetRamSizeReal())
return m_pRAM + address; return m_ram + address;
if (m_pEXRAM) if (m_exram)
{ {
if ((address >> 28) == 0x1 && (address & 0x0fffffff) < GetExRamSizeReal()) if ((address >> 28) == 0x1 && (address & 0x0fffffff) < GetExRamSizeReal())
return m_pEXRAM + (address & GetExRamMask()); return m_exram + (address & GetExRamMask());
} }
PanicAlertFmt("Unknown Pointer {:#010x} PC {:#010x} LR {:#010x}", address, PC, LR); PanicAlertFmt("Unknown Pointer {:#010x} PC {:#010x} LR {:#010x}", address, PC, LR);
return nullptr; return nullptr;
} }
u8 Read_U8(u32 address) u8 MemoryManager::Read_U8(u32 address) const
{ {
u8 value = 0; u8 value = 0;
CopyFromEmu(&value, address, sizeof(value)); CopyFromEmu(&value, address, sizeof(value));
return value; return value;
} }
u16 Read_U16(u32 address) u16 MemoryManager::Read_U16(u32 address) const
{ {
u16 value = 0; u16 value = 0;
CopyFromEmu(&value, address, sizeof(value)); CopyFromEmu(&value, address, sizeof(value));
return Common::swap16(value); return Common::swap16(value);
} }
u32 Read_U32(u32 address) u32 MemoryManager::Read_U32(u32 address) const
{ {
u32 value = 0; u32 value = 0;
CopyFromEmu(&value, address, sizeof(value)); CopyFromEmu(&value, address, sizeof(value));
return Common::swap32(value); return Common::swap32(value);
} }
u64 Read_U64(u32 address) u64 MemoryManager::Read_U64(u32 address) const
{ {
u64 value = 0; u64 value = 0;
CopyFromEmu(&value, address, sizeof(value)); CopyFromEmu(&value, address, sizeof(value));
return Common::swap64(value); return Common::swap64(value);
} }
void Write_U8(u8 value, u32 address) void MemoryManager::Write_U8(u8 value, u32 address)
{ {
CopyToEmu(address, &value, sizeof(value)); CopyToEmu(address, &value, sizeof(value));
} }
void Write_U16(u16 value, u32 address) void MemoryManager::Write_U16(u16 value, u32 address)
{ {
u16 swapped_value = Common::swap16(value); u16 swapped_value = Common::swap16(value);
CopyToEmu(address, &swapped_value, sizeof(swapped_value)); CopyToEmu(address, &swapped_value, sizeof(swapped_value));
} }
void Write_U32(u32 value, u32 address) void MemoryManager::Write_U32(u32 value, u32 address)
{ {
u32 swapped_value = Common::swap32(value); u32 swapped_value = Common::swap32(value);
CopyToEmu(address, &swapped_value, sizeof(swapped_value)); CopyToEmu(address, &swapped_value, sizeof(swapped_value));
} }
void Write_U64(u64 value, u32 address) void MemoryManager::Write_U64(u64 value, u32 address)
{ {
u64 swapped_value = Common::swap64(value); u64 swapped_value = Common::swap64(value);
CopyToEmu(address, &swapped_value, sizeof(swapped_value)); CopyToEmu(address, &swapped_value, sizeof(swapped_value));
} }
void Write_U32_Swap(u32 value, u32 address) void MemoryManager::Write_U32_Swap(u32 value, u32 address)
{ {
CopyToEmu(address, &value, sizeof(value)); CopyToEmu(address, &value, sizeof(value));
} }
void Write_U64_Swap(u64 value, u32 address) void MemoryManager::Write_U64_Swap(u64 value, u32 address)
{ {
CopyToEmu(address, &value, sizeof(value)); CopyToEmu(address, &value, sizeof(value));
} }

View File

@ -3,11 +3,14 @@
#pragma once #pragma once
#include <array>
#include <memory> #include <memory>
#include <string> #include <string>
#include <vector>
#include "Common/CommonTypes.h" #include "Common/CommonTypes.h"
#include "Common/MathUtil.h" #include "Common/MathUtil.h"
#include "Common/MemArena.h"
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
@ -20,35 +23,6 @@ class Mapping;
namespace Memory namespace Memory
{ {
// Base is a pointer to the base of the memory map. Yes, some MMU tricks
// are used to set up a full GC or Wii memory map in process memory.
// In 64-bit, this might point to "high memory" (above the 32-bit limit),
// so be sure to load it into a 64-bit register.
extern u8* physical_base;
extern u8* logical_base;
// This page table is used for a "soft MMU" implementation when
// setting up the full memory map in process memory isn't possible.
extern u8* physical_page_mappings_base;
extern u8* logical_page_mappings_base;
// The actual memory used for backing the memory map.
extern u8* m_pRAM;
extern u8* m_pEXRAM;
extern u8* m_pL1Cache;
extern u8* m_pFakeVMEM;
u32 GetRamSizeReal();
u32 GetRamSize();
u32 GetRamMask();
u32 GetFakeVMemSize();
u32 GetFakeVMemMask();
u32 GetL1CacheSize();
u32 GetL1CacheMask();
u32 GetExRamSizeReal();
u32 GetExRamSize();
u32 GetExRamMask();
constexpr u32 MEM1_BASE_ADDR = 0x80000000U; constexpr u32 MEM1_BASE_ADDR = 0x80000000U;
constexpr u32 MEM2_BASE_ADDR = 0x90000000U; constexpr u32 MEM2_BASE_ADDR = 0x90000000U;
constexpr u32 MEM1_SIZE_RETAIL = 0x01800000U; constexpr u32 MEM1_SIZE_RETAIL = 0x01800000U;
@ -56,11 +30,63 @@ constexpr u32 MEM1_SIZE_GDEV = 0x04000000U;
constexpr u32 MEM2_SIZE_RETAIL = 0x04000000U; constexpr u32 MEM2_SIZE_RETAIL = 0x04000000U;
constexpr u32 MEM2_SIZE_NDEV = 0x08000000U; constexpr u32 MEM2_SIZE_NDEV = 0x08000000U;
// MMIO mapping object. struct PhysicalMemoryRegion
extern std::unique_ptr<MMIO::Mapping> mmio_mapping; {
u8** out_pointer;
u32 physical_address;
u32 size;
enum : u32
{
ALWAYS = 0,
FAKE_VMEM = 1,
WII_ONLY = 2,
} flags;
u32 shm_position;
bool active;
};
struct LogicalMemoryView
{
void* mapped_pointer;
u32 mapped_size;
};
class MemoryManager
{
public:
MemoryManager();
MemoryManager(const MemoryManager& other) = delete;
MemoryManager(MemoryManager&& other) = delete;
MemoryManager& operator=(const MemoryManager& other) = delete;
MemoryManager& operator=(MemoryManager&& other) = delete;
~MemoryManager();
u32 GetRamSizeReal() const { return m_ram_size_real; }
u32 GetRamSize() const { return m_ram_size; }
u32 GetRamMask() const { return m_ram_mask; }
u32 GetFakeVMemSize() const { return m_fakevmem_size; }
u32 GetFakeVMemMask() const { return m_fakevmem_mask; }
u32 GetL1CacheSize() const { return m_l1_cache_size; }
u32 GetL1CacheMask() const { return m_l1_cache_mask; }
u32 GetExRamSizeReal() const { return m_exram_size_real; }
u32 GetExRamSize() const { return m_exram_size; }
u32 GetExRamMask() const { return m_exram_mask; }
u8* GetPhysicalBase() const { return m_physical_base; }
u8* GetLogicalBase() const { return m_logical_base; }
u8* GetPhysicalPageMappingsBase() const { return m_physical_page_mappings_base; }
u8* GetLogicalPageMappingsBase() const { return m_logical_page_mappings_base; }
// FIXME: these should not return their address, but AddressSpace wants that
u8*& GetRAM() { return m_ram; }
u8*& GetEXRAM() { return m_exram; }
u8* GetL1Cache() { return m_l1_cache; }
u8*& GetFakeVMEM() { return m_fake_vmem; }
MMIO::Mapping* GetMMIOMapping() const { return m_mmio_mapping.get(); }
// Init and Shutdown // Init and Shutdown
bool IsInitialized(); bool IsInitialized() const { return m_is_initialized; }
void Init(); void Init();
void Shutdown(); void Shutdown();
bool InitFastmemArena(); bool InitFastmemArena();
@ -74,15 +100,15 @@ void Clear();
// Routines to access physically addressed memory, designed for use by // Routines to access physically addressed memory, designed for use by
// emulated hardware outside the CPU. Use "Device_" prefix. // emulated hardware outside the CPU. Use "Device_" prefix.
std::string GetString(u32 em_address, size_t size = 0); std::string GetString(u32 em_address, size_t size = 0);
u8* GetPointer(u32 address); u8* GetPointer(u32 address) const;
u8* GetPointerForRange(u32 address, size_t size); u8* GetPointerForRange(u32 address, size_t size) const;
void CopyFromEmu(void* data, u32 address, size_t size); void CopyFromEmu(void* data, u32 address, size_t size) const;
void CopyToEmu(u32 address, const void* data, size_t size); void CopyToEmu(u32 address, const void* data, size_t size);
void Memset(u32 address, u8 value, size_t size); void Memset(u32 address, u8 value, size_t size);
u8 Read_U8(u32 address); u8 Read_U8(u32 address) const;
u16 Read_U16(u32 address); u16 Read_U16(u32 address) const;
u32 Read_U32(u32 address); u32 Read_U32(u32 address) const;
u64 Read_U64(u32 address); u64 Read_U64(u32 address) const;
void Write_U8(u8 var, u32 address); void Write_U8(u8 var, u32 address);
void Write_U16(u16 var, u32 address); void Write_U16(u16 var, u32 address);
void Write_U32(u32 var, u32 address); void Write_U32(u32 var, u32 address);
@ -92,7 +118,7 @@ void Write_U64_Swap(u64 var, u32 address);
// Templated functions for byteswapped copies. // Templated functions for byteswapped copies.
template <typename T> template <typename T>
void CopyFromEmuSwapped(T* data, u32 address, size_t size) void CopyFromEmuSwapped(T* data, u32 address, size_t size) const
{ {
const T* src = reinterpret_cast<T*>(GetPointerForRange(address, size)); const T* src = reinterpret_cast<T*>(GetPointerForRange(address, size));
@ -114,4 +140,104 @@ void CopyToEmuSwapped(u32 address, const T* data, size_t size)
for (size_t i = 0; i < size / sizeof(T); i++) for (size_t i = 0; i < size / sizeof(T); i++)
dest[i] = Common::FromBigEndian(data[i]); dest[i] = Common::FromBigEndian(data[i]);
} }
private:
// Base is a pointer to the base of the memory map. Yes, some MMU tricks
// are used to set up a full GC or Wii memory map in process memory.
// In 64-bit, this might point to "high memory" (above the 32-bit limit),
// so be sure to load it into a 64-bit register.
u8* m_physical_base = nullptr;
u8* m_logical_base = nullptr;
// This page table is used for a "soft MMU" implementation when
// setting up the full memory map in process memory isn't possible.
u8* m_physical_page_mappings_base = nullptr;
u8* m_logical_page_mappings_base = nullptr;
// The actual memory used for backing the memory map.
u8* m_ram;
u8* m_exram;
u8* m_l1_cache;
u8* m_fake_vmem;
// m_ram_size is the amount allocated by the emulator, whereas m_ram_size_real
// is what will be reported in lowmem, and thus used by emulated software.
// Note: Writing to lowmem is done by IPL. If using retail IPL, it will
// always be set to 24MB.
u32 m_ram_size_real;
u32 m_ram_size;
u32 m_ram_mask;
u32 m_fakevmem_size;
u32 m_fakevmem_mask;
u32 m_l1_cache_size;
u32 m_l1_cache_mask;
// m_exram_size is the amount allocated by the emulator, whereas m_exram_size_real
// is what gets used by emulated software. If using retail IOS, it will
// always be set to 64MB.
u32 m_exram_size_real;
u32 m_exram_size;
u32 m_exram_mask;
bool m_is_fastmem_arena_initialized = false;
// STATE_TO_SAVE
// Save the Init(), Shutdown() state
bool m_is_initialized = false;
// END STATE_TO_SAVE
// MMIO mapping object.
std::unique_ptr<MMIO::Mapping> m_mmio_mapping;
// The MemArena class
Common::MemArena m_arena;
// Dolphin allocates memory to represent four regions:
// - 32MB RAM (actually 24MB on hardware), available on GameCube and Wii
// - 64MB "EXRAM", RAM only available on Wii
// - 32MB FakeVMem, allocated in GameCube mode when MMU support is turned off.
// This is used to approximate the behavior of a common library which pages
// memory to and from the DSP's dedicated RAM. The DSP's RAM (ARAM) isn't
// directly addressable on GameCube.
// - 256KB Locked L1, to represent cache lines allocated out of the L1 data
// cache in Locked L1 mode. Dolphin does not emulate this hardware feature
// accurately; it just pretends there is extra memory at 0xE0000000.
//
// The 4GB starting at m_physical_base represents access from the CPU
// with address translation turned off. (This is only used by the CPU;
// other devices, like the GPU, use other rules, approximated by
// Memory::GetPointer.) This memory is laid out as follows:
// [0x00000000, 0x02000000) - 32MB RAM
// [0x02000000, 0x08000000) - Mirrors of 32MB RAM (not handled here)
// [0x08000000, 0x0C000000) - EFB "mapping" (not handled here)
// [0x0C000000, 0x0E000000) - MMIO etc. (not handled here)
// [0x10000000, 0x14000000) - 64MB RAM (Wii-only; slightly slower)
// [0x7E000000, 0x80000000) - FakeVMEM
// [0xE0000000, 0xE0040000) - 256KB locked L1
//
// The 4GB starting at m_logical_base represents access from the CPU
// with address translation turned on. This mapping is computed based
// on the BAT registers.
//
// Each of these 4GB regions is followed by 4GB of empty space so overflows
// in address computation in the JIT don't access the wrong memory.
//
// The neighboring mirrors of RAM ([0x02000000, 0x08000000), etc.) exist because
// the bus masks off the bits in question for RAM accesses; using them is a
// terrible idea because the CPU cache won't handle them correctly, but a
// few buggy games (notably Rogue Squadron 2) use them by accident. They
// aren't backed by memory mappings because they are used very rarely.
//
// Dolphin doesn't emulate the difference between cached and uncached access.
//
// TODO: The actual size of RAM is 24MB; the other 8MB shouldn't be backed by actual memory.
// TODO: Do we want to handle the mirrors of the GC RAM?
std::array<PhysicalMemoryRegion, 4> m_physical_regions;
std::vector<LogicalMemoryView> m_logical_mapped_entries;
std::array<void*, PowerPC::BAT_PAGE_COUNT> m_physical_page_mappings;
std::array<void*, PowerPC::BAT_PAGE_COUNT> m_logical_page_mappings;
void InitMMIO(bool is_wii);
};
} // namespace Memory } // namespace Memory

View File

@ -21,14 +21,25 @@
#include "Core/HW/WII_IPC.h" #include "Core/HW/WII_IPC.h"
#include "Core/IOS/ES/ES.h" #include "Core/IOS/ES/ES.h"
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/System.h"
#include "DiscIO/Volume.h" #include "DiscIO/Volume.h"
template <u32 addr> template <u32 addr>
class RegisterWrapper class RegisterWrapper
{ {
public: public:
operator u32() const { return Memory::mmio_mapping->Read<u32>(addr); } operator u32() const
void operator=(u32 rhs) { Memory::mmio_mapping->Write(addr, rhs); } {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return memory.GetMMIOMapping()->Read<u32>(addr);
}
void operator=(u32 rhs)
{
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.GetMMIOMapping()->Write(addr, rhs);
}
}; };
static RegisterWrapper<0x0D806000> DISR; static RegisterWrapper<0x0D806000> DISR;
static RegisterWrapper<0x0D806004> DICVR; static RegisterWrapper<0x0D806004> DICVR;
@ -76,7 +87,9 @@ std::optional<IPCReply> DIDevice::IOCtl(const IOCtlRequest& request)
// asynchronously. The rest are also treated as async to match this. Only one command can be // asynchronously. The rest are also treated as async to match this. Only one command can be
// executed at a time, so commands are queued until DVDInterface is ready to handle them. // executed at a time, so commands are queued until DVDInterface is ready to handle them.
const u8 command = Memory::Read_U8(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 command = memory.Read_U8(request.buffer_in);
if (request.request != command) if (request.request != command)
{ {
WARN_LOG_FMT(IOS_DI, WARN_LOG_FMT(IOS_DI,
@ -128,7 +141,9 @@ std::optional<DIDevice::DIResult> DIDevice::WriteIfFits(const IOCtlRequest& requ
} }
else else
{ {
Memory::Write_U32(value, request.buffer_out); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(value, request.buffer_out);
return DIResult::Success; return DIResult::Success;
} }
} }
@ -152,6 +167,9 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return DIResult::SecurityError; return DIResult::SecurityError;
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// DVDInterface's ExecuteCommand handles most of the work for most of these. // DVDInterface's ExecuteCommand handles most of the work for most of these.
// The IOCtl callback is used to generate a reply afterwards. // The IOCtl callback is used to generate a reply afterwards.
switch (static_cast<DIIoctl>(request.request)) switch (static_cast<DIIoctl>(request.request))
@ -169,7 +187,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return StartDMATransfer(0x20, request); return StartDMATransfer(0x20, request);
// TODO: Include an additional read that happens on Wii discs, or at least // TODO: Include an additional read that happens on Wii discs, or at least
// emulate its side effect of disabling DTK configuration // emulate its side effect of disabling DTK configuration
// if (Memory::Read_U32(request.buffer_out + 24) == 0x5d1c9ea3) { // Wii Magic // if (memory.Read_U32(request.buffer_out + 24) == 0x5d1c9ea3) { // Wii Magic
// if (!m_has_read_encryption_info) { // if (!m_has_read_encryption_info) {
// // Read 0x44 (=> 0x60) bytes starting from offset 8 or byte 0x20; // // Read 0x44 (=> 0x60) bytes starting from offset 8 or byte 0x20;
// // byte 0x60 is disable hashing and byte 0x61 is disable encryption // // byte 0x60 is disable hashing and byte 0x61 is disable encryption
@ -177,8 +195,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
// } // }
case DIIoctl::DVDLowRead: case DIIoctl::DVDLowRead:
{ {
const u32 length = Memory::Read_U32(request.buffer_in + 4); const u32 length = memory.Read_U32(request.buffer_in + 4);
const u32 position = Memory::Read_U32(request.buffer_in + 8); const u32 position = memory.Read_U32(request.buffer_in + 8);
INFO_LOG_FMT(IOS_DI, "DVDLowRead: offset {:#010x} (byte {:#011x}), length {:#x}", position, INFO_LOG_FMT(IOS_DI, "DVDLowRead: offset {:#010x} (byte {:#011x}), length {:#x}", position,
static_cast<u64>(position) << 2, length); static_cast<u64>(position) << 2, length);
if (m_current_partition == DiscIO::PARTITION_NONE) if (m_current_partition == DiscIO::PARTITION_NONE)
@ -220,7 +238,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return DIResult::BadArgument; return DIResult::BadArgument;
case DIIoctl::DVDLowReadDvdPhysical: case DIIoctl::DVDLowReadDvdPhysical:
{ {
const u8 position = Memory::Read_U8(request.buffer_in + 7); const u8 position = memory.Read_U8(request.buffer_in + 7);
INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdPhysical: position {:#04x}", position); INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdPhysical: position {:#04x}", position);
DICMDBUF0 = 0xAD000000 | (position << 8); DICMDBUF0 = 0xAD000000 | (position << 8);
DICMDBUF1 = 0; DICMDBUF1 = 0;
@ -229,7 +247,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowReadDvdCopyright: case DIIoctl::DVDLowReadDvdCopyright:
{ {
const u8 position = Memory::Read_U8(request.buffer_in + 7); const u8 position = memory.Read_U8(request.buffer_in + 7);
INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdCopyright: position {:#04x}", position); INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdCopyright: position {:#04x}", position);
DICMDBUF0 = 0xAD010000 | (position << 8); DICMDBUF0 = 0xAD010000 | (position << 8);
DICMDBUF1 = 0; DICMDBUF1 = 0;
@ -238,7 +256,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowReadDvdDiscKey: case DIIoctl::DVDLowReadDvdDiscKey:
{ {
const u8 position = Memory::Read_U8(request.buffer_in + 7); const u8 position = memory.Read_U8(request.buffer_in + 7);
INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdDiscKey: position {:#04x}", position); INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdDiscKey: position {:#04x}", position);
DICMDBUF0 = 0xAD020000 | (position << 8); DICMDBUF0 = 0xAD020000 | (position << 8);
DICMDBUF1 = 0; DICMDBUF1 = 0;
@ -280,7 +298,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return DIResult::Success; return DIResult::Success;
case DIIoctl::DVDLowReset: case DIIoctl::DVDLowReset:
{ {
const bool spinup = Memory::Read_U32(request.buffer_in + 4); const bool spinup = memory.Read_U32(request.buffer_in + 4);
// The GPIO *disables* spinning up the drive. Normally handled via syscall 0x4e. // The GPIO *disables* spinning up the drive. Normally handled via syscall 0x4e.
const u32 old_gpio = HW_GPIO_OUT; const u32 old_gpio = HW_GPIO_OUT;
@ -318,8 +336,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return DIResult::Success; return DIResult::Success;
case DIIoctl::DVDLowUnencryptedRead: case DIIoctl::DVDLowUnencryptedRead:
{ {
const u32 length = Memory::Read_U32(request.buffer_in + 4); const u32 length = memory.Read_U32(request.buffer_in + 4);
const u32 position = Memory::Read_U32(request.buffer_in + 8); const u32 position = memory.Read_U32(request.buffer_in + 8);
const u32 end = position + (length >> 2); // a 32-bit offset const u32 end = position + (length >> 2); // a 32-bit offset
INFO_LOG_FMT(IOS_DI, "DVDLowUnencryptedRead: offset {:#010x} (byte {:#011x}), length {:#x}", INFO_LOG_FMT(IOS_DI, "DVDLowUnencryptedRead: offset {:#010x} (byte {:#011x}), length {:#x}",
position, static_cast<u64>(position) << 2, length); position, static_cast<u64>(position) << 2, length);
@ -395,8 +413,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowReportKey: case DIIoctl::DVDLowReportKey:
{ {
const u8 param1 = Memory::Read_U8(request.buffer_in + 7); const u8 param1 = memory.Read_U8(request.buffer_in + 7);
const u32 param2 = Memory::Read_U32(request.buffer_in + 8); const u32 param2 = memory.Read_U32(request.buffer_in + 8);
INFO_LOG_FMT(IOS_DI, "DVDLowReportKey: param1 {:#04x}, param2 {:#08x}", param1, param2); INFO_LOG_FMT(IOS_DI, "DVDLowReportKey: param1 {:#04x}, param2 {:#08x}", param1, param2);
DICMDBUF0 = 0xA4000000 | (param1 << 16); DICMDBUF0 = 0xA4000000 | (param1 << 16);
DICMDBUF1 = param2 & 0xFFFFFF; DICMDBUF1 = param2 & 0xFFFFFF;
@ -405,7 +423,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowSeek: case DIIoctl::DVDLowSeek:
{ {
const u32 position = Memory::Read_U32(request.buffer_in + 4); // 32-bit offset const u32 position = memory.Read_U32(request.buffer_in + 4); // 32-bit offset
INFO_LOG_FMT(IOS_DI, "DVDLowSeek: position {:#010x}, translated to {:#010x}", position, INFO_LOG_FMT(IOS_DI, "DVDLowSeek: position {:#010x}, translated to {:#010x}", position,
position); // TODO: do partition translation! position); // TODO: do partition translation!
DICMDBUF0 = 0xAB000000; DICMDBUF0 = 0xAB000000;
@ -414,10 +432,10 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowReadDvd: case DIIoctl::DVDLowReadDvd:
{ {
const u8 flag1 = Memory::Read_U8(request.buffer_in + 7); const u8 flag1 = memory.Read_U8(request.buffer_in + 7);
const u8 flag2 = Memory::Read_U8(request.buffer_in + 11); const u8 flag2 = memory.Read_U8(request.buffer_in + 11);
const u32 length = Memory::Read_U32(request.buffer_in + 12); const u32 length = memory.Read_U32(request.buffer_in + 12);
const u32 position = Memory::Read_U32(request.buffer_in + 16); const u32 position = memory.Read_U32(request.buffer_in + 16);
INFO_LOG_FMT(IOS_DI, "DVDLowReadDvd({}, {}): position {:#08x}, length {:#08x}", flag1, flag2, INFO_LOG_FMT(IOS_DI, "DVDLowReadDvd({}, {}): position {:#08x}, length {:#08x}", flag1, flag2,
position, length); position, length);
DICMDBUF0 = 0xD0000000 | ((flag1 & 1) << 7) | ((flag2 & 1) << 6); DICMDBUF0 = 0xD0000000 | ((flag1 & 1) << 7) | ((flag2 & 1) << 6);
@ -427,9 +445,9 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowReadDvdConfig: case DIIoctl::DVDLowReadDvdConfig:
{ {
const u8 flag1 = Memory::Read_U8(request.buffer_in + 7); const u8 flag1 = memory.Read_U8(request.buffer_in + 7);
const u8 param2 = Memory::Read_U8(request.buffer_in + 11); const u8 param2 = memory.Read_U8(request.buffer_in + 11);
const u32 position = Memory::Read_U32(request.buffer_in + 12); const u32 position = memory.Read_U32(request.buffer_in + 12);
INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdConfig({}, {}): position {:#08x}", flag1, param2, position); INFO_LOG_FMT(IOS_DI, "DVDLowReadDvdConfig({}, {}): position {:#08x}", flag1, param2, position);
DICMDBUF0 = 0xD1000000 | ((flag1 & 1) << 16) | param2; DICMDBUF0 = 0xD1000000 | ((flag1 & 1) << 16) | param2;
DICMDBUF1 = position & 0xFFFFFF; DICMDBUF1 = position & 0xFFFFFF;
@ -442,8 +460,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return StartImmediateTransfer(request); return StartImmediateTransfer(request);
case DIIoctl::DVDLowOffset: case DIIoctl::DVDLowOffset:
{ {
const u8 flag = Memory::Read_U8(request.buffer_in + 7); const u8 flag = memory.Read_U8(request.buffer_in + 7);
const u32 offset = Memory::Read_U32(request.buffer_in + 8); const u32 offset = memory.Read_U32(request.buffer_in + 8);
INFO_LOG_FMT(IOS_DI, "DVDLowOffset({}): offset {:#010x}", flag, offset); INFO_LOG_FMT(IOS_DI, "DVDLowOffset({}): offset {:#010x}", flag, offset);
DICMDBUF0 = 0xD9000000 | ((flag & 1) << 16); DICMDBUF0 = 0xD9000000 | ((flag & 1) << 16);
DICMDBUF1 = offset; DICMDBUF1 = offset;
@ -463,15 +481,15 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return StartImmediateTransfer(request); return StartImmediateTransfer(request);
case DIIoctl::DVDLowSetMaximumRotation: case DIIoctl::DVDLowSetMaximumRotation:
{ {
const u8 speed = Memory::Read_U8(request.buffer_in + 7); const u8 speed = memory.Read_U8(request.buffer_in + 7);
INFO_LOG_FMT(IOS_DI, "DVDLowSetMaximumRotation: speed {}", speed); INFO_LOG_FMT(IOS_DI, "DVDLowSetMaximumRotation: speed {}", speed);
DICMDBUF0 = 0xDD000000 | ((speed & 3) << 16); DICMDBUF0 = 0xDD000000 | ((speed & 3) << 16);
return StartImmediateTransfer(request, false); return StartImmediateTransfer(request, false);
} }
case DIIoctl::DVDLowSerMeasControl: case DIIoctl::DVDLowSerMeasControl:
{ {
const u8 flag1 = Memory::Read_U8(request.buffer_in + 7); const u8 flag1 = memory.Read_U8(request.buffer_in + 7);
const u8 flag2 = Memory::Read_U8(request.buffer_in + 11); const u8 flag2 = memory.Read_U8(request.buffer_in + 11);
INFO_LOG_FMT(IOS_DI, "DVDLowSerMeasControl({}, {})", flag1, flag2); INFO_LOG_FMT(IOS_DI, "DVDLowSerMeasControl({}, {})", flag1, flag2);
DICMDBUF0 = 0xDF000000 | ((flag1 & 1) << 17) | ((flag2 & 1) << 16); DICMDBUF0 = 0xDF000000 | ((flag1 & 1) << 17) | ((flag2 & 1) << 16);
return StartDMATransfer(0x20, request); return StartDMATransfer(0x20, request);
@ -482,9 +500,9 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
return StartImmediateTransfer(request); return StartImmediateTransfer(request);
case DIIoctl::DVDLowAudioStream: case DIIoctl::DVDLowAudioStream:
{ {
const u8 mode = Memory::Read_U8(request.buffer_in + 7); const u8 mode = memory.Read_U8(request.buffer_in + 7);
const u32 length = Memory::Read_U32(request.buffer_in + 8); const u32 length = memory.Read_U32(request.buffer_in + 8);
const u32 position = Memory::Read_U32(request.buffer_in + 12); const u32 position = memory.Read_U32(request.buffer_in + 12);
INFO_LOG_FMT(IOS_DI, "DVDLowAudioStream({}): offset {:#010x} (byte {:#011x}), length {:#x}", INFO_LOG_FMT(IOS_DI, "DVDLowAudioStream({}): offset {:#010x} (byte {:#011x}), length {:#x}",
mode, position, static_cast<u64>(position) << 2, length); mode, position, static_cast<u64>(position) << 2, length);
DICMDBUF0 = 0xE1000000 | ((mode & 3) << 16); DICMDBUF0 = 0xE1000000 | ((mode & 3) << 16);
@ -494,7 +512,7 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowRequestAudioStatus: case DIIoctl::DVDLowRequestAudioStatus:
{ {
const u8 mode = Memory::Read_U8(request.buffer_in + 7); const u8 mode = memory.Read_U8(request.buffer_in + 7);
INFO_LOG_FMT(IOS_DI, "DVDLowRequestAudioStatus({})", mode); INFO_LOG_FMT(IOS_DI, "DVDLowRequestAudioStatus({})", mode);
DICMDBUF0 = 0xE2000000 | ((mode & 3) << 16); DICMDBUF0 = 0xE2000000 | ((mode & 3) << 16);
DICMDBUF1 = 0; DICMDBUF1 = 0;
@ -504,8 +522,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowStopMotor: case DIIoctl::DVDLowStopMotor:
{ {
const u8 eject = Memory::Read_U8(request.buffer_in + 7); const u8 eject = memory.Read_U8(request.buffer_in + 7);
const u8 kill = Memory::Read_U8(request.buffer_in + 11); const u8 kill = memory.Read_U8(request.buffer_in + 11);
INFO_LOG_FMT(IOS_DI, "DVDLowStopMotor({}, {})", eject, kill); INFO_LOG_FMT(IOS_DI, "DVDLowStopMotor({}, {})", eject, kill);
DICMDBUF0 = 0xE3000000 | ((eject & 1) << 17) | ((kill & 1) << 20); DICMDBUF0 = 0xE3000000 | ((eject & 1) << 17) | ((kill & 1) << 20);
DICMDBUF1 = 0; DICMDBUF1 = 0;
@ -513,8 +531,8 @@ std::optional<DIDevice::DIResult> DIDevice::StartIOCtl(const IOCtlRequest& reque
} }
case DIIoctl::DVDLowAudioBufferConfig: case DIIoctl::DVDLowAudioBufferConfig:
{ {
const u8 enable = Memory::Read_U8(request.buffer_in + 7); const u8 enable = memory.Read_U8(request.buffer_in + 7);
const u8 buffer_size = Memory::Read_U8(request.buffer_in + 11); const u8 buffer_size = memory.Read_U8(request.buffer_in + 11);
INFO_LOG_FMT(IOS_DI, "DVDLowAudioBufferConfig: {}, buffer size {}", INFO_LOG_FMT(IOS_DI, "DVDLowAudioBufferConfig: {}, buffer size {}",
enable ? "enabled" : "disabled", buffer_size); enable ? "enabled" : "disabled", buffer_size);
DICMDBUF0 = 0xE4000000 | ((enable & 1) << 16) | (buffer_size & 0xf); DICMDBUF0 = 0xE4000000 | ((enable & 1) << 16) | (buffer_size & 0xf);
@ -643,9 +661,12 @@ void DIDevice::FinishDICommand(DIResult result)
return; return;
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
IOCtlRequest request{m_executing_command->m_request_address}; IOCtlRequest request{m_executing_command->m_request_address};
if (m_executing_command->m_copy_diimmbuf) if (m_executing_command->m_copy_diimmbuf)
Memory::Write_U32(DIIMMBUF, request.buffer_out); memory.Write_U32(DIIMMBUF, request.buffer_out);
m_ios.EnqueueIPCReply(request, static_cast<s32>(result)); m_ios.EnqueueIPCReply(request, static_cast<s32>(result));
@ -672,7 +693,11 @@ std::optional<IPCReply> DIDevice::IOCtlV(const IOCtlVRequest& request)
request.in_vectors[0].size); request.in_vectors[0].size);
return IPCReply{static_cast<s32>(DIResult::BadArgument)}; return IPCReply{static_cast<s32>(DIResult::BadArgument)};
} }
const u8 command = Memory::Read_U8(request.in_vectors[0].address);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 command = memory.Read_U8(request.in_vectors[0].address);
if (request.request != command) if (request.request != command)
{ {
WARN_LOG_FMT( WARN_LOG_FMT(
@ -708,7 +733,7 @@ std::optional<IPCReply> DIDevice::IOCtlV(const IOCtlVRequest& request)
} }
const u64 partition_offset = const u64 partition_offset =
static_cast<u64>(Memory::Read_U32(request.in_vectors[0].address + 4)) << 2; static_cast<u64>(memory.Read_U32(request.in_vectors[0].address + 4)) << 2;
ChangePartition(DiscIO::Partition(partition_offset)); ChangePartition(DiscIO::Partition(partition_offset));
INFO_LOG_FMT(IOS_DI, "DVDLowOpenPartition: partition_offset {:#011x}", partition_offset); INFO_LOG_FMT(IOS_DI, "DVDLowOpenPartition: partition_offset {:#011x}", partition_offset);
@ -716,10 +741,10 @@ std::optional<IPCReply> DIDevice::IOCtlV(const IOCtlVRequest& request)
// Read TMD to the buffer // Read TMD to the buffer
const ES::TMDReader tmd = DVDThread::GetTMD(m_current_partition); const ES::TMDReader tmd = DVDThread::GetTMD(m_current_partition);
const std::vector<u8>& raw_tmd = tmd.GetBytes(); const std::vector<u8>& raw_tmd = tmd.GetBytes();
Memory::CopyToEmu(request.io_vectors[0].address, raw_tmd.data(), raw_tmd.size()); memory.CopyToEmu(request.io_vectors[0].address, raw_tmd.data(), raw_tmd.size());
ReturnCode es_result = m_ios.GetES()->DIVerify(tmd, DVDThread::GetTicket(m_current_partition)); ReturnCode es_result = m_ios.GetES()->DIVerify(tmd, DVDThread::GetTicket(m_current_partition));
Memory::Write_U32(es_result, request.io_vectors[1].address); memory.Write_U32(es_result, request.io_vectors[1].address);
return_value = DIResult::Success; return_value = DIResult::Success;
break; break;

View File

@ -12,19 +12,24 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/HW/SystemTimers.h" #include "Core/HW/SystemTimers.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
Request::Request(const u32 address_) : address(address_) Request::Request(const u32 address_) : address(address_)
{ {
command = static_cast<IPCCommandType>(Memory::Read_U32(address)); auto& system = Core::System::GetInstance();
fd = Memory::Read_U32(address + 8); auto& memory = system.GetMemory();
command = static_cast<IPCCommandType>(memory.Read_U32(address));
fd = memory.Read_U32(address + 8);
} }
OpenRequest::OpenRequest(const u32 address_) : Request(address_) OpenRequest::OpenRequest(const u32 address_) : Request(address_)
{ {
path = Memory::GetString(Memory::Read_U32(address + 0xc)); auto& system = Core::System::GetInstance();
flags = static_cast<OpenMode>(Memory::Read_U32(address + 0x10)); auto& memory = system.GetMemory();
path = memory.GetString(memory.Read_U32(address + 0xc));
flags = static_cast<OpenMode>(memory.Read_U32(address + 0x10));
const Kernel* ios = GetIOS(); const Kernel* ios = GetIOS();
if (ios) if (ios)
{ {
@ -35,38 +40,46 @@ OpenRequest::OpenRequest(const u32 address_) : Request(address_)
ReadWriteRequest::ReadWriteRequest(const u32 address_) : Request(address_) ReadWriteRequest::ReadWriteRequest(const u32 address_) : Request(address_)
{ {
buffer = Memory::Read_U32(address + 0xc); auto& system = Core::System::GetInstance();
size = Memory::Read_U32(address + 0x10); auto& memory = system.GetMemory();
buffer = memory.Read_U32(address + 0xc);
size = memory.Read_U32(address + 0x10);
} }
SeekRequest::SeekRequest(const u32 address_) : Request(address_) SeekRequest::SeekRequest(const u32 address_) : Request(address_)
{ {
offset = Memory::Read_U32(address + 0xc); auto& system = Core::System::GetInstance();
mode = static_cast<SeekMode>(Memory::Read_U32(address + 0x10)); auto& memory = system.GetMemory();
offset = memory.Read_U32(address + 0xc);
mode = static_cast<SeekMode>(memory.Read_U32(address + 0x10));
} }
IOCtlRequest::IOCtlRequest(const u32 address_) : Request(address_) IOCtlRequest::IOCtlRequest(const u32 address_) : Request(address_)
{ {
request = Memory::Read_U32(address + 0x0c); auto& system = Core::System::GetInstance();
buffer_in = Memory::Read_U32(address + 0x10); auto& memory = system.GetMemory();
buffer_in_size = Memory::Read_U32(address + 0x14); request = memory.Read_U32(address + 0x0c);
buffer_out = Memory::Read_U32(address + 0x18); buffer_in = memory.Read_U32(address + 0x10);
buffer_out_size = Memory::Read_U32(address + 0x1c); buffer_in_size = memory.Read_U32(address + 0x14);
buffer_out = memory.Read_U32(address + 0x18);
buffer_out_size = memory.Read_U32(address + 0x1c);
} }
IOCtlVRequest::IOCtlVRequest(const u32 address_) : Request(address_) IOCtlVRequest::IOCtlVRequest(const u32 address_) : Request(address_)
{ {
request = Memory::Read_U32(address + 0x0c); auto& system = Core::System::GetInstance();
const u32 in_number = Memory::Read_U32(address + 0x10); auto& memory = system.GetMemory();
const u32 out_number = Memory::Read_U32(address + 0x14); request = memory.Read_U32(address + 0x0c);
const u32 vectors_base = Memory::Read_U32(address + 0x18); // address to vectors const u32 in_number = memory.Read_U32(address + 0x10);
const u32 out_number = memory.Read_U32(address + 0x14);
const u32 vectors_base = memory.Read_U32(address + 0x18); // address to vectors
u32 offset = 0; u32 offset = 0;
for (size_t i = 0; i < (in_number + out_number); ++i) for (size_t i = 0; i < (in_number + out_number); ++i)
{ {
IOVector vector; IOVector vector;
vector.address = Memory::Read_U32(vectors_base + offset); vector.address = memory.Read_U32(vectors_base + offset);
vector.size = Memory::Read_U32(vectors_base + offset + 4); vector.size = memory.Read_U32(vectors_base + offset + 4);
offset += 8; offset += 8;
if (i < in_number) if (i < in_number)
in_vectors.emplace_back(vector); in_vectors.emplace_back(vector);
@ -104,11 +117,14 @@ void IOCtlRequest::Log(std::string_view device_name, Common::Log::LogType type,
void IOCtlRequest::Dump(const std::string& description, Common::Log::LogType type, void IOCtlRequest::Dump(const std::string& description, Common::Log::LogType type,
Common::Log::LogLevel level) const Common::Log::LogLevel level) const
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
Log("===== " + description, type, level); Log("===== " + description, type, level);
GENERIC_LOG_FMT(type, level, "In buffer\n{}", GENERIC_LOG_FMT(type, level, "In buffer\n{}",
HexDump(Memory::GetPointer(buffer_in), buffer_in_size)); HexDump(memory.GetPointer(buffer_in), buffer_in_size));
GENERIC_LOG_FMT(type, level, "Out buffer\n{}", GENERIC_LOG_FMT(type, level, "Out buffer\n{}",
HexDump(Memory::GetPointer(buffer_out), buffer_out_size)); HexDump(memory.GetPointer(buffer_out), buffer_out_size));
} }
void IOCtlRequest::DumpUnknown(const std::string& description, Common::Log::LogType type, void IOCtlRequest::DumpUnknown(const std::string& description, Common::Log::LogType type,
@ -120,6 +136,9 @@ void IOCtlRequest::DumpUnknown(const std::string& description, Common::Log::LogT
void IOCtlVRequest::Dump(std::string_view description, Common::Log::LogType type, void IOCtlVRequest::Dump(std::string_view description, Common::Log::LogType type,
Common::Log::LogLevel level) const Common::Log::LogLevel level) const
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
GENERIC_LOG_FMT(type, level, "===== {} (fd {}) - IOCtlV {:#x} ({} in, {} io)", description, fd, GENERIC_LOG_FMT(type, level, "===== {} (fd {}) - IOCtlV {:#x} ({} in, {} io)", description, fd,
request, in_vectors.size(), io_vectors.size()); request, in_vectors.size(), io_vectors.size());
@ -127,7 +146,7 @@ void IOCtlVRequest::Dump(std::string_view description, Common::Log::LogType type
for (const auto& vector : in_vectors) for (const auto& vector : in_vectors)
{ {
GENERIC_LOG_FMT(type, level, "in[{}] (size={:#x}):\n{}", i++, vector.size, GENERIC_LOG_FMT(type, level, "in[{}] (size={:#x}):\n{}", i++, vector.size,
HexDump(Memory::GetPointer(vector.address), vector.size)); HexDump(memory.GetPointer(vector.address), vector.size));
} }
i = 0; i = 0;

View File

@ -19,6 +19,7 @@
#include "Core/Core.h" #include "Core/Core.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/Host.h" #include "Core/Host.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -48,8 +49,10 @@ IPCReply GetVersion(const IOCtlVRequest& request)
const auto length = std::min(size_t(request.io_vectors[0].size), Common::GetScmDescStr().size()); const auto length = std::min(size_t(request.io_vectors[0].size), Common::GetScmDescStr().size());
Memory::Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size); auto& system = Core::System::GetInstance();
Memory::CopyToEmu(request.io_vectors[0].address, Common::GetScmDescStr().data(), length); auto& memory = system.GetMemory();
memory.Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size);
memory.CopyToEmu(request.io_vectors[0].address, Common::GetScmDescStr().data(), length);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -71,7 +74,9 @@ IPCReply GetCPUSpeed(const IOCtlVRequest& request)
const u32 core_clock = u32(float(SystemTimers::GetTicksPerSecond()) * oc); const u32 core_clock = u32(float(SystemTimers::GetTicksPerSecond()) * oc);
Memory::Write_U32(core_clock, request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(core_clock, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -90,7 +95,10 @@ IPCReply GetSpeedLimit(const IOCtlVRequest& request)
} }
const u32 speed_percent = Config::Get(Config::MAIN_EMULATION_SPEED) * 100; const u32 speed_percent = Config::Get(Config::MAIN_EMULATION_SPEED) * 100;
Memory::Write_U32(speed_percent, request.io_vectors[0].address);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(speed_percent, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -108,7 +116,9 @@ IPCReply SetSpeedLimit(const IOCtlVRequest& request)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
} }
const float speed = float(Memory::Read_U32(request.in_vectors[0].address)) / 100.0f; auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const float speed = float(memory.Read_U32(request.in_vectors[0].address)) / 100.0f;
Config::SetCurrent(Config::MAIN_EMULATION_SPEED, speed); Config::SetCurrent(Config::MAIN_EMULATION_SPEED, speed);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -140,8 +150,10 @@ IPCReply GetRealProductCode(const IOCtlVRequest& request)
if (length == 0) if (length == 0)
return IPCReply(IPC_ENOENT); return IPCReply(IPC_ENOENT);
Memory::Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size); auto& system = Core::System::GetInstance();
Memory::CopyToEmu(request.io_vectors[0].address, code.c_str(), length); auto& memory = system.GetMemory();
memory.Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size);
memory.CopyToEmu(request.io_vectors[0].address, code.c_str(), length);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -153,8 +165,10 @@ IPCReply SetDiscordClient(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 0)) if (!request.HasNumberOfValidVectors(1, 0))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::string new_client_id = std::string new_client_id =
Memory::GetString(request.in_vectors[0].address, request.in_vectors[0].size); memory.GetString(request.in_vectors[0].address, request.in_vectors[0].size);
Host_UpdateDiscordClientID(new_client_id); Host_UpdateDiscordClientID(new_client_id);
@ -169,22 +183,24 @@ IPCReply SetDiscordPresence(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(10, 0)) if (!request.HasNumberOfValidVectors(10, 0))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
std::string details = auto& system = Core::System::GetInstance();
Memory::GetString(request.in_vectors[0].address, request.in_vectors[0].size); auto& memory = system.GetMemory();
std::string state = Memory::GetString(request.in_vectors[1].address, request.in_vectors[1].size);
std::string large_image_key =
Memory::GetString(request.in_vectors[2].address, request.in_vectors[2].size);
std::string large_image_text =
Memory::GetString(request.in_vectors[3].address, request.in_vectors[3].size);
std::string small_image_key =
Memory::GetString(request.in_vectors[4].address, request.in_vectors[4].size);
std::string small_image_text =
Memory::GetString(request.in_vectors[5].address, request.in_vectors[5].size);
int64_t start_timestamp = Memory::Read_U64(request.in_vectors[6].address); std::string details = memory.GetString(request.in_vectors[0].address, request.in_vectors[0].size);
int64_t end_timestamp = Memory::Read_U64(request.in_vectors[7].address); std::string state = memory.GetString(request.in_vectors[1].address, request.in_vectors[1].size);
int party_size = Memory::Read_U32(request.in_vectors[8].address); std::string large_image_key =
int party_max = Memory::Read_U32(request.in_vectors[9].address); memory.GetString(request.in_vectors[2].address, request.in_vectors[2].size);
std::string large_image_text =
memory.GetString(request.in_vectors[3].address, request.in_vectors[3].size);
std::string small_image_key =
memory.GetString(request.in_vectors[4].address, request.in_vectors[4].size);
std::string small_image_text =
memory.GetString(request.in_vectors[5].address, request.in_vectors[5].size);
int64_t start_timestamp = memory.Read_U64(request.in_vectors[6].address);
int64_t end_timestamp = memory.Read_U64(request.in_vectors[7].address);
int party_size = memory.Read_U32(request.in_vectors[8].address);
int party_max = memory.Read_U32(request.in_vectors[9].address);
bool ret = Host_UpdateDiscordPresenceRaw(details, state, large_image_key, large_image_text, bool ret = Host_UpdateDiscordPresenceRaw(details, state, large_image_key, large_image_text,
small_image_key, small_image_text, start_timestamp, small_image_key, small_image_text, start_timestamp,
@ -225,7 +241,11 @@ IPCReply DolphinDevice::GetElapsedTime(const IOCtlVRequest& request) const
// Return elapsed time instead of current timestamp to make buggy emulated code less likely to // Return elapsed time instead of current timestamp to make buggy emulated code less likely to
// have issues. // have issues.
const u32 milliseconds = static_cast<u32>(m_timer.ElapsedMs()); const u32 milliseconds = static_cast<u32>(m_timer.ElapsedMs());
Memory::Write_U32(milliseconds, request.io_vectors[0].address);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(milliseconds, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -241,11 +261,14 @@ IPCReply DolphinDevice::GetSystemTime(const IOCtlVRequest& request) const
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Write Unix timestamp in milliseconds to memory address // Write Unix timestamp in milliseconds to memory address
const u64 milliseconds = std::chrono::duration_cast<std::chrono::milliseconds>( const u64 milliseconds = std::chrono::duration_cast<std::chrono::milliseconds>(
std::chrono::system_clock::now().time_since_epoch()) std::chrono::system_clock::now().time_since_epoch())
.count(); .count();
Memory::Write_U64(milliseconds, request.io_vectors[0].address); memory.Write_U64(milliseconds, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }

View File

@ -202,9 +202,12 @@ IPCReply ESDevice::GetTitleDirectory(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1)) if (!request.HasNumberOfValidVectors(1, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
char* path = reinterpret_cast<char*>(Memory::GetPointer(request.io_vectors[0].address)); const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
char* path = reinterpret_cast<char*>(memory.GetPointer(request.io_vectors[0].address));
sprintf(path, "/title/%08x/%08x/data", static_cast<u32>(title_id >> 32), sprintf(path, "/title/%08x/%08x/data", static_cast<u32>(title_id >> 32),
static_cast<u32>(title_id)); static_cast<u32>(title_id));
@ -230,7 +233,10 @@ IPCReply ESDevice::GetTitleId(const IOCtlVRequest& request)
if (ret != IPC_SUCCESS) if (ret != IPC_SUCCESS)
return IPCReply(ret); return IPCReply(ret);
Memory::Write_U64(title_id, request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U64(title_id, request.io_vectors[0].address);
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETTITLEID: {:08x}/{:08x}", static_cast<u32>(title_id >> 32), INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETTITLEID: {:08x}/{:08x}", static_cast<u32>(title_id >> 32),
static_cast<u32>(title_id)); static_cast<u32>(title_id));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -278,7 +284,10 @@ IPCReply ESDevice::SetUID(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 8) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 8)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const s32 ret = CheckIsAllowedToSetUID(m_ios, uid, m_title_context.tmd); const s32 ret = CheckIsAllowedToSetUID(m_ios, uid, m_title_context.tmd);
if (ret < 0) if (ret < 0)
@ -707,7 +716,9 @@ IPCReply ESDevice::GetConsumption(const IOCtlVRequest& request)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
// This is at least what crediar's ES module does // This is at least what crediar's ES module does
Memory::Write_U32(0, request.io_vectors[1].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(0, request.io_vectors[1].address);
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETCONSUMPTION"); INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETCONSUMPTION");
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -717,12 +728,15 @@ std::optional<IPCReply> ESDevice::Launch(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(2, 0)) if (!request.HasNumberOfValidVectors(2, 0))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 view = Memory::Read_U32(request.in_vectors[1].address); auto& memory = system.GetMemory();
const u64 ticketid = Memory::Read_U64(request.in_vectors[1].address + 4);
const u32 devicetype = Memory::Read_U32(request.in_vectors[1].address + 12); const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const u64 titleid = Memory::Read_U64(request.in_vectors[1].address + 16); const u32 view = memory.Read_U32(request.in_vectors[1].address);
const u16 access = Memory::Read_U16(request.in_vectors[1].address + 24); const u64 ticketid = memory.Read_U64(request.in_vectors[1].address + 4);
const u32 devicetype = memory.Read_U32(request.in_vectors[1].address + 12);
const u64 titleid = memory.Read_U64(request.in_vectors[1].address + 16);
const u16 access = memory.Read_U16(request.in_vectors[1].address + 24);
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_LAUNCH {:016x} {:08x} {:016x} {:08x} {:016x} {:04x}", title_id, INFO_LOG_FMT(IOS_ES, "IOCTL_ES_LAUNCH {:016x} {:08x} {:016x} {:08x} {:016x} {:04x}", title_id,
view, ticketid, devicetype, titleid, access); view, ticketid, devicetype, titleid, access);
@ -936,8 +950,11 @@ IPCReply ESDevice::SetUpStreamKey(const Context& context, const IOCtlVRequest& r
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> tmd_bytes(request.in_vectors[1].size); std::vector<u8> tmd_bytes(request.in_vectors[1].size);
Memory::CopyFromEmu(tmd_bytes.data(), request.in_vectors[1].address, tmd_bytes.size()); memory.CopyFromEmu(tmd_bytes.data(), request.in_vectors[1].address, tmd_bytes.size());
const ES::TMDReader tmd{std::move(tmd_bytes)}; const ES::TMDReader tmd{std::move(tmd_bytes)};
if (!tmd.IsValid()) if (!tmd.IsValid())
@ -945,8 +962,8 @@ IPCReply ESDevice::SetUpStreamKey(const Context& context, const IOCtlVRequest& r
u32 handle; u32 handle;
const ReturnCode ret = const ReturnCode ret =
SetUpStreamKey(context.uid, Memory::GetPointer(request.in_vectors[0].address), tmd, &handle); SetUpStreamKey(context.uid, memory.GetPointer(request.in_vectors[0].address), tmd, &handle);
Memory::Write_U32(handle, request.io_vectors[0].address); memory.Write_U32(handle, request.io_vectors[0].address);
return IPCReply(ret); return IPCReply(ret);
} }
@ -955,7 +972,9 @@ IPCReply ESDevice::DeleteStreamKey(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u32 handle = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 handle = memory.Read_U32(request.in_vectors[0].address);
return IPCReply(m_ios.GetIOSC().DeleteObject(handle, PID_ES)); return IPCReply(m_ios.GetIOSC().DeleteObject(handle, PID_ES));
} }

View File

@ -15,6 +15,7 @@
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/IOS/IOSC.h" #include "Core/IOS/IOSC.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -34,7 +35,10 @@ IPCReply ESDevice::GetDeviceId(const IOCtlVRequest& request)
const ReturnCode ret = GetDeviceId(&device_id); const ReturnCode ret = GetDeviceId(&device_id);
if (ret != IPC_SUCCESS) if (ret != IPC_SUCCESS)
return IPCReply(ret); return IPCReply(ret);
Memory::Write_U32(device_id, request.io_vectors[0].address);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(device_id, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -43,11 +47,13 @@ IPCReply ESDevice::Encrypt(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(3, 2)) if (!request.HasNumberOfValidVectors(3, 2))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
u32 keyIndex = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u8* source = Memory::GetPointer(request.in_vectors[2].address); auto& memory = system.GetMemory();
u32 keyIndex = memory.Read_U32(request.in_vectors[0].address);
u8* source = memory.GetPointer(request.in_vectors[2].address);
u32 size = request.in_vectors[2].size; u32 size = request.in_vectors[2].size;
u8* iv = Memory::GetPointer(request.io_vectors[0].address); u8* iv = memory.GetPointer(request.io_vectors[0].address);
u8* destination = Memory::GetPointer(request.io_vectors[1].address); u8* destination = memory.GetPointer(request.io_vectors[1].address);
// TODO: Check whether the active title is allowed to encrypt. // TODO: Check whether the active title is allowed to encrypt.
@ -60,11 +66,13 @@ IPCReply ESDevice::Decrypt(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(3, 2)) if (!request.HasNumberOfValidVectors(3, 2))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
u32 keyIndex = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u8* source = Memory::GetPointer(request.in_vectors[2].address); auto& memory = system.GetMemory();
u32 keyIndex = memory.Read_U32(request.in_vectors[0].address);
u8* source = memory.GetPointer(request.in_vectors[2].address);
u32 size = request.in_vectors[2].size; u32 size = request.in_vectors[2].size;
u8* iv = Memory::GetPointer(request.io_vectors[0].address); u8* iv = memory.GetPointer(request.io_vectors[0].address);
u8* destination = Memory::GetPointer(request.io_vectors[1].address); u8* destination = memory.GetPointer(request.io_vectors[1].address);
// TODO: Check whether the active title is allowed to decrypt. // TODO: Check whether the active title is allowed to decrypt.
@ -92,8 +100,10 @@ IPCReply ESDevice::GetDeviceCertificate(const IOCtlVRequest& request)
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETDEVICECERT"); INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETDEVICECERT");
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const IOS::CertECC cert = m_ios.GetIOSC().GetDeviceCertificate(); const IOS::CertECC cert = m_ios.GetIOSC().GetDeviceCertificate();
Memory::CopyToEmu(request.io_vectors[0].address, &cert, sizeof(cert)); memory.CopyToEmu(request.io_vectors[0].address, &cert, sizeof(cert));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -103,10 +113,12 @@ IPCReply ESDevice::Sign(const IOCtlVRequest& request)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_SIGN"); INFO_LOG_FMT(IOS_ES, "IOCTL_ES_SIGN");
u8* ap_cert_out = Memory::GetPointer(request.io_vectors[1].address); auto& system = Core::System::GetInstance();
u8* data = Memory::GetPointer(request.in_vectors[0].address); auto& memory = system.GetMemory();
u8* ap_cert_out = memory.GetPointer(request.io_vectors[1].address);
u8* data = memory.GetPointer(request.in_vectors[0].address);
u32 data_size = request.in_vectors[0].size; u32 data_size = request.in_vectors[0].size;
u8* sig_out = Memory::GetPointer(request.io_vectors[0].address); u8* sig_out = memory.GetPointer(request.io_vectors[0].address);
if (!m_title_context.active) if (!m_title_context.active)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
@ -193,14 +205,17 @@ IPCReply ESDevice::VerifySign(const IOCtlVRequest& request)
if (request.in_vectors[1].size != sizeof(Common::ec::Signature)) if (request.in_vectors[1].size != sizeof(Common::ec::Signature))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> hash(request.in_vectors[0].size); std::vector<u8> hash(request.in_vectors[0].size);
Memory::CopyFromEmu(hash.data(), request.in_vectors[0].address, hash.size()); memory.CopyFromEmu(hash.data(), request.in_vectors[0].address, hash.size());
std::vector<u8> ecc_signature(request.in_vectors[1].size); std::vector<u8> ecc_signature(request.in_vectors[1].size);
Memory::CopyFromEmu(ecc_signature.data(), request.in_vectors[1].address, ecc_signature.size()); memory.CopyFromEmu(ecc_signature.data(), request.in_vectors[1].address, ecc_signature.size());
std::vector<u8> certs(request.in_vectors[2].size); std::vector<u8> certs(request.in_vectors[2].size);
Memory::CopyFromEmu(certs.data(), request.in_vectors[2].address, certs.size()); memory.CopyFromEmu(certs.data(), request.in_vectors[2].address, certs.size());
return IPCReply(VerifySign(hash, ecc_signature, certs)); return IPCReply(VerifySign(hash, ecc_signature, certs));
} }

View File

@ -10,6 +10,7 @@
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/IOS/FS/FileSystemProxy.h" #include "Core/IOS/FS/FileSystemProxy.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -56,8 +57,10 @@ IPCReply ESDevice::OpenContent(u32 uid, const IOCtlVRequest& request)
return ES_EINVAL; return ES_EINVAL;
} }
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 content_index = Memory::Read_U32(request.in_vectors[2].address); auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const u32 content_index = memory.Read_U32(request.in_vectors[2].address);
// TODO: check the ticket view, check permissions. // TODO: check the ticket view, check permissions.
const auto tmd = FindInstalledTMD(title_id, ticks); const auto tmd = FindInstalledTMD(title_id, ticks);
@ -74,7 +77,9 @@ IPCReply ESDevice::OpenActiveTitleContent(u32 caller_uid, const IOCtlVRequest& r
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32))
return ES_EINVAL; return ES_EINVAL;
const u32 content_index = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 content_index = memory.Read_U32(request.in_vectors[0].address);
if (!m_title_context.active) if (!m_title_context.active)
return ES_EINVAL; return ES_EINVAL;
@ -109,13 +114,15 @@ IPCReply ESDevice::ReadContent(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32))
return ES_EINVAL; return ES_EINVAL;
const u32 cfd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 cfd = memory.Read_U32(request.in_vectors[0].address);
const u32 size = request.io_vectors[0].size; const u32 size = request.io_vectors[0].size;
const u32 addr = request.io_vectors[0].address; const u32 addr = request.io_vectors[0].address;
INFO_LOG_FMT(IOS_ES, "ReadContent(uid={:#x}, cfd={}, size={}, addr={:08x})", uid, cfd, size, INFO_LOG_FMT(IOS_ES, "ReadContent(uid={:#x}, cfd={}, size={}, addr={:08x})", uid, cfd, size,
addr); addr);
return ReadContent(cfd, Memory::GetPointer(addr), size, uid, ticks); return ReadContent(cfd, memory.GetPointer(addr), size, uid, ticks);
}); });
} }
@ -142,7 +149,9 @@ IPCReply ESDevice::CloseContent(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u32))
return ES_EINVAL; return ES_EINVAL;
const u32 cfd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 cfd = memory.Read_U32(request.in_vectors[0].address);
return CloseContent(cfd, uid, ticks); return CloseContent(cfd, uid, ticks);
}); });
} }
@ -167,9 +176,11 @@ IPCReply ESDevice::SeekContent(u32 uid, const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(3, 0)) if (!request.HasNumberOfValidVectors(3, 0))
return ES_EINVAL; return ES_EINVAL;
const u32 cfd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 offset = Memory::Read_U32(request.in_vectors[1].address); auto& memory = system.GetMemory();
const auto mode = static_cast<SeekMode>(Memory::Read_U32(request.in_vectors[2].address)); const u32 cfd = memory.Read_U32(request.in_vectors[0].address);
const u32 offset = memory.Read_U32(request.in_vectors[1].address);
const auto mode = static_cast<SeekMode>(memory.Read_U32(request.in_vectors[2].address));
return SeekContent(cfd, offset, mode, uid, ticks); return SeekContent(cfd, offset, mode, uid, ticks);
}); });

View File

@ -10,6 +10,7 @@
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -21,7 +22,9 @@ IPCReply ESDevice::GetStoredContentsCount(const ES::TMDReader& tmd, const IOCtlV
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u16 num_contents = static_cast<u16>(GetStoredContentsFromTMD(tmd).size()); const u16 num_contents = static_cast<u16>(GetStoredContentsFromTMD(tmd).size());
Memory::Write_U32(num_contents, request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(num_contents, request.io_vectors[0].address);
INFO_LOG_FMT(IOS_ES, "GetStoredContentsCount ({:#x}): {} content(s) for {:016x}", INFO_LOG_FMT(IOS_ES, "GetStoredContentsCount ({:#x}): {} content(s) for {:016x}",
request.request, num_contents, tmd.GetTitleId()); request.request, num_contents, tmd.GetTitleId());
@ -35,16 +38,18 @@ IPCReply ESDevice::GetStoredContents(const ES::TMDReader& tmd, const IOCtlVReque
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (request.in_vectors[1].size != sizeof(u32) || if (request.in_vectors[1].size != sizeof(u32) ||
request.io_vectors[0].size != Memory::Read_U32(request.in_vectors[1].address) * sizeof(u32)) request.io_vectors[0].size != memory.Read_U32(request.in_vectors[1].address) * sizeof(u32))
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
const auto contents = GetStoredContentsFromTMD(tmd); const auto contents = GetStoredContentsFromTMD(tmd);
const u32 max_content_count = Memory::Read_U32(request.in_vectors[1].address); const u32 max_content_count = memory.Read_U32(request.in_vectors[1].address);
for (u32 i = 0; i < std::min(static_cast<u32>(contents.size()), max_content_count); ++i) for (u32 i = 0; i < std::min(static_cast<u32>(contents.size()), max_content_count); ++i)
Memory::Write_U32(contents[i].id, request.io_vectors[0].address + i * sizeof(u32)); memory.Write_U32(contents[i].id, request.io_vectors[0].address + i * sizeof(u32));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -54,7 +59,9 @@ IPCReply ESDevice::GetStoredContentsCount(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u64)) if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u64))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(title_id); const ES::TMDReader tmd = FindInstalledTMD(title_id);
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(FS_ENOENT); return IPCReply(FS_ENOENT);
@ -66,7 +73,9 @@ IPCReply ESDevice::GetStoredContents(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(2, 1) || request.in_vectors[0].size != sizeof(u64)) if (!request.HasNumberOfValidVectors(2, 1) || request.in_vectors[0].size != sizeof(u64))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(title_id); const ES::TMDReader tmd = FindInstalledTMD(title_id);
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(FS_ENOENT); return IPCReply(FS_ENOENT);
@ -79,7 +88,9 @@ IPCReply ESDevice::GetTMDStoredContentsCount(const IOCtlVRequest& request)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
std::vector<u8> tmd_bytes(request.in_vectors[0].size); std::vector<u8> tmd_bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size()); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size());
return GetStoredContentsCount(ES::TMDReader{std::move(tmd_bytes)}, request); return GetStoredContentsCount(ES::TMDReader{std::move(tmd_bytes)}, request);
} }
@ -89,7 +100,9 @@ IPCReply ESDevice::GetTMDStoredContents(const IOCtlVRequest& request)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
std::vector<u8> tmd_bytes(request.in_vectors[0].size); std::vector<u8> tmd_bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size()); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size());
const ES::TMDReader tmd{std::move(tmd_bytes)}; const ES::TMDReader tmd{std::move(tmd_bytes)};
if (!tmd.IsValid()) if (!tmd.IsValid())
@ -112,7 +125,9 @@ IPCReply ESDevice::GetTitleCount(const std::vector<u64>& titles, const IOCtlVReq
if (!request.HasNumberOfValidVectors(0, 1) || request.io_vectors[0].size != 4) if (!request.HasNumberOfValidVectors(0, 1) || request.io_vectors[0].size != 4)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::Write_U32(static_cast<u32>(titles.size()), request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(static_cast<u32>(titles.size()), request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -122,10 +137,12 @@ IPCReply ESDevice::GetTitles(const std::vector<u64>& titles, const IOCtlVRequest
if (!request.HasNumberOfValidVectors(1, 1)) if (!request.HasNumberOfValidVectors(1, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const size_t max_count = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const size_t max_count = memory.Read_U32(request.in_vectors[0].address);
for (size_t i = 0; i < std::min(max_count, titles.size()); i++) for (size_t i = 0; i < std::min(max_count, titles.size()); i++)
{ {
Memory::Write_U64(titles[i], request.io_vectors[0].address + static_cast<u32>(i) * sizeof(u64)); memory.Write_U64(titles[i], request.io_vectors[0].address + static_cast<u32>(i) * sizeof(u64));
INFO_LOG_FMT(IOS_ES, " title {:016x}", titles[i]); INFO_LOG_FMT(IOS_ES, " title {:016x}", titles[i]);
} }
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -148,13 +165,15 @@ IPCReply ESDevice::GetStoredTMDSize(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1)) if (!request.HasNumberOfValidVectors(1, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(title_id); const ES::TMDReader tmd = FindInstalledTMD(title_id);
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(FS_ENOENT); return IPCReply(FS_ENOENT);
const u32 tmd_size = static_cast<u32>(tmd.GetBytes().size()); const u32 tmd_size = static_cast<u32>(tmd.GetBytes().size());
Memory::Write_U32(tmd_size, request.io_vectors[0].address); memory.Write_U32(tmd_size, request.io_vectors[0].address);
INFO_LOG_FMT(IOS_ES, "GetStoredTMDSize: {} bytes for {:016x}", tmd_size, title_id); INFO_LOG_FMT(IOS_ES, "GetStoredTMDSize: {} bytes for {:016x}", tmd_size, title_id);
@ -166,19 +185,21 @@ IPCReply ESDevice::GetStoredTMD(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(2, 1)) if (!request.HasNumberOfValidVectors(2, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(title_id); const ES::TMDReader tmd = FindInstalledTMD(title_id);
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(FS_ENOENT); return IPCReply(FS_ENOENT);
// TODO: actually use this param in when writing to the outbuffer :/ // TODO: actually use this param in when writing to the outbuffer :/
const u32 MaxCount = Memory::Read_U32(request.in_vectors[1].address); const u32 MaxCount = memory.Read_U32(request.in_vectors[1].address);
const std::vector<u8>& raw_tmd = tmd.GetBytes(); const std::vector<u8>& raw_tmd = tmd.GetBytes();
if (raw_tmd.size() != request.io_vectors[0].size) if (raw_tmd.size() != request.io_vectors[0].size)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::CopyToEmu(request.io_vectors[0].address, raw_tmd.data(), raw_tmd.size()); memory.CopyToEmu(request.io_vectors[0].address, raw_tmd.data(), raw_tmd.size());
INFO_LOG_FMT(IOS_ES, "GetStoredTMD: title {:016x} (buffer size: {})", title_id, MaxCount); INFO_LOG_FMT(IOS_ES, "GetStoredTMD: title {:016x} (buffer size: {})", title_id, MaxCount);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -204,7 +225,9 @@ IPCReply ESDevice::GetBoot2Version(const IOCtlVRequest& request)
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETBOOT2VERSION"); INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETBOOT2VERSION");
// as of 26/02/2012, this was latest bootmii version // as of 26/02/2012, this was latest bootmii version
Memory::Write_U32(4, request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(4, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -214,7 +237,9 @@ IPCReply ESDevice::GetSharedContentsCount(const IOCtlVRequest& request) const
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u32 count = GetSharedContentsCount(); const u32 count = GetSharedContentsCount();
Memory::Write_U32(count, request.io_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(count, request.io_vectors[0].address);
INFO_LOG_FMT(IOS_ES, "GetSharedContentsCount: {} contents", count); INFO_LOG_FMT(IOS_ES, "GetSharedContentsCount: {} contents", count);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -225,13 +250,15 @@ IPCReply ESDevice::GetSharedContents(const IOCtlVRequest& request) const
if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u32 max_count = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 max_count = memory.Read_U32(request.in_vectors[0].address);
if (request.io_vectors[0].size != 20 * max_count) if (request.io_vectors[0].size != 20 * max_count)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const std::vector<std::array<u8, 20>> hashes = GetSharedContents(); const std::vector<std::array<u8, 20>> hashes = GetSharedContents();
const u32 count = std::min(static_cast<u32>(hashes.size()), max_count); const u32 count = std::min(static_cast<u32>(hashes.size()), max_count);
Memory::CopyToEmu(request.io_vectors[0].address, hashes.data(), 20 * count); memory.CopyToEmu(request.io_vectors[0].address, hashes.data(), 20 * count);
INFO_LOG_FMT(IOS_ES, "GetSharedContents: {} contents ({} requested)", count, max_count); INFO_LOG_FMT(IOS_ES, "GetSharedContents: {} contents ({} requested)", count, max_count);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);

View File

@ -19,6 +19,7 @@
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/IOS/FS/FileSystem.h" #include "Core/IOS/FS/FileSystem.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -97,10 +98,13 @@ IPCReply ESDevice::ImportTicket(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(3, 0)) if (!request.HasNumberOfValidVectors(3, 0))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> bytes(request.in_vectors[0].size); std::vector<u8> bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(bytes.data(), request.in_vectors[0].address, request.in_vectors[0].size); memory.CopyFromEmu(bytes.data(), request.in_vectors[0].address, request.in_vectors[0].size);
std::vector<u8> cert_chain(request.in_vectors[1].size); std::vector<u8> cert_chain(request.in_vectors[1].size);
Memory::CopyFromEmu(cert_chain.data(), request.in_vectors[1].address, request.in_vectors[1].size); memory.CopyFromEmu(cert_chain.data(), request.in_vectors[1].address, request.in_vectors[1].size);
return IPCReply(ImportTicket(bytes, cert_chain)); return IPCReply(ImportTicket(bytes, cert_chain));
} }
@ -187,8 +191,11 @@ IPCReply ESDevice::ImportTmd(Context& context, const IOCtlVRequest& request)
if (!ES::IsValidTMDSize(request.in_vectors[0].size)) if (!ES::IsValidTMDSize(request.in_vectors[0].size))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> tmd(request.in_vectors[0].size); std::vector<u8> tmd(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd.data(), request.in_vectors[0].address, request.in_vectors[0].size); memory.CopyFromEmu(tmd.data(), request.in_vectors[0].address, request.in_vectors[0].size);
return IPCReply(ImportTmd(context, tmd, m_title_context.tmd.GetTitleId(), return IPCReply(ImportTmd(context, tmd, m_title_context.tmd.GetTitleId(),
m_title_context.tmd.GetTitleFlags())); m_title_context.tmd.GetTitleFlags()));
} }
@ -273,10 +280,13 @@ IPCReply ESDevice::ImportTitleInit(Context& context, const IOCtlVRequest& reques
if (!ES::IsValidTMDSize(request.in_vectors[0].size)) if (!ES::IsValidTMDSize(request.in_vectors[0].size))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> tmd(request.in_vectors[0].size); std::vector<u8> tmd(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd.data(), request.in_vectors[0].address, request.in_vectors[0].size); memory.CopyFromEmu(tmd.data(), request.in_vectors[0].address, request.in_vectors[0].size);
std::vector<u8> certs(request.in_vectors[1].size); std::vector<u8> certs(request.in_vectors[1].size);
Memory::CopyFromEmu(certs.data(), request.in_vectors[1].address, request.in_vectors[1].size); memory.CopyFromEmu(certs.data(), request.in_vectors[1].address, request.in_vectors[1].size);
return IPCReply(ImportTitleInit(context, tmd, certs)); return IPCReply(ImportTitleInit(context, tmd, certs));
} }
@ -328,8 +338,11 @@ IPCReply ESDevice::ImportContentBegin(Context& context, const IOCtlVRequest& req
if (!request.HasNumberOfValidVectors(2, 0)) if (!request.HasNumberOfValidVectors(2, 0))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u32 content_id = Memory::Read_U32(request.in_vectors[1].address); auto& memory = system.GetMemory();
u64 title_id = memory.Read_U64(request.in_vectors[0].address);
u32 content_id = memory.Read_U32(request.in_vectors[1].address);
return IPCReply(ImportContentBegin(context, title_id, content_id)); return IPCReply(ImportContentBegin(context, title_id, content_id));
} }
@ -347,8 +360,11 @@ IPCReply ESDevice::ImportContentData(Context& context, const IOCtlVRequest& requ
if (!request.HasNumberOfValidVectors(2, 0)) if (!request.HasNumberOfValidVectors(2, 0))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
u32 content_fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u8* data_start = Memory::GetPointer(request.in_vectors[1].address); auto& memory = system.GetMemory();
u32 content_fd = memory.Read_U32(request.in_vectors[0].address);
u8* data_start = memory.GetPointer(request.in_vectors[1].address);
return IPCReply(ImportContentData(context, content_fd, data_start, request.in_vectors[1].size)); return IPCReply(ImportContentData(context, content_fd, data_start, request.in_vectors[1].size));
} }
@ -430,7 +446,10 @@ IPCReply ESDevice::ImportContentEnd(Context& context, const IOCtlVRequest& reque
if (!request.HasNumberOfValidVectors(1, 0)) if (!request.HasNumberOfValidVectors(1, 0))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
u32 content_fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 content_fd = memory.Read_U32(request.in_vectors[0].address);
return IPCReply(ImportContentEnd(context, content_fd)); return IPCReply(ImportContentEnd(context, content_fd));
} }
@ -542,7 +561,10 @@ IPCReply ESDevice::DeleteTitle(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 8) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 8)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
return IPCReply(DeleteTitle(title_id)); return IPCReply(DeleteTitle(title_id));
} }
@ -595,7 +617,10 @@ IPCReply ESDevice::DeleteTicket(const IOCtlVRequest& request)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
return IPCReply(DeleteTicket(Memory::GetPointer(request.in_vectors[0].address)));
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return IPCReply(DeleteTicket(memory.GetPointer(request.in_vectors[0].address)));
} }
ReturnCode ESDevice::DeleteTitleContent(u64 title_id) const ReturnCode ESDevice::DeleteTitleContent(u64 title_id) const
@ -621,7 +646,10 @@ IPCReply ESDevice::DeleteTitleContent(const IOCtlVRequest& request)
{ {
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u64)) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sizeof(u64))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
return IPCReply(DeleteTitleContent(Memory::Read_U64(request.in_vectors[0].address)));
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return IPCReply(DeleteTitleContent(memory.Read_U64(request.in_vectors[0].address)));
} }
ReturnCode ESDevice::DeleteContent(u64 title_id, u32 content_id) const ReturnCode ESDevice::DeleteContent(u64 title_id, u32 content_id) const
@ -649,8 +677,11 @@ IPCReply ESDevice::DeleteContent(const IOCtlVRequest& request)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
return IPCReply(DeleteContent(Memory::Read_U64(request.in_vectors[0].address),
Memory::Read_U32(request.in_vectors[1].address))); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return IPCReply(DeleteContent(memory.Read_U64(request.in_vectors[0].address),
memory.Read_U32(request.in_vectors[1].address)));
} }
ReturnCode ESDevice::ExportTitleInit(Context& context, u64 title_id, u8* tmd_bytes, u32 tmd_size, ReturnCode ESDevice::ExportTitleInit(Context& context, u64 title_id, u8* tmd_bytes, u32 tmd_size,
@ -687,8 +718,11 @@ IPCReply ESDevice::ExportTitleInit(Context& context, const IOCtlVRequest& reques
if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != 8) if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != 8)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u8* tmd_bytes = Memory::GetPointer(request.io_vectors[0].address); auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
u8* tmd_bytes = memory.GetPointer(request.io_vectors[0].address);
const u32 tmd_size = request.io_vectors[0].size; const u32 tmd_size = request.io_vectors[0].size;
return IPCReply(ExportTitleInit(context, title_id, tmd_bytes, tmd_size, return IPCReply(ExportTitleInit(context, title_id, tmd_bytes, tmd_size,
@ -733,8 +767,11 @@ IPCReply ESDevice::ExportContentBegin(Context& context, const IOCtlVRequest& req
request.in_vectors[1].size != 4) request.in_vectors[1].size != 4)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 content_id = Memory::Read_U32(request.in_vectors[1].address); auto& memory = system.GetMemory();
const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const u32 content_id = memory.Read_U32(request.in_vectors[1].address);
return IPCReply(ExportContentBegin(context, title_id, content_id)); return IPCReply(ExportContentBegin(context, title_id, content_id));
} }
@ -781,8 +818,11 @@ IPCReply ESDevice::ExportContentData(Context& context, const IOCtlVRequest& requ
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
const u32 content_fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
u8* data = Memory::GetPointer(request.io_vectors[0].address); auto& memory = system.GetMemory();
const u32 content_fd = memory.Read_U32(request.in_vectors[0].address);
u8* data = memory.GetPointer(request.io_vectors[0].address);
const u32 bytes_to_read = request.io_vectors[0].size; const u32 bytes_to_read = request.io_vectors[0].size;
return IPCReply(ExportContentData(context, content_fd, data, bytes_to_read)); return IPCReply(ExportContentData(context, content_fd, data, bytes_to_read));
@ -800,7 +840,10 @@ IPCReply ESDevice::ExportContentEnd(Context& context, const IOCtlVRequest& reque
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 4) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != 4)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u32 content_fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 content_fd = memory.Read_U32(request.in_vectors[0].address);
return IPCReply(ExportContentEnd(context, content_fd)); return IPCReply(ExportContentEnd(context, content_fd));
} }
@ -857,7 +900,11 @@ IPCReply ESDevice::DeleteSharedContent(const IOCtlVRequest& request)
std::array<u8, 20> sha1; std::array<u8, 20> sha1;
if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sha1.size()) if (!request.HasNumberOfValidVectors(1, 0) || request.in_vectors[0].size != sha1.size())
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::CopyFromEmu(sha1.data(), request.in_vectors[0].address, request.in_vectors[0].size);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(sha1.data(), request.in_vectors[0].address, request.in_vectors[0].size);
return IPCReply(DeleteSharedContent(sha1)); return IPCReply(DeleteSharedContent(sha1));
} }
} // namespace IOS::HLE } // namespace IOS::HLE

View File

@ -16,6 +16,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/IOS/VersionInfo.h" #include "Core/IOS/VersionInfo.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -40,7 +41,10 @@ IPCReply ESDevice::GetTicketViewCount(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1)) if (!request.HasNumberOfValidVectors(1, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 TitleID = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 TitleID = memory.Read_U64(request.in_vectors[0].address);
const ES::TicketReader ticket = FindSignedTicket(TitleID); const ES::TicketReader ticket = FindSignedTicket(TitleID);
u32 view_count = ticket.IsValid() ? static_cast<u32>(ticket.GetNumberOfTickets()) : 0; u32 view_count = ticket.IsValid() ? static_cast<u32>(ticket.GetNumberOfTickets()) : 0;
@ -59,7 +63,7 @@ IPCReply ESDevice::GetTicketViewCount(const IOCtlVRequest& request)
INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETVIEWCNT for titleID: {:016x} (View Count = {})", TitleID, INFO_LOG_FMT(IOS_ES, "IOCTL_ES_GETVIEWCNT for titleID: {:016x} (View Count = {})", TitleID,
view_count); view_count);
Memory::Write_U32(view_count, request.io_vectors[0].address); memory.Write_U32(view_count, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -68,8 +72,11 @@ IPCReply ESDevice::GetTicketViews(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(2, 1)) if (!request.HasNumberOfValidVectors(2, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 TitleID = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 maxViews = Memory::Read_U32(request.in_vectors[1].address); auto& memory = system.GetMemory();
const u64 TitleID = memory.Read_U64(request.in_vectors[0].address);
const u32 maxViews = memory.Read_U32(request.in_vectors[1].address);
const ES::TicketReader ticket = FindSignedTicket(TitleID); const ES::TicketReader ticket = FindSignedTicket(TitleID);
@ -83,13 +90,13 @@ IPCReply ESDevice::GetTicketViews(const IOCtlVRequest& request)
for (u32 view = 0; view < number_of_views; ++view) for (u32 view = 0; view < number_of_views; ++view)
{ {
const std::vector<u8> ticket_view = ticket.GetRawTicketView(view); const std::vector<u8> ticket_view = ticket.GetRawTicketView(view);
Memory::CopyToEmu(request.io_vectors[0].address + view * sizeof(ES::TicketView), memory.CopyToEmu(request.io_vectors[0].address + view * sizeof(ES::TicketView),
ticket_view.data(), ticket_view.size()); ticket_view.data(), ticket_view.size());
} }
} }
else if (ShouldReturnFakeViewsForIOSes(TitleID, m_title_context)) else if (ShouldReturnFakeViewsForIOSes(TitleID, m_title_context))
{ {
Memory::Memset(request.io_vectors[0].address, 0, sizeof(ES::TicketView)); memory.Memset(request.io_vectors[0].address, 0, sizeof(ES::TicketView));
WARN_LOG_FMT(IOS_ES, "GetViews: Faking IOS title {:016x} being present", TitleID); WARN_LOG_FMT(IOS_ES, "GetViews: Faking IOS title {:016x} being present", TitleID);
} }
@ -152,8 +159,11 @@ IPCReply ESDevice::GetV0TicketFromView(const IOCtlVRequest& request)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
return IPCReply(GetTicketFromView(Memory::GetPointer(request.in_vectors[0].address),
Memory::GetPointer(request.io_vectors[0].address), nullptr, 0)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return IPCReply(GetTicketFromView(memory.GetPointer(request.in_vectors[0].address),
memory.GetPointer(request.io_vectors[0].address), nullptr, 0));
} }
IPCReply ESDevice::GetTicketSizeFromView(const IOCtlVRequest& request) IPCReply ESDevice::GetTicketSizeFromView(const IOCtlVRequest& request)
@ -165,9 +175,12 @@ IPCReply ESDevice::GetTicketSizeFromView(const IOCtlVRequest& request)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
const ReturnCode ret = GetTicketFromView(Memory::GetPointer(request.in_vectors[0].address),
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const ReturnCode ret = GetTicketFromView(memory.GetPointer(request.in_vectors[0].address),
nullptr, &ticket_size, std::nullopt); nullptr, &ticket_size, std::nullopt);
Memory::Write_U32(ticket_size, request.io_vectors[0].address); memory.Write_U32(ticket_size, request.io_vectors[0].address);
return IPCReply(ret); return IPCReply(ret);
} }
@ -180,12 +193,15 @@ IPCReply ESDevice::GetTicketFromView(const IOCtlVRequest& request)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
u32 ticket_size = Memory::Read_U32(request.in_vectors[1].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 ticket_size = memory.Read_U32(request.in_vectors[1].address);
if (ticket_size != request.io_vectors[0].size) if (ticket_size != request.io_vectors[0].size)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
return IPCReply(GetTicketFromView(Memory::GetPointer(request.in_vectors[0].address), return IPCReply(GetTicketFromView(memory.GetPointer(request.in_vectors[0].address),
Memory::GetPointer(request.io_vectors[0].address), &ticket_size, memory.GetPointer(request.io_vectors[0].address), &ticket_size,
std::nullopt)); std::nullopt));
} }
@ -194,14 +210,17 @@ IPCReply ESDevice::GetTMDViewSize(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1)) if (!request.HasNumberOfValidVectors(1, 1))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u64 TitleID = Memory::Read_U64(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u64 TitleID = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(TitleID); const ES::TMDReader tmd = FindInstalledTMD(TitleID);
if (!tmd.IsValid()) if (!tmd.IsValid())
return IPCReply(FS_ENOENT); return IPCReply(FS_ENOENT);
const u32 view_size = static_cast<u32>(tmd.GetRawView().size()); const u32 view_size = static_cast<u32>(tmd.GetRawView().size());
Memory::Write_U32(view_size, request.io_vectors[0].address); memory.Write_U32(view_size, request.io_vectors[0].address);
INFO_LOG_FMT(IOS_ES, "GetTMDViewSize: {} bytes for title {:016x}", view_size, TitleID); INFO_LOG_FMT(IOS_ES, "GetTMDViewSize: {} bytes for title {:016x}", view_size, TitleID);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -209,15 +228,18 @@ IPCReply ESDevice::GetTMDViewSize(const IOCtlVRequest& request)
IPCReply ESDevice::GetTMDViews(const IOCtlVRequest& request) IPCReply ESDevice::GetTMDViews(const IOCtlVRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (!request.HasNumberOfValidVectors(2, 1) || if (!request.HasNumberOfValidVectors(2, 1) ||
request.in_vectors[0].size != sizeof(ES::TMDHeader::title_id) || request.in_vectors[0].size != sizeof(ES::TMDHeader::title_id) ||
request.in_vectors[1].size != sizeof(u32) || request.in_vectors[1].size != sizeof(u32) ||
Memory::Read_U32(request.in_vectors[1].address) != request.io_vectors[0].size) memory.Read_U32(request.in_vectors[1].address) != request.io_vectors[0].size)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
const u64 title_id = Memory::Read_U64(request.in_vectors[0].address); const u64 title_id = memory.Read_U64(request.in_vectors[0].address);
const ES::TMDReader tmd = FindInstalledTMD(title_id); const ES::TMDReader tmd = FindInstalledTMD(title_id);
if (!tmd.IsValid()) if (!tmd.IsValid())
@ -227,7 +249,7 @@ IPCReply ESDevice::GetTMDViews(const IOCtlVRequest& request)
if (request.io_vectors[0].size < raw_view.size()) if (request.io_vectors[0].size < raw_view.size())
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::CopyToEmu(request.io_vectors[0].address, raw_view.data(), raw_view.size()); memory.CopyToEmu(request.io_vectors[0].address, raw_view.data(), raw_view.size());
INFO_LOG_FMT(IOS_ES, "GetTMDView: {} bytes for title {:016x}", raw_view.size(), title_id); INFO_LOG_FMT(IOS_ES, "GetTMDView: {} bytes for title {:016x}", raw_view.size(), title_id);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -245,13 +267,16 @@ IPCReply ESDevice::DIGetTMDViewSize(const IOCtlVRequest& request)
if (request.io_vectors[0].size != sizeof(u32)) if (request.io_vectors[0].size != sizeof(u32))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const bool has_tmd = request.in_vectors[0].size != 0; const bool has_tmd = request.in_vectors[0].size != 0;
size_t tmd_view_size = 0; size_t tmd_view_size = 0;
if (has_tmd) if (has_tmd)
{ {
std::vector<u8> tmd_bytes(request.in_vectors[0].size); std::vector<u8> tmd_bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size()); memory.CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size());
const ES::TMDReader tmd{std::move(tmd_bytes)}; const ES::TMDReader tmd{std::move(tmd_bytes)};
// Yes, this returns -1017, not ES_INVALID_TMD. // Yes, this returns -1017, not ES_INVALID_TMD.
@ -270,7 +295,7 @@ IPCReply ESDevice::DIGetTMDViewSize(const IOCtlVRequest& request)
tmd_view_size = m_title_context.tmd.GetRawView().size(); tmd_view_size = m_title_context.tmd.GetRawView().size();
} }
Memory::Write_U32(static_cast<u32>(tmd_view_size), request.io_vectors[0].address); memory.Write_U32(static_cast<u32>(tmd_view_size), request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -283,9 +308,12 @@ IPCReply ESDevice::DIGetTMDView(const IOCtlVRequest& request)
if (request.in_vectors[0].size >= 4 * 1024 * 1024) if (request.in_vectors[0].size >= 4 * 1024 * 1024)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Check whether the TMD view size is consistent. // Check whether the TMD view size is consistent.
if (request.in_vectors[1].size != sizeof(u32) || if (request.in_vectors[1].size != sizeof(u32) ||
Memory::Read_U32(request.in_vectors[1].address) != request.io_vectors[0].size) memory.Read_U32(request.in_vectors[1].address) != request.io_vectors[0].size)
{ {
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
} }
@ -296,7 +324,7 @@ IPCReply ESDevice::DIGetTMDView(const IOCtlVRequest& request)
if (has_tmd) if (has_tmd)
{ {
std::vector<u8> tmd_bytes(request.in_vectors[0].size); std::vector<u8> tmd_bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size()); memory.CopyFromEmu(tmd_bytes.data(), request.in_vectors[0].address, tmd_bytes.size());
const ES::TMDReader tmd{std::move(tmd_bytes)}; const ES::TMDReader tmd{std::move(tmd_bytes)};
if (!tmd.IsValid()) if (!tmd.IsValid())
@ -316,7 +344,7 @@ IPCReply ESDevice::DIGetTMDView(const IOCtlVRequest& request)
if (tmd_view.size() > request.io_vectors[0].size) if (tmd_view.size() > request.io_vectors[0].size)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::CopyToEmu(request.io_vectors[0].address, tmd_view.data(), tmd_view.size()); memory.CopyToEmu(request.io_vectors[0].address, tmd_view.data(), tmd_view.size());
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -334,6 +362,9 @@ IPCReply ESDevice::DIGetTicketView(const IOCtlVRequest& request)
if (!has_ticket_vector && request.in_vectors[0].size != 0) if (!has_ticket_vector && request.in_vectors[0].size != 0)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::vector<u8> view; std::vector<u8> view;
// If no ticket was passed in, IOS returns the ticket view for the current title. // If no ticket was passed in, IOS returns the ticket view for the current title.
@ -348,13 +379,13 @@ IPCReply ESDevice::DIGetTicketView(const IOCtlVRequest& request)
else else
{ {
std::vector<u8> ticket_bytes(request.in_vectors[0].size); std::vector<u8> ticket_bytes(request.in_vectors[0].size);
Memory::CopyFromEmu(ticket_bytes.data(), request.in_vectors[0].address, ticket_bytes.size()); memory.CopyFromEmu(ticket_bytes.data(), request.in_vectors[0].address, ticket_bytes.size());
const ES::TicketReader ticket{std::move(ticket_bytes)}; const ES::TicketReader ticket{std::move(ticket_bytes)};
view = ticket.GetRawTicketView(0); view = ticket.GetRawTicketView(0);
} }
Memory::CopyToEmu(request.io_vectors[0].address, view.data(), view.size()); memory.CopyToEmu(request.io_vectors[0].address, view.data(), view.size());
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -366,7 +397,9 @@ IPCReply ESDevice::DIGetTMDSize(const IOCtlVRequest& request)
if (!m_title_context.active) if (!m_title_context.active)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::Write_U32(static_cast<u32>(m_title_context.tmd.GetBytes().size()), auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(static_cast<u32>(m_title_context.tmd.GetBytes().size()),
request.io_vectors[0].address); request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -376,7 +409,10 @@ IPCReply ESDevice::DIGetTMD(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32)) if (!request.HasNumberOfValidVectors(1, 1) || request.in_vectors[0].size != sizeof(u32))
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
const u32 tmd_size = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 tmd_size = memory.Read_U32(request.in_vectors[0].address);
if (tmd_size != request.io_vectors[0].size) if (tmd_size != request.io_vectors[0].size)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
@ -388,7 +424,7 @@ IPCReply ESDevice::DIGetTMD(const IOCtlVRequest& request)
if (static_cast<u32>(tmd_bytes.size()) > tmd_size) if (static_cast<u32>(tmd_bytes.size()) > tmd_size)
return IPCReply(ES_EINVAL); return IPCReply(ES_EINVAL);
Memory::CopyToEmu(request.io_vectors[0].address, tmd_bytes.data(), tmd_bytes.size()); memory.CopyToEmu(request.io_vectors[0].address, tmd_bytes.data(), tmd_bytes.size());
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
} // namespace IOS::HLE } // namespace IOS::HLE

View File

@ -16,6 +16,7 @@
#include "Core/HW/SystemTimers.h" #include "Core/HW/SystemTimers.h"
#include "Core/IOS/FS/FileSystem.h" #include "Core/IOS/FS/FileSystem.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -312,7 +313,9 @@ bool FSDevice::HasCacheForFile(u64 fd, u32 offset) const
std::optional<IPCReply> FSDevice::Read(const ReadWriteRequest& request) std::optional<IPCReply> FSDevice::Read(const ReadWriteRequest& request)
{ {
return MakeIPCReply([&](Ticks t) { return MakeIPCReply([&](Ticks t) {
return Read(request.fd, Memory::GetPointer(request.buffer), request.size, request.buffer, t); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return Read(request.fd, memory.GetPointer(request.buffer), request.size, request.buffer, t);
}); });
} }
@ -340,7 +343,9 @@ s32 FSDevice::Read(u64 fd, u8* data, u32 size, std::optional<u32> ipc_buffer_add
std::optional<IPCReply> FSDevice::Write(const ReadWriteRequest& request) std::optional<IPCReply> FSDevice::Write(const ReadWriteRequest& request)
{ {
return MakeIPCReply([&](Ticks t) { return MakeIPCReply([&](Ticks t) {
return Write(request.fd, Memory::GetPointer(request.buffer), request.size, request.buffer, t); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return Write(request.fd, memory.GetPointer(request.buffer), request.size, request.buffer, t);
}); });
} }
@ -422,8 +427,11 @@ static Result<T> GetParams(const IOCtlRequest& request)
if (request.buffer_in_size < sizeof(T)) if (request.buffer_in_size < sizeof(T))
return ResultCode::Invalid; return ResultCode::Invalid;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
T params; T params;
Memory::CopyFromEmu(&params, request.buffer_in, sizeof(params)); memory.CopyFromEmu(&params, request.buffer_in, sizeof(params));
return params; return params;
} }
@ -498,6 +506,9 @@ IPCReply FSDevice::GetStats(const Handle& handle, const IOCtlRequest& request)
if (!stats) if (!stats)
return IPCReply(ConvertResult(stats.Error())); return IPCReply(ConvertResult(stats.Error()));
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
ISFSNandStats out; ISFSNandStats out;
out.cluster_size = stats->cluster_size; out.cluster_size = stats->cluster_size;
out.free_clusters = stats->free_clusters; out.free_clusters = stats->free_clusters;
@ -506,7 +517,7 @@ IPCReply FSDevice::GetStats(const Handle& handle, const IOCtlRequest& request)
out.reserved_clusters = stats->reserved_clusters; out.reserved_clusters = stats->reserved_clusters;
out.free_inodes = stats->free_inodes; out.free_inodes = stats->free_inodes;
out.used_inodes = stats->used_inodes; out.used_inodes = stats->used_inodes;
Memory::CopyToEmu(request.buffer_out, &out, sizeof(out)); memory.CopyToEmu(request.buffer_out, &out, sizeof(out));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -530,28 +541,31 @@ IPCReply FSDevice::ReadDirectory(const Handle& handle, const IOCtlVRequest& requ
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 file_list_address, file_count_address, max_count; u32 file_list_address, file_count_address, max_count;
if (request.in_vectors.size() == 2) if (request.in_vectors.size() == 2)
{ {
if (request.in_vectors[1].size != 4 || request.io_vectors[1].size != 4) if (request.in_vectors[1].size != 4 || request.io_vectors[1].size != 4)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
max_count = Memory::Read_U32(request.in_vectors[1].address); max_count = memory.Read_U32(request.in_vectors[1].address);
file_count_address = request.io_vectors[1].address; file_count_address = request.io_vectors[1].address;
file_list_address = request.io_vectors[0].address; file_list_address = request.io_vectors[0].address;
if (request.io_vectors[0].size != 13 * max_count) if (request.io_vectors[0].size != 13 * max_count)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
Memory::Write_U32(max_count, file_count_address); memory.Write_U32(max_count, file_count_address);
} }
else else
{ {
if (request.io_vectors[0].size != 4) if (request.io_vectors[0].size != 4)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
max_count = Memory::Read_U32(request.io_vectors[0].address); max_count = memory.Read_U32(request.io_vectors[0].address);
file_count_address = request.io_vectors[0].address; file_count_address = request.io_vectors[0].address;
file_list_address = 0; file_list_address = 0;
} }
const std::string directory = Memory::GetString(request.in_vectors[0].address, 64); const std::string directory = memory.GetString(request.in_vectors[0].address, 64);
const Result<std::vector<std::string>> list = const Result<std::vector<std::string>> list =
m_ios.GetFS()->ReadDirectory(handle.uid, handle.gid, directory); m_ios.GetFS()->ReadDirectory(handle.uid, handle.gid, directory);
LogResult(list, "ReadDirectory({})", directory); LogResult(list, "ReadDirectory({})", directory);
@ -560,19 +574,19 @@ IPCReply FSDevice::ReadDirectory(const Handle& handle, const IOCtlVRequest& requ
if (!file_list_address) if (!file_list_address)
{ {
Memory::Write_U32(static_cast<u32>(list->size()), file_count_address); memory.Write_U32(static_cast<u32>(list->size()), file_count_address);
return GetFSReply(IPC_SUCCESS); return GetFSReply(IPC_SUCCESS);
} }
for (size_t i = 0; i < list->size() && i < max_count; ++i) for (size_t i = 0; i < list->size() && i < max_count; ++i)
{ {
Memory::Memset(file_list_address, 0, 13); memory.Memset(file_list_address, 0, 13);
Memory::CopyToEmu(file_list_address, (*list)[i].data(), (*list)[i].size()); memory.CopyToEmu(file_list_address, (*list)[i].data(), (*list)[i].size());
Memory::Write_U8(0, file_list_address + 12); memory.Write_U8(0, file_list_address + 12);
file_list_address += static_cast<u32>((*list)[i].size()) + 1; file_list_address += static_cast<u32>((*list)[i].size()) + 1;
} }
// Write the actual number of entries in the buffer. // Write the actual number of entries in the buffer.
Memory::Write_U32(std::min(max_count, static_cast<u32>(list->size())), file_count_address); memory.Write_U32(std::min(max_count, static_cast<u32>(list->size())), file_count_address);
return GetFSReply(IPC_SUCCESS); return GetFSReply(IPC_SUCCESS);
} }
@ -593,7 +607,10 @@ IPCReply FSDevice::GetAttribute(const Handle& handle, const IOCtlRequest& reques
if (request.buffer_in_size < 64 || request.buffer_out_size < sizeof(ISFSParams)) if (request.buffer_in_size < 64 || request.buffer_out_size < sizeof(ISFSParams))
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
const std::string path = Memory::GetString(request.buffer_in, 64); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string path = memory.GetString(request.buffer_in, 64);
const auto ticks = EstimateFileLookupTicks(path, FileLookupMode::Split); const auto ticks = EstimateFileLookupTicks(path, FileLookupMode::Split);
const Result<Metadata> metadata = m_ios.GetFS()->GetMetadata(handle.uid, handle.gid, path); const Result<Metadata> metadata = m_ios.GetFS()->GetMetadata(handle.uid, handle.gid, path);
LogResult(metadata, "GetMetadata({})", path); LogResult(metadata, "GetMetadata({})", path);
@ -608,7 +625,7 @@ IPCReply FSDevice::GetAttribute(const Handle& handle, const IOCtlRequest& reques
out.gid = metadata->gid; out.gid = metadata->gid;
out.attribute = metadata->attribute; out.attribute = metadata->attribute;
out.modes = metadata->modes; out.modes = metadata->modes;
Memory::CopyToEmu(request.buffer_out, &out, sizeof(out)); memory.CopyToEmu(request.buffer_out, &out, sizeof(out));
return GetFSReply(IPC_SUCCESS, ticks); return GetFSReply(IPC_SUCCESS, ticks);
} }
@ -627,7 +644,10 @@ IPCReply FSDevice::DeleteFile(const Handle& handle, const IOCtlRequest& request)
if (request.buffer_in_size < 64) if (request.buffer_in_size < 64)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
const std::string path = Memory::GetString(request.buffer_in, 64); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string path = memory.GetString(request.buffer_in, 64);
return MakeIPCReply( return MakeIPCReply(
[&](Ticks ticks) { return ConvertResult(DeleteFile(handle.uid, handle.gid, path, ticks)); }); [&](Ticks ticks) { return ConvertResult(DeleteFile(handle.uid, handle.gid, path, ticks)); });
} }
@ -648,8 +668,11 @@ IPCReply FSDevice::RenameFile(const Handle& handle, const IOCtlRequest& request)
if (request.buffer_in_size < 64 * 2) if (request.buffer_in_size < 64 * 2)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
const std::string old_path = Memory::GetString(request.buffer_in, 64); auto& system = Core::System::GetInstance();
const std::string new_path = Memory::GetString(request.buffer_in + 64, 64); auto& memory = system.GetMemory();
const std::string old_path = memory.GetString(request.buffer_in, 64);
const std::string new_path = memory.GetString(request.buffer_in + 64, 64);
return MakeIPCReply([&](Ticks ticks) { return MakeIPCReply([&](Ticks ticks) {
return ConvertResult(RenameFile(handle.uid, handle.gid, old_path, new_path, ticks)); return ConvertResult(RenameFile(handle.uid, handle.gid, old_path, new_path, ticks));
}); });
@ -699,10 +722,13 @@ IPCReply FSDevice::GetFileStats(const Handle& handle, const IOCtlRequest& reques
if (!status) if (!status)
return ConvertResult(status.Error()); return ConvertResult(status.Error());
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
ISFSFileStats out; ISFSFileStats out;
out.size = status->size; out.size = status->size;
out.seek_position = status->offset; out.seek_position = status->offset;
Memory::CopyToEmu(request.buffer_out, &out, sizeof(out)); memory.CopyToEmu(request.buffer_out, &out, sizeof(out));
return IPC_SUCCESS; return IPC_SUCCESS;
}); });
} }
@ -727,14 +753,17 @@ IPCReply FSDevice::GetUsage(const Handle& handle, const IOCtlVRequest& request)
return GetFSReply(ConvertResult(ResultCode::Invalid)); return GetFSReply(ConvertResult(ResultCode::Invalid));
} }
const std::string directory = Memory::GetString(request.in_vectors[0].address, 64); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string directory = memory.GetString(request.in_vectors[0].address, 64);
const Result<DirectoryStats> stats = m_ios.GetFS()->GetDirectoryStats(directory); const Result<DirectoryStats> stats = m_ios.GetFS()->GetDirectoryStats(directory);
LogResult(stats, "GetDirectoryStats({})", directory); LogResult(stats, "GetDirectoryStats({})", directory);
if (!stats) if (!stats)
return GetFSReply(ConvertResult(stats.Error())); return GetFSReply(ConvertResult(stats.Error()));
Memory::Write_U32(stats->used_clusters, request.io_vectors[0].address); memory.Write_U32(stats->used_clusters, request.io_vectors[0].address);
Memory::Write_U32(stats->used_inodes, request.io_vectors[1].address); memory.Write_U32(stats->used_inodes, request.io_vectors[1].address);
return GetFSReply(IPC_SUCCESS); return GetFSReply(IPC_SUCCESS);
} }

View File

@ -104,6 +104,9 @@ constexpr u32 PLACEHOLDER = 0xDEADBEEF;
static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type) static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
auto target_imv = std::find_if( auto target_imv = std::find_if(
GetMemoryValues().begin(), GetMemoryValues().end(), GetMemoryValues().begin(), GetMemoryValues().end(),
[&](const MemoryValues& imv) { return imv.ios_number == (ios_title_id & 0xffff); }); [&](const MemoryValues& imv) { return imv.ios_number == (ios_title_id & 0xffff); });
@ -116,7 +119,7 @@ static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type)
if (setup_type == MemorySetupType::IOSReload) if (setup_type == MemorySetupType::IOSReload)
{ {
Memory::Write_U32(target_imv->ios_version, ADDR_IOS_VERSION); memory.Write_U32(target_imv->ios_version, ADDR_IOS_VERSION);
// These values are written by the IOS kernel as part of its boot process (for IOS28 and newer). // These values are written by the IOS kernel as part of its boot process (for IOS28 and newer).
// //
@ -127,15 +130,15 @@ static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type)
// the new IOS either updates the range (>= IOS28) or inherits it (< IOS28). // the new IOS either updates the range (>= IOS28) or inherits it (< IOS28).
// //
// We can skip this convoluted process and just write the correct range directly. // We can skip this convoluted process and just write the correct range directly.
Memory::Write_U32(target_imv->mem2_physical_size, ADDR_MEM2_SIZE); memory.Write_U32(target_imv->mem2_physical_size, ADDR_MEM2_SIZE);
Memory::Write_U32(target_imv->mem2_simulated_size, ADDR_MEM2_SIM_SIZE); memory.Write_U32(target_imv->mem2_simulated_size, ADDR_MEM2_SIM_SIZE);
Memory::Write_U32(target_imv->mem2_end, ADDR_MEM2_END); memory.Write_U32(target_imv->mem2_end, ADDR_MEM2_END);
Memory::Write_U32(target_imv->mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN); memory.Write_U32(target_imv->mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN);
Memory::Write_U32(target_imv->mem2_arena_end, ADDR_MEM2_ARENA_END); memory.Write_U32(target_imv->mem2_arena_end, ADDR_MEM2_ARENA_END);
Memory::Write_U32(target_imv->ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN); memory.Write_U32(target_imv->ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN);
Memory::Write_U32(target_imv->ipc_buffer_end, ADDR_IPC_BUFFER_END); memory.Write_U32(target_imv->ipc_buffer_end, ADDR_IPC_BUFFER_END);
Memory::Write_U32(target_imv->ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN); memory.Write_U32(target_imv->ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN);
Memory::Write_U32(target_imv->ios_reserved_end, ADDR_IOS_RESERVED_END); memory.Write_U32(target_imv->ios_reserved_end, ADDR_IOS_RESERVED_END);
RAMOverrideForIOSMemoryValues(setup_type); RAMOverrideForIOSMemoryValues(setup_type);
@ -146,40 +149,40 @@ static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type)
// and system information (see below). // and system information (see below).
constexpr u32 LOW_MEM1_REGION_START = 0; constexpr u32 LOW_MEM1_REGION_START = 0;
constexpr u32 LOW_MEM1_REGION_SIZE = 0x3fff; constexpr u32 LOW_MEM1_REGION_SIZE = 0x3fff;
Memory::Memset(LOW_MEM1_REGION_START, 0, LOW_MEM1_REGION_SIZE); memory.Memset(LOW_MEM1_REGION_START, 0, LOW_MEM1_REGION_SIZE);
Memory::Write_U32(target_imv->mem1_physical_size, ADDR_MEM1_SIZE); memory.Write_U32(target_imv->mem1_physical_size, ADDR_MEM1_SIZE);
Memory::Write_U32(target_imv->mem1_simulated_size, ADDR_MEM1_SIM_SIZE); memory.Write_U32(target_imv->mem1_simulated_size, ADDR_MEM1_SIM_SIZE);
Memory::Write_U32(target_imv->mem1_end, ADDR_MEM1_END); memory.Write_U32(target_imv->mem1_end, ADDR_MEM1_END);
Memory::Write_U32(target_imv->mem1_arena_begin, ADDR_MEM1_ARENA_BEGIN); memory.Write_U32(target_imv->mem1_arena_begin, ADDR_MEM1_ARENA_BEGIN);
Memory::Write_U32(target_imv->mem1_arena_end, ADDR_MEM1_ARENA_END); memory.Write_U32(target_imv->mem1_arena_end, ADDR_MEM1_ARENA_END);
Memory::Write_U32(PLACEHOLDER, ADDR_PH1); memory.Write_U32(PLACEHOLDER, ADDR_PH1);
Memory::Write_U32(target_imv->mem2_physical_size, ADDR_MEM2_SIZE); memory.Write_U32(target_imv->mem2_physical_size, ADDR_MEM2_SIZE);
Memory::Write_U32(target_imv->mem2_simulated_size, ADDR_MEM2_SIM_SIZE); memory.Write_U32(target_imv->mem2_simulated_size, ADDR_MEM2_SIM_SIZE);
Memory::Write_U32(target_imv->mem2_end, ADDR_MEM2_END); memory.Write_U32(target_imv->mem2_end, ADDR_MEM2_END);
Memory::Write_U32(target_imv->mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN); memory.Write_U32(target_imv->mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN);
Memory::Write_U32(target_imv->mem2_arena_end, ADDR_MEM2_ARENA_END); memory.Write_U32(target_imv->mem2_arena_end, ADDR_MEM2_ARENA_END);
Memory::Write_U32(PLACEHOLDER, ADDR_PH2); memory.Write_U32(PLACEHOLDER, ADDR_PH2);
Memory::Write_U32(target_imv->ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN); memory.Write_U32(target_imv->ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN);
Memory::Write_U32(target_imv->ipc_buffer_end, ADDR_IPC_BUFFER_END); memory.Write_U32(target_imv->ipc_buffer_end, ADDR_IPC_BUFFER_END);
Memory::Write_U32(target_imv->hollywood_revision, ADDR_HOLLYWOOD_REVISION); memory.Write_U32(target_imv->hollywood_revision, ADDR_HOLLYWOOD_REVISION);
Memory::Write_U32(PLACEHOLDER, ADDR_PH3); memory.Write_U32(PLACEHOLDER, ADDR_PH3);
Memory::Write_U32(target_imv->ios_version, ADDR_IOS_VERSION); memory.Write_U32(target_imv->ios_version, ADDR_IOS_VERSION);
Memory::Write_U32(target_imv->ios_date, ADDR_IOS_DATE); memory.Write_U32(target_imv->ios_date, ADDR_IOS_DATE);
Memory::Write_U32(target_imv->ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN); memory.Write_U32(target_imv->ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN);
Memory::Write_U32(target_imv->ios_reserved_end, ADDR_IOS_RESERVED_END); memory.Write_U32(target_imv->ios_reserved_end, ADDR_IOS_RESERVED_END);
Memory::Write_U32(PLACEHOLDER, ADDR_PH4); memory.Write_U32(PLACEHOLDER, ADDR_PH4);
Memory::Write_U32(PLACEHOLDER, ADDR_PH5); memory.Write_U32(PLACEHOLDER, ADDR_PH5);
Memory::Write_U32(target_imv->ram_vendor, ADDR_RAM_VENDOR); memory.Write_U32(target_imv->ram_vendor, ADDR_RAM_VENDOR);
Memory::Write_U8(0xDE, ADDR_BOOT_FLAG); memory.Write_U8(0xDE, ADDR_BOOT_FLAG);
Memory::Write_U8(0xAD, ADDR_APPLOADER_FLAG); memory.Write_U8(0xAD, ADDR_APPLOADER_FLAG);
Memory::Write_U16(0xBEEF, ADDR_DEVKIT_BOOT_PROGRAM_VERSION); memory.Write_U16(0xBEEF, ADDR_DEVKIT_BOOT_PROGRAM_VERSION);
Memory::Write_U32(target_imv->sysmenu_sync, ADDR_SYSMENU_SYNC); memory.Write_U32(target_imv->sysmenu_sync, ADDR_SYSMENU_SYNC);
Memory::Write_U32(target_imv->mem1_physical_size, ADDR_LEGACY_MEM_SIZE); memory.Write_U32(target_imv->mem1_physical_size, ADDR_LEGACY_MEM_SIZE);
Memory::Write_U32(target_imv->mem1_arena_begin, ADDR_LEGACY_ARENA_LOW); memory.Write_U32(target_imv->mem1_arena_begin, ADDR_LEGACY_ARENA_LOW);
Memory::Write_U32(target_imv->mem1_arena_end, ADDR_LEGACY_ARENA_HIGH); memory.Write_U32(target_imv->mem1_arena_end, ADDR_LEGACY_ARENA_HIGH);
Memory::Write_U32(target_imv->mem1_simulated_size, ADDR_LEGACY_MEM_SIM_SIZE); memory.Write_U32(target_imv->mem1_simulated_size, ADDR_LEGACY_MEM_SIM_SIZE);
RAMOverrideForIOSMemoryValues(setup_type); RAMOverrideForIOSMemoryValues(setup_type);
@ -193,14 +196,18 @@ static bool SetupMemory(u64 ios_title_id, MemorySetupType setup_type)
static void ResetAndPausePPC() static void ResetAndPausePPC()
{ {
// This should be cleared when the PPC is released so that the write is not observable. // This should be cleared when the PPC is released so that the write is not observable.
Memory::Write_U32(0x48000000, 0x00000000); // b 0x0 auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(0x48000000, 0x00000000); // b 0x0
PowerPC::Reset(); PowerPC::Reset();
PC = 0; PC = 0;
} }
static void ReleasePPC() static void ReleasePPC()
{ {
Memory::Write_U32(0, 0); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(0, 0);
// HLE the bootstub that jumps to 0x3400. // HLE the bootstub that jumps to 0x3400.
// NAND titles start with address translation off at 0x3400 (via the PPC bootstub) // NAND titles start with address translation off at 0x3400 (via the PPC bootstub)
// The state of other CPU registers (like the BAT registers) doesn't matter much // The state of other CPU registers (like the BAT registers) doesn't matter much
@ -210,7 +217,9 @@ static void ReleasePPC()
static void ReleasePPCAncast() static void ReleasePPCAncast()
{ {
Memory::Write_U32(0, 0); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(0, 0);
// On a real console the Espresso verifies and decrypts the Ancast image, // On a real console the Espresso verifies and decrypts the Ancast image,
// then jumps to the decrypted ancast body. // then jumps to the decrypted ancast body.
// The Ancast loader already did this, so just jump to the decrypted body. // The Ancast loader already did this, so just jump to the decrypted body.
@ -223,19 +232,22 @@ void RAMOverrideForIOSMemoryValues(MemorySetupType setup_type)
if (!Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE)) if (!Config::Get(Config::MAIN_RAM_OVERRIDE_ENABLE))
return; return;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Some unstated constants that can be inferred. // Some unstated constants that can be inferred.
const u32 ipc_buffer_size = const u32 ipc_buffer_size =
Memory::Read_U32(ADDR_IPC_BUFFER_END) - Memory::Read_U32(ADDR_IPC_BUFFER_BEGIN); memory.Read_U32(ADDR_IPC_BUFFER_END) - memory.Read_U32(ADDR_IPC_BUFFER_BEGIN);
const u32 ios_reserved_size = const u32 ios_reserved_size =
Memory::Read_U32(ADDR_IOS_RESERVED_END) - Memory::Read_U32(ADDR_IOS_RESERVED_BEGIN); memory.Read_U32(ADDR_IOS_RESERVED_END) - memory.Read_U32(ADDR_IOS_RESERVED_BEGIN);
const u32 mem1_physical_size = Memory::GetRamSizeReal(); const u32 mem1_physical_size = memory.GetRamSizeReal();
const u32 mem1_simulated_size = Memory::GetRamSizeReal(); const u32 mem1_simulated_size = memory.GetRamSizeReal();
const u32 mem1_end = Memory::MEM1_BASE_ADDR + mem1_simulated_size; const u32 mem1_end = Memory::MEM1_BASE_ADDR + mem1_simulated_size;
const u32 mem1_arena_begin = 0; const u32 mem1_arena_begin = 0;
const u32 mem1_arena_end = mem1_end; const u32 mem1_arena_end = mem1_end;
const u32 mem2_physical_size = Memory::GetExRamSizeReal(); const u32 mem2_physical_size = memory.GetExRamSizeReal();
const u32 mem2_simulated_size = Memory::GetExRamSizeReal(); const u32 mem2_simulated_size = memory.GetExRamSizeReal();
const u32 mem2_end = Memory::MEM2_BASE_ADDR + mem2_simulated_size - ios_reserved_size; const u32 mem2_end = Memory::MEM2_BASE_ADDR + mem2_simulated_size - ios_reserved_size;
const u32 mem2_arena_begin = Memory::MEM2_BASE_ADDR + 0x800U; const u32 mem2_arena_begin = Memory::MEM2_BASE_ADDR + 0x800U;
const u32 mem2_arena_end = mem2_end - ipc_buffer_size; const u32 mem2_arena_end = mem2_end - ipc_buffer_size;
@ -247,31 +259,33 @@ void RAMOverrideForIOSMemoryValues(MemorySetupType setup_type)
if (setup_type == MemorySetupType::Full) if (setup_type == MemorySetupType::Full)
{ {
// Overwriting these after the game's apploader sets them would be bad // Overwriting these after the game's apploader sets them would be bad
Memory::Write_U32(mem1_physical_size, ADDR_MEM1_SIZE); memory.Write_U32(mem1_physical_size, ADDR_MEM1_SIZE);
Memory::Write_U32(mem1_simulated_size, ADDR_MEM1_SIM_SIZE); memory.Write_U32(mem1_simulated_size, ADDR_MEM1_SIM_SIZE);
Memory::Write_U32(mem1_end, ADDR_MEM1_END); memory.Write_U32(mem1_end, ADDR_MEM1_END);
Memory::Write_U32(mem1_arena_begin, ADDR_MEM1_ARENA_BEGIN); memory.Write_U32(mem1_arena_begin, ADDR_MEM1_ARENA_BEGIN);
Memory::Write_U32(mem1_arena_end, ADDR_MEM1_ARENA_END); memory.Write_U32(mem1_arena_end, ADDR_MEM1_ARENA_END);
Memory::Write_U32(mem1_physical_size, ADDR_LEGACY_MEM_SIZE); memory.Write_U32(mem1_physical_size, ADDR_LEGACY_MEM_SIZE);
Memory::Write_U32(mem1_arena_begin, ADDR_LEGACY_ARENA_LOW); memory.Write_U32(mem1_arena_begin, ADDR_LEGACY_ARENA_LOW);
Memory::Write_U32(mem1_arena_end, ADDR_LEGACY_ARENA_HIGH); memory.Write_U32(mem1_arena_end, ADDR_LEGACY_ARENA_HIGH);
Memory::Write_U32(mem1_simulated_size, ADDR_LEGACY_MEM_SIM_SIZE); memory.Write_U32(mem1_simulated_size, ADDR_LEGACY_MEM_SIM_SIZE);
} }
Memory::Write_U32(mem2_physical_size, ADDR_MEM2_SIZE); memory.Write_U32(mem2_physical_size, ADDR_MEM2_SIZE);
Memory::Write_U32(mem2_simulated_size, ADDR_MEM2_SIM_SIZE); memory.Write_U32(mem2_simulated_size, ADDR_MEM2_SIM_SIZE);
Memory::Write_U32(mem2_end, ADDR_MEM2_END); memory.Write_U32(mem2_end, ADDR_MEM2_END);
Memory::Write_U32(mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN); memory.Write_U32(mem2_arena_begin, ADDR_MEM2_ARENA_BEGIN);
Memory::Write_U32(mem2_arena_end, ADDR_MEM2_ARENA_END); memory.Write_U32(mem2_arena_end, ADDR_MEM2_ARENA_END);
Memory::Write_U32(ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN); memory.Write_U32(ipc_buffer_begin, ADDR_IPC_BUFFER_BEGIN);
Memory::Write_U32(ipc_buffer_end, ADDR_IPC_BUFFER_END); memory.Write_U32(ipc_buffer_end, ADDR_IPC_BUFFER_END);
Memory::Write_U32(ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN); memory.Write_U32(ios_reserved_begin, ADDR_IOS_RESERVED_BEGIN);
Memory::Write_U32(ios_reserved_end, ADDR_IOS_RESERVED_END); memory.Write_U32(ios_reserved_end, ADDR_IOS_RESERVED_END);
} }
void WriteReturnValue(s32 value, u32 address) void WriteReturnValue(s32 value, u32 address)
{ {
Memory::Write_U32(static_cast<u32>(value), address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(static_cast<u32>(value), address);
} }
Kernel::Kernel(IOSC::ConsoleType console_type) : m_iosc(console_type) Kernel::Kernel(IOSC::ConsoleType console_type) : m_iosc(console_type)
@ -751,13 +765,14 @@ void Kernel::EnqueueIPCRequest(u32 address)
void Kernel::EnqueueIPCReply(const Request& request, const s32 return_value, s64 cycles_in_future, void Kernel::EnqueueIPCReply(const Request& request, const s32 return_value, s64 cycles_in_future,
CoreTiming::FromThread from) CoreTiming::FromThread from)
{ {
Memory::Write_U32(static_cast<u32>(return_value), request.address + 4); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(static_cast<u32>(return_value), request.address + 4);
// IOS writes back the command that was responded to in the FD field. // IOS writes back the command that was responded to in the FD field.
Memory::Write_U32(request.command, request.address + 8); memory.Write_U32(request.command, request.address + 8);
// IOS also overwrites the command type with the reply type. // IOS also overwrites the command type with the reply type.
Memory::Write_U32(IPC_REPLY, request.address); memory.Write_U32(IPC_REPLY, request.address);
Core::System::GetInstance().GetCoreTiming().ScheduleEvent(cycles_in_future, s_event_enqueue, system.GetCoreTiming().ScheduleEvent(cycles_in_future, s_event_enqueue, request.address, from);
request.address, from);
} }
void Kernel::HandleIPCEvent(u64 userdata) void Kernel::HandleIPCEvent(u64 userdata)

View File

@ -24,6 +24,7 @@
#include "Core/Host.h" #include "Core/Host.h"
#include "Core/PowerPC/PPCSymbolDB.h" #include "Core/PowerPC/PPCSymbolDB.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
namespace IOS::HLE::MIOS namespace IOS::HLE::MIOS
{ {
@ -32,7 +33,9 @@ static void ReinitHardware()
SConfig::GetInstance().bWii = false; SConfig::GetInstance().bWii = false;
// IOS clears mem2 and overwrites it with pseudo-random data (for security). // IOS clears mem2 and overwrites it with pseudo-random data (for security).
std::memset(Memory::m_pEXRAM, 0, Memory::GetExRamSizeReal()); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::memset(memory.GetEXRAM(), 0, memory.GetExRamSizeReal());
// MIOS appears to only reset the DI and the PPC. // MIOS appears to only reset the DI and the PPC.
// HACK However, resetting DI will reset the DTK config, which is set by the system menu // HACK However, resetting DI will reset the DTK config, which is set by the system menu
// (and not by MIOS), causing games that use DTK to break. Perhaps MIOS doesn't actually // (and not by MIOS), causing games that use DTK to break. Perhaps MIOS doesn't actually
@ -52,8 +55,10 @@ constexpr u32 ADDRESS_INIT_SEMAPHORE = 0x30f8;
bool Load() bool Load()
{ {
Memory::Write_U32(0x00000000, ADDRESS_INIT_SEMAPHORE); auto& system = Core::System::GetInstance();
Memory::Write_U32(0x09142001, 0x3180); auto& memory = system.GetMemory();
memory.Write_U32(0x00000000, ADDRESS_INIT_SEMAPHORE);
memory.Write_U32(0x09142001, 0x3180);
ReinitHardware(); ReinitHardware();
NOTICE_LOG_FMT(IOS, "Reinitialised hardware."); NOTICE_LOG_FMT(IOS, "Reinitialised hardware.");
@ -79,11 +84,11 @@ bool Load()
// IOS writes 0 to 0x30f8 before bootstrapping the PPC. Once started, the IPL eventually writes // IOS writes 0 to 0x30f8 before bootstrapping the PPC. Once started, the IPL eventually writes
// 0xdeadbeef there, then waits for it to be cleared by IOS before continuing. // 0xdeadbeef there, then waits for it to be cleared by IOS before continuing.
while (Memory::Read_U32(ADDRESS_INIT_SEMAPHORE) != 0xdeadbeef) while (memory.Read_U32(ADDRESS_INIT_SEMAPHORE) != 0xdeadbeef)
PowerPC::SingleStep(); PowerPC::SingleStep();
PowerPC::SetMode(core_mode); PowerPC::SetMode(core_mode);
Memory::Write_U32(0x00000000, ADDRESS_INIT_SEMAPHORE); memory.Write_U32(0x00000000, ADDRESS_INIT_SEMAPHORE);
NOTICE_LOG_FMT(IOS, "IPL ready."); NOTICE_LOG_FMT(IOS, "IPL ready.");
SConfig::GetInstance().m_is_mios = true; SConfig::GetInstance().m_is_mios = true;
DVDInterface::UpdateRunningGameMetadata(); DVDInterface::UpdateRunningGameMetadata();

View File

@ -31,6 +31,7 @@
#include "Core/IOS/Network/ICMP.h" #include "Core/IOS/Network/ICMP.h"
#include "Core/IOS/Network/MACUtils.h" #include "Core/IOS/Network/MACUtils.h"
#include "Core/IOS/Network/Socket.h" #include "Core/IOS/Network/Socket.h"
#include "Core/System.h"
#ifdef _WIN32 #ifdef _WIN32
#include <iphlpapi.h> #include <iphlpapi.h>
@ -381,9 +382,12 @@ IPCReply NetIPTopDevice::HandleInitInterfaceRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleSocketRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleSocketRequest(const IOCtlRequest& request)
{ {
const u32 af = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
const u32 type = Memory::Read_U32(request.buffer_in + 4); auto& memory = system.GetMemory();
const u32 prot = Memory::Read_U32(request.buffer_in + 8);
const u32 af = memory.Read_U32(request.buffer_in);
const u32 type = memory.Read_U32(request.buffer_in + 4);
const u32 prot = memory.Read_U32(request.buffer_in + 8);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
const s32 return_value = sm.NewSocket(af, type, prot); const s32 return_value = sm.NewSocket(af, type, prot);
@ -398,7 +402,10 @@ IPCReply NetIPTopDevice::HandleSocketRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleICMPSocketRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleICMPSocketRequest(const IOCtlRequest& request)
{ {
const u32 pf = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 pf = memory.Read_U32(request.buffer_in);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
const s32 return_value = sm.NewSocket(pf, SOCK_RAW, IPPROTO_ICMP); const s32 return_value = sm.NewSocket(pf, SOCK_RAW, IPPROTO_ICMP);
@ -408,7 +415,10 @@ IPCReply NetIPTopDevice::HandleICMPSocketRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleCloseRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleCloseRequest(const IOCtlRequest& request)
{ {
const u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 fd = memory.Read_U32(request.buffer_in);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
const s32 return_value = sm.DeleteSocket(fd); const s32 return_value = sm.DeleteSocket(fd);
const char* const close_fn = const char* const close_fn =
@ -421,7 +431,10 @@ IPCReply NetIPTopDevice::HandleCloseRequest(const IOCtlRequest& request)
std::optional<IPCReply> NetIPTopDevice::HandleDoSockRequest(const IOCtlRequest& request) std::optional<IPCReply> NetIPTopDevice::HandleDoSockRequest(const IOCtlRequest& request)
{ {
const u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 fd = memory.Read_U32(request.buffer_in);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
sm.DoSock(fd, request, static_cast<NET_IOCTL>(request.request)); sm.DoSock(fd, request, static_cast<NET_IOCTL>(request.request));
return std::nullopt; return std::nullopt;
@ -436,8 +449,11 @@ IPCReply NetIPTopDevice::HandleShutdownRequest(const IOCtlRequest& request)
return IPCReply(-SO_EINVAL); return IPCReply(-SO_EINVAL);
} }
const u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
const u32 how = Memory::Read_U32(request.buffer_in + 4); auto& memory = system.GetMemory();
const u32 fd = memory.Read_U32(request.buffer_in);
const u32 how = memory.Read_U32(request.buffer_in + 4);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
const s32 return_value = sm.ShutdownSocket(fd, how); const s32 return_value = sm.ShutdownSocket(fd, how);
@ -447,8 +463,11 @@ IPCReply NetIPTopDevice::HandleShutdownRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleListenRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleListenRequest(const IOCtlRequest& request)
{ {
u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
u32 BACKLOG = Memory::Read_U32(request.buffer_in + 0x04); auto& memory = system.GetMemory();
u32 fd = memory.Read_U32(request.buffer_in);
u32 BACKLOG = memory.Read_U32(request.buffer_in + 0x04);
u32 ret = listen(WiiSockMan::GetInstance().GetHostSocket(fd), BACKLOG); u32 ret = listen(WiiSockMan::GetInstance().GetHostSocket(fd), BACKLOG);
request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24); request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24);
@ -457,9 +476,12 @@ IPCReply NetIPTopDevice::HandleListenRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleGetSockOptRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleGetSockOptRequest(const IOCtlRequest& request)
{ {
u32 fd = Memory::Read_U32(request.buffer_out); auto& system = Core::System::GetInstance();
u32 level = Memory::Read_U32(request.buffer_out + 4); auto& memory = system.GetMemory();
u32 optname = Memory::Read_U32(request.buffer_out + 8);
u32 fd = memory.Read_U32(request.buffer_out);
u32 level = memory.Read_U32(request.buffer_out + 4);
u32 optname = memory.Read_U32(request.buffer_out + 8);
request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24); request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24);
@ -474,15 +496,15 @@ IPCReply NetIPTopDevice::HandleGetSockOptRequest(const IOCtlRequest& request)
(char*)&optval, (socklen_t*)&optlen); (char*)&optval, (socklen_t*)&optlen);
const s32 return_value = WiiSockMan::GetNetErrorCode(ret, "SO_GETSOCKOPT", false); const s32 return_value = WiiSockMan::GetNetErrorCode(ret, "SO_GETSOCKOPT", false);
Memory::Write_U32(optlen, request.buffer_out + 0xC); memory.Write_U32(optlen, request.buffer_out + 0xC);
Memory::CopyToEmu(request.buffer_out + 0x10, optval, optlen); memory.CopyToEmu(request.buffer_out + 0x10, optval, optlen);
if (optname == SO_ERROR) if (optname == SO_ERROR)
{ {
s32 last_error = WiiSockMan::GetInstance().GetLastNetError(); s32 last_error = WiiSockMan::GetInstance().GetLastNetError();
Memory::Write_U32(sizeof(s32), request.buffer_out + 0xC); memory.Write_U32(sizeof(s32), request.buffer_out + 0xC);
Memory::Write_U32(last_error, request.buffer_out + 0x10); memory.Write_U32(last_error, request.buffer_out + 0x10);
} }
return IPCReply(return_value); return IPCReply(return_value);
@ -490,13 +512,16 @@ IPCReply NetIPTopDevice::HandleGetSockOptRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleSetSockOptRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleSetSockOptRequest(const IOCtlRequest& request)
{ {
const u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
const u32 level = Memory::Read_U32(request.buffer_in + 4); auto& memory = system.GetMemory();
const u32 optname = Memory::Read_U32(request.buffer_in + 8);
u32 optlen = Memory::Read_U32(request.buffer_in + 0xc); const u32 fd = memory.Read_U32(request.buffer_in);
const u32 level = memory.Read_U32(request.buffer_in + 4);
const u32 optname = memory.Read_U32(request.buffer_in + 8);
u32 optlen = memory.Read_U32(request.buffer_in + 0xc);
u8 optval[20]; u8 optval[20];
optlen = std::min(optlen, (u32)sizeof(optval)); optlen = std::min(optlen, (u32)sizeof(optval));
Memory::CopyFromEmu(optval, request.buffer_in + 0x10, optlen); memory.CopyFromEmu(optval, request.buffer_in + 0x10, optlen);
INFO_LOG_FMT(IOS_NET, INFO_LOG_FMT(IOS_NET,
"IOCTL_SO_SETSOCKOPT({:08x}, {:08x}, {:08x}, {:08x}) " "IOCTL_SO_SETSOCKOPT({:08x}, {:08x}, {:08x}, {:08x}) "
@ -525,7 +550,10 @@ IPCReply NetIPTopDevice::HandleSetSockOptRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleGetSockNameRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleGetSockNameRequest(const IOCtlRequest& request)
{ {
u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 fd = memory.Read_U32(request.buffer_in);
request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24); request.Log(GetDeviceName(), Common::Log::LogType::IOS_WC24);
@ -537,12 +565,12 @@ IPCReply NetIPTopDevice::HandleGetSockNameRequest(const IOCtlRequest& request)
WARN_LOG_FMT(IOS_NET, "IOCTL_SO_GETSOCKNAME output buffer is too small. Truncating"); WARN_LOG_FMT(IOS_NET, "IOCTL_SO_GETSOCKNAME output buffer is too small. Truncating");
if (request.buffer_out_size > 0) if (request.buffer_out_size > 0)
Memory::Write_U8(request.buffer_out_size, request.buffer_out); memory.Write_U8(request.buffer_out_size, request.buffer_out);
if (request.buffer_out_size > 1) if (request.buffer_out_size > 1)
Memory::Write_U8(sa.sa_family & 0xFF, request.buffer_out + 1); memory.Write_U8(sa.sa_family & 0xFF, request.buffer_out + 1);
if (request.buffer_out_size > 2) if (request.buffer_out_size > 2)
{ {
Memory::CopyToEmu(request.buffer_out + 2, &sa.sa_data, memory.CopyToEmu(request.buffer_out + 2, &sa.sa_data,
std::min<size_t>(sizeof(sa.sa_data), request.buffer_out_size - 2)); std::min<size_t>(sizeof(sa.sa_data), request.buffer_out_size - 2));
} }
@ -551,7 +579,10 @@ IPCReply NetIPTopDevice::HandleGetSockNameRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleGetPeerNameRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleGetPeerNameRequest(const IOCtlRequest& request)
{ {
u32 fd = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 fd = memory.Read_U32(request.buffer_in);
sockaddr sa; sockaddr sa;
socklen_t sa_len = sizeof(sa); socklen_t sa_len = sizeof(sa);
@ -561,12 +592,12 @@ IPCReply NetIPTopDevice::HandleGetPeerNameRequest(const IOCtlRequest& request)
WARN_LOG_FMT(IOS_NET, "IOCTL_SO_GETPEERNAME output buffer is too small. Truncating"); WARN_LOG_FMT(IOS_NET, "IOCTL_SO_GETPEERNAME output buffer is too small. Truncating");
if (request.buffer_out_size > 0) if (request.buffer_out_size > 0)
Memory::Write_U8(request.buffer_out_size, request.buffer_out); memory.Write_U8(request.buffer_out_size, request.buffer_out);
if (request.buffer_out_size > 1) if (request.buffer_out_size > 1)
Memory::Write_U8(AF_INET, request.buffer_out + 1); memory.Write_U8(AF_INET, request.buffer_out + 1);
if (request.buffer_out_size > 2) if (request.buffer_out_size > 2)
{ {
Memory::CopyToEmu(request.buffer_out + 2, &sa.sa_data, memory.CopyToEmu(request.buffer_out + 2, &sa.sa_data,
std::min<size_t>(sizeof(sa.sa_data), request.buffer_out_size - 2)); std::min<size_t>(sizeof(sa.sa_data), request.buffer_out_size - 2));
} }
@ -585,7 +616,10 @@ IPCReply NetIPTopDevice::HandleGetHostIDRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleInetAToNRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleInetAToNRequest(const IOCtlRequest& request)
{ {
const std::string hostname = Memory::GetString(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string hostname = memory.GetString(request.buffer_in);
struct hostent* remoteHost = gethostbyname(hostname.c_str()); struct hostent* remoteHost = gethostbyname(hostname.c_str());
if (remoteHost == nullptr || remoteHost->h_addr_list == nullptr || if (remoteHost == nullptr || remoteHost->h_addr_list == nullptr ||
@ -601,7 +635,7 @@ IPCReply NetIPTopDevice::HandleInetAToNRequest(const IOCtlRequest& request)
} }
const auto ip = Common::swap32(reinterpret_cast<u8*>(remoteHost->h_addr_list[0])); const auto ip = Common::swap32(reinterpret_cast<u8*>(remoteHost->h_addr_list[0]));
Memory::Write_U32(ip, request.buffer_out); memory.Write_U32(ip, request.buffer_out);
INFO_LOG_FMT(IOS_NET, INFO_LOG_FMT(IOS_NET,
"IOCTL_SO_INETATON = 0 " "IOCTL_SO_INETATON = 0 "
@ -614,24 +648,30 @@ IPCReply NetIPTopDevice::HandleInetAToNRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleInetPToNRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleInetPToNRequest(const IOCtlRequest& request)
{ {
const std::string address = Memory::GetString(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string address = memory.GetString(request.buffer_in);
INFO_LOG_FMT(IOS_NET, "IOCTL_SO_INETPTON (Translating: {})", address); INFO_LOG_FMT(IOS_NET, "IOCTL_SO_INETPTON (Translating: {})", address);
return IPCReply(inet_pton(address.c_str(), Memory::GetPointer(request.buffer_out + 4))); return IPCReply(inet_pton(address.c_str(), memory.GetPointer(request.buffer_out + 4)));
} }
IPCReply NetIPTopDevice::HandleInetNToPRequest(const IOCtlRequest& request) IPCReply NetIPTopDevice::HandleInetNToPRequest(const IOCtlRequest& request)
{ {
// u32 af = Memory::Read_U32(BufferIn); auto& system = Core::System::GetInstance();
// u32 validAddress = Memory::Read_U32(request.buffer_in + 4); auto& memory = system.GetMemory();
// u32 src = Memory::Read_U32(request.buffer_in + 8);
// u32 af = memory.Read_U32(BufferIn);
// u32 validAddress = memory.Read_U32(request.buffer_in + 4);
// u32 src = memory.Read_U32(request.buffer_in + 8);
char ip_s[16]; char ip_s[16];
sprintf(ip_s, "%i.%i.%i.%i", Memory::Read_U8(request.buffer_in + 8), sprintf(ip_s, "%i.%i.%i.%i", memory.Read_U8(request.buffer_in + 8),
Memory::Read_U8(request.buffer_in + 8 + 1), Memory::Read_U8(request.buffer_in + 8 + 2), memory.Read_U8(request.buffer_in + 8 + 1), memory.Read_U8(request.buffer_in + 8 + 2),
Memory::Read_U8(request.buffer_in + 8 + 3)); memory.Read_U8(request.buffer_in + 8 + 3));
INFO_LOG_FMT(IOS_NET, "IOCTL_SO_INETNTOP {}", ip_s); INFO_LOG_FMT(IOS_NET, "IOCTL_SO_INETNTOP {}", ip_s);
Memory::CopyToEmu(request.buffer_out, reinterpret_cast<u8*>(ip_s), std::strlen(ip_s)); memory.CopyToEmu(request.buffer_out, reinterpret_cast<u8*>(ip_s), std::strlen(ip_s));
return IPCReply(0); return IPCReply(0);
} }
@ -642,8 +682,11 @@ std::optional<IPCReply> NetIPTopDevice::HandlePollRequest(const IOCtlRequest& re
if (!request.buffer_in || !request.buffer_out) if (!request.buffer_in || !request.buffer_out)
return IPCReply(-SO_EINVAL); return IPCReply(-SO_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Negative timeout indicates wait forever // Negative timeout indicates wait forever
const s64 timeout = static_cast<s64>(Memory::Read_U64(request.buffer_in)); const s64 timeout = static_cast<s64>(memory.Read_U64(request.buffer_in));
const u32 nfds = request.buffer_out_size / 0xc; const u32 nfds = request.buffer_out_size / 0xc;
if (nfds == 0 || nfds > WII_SOCKET_FD_MAX) if (nfds == 0 || nfds > WII_SOCKET_FD_MAX)
@ -656,9 +699,9 @@ std::optional<IPCReply> NetIPTopDevice::HandlePollRequest(const IOCtlRequest& re
for (u32 i = 0; i < nfds; ++i) for (u32 i = 0; i < nfds; ++i)
{ {
const s32 wii_fd = Memory::Read_U32(request.buffer_out + 0xc * i); const s32 wii_fd = memory.Read_U32(request.buffer_out + 0xc * i);
ufds[i].fd = sm.GetHostSocket(wii_fd); // fd ufds[i].fd = sm.GetHostSocket(wii_fd); // fd
const int events = Memory::Read_U32(request.buffer_out + 0xc * i + 4); // events const int events = memory.Read_U32(request.buffer_out + 0xc * i + 4); // events
ufds[i].revents = 0; ufds[i].revents = 0;
// Translate Wii to native events // Translate Wii to native events
@ -686,7 +729,10 @@ IPCReply NetIPTopDevice::HandleGetHostByNameRequest(const IOCtlRequest& request)
return IPCReply(-1); return IPCReply(-1);
} }
const std::string hostname = Memory::GetString(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const std::string hostname = memory.GetString(request.buffer_in);
hostent* remoteHost = gethostbyname(hostname.c_str()); hostent* remoteHost = gethostbyname(hostname.c_str());
INFO_LOG_FMT(IOS_NET, INFO_LOG_FMT(IOS_NET,
@ -721,9 +767,8 @@ IPCReply NetIPTopDevice::HandleGetHostByNameRequest(const IOCtlRequest& request)
ERROR_LOG_FMT(IOS_NET, "Hostname too long in IOCTL_SO_GETHOSTBYNAME"); ERROR_LOG_FMT(IOS_NET, "Hostname too long in IOCTL_SO_GETHOSTBYNAME");
return IPCReply(-1); return IPCReply(-1);
} }
Memory::CopyToEmu(request.buffer_out + GETHOSTBYNAME_STRUCT_SIZE, remoteHost->h_name, memory.CopyToEmu(request.buffer_out + GETHOSTBYNAME_STRUCT_SIZE, remoteHost->h_name, name_length);
name_length); memory.Write_U32(request.buffer_out + GETHOSTBYNAME_STRUCT_SIZE, request.buffer_out);
Memory::Write_U32(request.buffer_out + GETHOSTBYNAME_STRUCT_SIZE, request.buffer_out);
// IP address list; located at offset 0x110. // IP address list; located at offset 0x110.
u32 num_ip_addr = 0; u32 num_ip_addr = 0;
@ -736,30 +781,30 @@ IPCReply NetIPTopDevice::HandleGetHostByNameRequest(const IOCtlRequest& request)
for (u32 i = 0; i < num_ip_addr; ++i) for (u32 i = 0; i < num_ip_addr; ++i)
{ {
u32 addr = request.buffer_out + GETHOSTBYNAME_IP_LIST_OFFSET + i * 4; u32 addr = request.buffer_out + GETHOSTBYNAME_IP_LIST_OFFSET + i * 4;
Memory::Write_U32_Swap(*(u32*)(remoteHost->h_addr_list[i]), addr); memory.Write_U32_Swap(*(u32*)(remoteHost->h_addr_list[i]), addr);
} }
// List of pointers to IP addresses; located at offset 0x340. // List of pointers to IP addresses; located at offset 0x340.
// This must be exact: PPC code to convert the struct hardcodes // This must be exact: PPC code to convert the struct hardcodes
// this offset. // this offset.
static const u32 GETHOSTBYNAME_IP_PTR_LIST_OFFSET = 0x340; static const u32 GETHOSTBYNAME_IP_PTR_LIST_OFFSET = 0x340;
Memory::Write_U32(request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET, request.buffer_out + 12); memory.Write_U32(request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET, request.buffer_out + 12);
for (u32 i = 0; i < num_ip_addr; ++i) for (u32 i = 0; i < num_ip_addr; ++i)
{ {
u32 addr = request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + i * 4; u32 addr = request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + i * 4;
Memory::Write_U32(request.buffer_out + GETHOSTBYNAME_IP_LIST_OFFSET + i * 4, addr); memory.Write_U32(request.buffer_out + GETHOSTBYNAME_IP_LIST_OFFSET + i * 4, addr);
} }
Memory::Write_U32(0, request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + num_ip_addr * 4); memory.Write_U32(0, request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + num_ip_addr * 4);
// Aliases - empty. (Hardware doesn't return anything.) // Aliases - empty. (Hardware doesn't return anything.)
Memory::Write_U32(request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + num_ip_addr * 4, memory.Write_U32(request.buffer_out + GETHOSTBYNAME_IP_PTR_LIST_OFFSET + num_ip_addr * 4,
request.buffer_out + 4); request.buffer_out + 4);
// Returned struct must be ipv4. // Returned struct must be ipv4.
ASSERT_MSG(IOS_NET, remoteHost->h_addrtype == AF_INET && remoteHost->h_length == sizeof(u32), ASSERT_MSG(IOS_NET, remoteHost->h_addrtype == AF_INET && remoteHost->h_length == sizeof(u32),
"returned host info is not IPv4"); "returned host info is not IPv4");
Memory::Write_U16(AF_INET, request.buffer_out + 8); memory.Write_U16(AF_INET, request.buffer_out + 8);
Memory::Write_U16(sizeof(u32), request.buffer_out + 10); memory.Write_U16(sizeof(u32), request.buffer_out + 10);
return IPCReply(0); return IPCReply(0);
} }
@ -772,10 +817,13 @@ IPCReply NetIPTopDevice::HandleICMPCancelRequest(const IOCtlRequest& request)
IPCReply NetIPTopDevice::HandleGetInterfaceOptRequest(const IOCtlVRequest& request) IPCReply NetIPTopDevice::HandleGetInterfaceOptRequest(const IOCtlVRequest& request)
{ {
const u32 param = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 param2 = Memory::Read_U32(request.in_vectors[0].address + 4); auto& memory = system.GetMemory();
const u32 param3 = Memory::Read_U32(request.io_vectors[0].address);
const u32 param4 = Memory::Read_U32(request.io_vectors[1].address); const u32 param = memory.Read_U32(request.in_vectors[0].address);
const u32 param2 = memory.Read_U32(request.in_vectors[0].address + 4);
const u32 param3 = memory.Read_U32(request.io_vectors[0].address);
const u32 param4 = memory.Read_U32(request.io_vectors[1].address);
u32 param5 = 0; u32 param5 = 0;
if (param != 0xfffe) if (param != 0xfffe)
@ -786,7 +834,7 @@ IPCReply NetIPTopDevice::HandleGetInterfaceOptRequest(const IOCtlVRequest& reque
if (request.io_vectors[0].size >= 8) if (request.io_vectors[0].size >= 8)
{ {
param5 = Memory::Read_U32(request.io_vectors[0].address + 4); param5 = memory.Read_U32(request.io_vectors[0].address + 4);
} }
INFO_LOG_FMT(IOS_NET, INFO_LOG_FMT(IOS_NET,
@ -893,63 +941,63 @@ IPCReply NetIPTopDevice::HandleGetInterfaceOptRequest(const IOCtlVRequest& reque
INFO_LOG_FMT(IOS_NET, "Primary DNS: {:X}", address); INFO_LOG_FMT(IOS_NET, "Primary DNS: {:X}", address);
INFO_LOG_FMT(IOS_NET, "Secondary DNS: {:X}", default_backup_dns_resolver); INFO_LOG_FMT(IOS_NET, "Secondary DNS: {:X}", default_backup_dns_resolver);
Memory::Write_U32(address, request.io_vectors[0].address); memory.Write_U32(address, request.io_vectors[0].address);
Memory::Write_U32(default_backup_dns_resolver, request.io_vectors[0].address + 4); memory.Write_U32(default_backup_dns_resolver, request.io_vectors[0].address + 4);
break; break;
} }
case 0x1003: // error case 0x1003: // error
Memory::Write_U32(0, request.io_vectors[0].address); memory.Write_U32(0, request.io_vectors[0].address);
break; break;
case 0x1004: // mac address case 0x1004: // mac address
{ {
const Common::MACAddress address = IOS::Net::GetMACAddress(); const Common::MACAddress address = IOS::Net::GetMACAddress();
Memory::CopyToEmu(request.io_vectors[0].address, address.data(), address.size()); memory.CopyToEmu(request.io_vectors[0].address, address.data(), address.size());
break; break;
} }
case 0x1005: // link state case 0x1005: // link state
Memory::Write_U32(1, request.io_vectors[0].address); memory.Write_U32(1, request.io_vectors[0].address);
break; break;
case 0x3001: // hardcoded value case 0x3001: // hardcoded value
Memory::Write_U32(0x10, request.io_vectors[0].address); memory.Write_U32(0x10, request.io_vectors[0].address);
break; break;
case 0x4002: // ip addr numberHandle case 0x4002: // ip addr numberHandle
Memory::Write_U32(1, request.io_vectors[0].address); memory.Write_U32(1, request.io_vectors[0].address);
break; break;
case 0x4003: // ip addr table case 0x4003: // ip addr table
{ {
// XXX: this isn't exactly right; the buffer can be larger than 12 bytes, in which case // XXX: this isn't exactly right; the buffer can be larger than 12 bytes, in which case
// SO can write 12 more bytes. // SO can write 12 more bytes.
Memory::Write_U32(0xC, request.io_vectors[1].address); memory.Write_U32(0xC, request.io_vectors[1].address);
const DefaultInterface interface = GetSystemDefaultInterfaceOrFallback(); const DefaultInterface interface = GetSystemDefaultInterfaceOrFallback();
Memory::Write_U32(Common::swap32(interface.inet), request.io_vectors[0].address); memory.Write_U32(Common::swap32(interface.inet), request.io_vectors[0].address);
Memory::Write_U32(Common::swap32(interface.netmask), request.io_vectors[0].address + 4); memory.Write_U32(Common::swap32(interface.netmask), request.io_vectors[0].address + 4);
Memory::Write_U32(Common::swap32(interface.broadcast), request.io_vectors[0].address + 8); memory.Write_U32(Common::swap32(interface.broadcast), request.io_vectors[0].address + 8);
break; break;
} }
case 0x4005: // hardcoded value case 0x4005: // hardcoded value
Memory::Write_U32(0x20, request.io_vectors[0].address); memory.Write_U32(0x20, request.io_vectors[0].address);
break; break;
case 0x6003: // hardcoded value case 0x6003: // hardcoded value
Memory::Write_U32(0x80, request.io_vectors[0].address); memory.Write_U32(0x80, request.io_vectors[0].address);
break; break;
case 0x600a: // hardcoded value case 0x600a: // hardcoded value
Memory::Write_U32(0x80, request.io_vectors[0].address); memory.Write_U32(0x80, request.io_vectors[0].address);
break; break;
case 0x600c: // hardcoded value case 0x600c: // hardcoded value
Memory::Write_U32(0x80, request.io_vectors[0].address); memory.Write_U32(0x80, request.io_vectors[0].address);
break; break;
case 0xb002: // hardcoded value case 0xb002: // hardcoded value
Memory::Write_U32(2, request.io_vectors[0].address); memory.Write_U32(2, request.io_vectors[0].address);
break; break;
default: default:
@ -962,7 +1010,10 @@ IPCReply NetIPTopDevice::HandleGetInterfaceOptRequest(const IOCtlVRequest& reque
std::optional<IPCReply> NetIPTopDevice::HandleSendToRequest(const IOCtlVRequest& request) std::optional<IPCReply> NetIPTopDevice::HandleSendToRequest(const IOCtlVRequest& request)
{ {
u32 fd = Memory::Read_U32(request.in_vectors[1].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 fd = memory.Read_U32(request.in_vectors[1].address);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
sm.DoSock(fd, request, IOCTLV_SO_SENDTO); sm.DoSock(fd, request, IOCTLV_SO_SENDTO);
return std::nullopt; return std::nullopt;
@ -970,7 +1021,10 @@ std::optional<IPCReply> NetIPTopDevice::HandleSendToRequest(const IOCtlVRequest&
std::optional<IPCReply> NetIPTopDevice::HandleRecvFromRequest(const IOCtlVRequest& request) std::optional<IPCReply> NetIPTopDevice::HandleRecvFromRequest(const IOCtlVRequest& request)
{ {
u32 fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 fd = memory.Read_U32(request.in_vectors[0].address);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
sm.DoSock(fd, request, IOCTLV_SO_RECVFROM); sm.DoSock(fd, request, IOCTLV_SO_RECVFROM);
return std::nullopt; return std::nullopt;
@ -978,16 +1032,19 @@ std::optional<IPCReply> NetIPTopDevice::HandleRecvFromRequest(const IOCtlVReques
IPCReply NetIPTopDevice::HandleGetAddressInfoRequest(const IOCtlVRequest& request) IPCReply NetIPTopDevice::HandleGetAddressInfoRequest(const IOCtlVRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
addrinfo hints; addrinfo hints;
const bool hints_valid = request.in_vectors.size() > 2 && request.in_vectors[2].size; const bool hints_valid = request.in_vectors.size() > 2 && request.in_vectors[2].size;
if (hints_valid) if (hints_valid)
{ {
hints.ai_flags = Memory::Read_U32(request.in_vectors[2].address); hints.ai_flags = memory.Read_U32(request.in_vectors[2].address);
hints.ai_family = Memory::Read_U32(request.in_vectors[2].address + 0x4); hints.ai_family = memory.Read_U32(request.in_vectors[2].address + 0x4);
hints.ai_socktype = Memory::Read_U32(request.in_vectors[2].address + 0x8); hints.ai_socktype = memory.Read_U32(request.in_vectors[2].address + 0x8);
hints.ai_protocol = Memory::Read_U32(request.in_vectors[2].address + 0xC); hints.ai_protocol = memory.Read_U32(request.in_vectors[2].address + 0xC);
hints.ai_addrlen = Memory::Read_U32(request.in_vectors[2].address + 0x10); hints.ai_addrlen = memory.Read_U32(request.in_vectors[2].address + 0x10);
hints.ai_canonname = nullptr; hints.ai_canonname = nullptr;
hints.ai_addr = nullptr; hints.ai_addr = nullptr;
hints.ai_next = nullptr; hints.ai_next = nullptr;
@ -999,7 +1056,7 @@ IPCReply NetIPTopDevice::HandleGetAddressInfoRequest(const IOCtlVRequest& reques
const char* pNodeName = nullptr; const char* pNodeName = nullptr;
if (!request.in_vectors.empty() && request.in_vectors[0].size > 0) if (!request.in_vectors.empty() && request.in_vectors[0].size > 0)
{ {
nodeNameStr = Memory::GetString(request.in_vectors[0].address, request.in_vectors[0].size); nodeNameStr = memory.GetString(request.in_vectors[0].address, request.in_vectors[0].size);
pNodeName = nodeNameStr.c_str(); pNodeName = nodeNameStr.c_str();
} }
@ -1007,7 +1064,7 @@ IPCReply NetIPTopDevice::HandleGetAddressInfoRequest(const IOCtlVRequest& reques
const char* pServiceName = nullptr; const char* pServiceName = nullptr;
if (request.in_vectors.size() > 1 && request.in_vectors[1].size > 0) if (request.in_vectors.size() > 1 && request.in_vectors[1].size > 0)
{ {
serviceNameStr = Memory::GetString(request.in_vectors[1].address, request.in_vectors[1].size); serviceNameStr = memory.GetString(request.in_vectors[1].address, request.in_vectors[1].size);
pServiceName = serviceNameStr.c_str(); pServiceName = serviceNameStr.c_str();
} }
@ -1020,35 +1077,35 @@ IPCReply NetIPTopDevice::HandleGetAddressInfoRequest(const IOCtlVRequest& reques
constexpr size_t WII_ADDR_INFO_SIZE = 0x20; constexpr size_t WII_ADDR_INFO_SIZE = 0x20;
for (addrinfo* result_iter = result; result_iter != nullptr; result_iter = result_iter->ai_next) for (addrinfo* result_iter = result; result_iter != nullptr; result_iter = result_iter->ai_next)
{ {
Memory::Write_U32(result_iter->ai_flags, addr); memory.Write_U32(result_iter->ai_flags, addr);
Memory::Write_U32(result_iter->ai_family, addr + 0x04); memory.Write_U32(result_iter->ai_family, addr + 0x04);
Memory::Write_U32(result_iter->ai_socktype, addr + 0x08); memory.Write_U32(result_iter->ai_socktype, addr + 0x08);
Memory::Write_U32(result_iter->ai_protocol, addr + 0x0C); memory.Write_U32(result_iter->ai_protocol, addr + 0x0C);
Memory::Write_U32((u32)result_iter->ai_addrlen, addr + 0x10); memory.Write_U32((u32)result_iter->ai_addrlen, addr + 0x10);
// what to do? where to put? the buffer of 0x834 doesn't allow space for this // what to do? where to put? the buffer of 0x834 doesn't allow space for this
Memory::Write_U32(/*result->ai_cannonname*/ 0, addr + 0x14); memory.Write_U32(/*result->ai_cannonname*/ 0, addr + 0x14);
if (result_iter->ai_addr) if (result_iter->ai_addr)
{ {
Memory::Write_U32(sockoffset, addr + 0x18); memory.Write_U32(sockoffset, addr + 0x18);
Memory::Write_U8(result_iter->ai_addrlen & 0xFF, sockoffset); memory.Write_U8(result_iter->ai_addrlen & 0xFF, sockoffset);
Memory::Write_U8(result_iter->ai_addr->sa_family & 0xFF, sockoffset + 0x01); memory.Write_U8(result_iter->ai_addr->sa_family & 0xFF, sockoffset + 0x01);
Memory::CopyToEmu(sockoffset + 0x2, result_iter->ai_addr->sa_data, memory.CopyToEmu(sockoffset + 0x2, result_iter->ai_addr->sa_data,
sizeof(result_iter->ai_addr->sa_data)); sizeof(result_iter->ai_addr->sa_data));
sockoffset += 0x1C; sockoffset += 0x1C;
} }
else else
{ {
Memory::Write_U32(0, addr + 0x18); memory.Write_U32(0, addr + 0x18);
} }
if (result_iter->ai_next) if (result_iter->ai_next)
{ {
Memory::Write_U32(addr + WII_ADDR_INFO_SIZE, addr + 0x1C); memory.Write_U32(addr + WII_ADDR_INFO_SIZE, addr + 0x1C);
} }
else else
{ {
Memory::Write_U32(0, addr + 0x1C); memory.Write_U32(0, addr + 0x1C);
} }
addr += WII_ADDR_INFO_SIZE; addr += WII_ADDR_INFO_SIZE;
@ -1075,19 +1132,22 @@ IPCReply NetIPTopDevice::HandleICMPPingRequest(const IOCtlVRequest& request)
u32 ip; u32 ip;
} ip_info; } ip_info;
const u32 fd = Memory::Read_U32(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u32 num_ip = Memory::Read_U32(request.in_vectors[0].address + 4); auto& memory = system.GetMemory();
const u64 timeout = Memory::Read_U64(request.in_vectors[0].address + 8);
const u32 fd = memory.Read_U32(request.in_vectors[0].address);
const u32 num_ip = memory.Read_U32(request.in_vectors[0].address + 4);
const u64 timeout = memory.Read_U64(request.in_vectors[0].address + 8);
if (num_ip != 1) if (num_ip != 1)
{ {
INFO_LOG_FMT(IOS_NET, "IOCTLV_SO_ICMPPING {} IPs", num_ip); INFO_LOG_FMT(IOS_NET, "IOCTLV_SO_ICMPPING {} IPs", num_ip);
} }
ip_info.length = Memory::Read_U8(request.in_vectors[0].address + 16); ip_info.length = memory.Read_U8(request.in_vectors[0].address + 16);
ip_info.addr_family = Memory::Read_U8(request.in_vectors[0].address + 17); ip_info.addr_family = memory.Read_U8(request.in_vectors[0].address + 17);
ip_info.icmp_id = Memory::Read_U16(request.in_vectors[0].address + 18); ip_info.icmp_id = memory.Read_U16(request.in_vectors[0].address + 18);
ip_info.ip = Memory::Read_U32(request.in_vectors[0].address + 20); ip_info.ip = memory.Read_U32(request.in_vectors[0].address + 20);
if (ip_info.length != 8 || ip_info.addr_family != AF_INET) if (ip_info.length != 8 || ip_info.addr_family != AF_INET)
{ {
@ -1110,7 +1170,7 @@ IPCReply NetIPTopDevice::HandleICMPPingRequest(const IOCtlVRequest& request)
if (request.in_vectors.size() > 1 && request.in_vectors[1].size == sizeof(data)) if (request.in_vectors.size() > 1 && request.in_vectors[1].size == sizeof(data))
{ {
Memory::CopyFromEmu(data, request.in_vectors[1].address, request.in_vectors[1].size); memory.CopyFromEmu(data, request.in_vectors[1].address, request.in_vectors[1].size);
} }
else else
{ {

View File

@ -23,6 +23,7 @@
#include "Core/IOS/Network/KD/VFF/VFFUtil.h" #include "Core/IOS/Network/KD/VFF/VFFUtil.h"
#include "Core/IOS/Network/Socket.h" #include "Core/IOS/Network/Socket.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -232,12 +233,14 @@ NWC24::ErrorCode NetKDRequestDevice::KDDownload(const u16 entry_index,
IPCReply NetKDRequestDevice::HandleNWC24DownloadNowEx(const IOCtlRequest& request) IPCReply NetKDRequestDevice::HandleNWC24DownloadNowEx(const IOCtlRequest& request)
{ {
const u32 flags = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 flags = memory.Read_U32(request.buffer_in);
// Nintendo converts the entry ID between a u32 and u16 // Nintendo converts the entry ID between a u32 and u16
// several times, presumably for alignment purposes. // several times, presumably for alignment purposes.
// We'll skip past buffer_in+4 and keep the entry index as a u16. // We'll skip past buffer_in+4 and keep the entry index as a u16.
const u16 entry_index = Memory::Read_U16(request.buffer_in + 6); const u16 entry_index = memory.Read_U16(request.buffer_in + 6);
const u32 subtask_bitmask = Memory::Read_U32(request.buffer_in + 8); const u32 subtask_bitmask = memory.Read_U32(request.buffer_in + 8);
INFO_LOG_FMT(IOS_WC24, INFO_LOG_FMT(IOS_WC24,
"NET_KD_REQ: IOCTL_NWC24_DOWNLOAD_NOW_EX - NI - flags: {}, index: {}, bitmask: {}", "NET_KD_REQ: IOCTL_NWC24_DOWNLOAD_NOW_EX - NI - flags: {}, index: {}, bitmask: {}",
@ -315,6 +318,8 @@ std::optional<IPCReply> NetKDRequestDevice::IOCtl(const IOCtlRequest& request)
IOCTL_NWC24_REQUEST_SHUTDOWN = 0x28, IOCTL_NWC24_REQUEST_SHUTDOWN = 0x28,
}; };
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
s32 return_value = 0; s32 return_value = 0;
switch (request.request) switch (request.request)
{ {
@ -335,7 +340,7 @@ std::optional<IPCReply> NetKDRequestDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_NWC24_STARTUP_SOCKET: // NWC24iStartupSocket case IOCTL_NWC24_STARTUP_SOCKET: // NWC24iStartupSocket
WriteReturnValue(0, request.buffer_out); WriteReturnValue(0, request.buffer_out);
Memory::Write_U32(0, request.buffer_out + 4); memory.Write_U32(0, request.buffer_out + 4);
return_value = 0; return_value = 0;
INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_STARTUP_SOCKET - NI"); INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_STARTUP_SOCKET - NI");
break; break;
@ -356,7 +361,7 @@ std::optional<IPCReply> NetKDRequestDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_NWC24_REQUEST_REGISTER_USER_ID: case IOCTL_NWC24_REQUEST_REGISTER_USER_ID:
INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_REGISTER_USER_ID"); INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_REGISTER_USER_ID");
WriteReturnValue(0, request.buffer_out); WriteReturnValue(0, request.buffer_out);
Memory::Write_U32(0, request.buffer_out + 4); memory.Write_U32(0, request.buffer_out + 4);
break; break;
case IOCTL_NWC24_REQUEST_GENERATED_USER_ID: // (Input: none, Output: 32 bytes) case IOCTL_NWC24_REQUEST_GENERATED_USER_ID: // (Input: none, Output: 32 bytes)
@ -409,8 +414,8 @@ std::optional<IPCReply> NetKDRequestDevice::IOCtl(const IOCtlRequest& request)
{ {
WriteReturnValue(NWC24::WC24_ERR_ID_REGISTERED, request.buffer_out); WriteReturnValue(NWC24::WC24_ERR_ID_REGISTERED, request.buffer_out);
} }
Memory::Write_U64(config.Id(), request.buffer_out + 4); memory.Write_U64(config.Id(), request.buffer_out + 4);
Memory::Write_U32(u32(config.CreationStage()), request.buffer_out + 0xC); memory.Write_U32(u32(config.CreationStage()), request.buffer_out + 0xC);
break; break;
case IOCTL_NWC24_GET_SCHEDULER_STAT: case IOCTL_NWC24_GET_SCHEDULER_STAT:
@ -435,7 +440,7 @@ std::optional<IPCReply> NetKDRequestDevice::IOCtl(const IOCtlRequest& request)
} }
INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_SHUTDOWN"); INFO_LOG_FMT(IOS_WC24, "NET_KD_REQ: IOCTL_NWC24_REQUEST_SHUTDOWN");
[[maybe_unused]] const u32 event = Memory::Read_U32(request.buffer_in); [[maybe_unused]] const u32 event = memory.Read_U32(request.buffer_in);
// TODO: Advertise shutdown event // TODO: Advertise shutdown event
// TODO: Shutdown USB keyboard LEDs if event == 3 // TODO: Shutdown USB keyboard LEDs if event == 3
// TODO: IOCTLV_NCD_SETCONFIG // TODO: IOCTLV_NCD_SETCONFIG

View File

@ -8,6 +8,7 @@
#include "Common/CommonTypes.h" #include "Common/CommonTypes.h"
#include "Core/HW/EXI/EXI_DeviceIPL.h" #include "Core/HW/EXI/EXI_DeviceIPL.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -34,36 +35,39 @@ std::optional<IPCReply> NetKDTimeDevice::IOCtl(const IOCtlRequest& request)
// TODO Writes stuff to /shared2/nwc24/misc.bin // TODO Writes stuff to /shared2/nwc24/misc.bin
u32 update_misc = 0; u32 update_misc = 0;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (request.request) switch (request.request)
{ {
case IOCTL_NW24_GET_UNIVERSAL_TIME: case IOCTL_NW24_GET_UNIVERSAL_TIME:
{ {
const u64 adjusted_utc = GetAdjustedUTC(); const u64 adjusted_utc = GetAdjustedUTC();
Memory::Write_U64(adjusted_utc, request.buffer_out + 4); memory.Write_U64(adjusted_utc, request.buffer_out + 4);
INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_GET_UNIVERSAL_TIME = {}, time = {}", result, adjusted_utc); INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_GET_UNIVERSAL_TIME = {}, time = {}", result, adjusted_utc);
} }
break; break;
case IOCTL_NW24_SET_UNIVERSAL_TIME: case IOCTL_NW24_SET_UNIVERSAL_TIME:
{ {
const u64 adjusted_utc = Memory::Read_U64(request.buffer_in); const u64 adjusted_utc = memory.Read_U64(request.buffer_in);
SetAdjustedUTC(adjusted_utc); SetAdjustedUTC(adjusted_utc);
update_misc = Memory::Read_U32(request.buffer_in + 8); update_misc = memory.Read_U32(request.buffer_in + 8);
INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_SET_UNIVERSAL_TIME ({}, {}) = {}", adjusted_utc, update_misc, INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_SET_UNIVERSAL_TIME ({}, {}) = {}", adjusted_utc, update_misc,
result); result);
} }
break; break;
case IOCTL_NW24_SET_RTC_COUNTER: case IOCTL_NW24_SET_RTC_COUNTER:
rtc = Memory::Read_U32(request.buffer_in); rtc = memory.Read_U32(request.buffer_in);
update_misc = Memory::Read_U32(request.buffer_in + 4); update_misc = memory.Read_U32(request.buffer_in + 4);
INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_SET_RTC_COUNTER ({}, {}) = {}", rtc, update_misc, result); INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_SET_RTC_COUNTER ({}, {}) = {}", rtc, update_misc, result);
break; break;
case IOCTL_NW24_GET_TIME_DIFF: case IOCTL_NW24_GET_TIME_DIFF:
{ {
const u64 time_diff = GetAdjustedUTC() - rtc; const u64 time_diff = GetAdjustedUTC() - rtc;
Memory::Write_U64(time_diff, request.buffer_out + 4); memory.Write_U64(time_diff, request.buffer_out + 4);
INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_GET_TIME_DIFF = {}, time_diff = {}", result, time_diff); INFO_LOG_FMT(IOS_WC24, "IOCTL_NW24_GET_TIME_DIFF = {}, time_diff = {}", result, time_diff);
} }
break; break;
@ -79,7 +83,7 @@ std::optional<IPCReply> NetKDTimeDevice::IOCtl(const IOCtlRequest& request)
} }
// write return values // write return values
Memory::Write_U32(common_result, request.buffer_out); memory.Write_U32(common_result, request.buffer_out);
return IPCReply(result); return IPCReply(result);
} }

View File

@ -11,6 +11,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Network/MACUtils.h" #include "Core/IOS/Network/MACUtils.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -32,6 +33,9 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
u32 common_result = 0; u32 common_result = 0;
u32 common_vector = 0; u32 common_vector = 0;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (request.request) switch (request.request)
{ {
case IOCTLV_NCD_LOCKWIRELESSDRIVER: case IOCTLV_NCD_LOCKWIRELESSDRIVER:
@ -52,7 +56,7 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
// We will just write the value of the file descriptor. // We will just write the value of the file descriptor.
// The value will be positive so this will work fine. // The value will be positive so this will work fine.
m_ipc_fd = request.fd; m_ipc_fd = request.fd;
Memory::Write_U32(request.fd, request.io_vectors[0].address + 4); memory.Write_U32(request.fd, request.io_vectors[0].address + 4);
} }
break; break;
@ -67,7 +71,7 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
if (request.io_vectors[0].size < sizeof(u32)) if (request.io_vectors[0].size < sizeof(u32))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const u32 request_handle = Memory::Read_U32(request.in_vectors[0].address); const u32 request_handle = memory.Read_U32(request.in_vectors[0].address);
if (m_ipc_fd == request_handle) if (m_ipc_fd == request_handle)
{ {
m_ipc_fd = 0; m_ipc_fd = 0;
@ -107,7 +111,7 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
case IOCTLV_NCD_GETLINKSTATUS: case IOCTLV_NCD_GETLINKSTATUS:
INFO_LOG_FMT(IOS_NET, "NET_NCD_MANAGE: IOCTLV_NCD_GETLINKSTATUS"); INFO_LOG_FMT(IOS_NET, "NET_NCD_MANAGE: IOCTLV_NCD_GETLINKSTATUS");
// Always connected // Always connected
Memory::Write_U32(Net::ConnectionSettings::LINK_WIRED, request.io_vectors.at(0).address + 4); memory.Write_U32(Net::ConnectionSettings::LINK_WIRED, request.io_vectors.at(0).address + 4);
break; break;
case IOCTLV_NCD_GETWIRELESSMACADDRESS: case IOCTLV_NCD_GETWIRELESSMACADDRESS:
@ -115,7 +119,7 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
INFO_LOG_FMT(IOS_NET, "NET_NCD_MANAGE: IOCTLV_NCD_GETWIRELESSMACADDRESS"); INFO_LOG_FMT(IOS_NET, "NET_NCD_MANAGE: IOCTLV_NCD_GETWIRELESSMACADDRESS");
const Common::MACAddress address = IOS::Net::GetMACAddress(); const Common::MACAddress address = IOS::Net::GetMACAddress();
Memory::CopyToEmu(request.io_vectors.at(1).address, address.data(), address.size()); memory.CopyToEmu(request.io_vectors.at(1).address, address.data(), address.size());
break; break;
} }
@ -124,10 +128,10 @@ std::optional<IPCReply> NetNCDManageDevice::IOCtlV(const IOCtlVRequest& request)
break; break;
} }
Memory::Write_U32(common_result, request.io_vectors.at(common_vector).address); memory.Write_U32(common_result, request.io_vectors.at(common_vector).address);
if (common_vector == 1) if (common_vector == 1)
{ {
Memory::Write_U32(common_result, request.io_vectors.at(common_vector).address + 4); memory.Write_U32(common_result, request.io_vectors.at(common_vector).address + 4);
} }
return IPCReply(return_value); return IPCReply(return_value);
} }

View File

@ -12,6 +12,7 @@
#include "Core/IOS/FS/FileSystem.h" #include "Core/IOS/FS/FileSystem.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/IOS/Uids.h" #include "Core/IOS/Uids.h"
#include "Core/System.h"
namespace IOS::HLE::Net namespace IOS::HLE::Net
{ {
@ -53,11 +54,15 @@ void WiiNetConfig::ResetConfig(FS::FileSystem* fs)
void WiiNetConfig::WriteToMem(const u32 address) const void WiiNetConfig::WriteToMem(const u32 address) const
{ {
Memory::CopyToEmu(address, &m_data, sizeof(m_data)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(address, &m_data, sizeof(m_data));
} }
void WiiNetConfig::ReadFromMem(const u32 address) void WiiNetConfig::ReadFromMem(const u32 address)
{ {
Memory::CopyFromEmu(&m_data, address, sizeof(m_data)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(&m_data, address, sizeof(m_data));
} }
} // namespace IOS::HLE::Net } // namespace IOS::HLE::Net

View File

@ -20,6 +20,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Network/Socket.h" #include "Core/IOS/Network/Socket.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -225,12 +226,15 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
if (Core::WantsDeterminism()) if (Core::WantsDeterminism())
return IPCReply(IPC_EACCES); return IPCReply(IPC_EACCES);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (request.request) switch (request.request)
{ {
case IOCTLV_NET_SSL_NEW: case IOCTLV_NET_SSL_NEW:
{ {
int verifyOption = Memory::Read_U32(BufferOut); int verifyOption = memory.Read_U32(BufferOut);
std::string hostname = Memory::GetString(BufferOut2, BufferOutSize2); std::string hostname = memory.GetString(BufferOut2, BufferOutSize2);
int freeSSL = GetSSLFreeID(); int freeSSL = GetSSLFreeID();
if (freeSSL) if (freeSSL)
@ -290,7 +294,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_SHUTDOWN: case IOCTLV_NET_SSL_SHUTDOWN:
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
@ -335,17 +339,17 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3, BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3,
BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3); BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3);
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
int ret = int ret =
mbedtls_x509_crt_parse_der(&ssl->cacert, Memory::GetPointer(BufferOut2), BufferOutSize2); mbedtls_x509_crt_parse_der(&ssl->cacert, memory.GetPointer(BufferOut2), BufferOutSize2);
if (Config::Get(Config::MAIN_NETWORK_SSL_DUMP_ROOT_CA)) if (Config::Get(Config::MAIN_NETWORK_SSL_DUMP_ROOT_CA))
{ {
std::string filename = File::GetUserPath(D_DUMPSSL_IDX) + ssl->hostname + "_rootca.der"; std::string filename = File::GetUserPath(D_DUMPSSL_IDX) + ssl->hostname + "_rootca.der";
File::IOFile(filename, "wb").WriteBytes(Memory::GetPointer(BufferOut2), BufferOutSize2); File::IOFile(filename, "wb").WriteBytes(memory.GetPointer(BufferOut2), BufferOutSize2);
} }
if (ret) if (ret)
@ -376,7 +380,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3, BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3,
BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3); BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3);
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
@ -423,7 +427,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3, BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3,
BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3); BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3);
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
@ -442,7 +446,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_SETBUILTINROOTCA: case IOCTLV_NET_SSL_SETBUILTINROOTCA:
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
@ -480,12 +484,12 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_CONNECT: case IOCTLV_NET_SSL_CONNECT:
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WII_SSL* ssl = &_SSL[sslID]; WII_SSL* ssl = &_SSL[sslID];
mbedtls_ssl_setup(&ssl->ctx, &ssl->config); mbedtls_ssl_setup(&ssl->ctx, &ssl->config);
ssl->sockfd = Memory::Read_U32(BufferOut2); ssl->sockfd = memory.Read_U32(BufferOut2);
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
ssl->hostfd = sm.GetHostSocket(ssl->sockfd); ssl->hostfd = sm.GetHostSocket(ssl->sockfd);
INFO_LOG_FMT(IOS_SSL, "IOCTLV_NET_SSL_CONNECT socket = {}", ssl->sockfd); INFO_LOG_FMT(IOS_SSL, "IOCTLV_NET_SSL_CONNECT socket = {}", ssl->sockfd);
@ -507,7 +511,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_DOHANDSHAKE: case IOCTLV_NET_SSL_DOHANDSHAKE:
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
@ -522,7 +526,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_WRITE: case IOCTLV_NET_SSL_WRITE:
{ {
const int sslID = Memory::Read_U32(BufferOut) - 1; const int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
@ -540,13 +544,13 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
"BufferOut2: ({:08x}, {}), BufferOut3: ({:08x}, {})", "BufferOut2: ({:08x}, {}), BufferOut3: ({:08x}, {})",
BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3, BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3,
BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3); BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3);
INFO_LOG_FMT(IOS_SSL, "{}", Memory::GetString(BufferOut2)); INFO_LOG_FMT(IOS_SSL, "{}", memory.GetString(BufferOut2));
break; break;
} }
case IOCTLV_NET_SSL_READ: case IOCTLV_NET_SSL_READ:
{ {
int ret = 0; int ret = 0;
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WiiSockMan& sm = WiiSockMan::GetInstance(); WiiSockMan& sm = WiiSockMan::GetInstance();
@ -569,7 +573,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
} }
case IOCTLV_NET_SSL_SETROOTCADEFAULT: case IOCTLV_NET_SSL_SETROOTCADEFAULT:
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WriteReturnValue(SSL_OK, BufferIn); WriteReturnValue(SSL_OK, BufferIn);
@ -597,7 +601,7 @@ std::optional<IPCReply> NetSSLDevice::IOCtlV(const IOCtlVRequest& request)
BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3, BufferIn, BufferInSize, BufferIn2, BufferInSize2, BufferIn3, BufferInSize3,
BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3); BufferOut, BufferOutSize, BufferOut2, BufferOutSize2, BufferOut3, BufferOutSize3);
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
WriteReturnValue(SSL_OK, BufferIn); WriteReturnValue(SSL_OK, BufferIn);

View File

@ -26,6 +26,7 @@
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
#ifdef _WIN32 #ifdef _WIN32
#define ERRORCODE(name) WSA##name #define ERRORCODE(name) WSA##name
@ -244,6 +245,9 @@ s32 WiiSocket::FCntl(u32 cmd, u32 arg)
void WiiSocket::Update(bool read, bool write, bool except) void WiiSocket::Update(bool read, bool write, bool except)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
auto it = pending_sockops.begin(); auto it = pending_sockops.begin();
while (it != pending_sockops.end()) while (it != pending_sockops.end())
{ {
@ -257,15 +261,15 @@ void WiiSocket::Update(bool read, bool write, bool except)
{ {
case IOCTL_SO_FCNTL: case IOCTL_SO_FCNTL:
{ {
u32 cmd = Memory::Read_U32(ioctl.buffer_in + 4); u32 cmd = memory.Read_U32(ioctl.buffer_in + 4);
u32 arg = Memory::Read_U32(ioctl.buffer_in + 8); u32 arg = memory.Read_U32(ioctl.buffer_in + 8);
ReturnValue = FCntl(cmd, arg); ReturnValue = FCntl(cmd, arg);
break; break;
} }
case IOCTL_SO_BIND: case IOCTL_SO_BIND:
{ {
sockaddr_in local_name; sockaddr_in local_name;
const u8* addr = Memory::GetPointer(ioctl.buffer_in + 8); const u8* addr = memory.GetPointer(ioctl.buffer_in + 8);
WiiSockMan::ToNativeAddrIn(addr, &local_name); WiiSockMan::ToNativeAddrIn(addr, &local_name);
int ret = bind(fd, (sockaddr*)&local_name, sizeof(local_name)); int ret = bind(fd, (sockaddr*)&local_name, sizeof(local_name));
@ -278,7 +282,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
case IOCTL_SO_CONNECT: case IOCTL_SO_CONNECT:
{ {
sockaddr_in local_name; sockaddr_in local_name;
const u8* addr = Memory::GetPointer(ioctl.buffer_in + 8); const u8* addr = memory.GetPointer(ioctl.buffer_in + 8);
WiiSockMan::ToNativeAddrIn(addr, &local_name); WiiSockMan::ToNativeAddrIn(addr, &local_name);
int ret = connect(fd, (sockaddr*)&local_name, sizeof(local_name)); int ret = connect(fd, (sockaddr*)&local_name, sizeof(local_name));
@ -295,7 +299,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
if (ioctl.buffer_out_size > 0) if (ioctl.buffer_out_size > 0)
{ {
sockaddr_in local_name; sockaddr_in local_name;
u8* addr = Memory::GetPointer(ioctl.buffer_out); u8* addr = memory.GetPointer(ioctl.buffer_out);
WiiSockMan::ToNativeAddrIn(addr, &local_name); WiiSockMan::ToNativeAddrIn(addr, &local_name);
socklen_t addrlen = sizeof(sockaddr_in); socklen_t addrlen = sizeof(sockaddr_in);
@ -375,7 +379,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
if (it->is_ssl) if (it->is_ssl)
{ {
int sslID = Memory::Read_U32(BufferOut) - 1; int sslID = memory.Read_U32(BufferOut) - 1;
if (IsSSLIDValid(sslID)) if (IsSSLIDValid(sslID))
{ {
switch (it->ssl_type) switch (it->ssl_type)
@ -479,11 +483,11 @@ void WiiSocket::Update(bool read, bool write, bool except)
{ {
WII_SSL* ssl = &NetSSLDevice::_SSL[sslID]; WII_SSL* ssl = &NetSSLDevice::_SSL[sslID];
const int ret = const int ret =
mbedtls_ssl_write(&ssl->ctx, Memory::GetPointer(BufferOut2), BufferOutSize2); mbedtls_ssl_write(&ssl->ctx, memory.GetPointer(BufferOut2), BufferOutSize2);
if (ret >= 0) if (ret >= 0)
{ {
PowerPC::debug_interface.NetworkLogger()->LogSSLWrite(Memory::GetPointer(BufferOut2), PowerPC::debug_interface.NetworkLogger()->LogSSLWrite(memory.GetPointer(BufferOut2),
ret, ssl->hostfd); ret, ssl->hostfd);
// Return bytes written or SSL_ERR_ZERO if none // Return bytes written or SSL_ERR_ZERO if none
WriteReturnValue((ret == 0) ? SSL_ERR_ZERO : ret, BufferIn); WriteReturnValue((ret == 0) ? SSL_ERR_ZERO : ret, BufferIn);
@ -513,11 +517,11 @@ void WiiSocket::Update(bool read, bool write, bool except)
{ {
WII_SSL* ssl = &NetSSLDevice::_SSL[sslID]; WII_SSL* ssl = &NetSSLDevice::_SSL[sslID];
const int ret = const int ret =
mbedtls_ssl_read(&ssl->ctx, Memory::GetPointer(BufferIn2), BufferInSize2); mbedtls_ssl_read(&ssl->ctx, memory.GetPointer(BufferIn2), BufferInSize2);
if (ret >= 0) if (ret >= 0)
{ {
PowerPC::debug_interface.NetworkLogger()->LogSSLRead(Memory::GetPointer(BufferIn2), PowerPC::debug_interface.NetworkLogger()->LogSSLRead(memory.GetPointer(BufferIn2),
ret, ssl->hostfd); ret, ssl->hostfd);
// Return bytes read or SSL_ERR_ZERO if none // Return bytes read or SSL_ERR_ZERO if none
WriteReturnValue((ret == 0) ? SSL_ERR_ZERO : ret, BufferIn); WriteReturnValue((ret == 0) ? SSL_ERR_ZERO : ret, BufferIn);
@ -568,11 +572,11 @@ void WiiSocket::Update(bool read, bool write, bool except)
break; break;
} }
u32 flags = Memory::Read_U32(BufferIn2 + 0x04); u32 flags = memory.Read_U32(BufferIn2 + 0x04);
u32 has_destaddr = Memory::Read_U32(BufferIn2 + 0x08); u32 has_destaddr = memory.Read_U32(BufferIn2 + 0x08);
// Not a string, Windows requires a const char* for sendto // Not a string, Windows requires a const char* for sendto
const char* data = (const char*)Memory::GetPointer(BufferIn); const char* data = (const char*)memory.GetPointer(BufferIn);
// Act as non blocking when SO_MSG_NONBLOCK is specified // Act as non blocking when SO_MSG_NONBLOCK is specified
forceNonBlock = ((flags & SO_MSG_NONBLOCK) == SO_MSG_NONBLOCK); forceNonBlock = ((flags & SO_MSG_NONBLOCK) == SO_MSG_NONBLOCK);
@ -582,7 +586,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
sockaddr_in local_name = {0}; sockaddr_in local_name = {0};
if (has_destaddr) if (has_destaddr)
{ {
const u8* addr = Memory::GetPointer(BufferIn2 + 0x0C); const u8* addr = memory.GetPointer(BufferIn2 + 0x0C);
WiiSockMan::ToNativeAddrIn(addr, &local_name); WiiSockMan::ToNativeAddrIn(addr, &local_name);
} }
@ -615,9 +619,9 @@ void WiiSocket::Update(bool read, bool write, bool except)
break; break;
} }
u32 flags = Memory::Read_U32(BufferIn + 0x04); u32 flags = memory.Read_U32(BufferIn + 0x04);
// Not a string, Windows requires a char* for recvfrom // Not a string, Windows requires a char* for recvfrom
char* data = (char*)Memory::GetPointer(BufferOut); char* data = (char*)memory.GetPointer(BufferOut);
int data_len = BufferOutSize; int data_len = BufferOutSize;
sockaddr_in local_name; sockaddr_in local_name;
@ -625,7 +629,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
if (BufferOutSize2 != 0) if (BufferOutSize2 != 0)
{ {
const u8* addr = Memory::GetPointer(BufferOut2); const u8* addr = memory.GetPointer(BufferOut2);
WiiSockMan::ToNativeAddrIn(addr, &local_name); WiiSockMan::ToNativeAddrIn(addr, &local_name);
} }
@ -662,7 +666,7 @@ void WiiSocket::Update(bool read, bool write, bool except)
if (BufferOutSize2 != 0) if (BufferOutSize2 != 0)
{ {
u8* addr = Memory::GetPointer(BufferOut2); u8* addr = memory.GetPointer(BufferOut2);
WiiSockMan::ToWiiAddrIn(local_name, addr, addrlen); WiiSockMan::ToWiiAddrIn(local_name, addr, addrlen);
} }
break; break;
@ -1008,10 +1012,13 @@ void WiiSockMan::UpdatePollCommands()
pcmd.timeout = std::max<s64>(0, pcmd.timeout - elapsed); pcmd.timeout = std::max<s64>(0, pcmd.timeout - elapsed);
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
pending_polls.erase( pending_polls.erase(
std::remove_if( std::remove_if(
pending_polls.begin(), pending_polls.end(), pending_polls.begin(), pending_polls.end(),
[this](PollCommand& pcmd) { [&memory, this](PollCommand& pcmd) {
const auto request = Request(pcmd.request_addr); const auto request = Request(pcmd.request_addr);
auto& pfds = pcmd.wii_fds; auto& pfds = pcmd.wii_fds;
int ret = 0; int ret = 0;
@ -1030,7 +1037,7 @@ void WiiSockMan::UpdatePollCommands()
std::iota(original_order.begin(), original_order.end(), 0); std::iota(original_order.begin(), original_order.end(), 0);
// Select indices with valid fds // Select indices with valid fds
auto mid = std::partition(original_order.begin(), original_order.end(), [&](auto i) { auto mid = std::partition(original_order.begin(), original_order.end(), [&](auto i) {
return GetHostSocket(Memory::Read_U32(pcmd.buffer_out + 0xc * i)) >= 0; return GetHostSocket(memory.Read_U32(pcmd.buffer_out + 0xc * i)) >= 0;
}); });
const auto n_valid = std::distance(original_order.begin(), mid); const auto n_valid = std::distance(original_order.begin(), mid);
@ -1057,9 +1064,9 @@ void WiiSockMan::UpdatePollCommands()
const int revents = ConvertEvents(pfds[i].revents, ConvertDirection::NativeToWii); const int revents = ConvertEvents(pfds[i].revents, ConvertDirection::NativeToWii);
// No need to change fd or events as they are input only. // No need to change fd or events as they are input only.
// Memory::Write_U32(ufds[i].fd, request.buffer_out + 0xc*i); //fd // memory.Write_U32(ufds[i].fd, request.buffer_out + 0xc*i); //fd
// Memory::Write_U32(events, request.buffer_out + 0xc*i + 4); //events // memory.Write_U32(events, request.buffer_out + 0xc*i + 4); //events
Memory::Write_U32(revents, pcmd.buffer_out + 0xc * i + 8); // revents memory.Write_U32(revents, pcmd.buffer_out + 0xc * i + 8); // revents
DEBUG_LOG_FMT(IOS_NET, DEBUG_LOG_FMT(IOS_NET,
"IOCTL_SO_POLL socket {} wevents {:08X} events {:08X} revents {:08X}", "IOCTL_SO_POLL socket {} wevents {:08X} events {:08X} revents {:08X}",
i, revents, pfds[i].events, pfds[i].revents); i, revents, pfds[i].events, pfds[i].revents);

View File

@ -14,6 +14,7 @@
#include "Core/DolphinAnalytics.h" #include "Core/DolphinAnalytics.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Network/MACUtils.h" #include "Core/IOS/Network/MACUtils.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -233,7 +234,9 @@ IPCReply NetWDCommandDevice::SetLinkState(const IOCtlVRequest& request)
if (!vector || vector->address == 0) if (!vector || vector->address == 0)
return IPCReply(u32(ResultCode::IllegalParameter)); return IPCReply(u32(ResultCode::IllegalParameter));
const u32 state = Memory::Read_U32(vector->address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 state = memory.Read_U32(vector->address);
INFO_LOG_FMT(IOS_NET, "WD_SetLinkState called (state={}, mode={})", state, m_mode); INFO_LOG_FMT(IOS_NET, "WD_SetLinkState called (state={}, mode={})", state, m_mode);
if (state == 0) if (state == 0)
@ -279,8 +282,11 @@ IPCReply NetWDCommandDevice::Disassociate(const IOCtlVRequest& request)
if (!vector || vector->address == 0) if (!vector || vector->address == 0)
return IPCReply(u32(ResultCode::IllegalParameter)); return IPCReply(u32(ResultCode::IllegalParameter));
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
Common::MACAddress mac; Common::MACAddress mac;
Memory::CopyFromEmu(mac.data(), vector->address, mac.size()); memory.CopyFromEmu(mac.data(), vector->address, mac.size());
INFO_LOG_FMT(IOS_NET, "WD_Disassociate: MAC {}", Common::MacAddressToString(mac)); INFO_LOG_FMT(IOS_NET, "WD_Disassociate: MAC {}", Common::MacAddressToString(mac));
@ -309,7 +315,9 @@ IPCReply NetWDCommandDevice::GetInfo(const IOCtlVRequest& request) const
if (!vector || vector->address == 0) if (!vector || vector->address == 0)
return IPCReply(u32(ResultCode::IllegalParameter)); return IPCReply(u32(ResultCode::IllegalParameter));
Memory::CopyToEmu(vector->address, &m_info, sizeof(m_info)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(vector->address, &m_info, sizeof(m_info));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -332,9 +340,11 @@ std::optional<IPCReply> NetWDCommandDevice::IOCtlV(const IOCtlVRequest& request)
{ {
// Gives parameters detailing type of scan and what to match // Gives parameters detailing type of scan and what to match
// XXX - unused // XXX - unused
// ScanInfo *scan = (ScanInfo *)Memory::GetPointer(request.in_vectors.at(0).m_Address); // ScanInfo *scan = (ScanInfo *)memory.GetPointer(request.in_vectors.at(0).m_Address);
u16* results = (u16*)Memory::GetPointer(request.io_vectors.at(0).address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u16* results = (u16*)memory.GetPointer(request.io_vectors.at(0).address);
// first u16 indicates number of BSSInfo following // first u16 indicates number of BSSInfo following
results[0] = Common::swap16(1); results[0] = Common::swap16(1);

View File

@ -21,6 +21,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/IOS/VersionInfo.h" #include "Core/IOS/VersionInfo.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -128,7 +129,9 @@ std::optional<IPCReply> SDIOSlot0Device::Close(u32 fd)
std::optional<IPCReply> SDIOSlot0Device::IOCtl(const IOCtlRequest& request) std::optional<IPCReply> SDIOSlot0Device::IOCtl(const IOCtlRequest& request)
{ {
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Memset(request.buffer_out, 0, request.buffer_out_size);
switch (request.request) switch (request.request)
{ {
@ -187,15 +190,18 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
u32 pad0; u32 pad0;
} req; } req;
req.command = Memory::Read_U32(buffer_in + 0); auto& system = Core::System::GetInstance();
req.type = Memory::Read_U32(buffer_in + 4); auto& memory = system.GetMemory();
req.resp = Memory::Read_U32(buffer_in + 8);
req.arg = Memory::Read_U32(buffer_in + 12); req.command = memory.Read_U32(buffer_in + 0);
req.blocks = Memory::Read_U32(buffer_in + 16); req.type = memory.Read_U32(buffer_in + 4);
req.bsize = Memory::Read_U32(buffer_in + 20); req.resp = memory.Read_U32(buffer_in + 8);
req.addr = Memory::Read_U32(buffer_in + 24); req.arg = memory.Read_U32(buffer_in + 12);
req.isDMA = Memory::Read_U32(buffer_in + 28); req.blocks = memory.Read_U32(buffer_in + 16);
req.pad0 = Memory::Read_U32(buffer_in + 32); req.bsize = memory.Read_U32(buffer_in + 20);
req.addr = memory.Read_U32(buffer_in + 24);
req.isDMA = memory.Read_U32(buffer_in + 28);
req.pad0 = memory.Read_U32(buffer_in + 32);
// Note: req.addr is the virtual address of _rwBuffer // Note: req.addr is the virtual address of _rwBuffer
@ -206,19 +212,19 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
case GO_IDLE_STATE: case GO_IDLE_STATE:
INFO_LOG_FMT(IOS_SD, "GO_IDLE_STATE"); INFO_LOG_FMT(IOS_SD, "GO_IDLE_STATE");
// Response is R1 (idle state) // Response is R1 (idle state)
Memory::Write_U32(0x00, buffer_out); memory.Write_U32(0x00, buffer_out);
break; break;
case SEND_RELATIVE_ADDR: case SEND_RELATIVE_ADDR:
// Technically RCA should be generated when asked and at power on...w/e :p // Technically RCA should be generated when asked and at power on...w/e :p
Memory::Write_U32(0x9f62, buffer_out); memory.Write_U32(0x9f62, buffer_out);
break; break;
case SELECT_CARD: case SELECT_CARD:
// This covers both select and deselect // This covers both select and deselect
// Differentiate by checking if rca is set in req.arg // Differentiate by checking if rca is set in req.arg
// If it is, it's a select and return 0x700 // If it is, it's a select and return 0x700
Memory::Write_U32((req.arg >> 16) ? 0x700 : 0x900, buffer_out); memory.Write_U32((req.arg >> 16) ? 0x700 : 0x900, buffer_out);
break; break;
case SEND_IF_COND: case SEND_IF_COND:
@ -227,45 +233,45 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
// voltage and the check pattern that were set in the command argument. // voltage and the check pattern that were set in the command argument.
// This command is used to differentiate between protocol v1 and v2. // This command is used to differentiate between protocol v1 and v2.
InitSDHC(); InitSDHC();
Memory::Write_U32(req.arg, buffer_out); memory.Write_U32(req.arg, buffer_out);
break; break;
case SEND_CSD: case SEND_CSD:
{ {
const std::array<u32, 4> csd = m_protocol == SDProtocol::V1 ? GetCSDv1() : GetCSDv2(); const std::array<u32, 4> csd = m_protocol == SDProtocol::V1 ? GetCSDv1() : GetCSDv2();
Memory::CopyToEmuSwapped(buffer_out, csd.data(), csd.size() * sizeof(u32)); memory.CopyToEmuSwapped(buffer_out, csd.data(), csd.size() * sizeof(u32));
} }
break; break;
case ALL_SEND_CID: case ALL_SEND_CID:
case SEND_CID: case SEND_CID:
INFO_LOG_FMT(IOS_SD, "(ALL_)SEND_CID"); INFO_LOG_FMT(IOS_SD, "(ALL_)SEND_CID");
Memory::Write_U32(0x80114d1c, buffer_out); memory.Write_U32(0x80114d1c, buffer_out);
Memory::Write_U32(0x80080000, buffer_out + 4); memory.Write_U32(0x80080000, buffer_out + 4);
Memory::Write_U32(0x8007b520, buffer_out + 8); memory.Write_U32(0x8007b520, buffer_out + 8);
Memory::Write_U32(0x80080000, buffer_out + 12); memory.Write_U32(0x80080000, buffer_out + 12);
break; break;
case SET_BLOCKLEN: case SET_BLOCKLEN:
m_block_length = req.arg; m_block_length = req.arg;
Memory::Write_U32(0x900, buffer_out); memory.Write_U32(0x900, buffer_out);
break; break;
case APP_CMD_NEXT: case APP_CMD_NEXT:
// Next cmd is going to be ACMD_* // Next cmd is going to be ACMD_*
Memory::Write_U32(0x920, buffer_out); memory.Write_U32(0x920, buffer_out);
break; break;
case ACMD_SETBUSWIDTH: case ACMD_SETBUSWIDTH:
// 0 = 1bit, 2 = 4bit // 0 = 1bit, 2 = 4bit
m_bus_width = (req.arg & 3); m_bus_width = (req.arg & 3);
Memory::Write_U32(0x920, buffer_out); memory.Write_U32(0x920, buffer_out);
break; break;
case ACMD_SENDOPCOND: case ACMD_SENDOPCOND:
// Sends host capacity support information (HCS) and asks the accessed card to send // Sends host capacity support information (HCS) and asks the accessed card to send
// its operating condition register (OCR) content // its operating condition register (OCR) content
Memory::Write_U32(GetOCRegister(), buffer_out); memory.Write_U32(GetOCRegister(), buffer_out);
break; break;
case READ_MULTIPLE_BLOCK: case READ_MULTIPLE_BLOCK:
@ -283,7 +289,7 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
if (!m_card.Seek(address, File::SeekOrigin::Begin)) if (!m_card.Seek(address, File::SeekOrigin::Begin))
ERROR_LOG_FMT(IOS_SD, "Seek failed"); ERROR_LOG_FMT(IOS_SD, "Seek failed");
if (m_card.ReadBytes(Memory::GetPointer(req.addr), size)) if (m_card.ReadBytes(memory.GetPointer(req.addr), size))
{ {
DEBUG_LOG_FMT(IOS_SD, "Outbuffer size {} got {}", rw_buffer_size, size); DEBUG_LOG_FMT(IOS_SD, "Outbuffer size {} got {}", rw_buffer_size, size);
} }
@ -295,7 +301,7 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
} }
} }
} }
Memory::Write_U32(0x900, buffer_out); memory.Write_U32(0x900, buffer_out);
break; break;
case WRITE_MULTIPLE_BLOCK: case WRITE_MULTIPLE_BLOCK:
@ -313,7 +319,7 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
if (!m_card.Seek(address, File::SeekOrigin::Begin)) if (!m_card.Seek(address, File::SeekOrigin::Begin))
ERROR_LOG_FMT(IOS_SD, "Seek failed"); ERROR_LOG_FMT(IOS_SD, "Seek failed");
if (!m_card.WriteBytes(Memory::GetPointer(req.addr), size)) if (!m_card.WriteBytes(memory.GetPointer(req.addr), size))
{ {
ERROR_LOG_FMT(IOS_SD, "Write Failed - error: {}, eof: {}", std::ferror(m_card.GetHandle()), ERROR_LOG_FMT(IOS_SD, "Write Failed - error: {}, eof: {}", std::ferror(m_card.GetHandle()),
std::feof(m_card.GetHandle())); std::feof(m_card.GetHandle()));
@ -321,7 +327,7 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
} }
} }
} }
Memory::Write_U32(0x900, buffer_out); memory.Write_U32(0x900, buffer_out);
break; break;
case EVENT_REGISTER: // async case EVENT_REGISTER: // async
@ -354,8 +360,11 @@ s32 SDIOSlot0Device::ExecuteCommand(const Request& request, u32 buffer_in, u32 b
IPCReply SDIOSlot0Device::WriteHCRegister(const IOCtlRequest& request) IPCReply SDIOSlot0Device::WriteHCRegister(const IOCtlRequest& request)
{ {
const u32 reg = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
const u32 val = Memory::Read_U32(request.buffer_in + 16); auto& memory = system.GetMemory();
const u32 reg = memory.Read_U32(request.buffer_in);
const u32 val = memory.Read_U32(request.buffer_in + 16);
INFO_LOG_FMT(IOS_SD, "IOCTL_WRITEHCR {:#010x} - {:#010x}", reg, val); INFO_LOG_FMT(IOS_SD, "IOCTL_WRITEHCR {:#010x} - {:#010x}", reg, val);
@ -386,7 +395,10 @@ IPCReply SDIOSlot0Device::WriteHCRegister(const IOCtlRequest& request)
IPCReply SDIOSlot0Device::ReadHCRegister(const IOCtlRequest& request) IPCReply SDIOSlot0Device::ReadHCRegister(const IOCtlRequest& request)
{ {
const u32 reg = Memory::Read_U32(request.buffer_in); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 reg = memory.Read_U32(request.buffer_in);
if (reg >= m_registers.size()) if (reg >= m_registers.size())
{ {
@ -398,7 +410,7 @@ IPCReply SDIOSlot0Device::ReadHCRegister(const IOCtlRequest& request)
INFO_LOG_FMT(IOS_SD, "IOCTL_READHCR {:#010x} - {:#010x}", reg, val); INFO_LOG_FMT(IOS_SD, "IOCTL_READHCR {:#010x} - {:#010x}", reg, val);
// Just reading the register // Just reading the register
Memory::Write_U32(val, request.buffer_out); memory.Write_U32(val, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -406,8 +418,11 @@ IPCReply SDIOSlot0Device::ResetCard(const IOCtlRequest& request)
{ {
INFO_LOG_FMT(IOS_SD, "IOCTL_RESETCARD"); INFO_LOG_FMT(IOS_SD, "IOCTL_RESETCARD");
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Returns 16bit RCA and 16bit 0s (meaning success) // Returns 16bit RCA and 16bit 0s (meaning success)
Memory::Write_U32(m_status, request.buffer_out); memory.Write_U32(m_status, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -416,9 +431,12 @@ IPCReply SDIOSlot0Device::SetClk(const IOCtlRequest& request)
{ {
INFO_LOG_FMT(IOS_SD, "IOCTL_SETCLK"); INFO_LOG_FMT(IOS_SD, "IOCTL_SETCLK");
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// libogc only sets it to 1 and makes sure the return isn't negative... // libogc only sets it to 1 and makes sure the return isn't negative...
// one half of the sdclk divisor: a power of two or zero. // one half of the sdclk divisor: a power of two or zero.
const u32 clock = Memory::Read_U32(request.buffer_in); const u32 clock = memory.Read_U32(request.buffer_in);
if (clock != 1) if (clock != 1)
INFO_LOG_FMT(IOS_SD, "Setting to {}, interesting", clock); INFO_LOG_FMT(IOS_SD, "Setting to {}, interesting", clock);
@ -427,7 +445,10 @@ IPCReply SDIOSlot0Device::SetClk(const IOCtlRequest& request)
std::optional<IPCReply> SDIOSlot0Device::SendCommand(const IOCtlRequest& request) std::optional<IPCReply> SDIOSlot0Device::SendCommand(const IOCtlRequest& request)
{ {
INFO_LOG_FMT(IOS_SD, "IOCTL_SENDCMD {:x} IPC:{:08x}", Memory::Read_U32(request.buffer_in), auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
INFO_LOG_FMT(IOS_SD, "IOCTL_SENDCMD {:x} IPC:{:08x}", memory.Read_U32(request.buffer_in),
request.address); request.address);
const s32 return_value = ExecuteCommand(request, request.buffer_in, request.buffer_in_size, 0, 0, const s32 return_value = ExecuteCommand(request, request.buffer_in, request.buffer_in_size, 0, 0,
@ -476,23 +497,31 @@ IPCReply SDIOSlot0Device::GetStatus(const IOCtlRequest& request)
(status & CARD_INSERTED) ? "inserted" : "not present", (status & CARD_INSERTED) ? "inserted" : "not present",
(status & CARD_INITIALIZED) ? " and initialized" : ""); (status & CARD_INITIALIZED) ? " and initialized" : "");
Memory::Write_U32(status, request.buffer_out); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(status, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
IPCReply SDIOSlot0Device::GetOCRegister(const IOCtlRequest& request) IPCReply SDIOSlot0Device::GetOCRegister(const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 ocr = GetOCRegister(); const u32 ocr = GetOCRegister();
INFO_LOG_FMT(IOS_SD, "IOCTL_GETOCR. Replying with ocr {:x}", ocr); INFO_LOG_FMT(IOS_SD, "IOCTL_GETOCR. Replying with ocr {:x}", ocr);
Memory::Write_U32(ocr, request.buffer_out); memory.Write_U32(ocr, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
IPCReply SDIOSlot0Device::SendCommand(const IOCtlVRequest& request) IPCReply SDIOSlot0Device::SendCommand(const IOCtlVRequest& request)
{ {
DEBUG_LOG_FMT(IOS_SD, "IOCTLV_SENDCMD {:#010x}", Memory::Read_U32(request.in_vectors[0].address)); auto& system = Core::System::GetInstance();
Memory::Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size); auto& memory = system.GetMemory();
DEBUG_LOG_FMT(IOS_SD, "IOCTLV_SENDCMD {:#010x}", memory.Read_U32(request.in_vectors[0].address));
memory.Memset(request.io_vectors[0].address, 0, request.io_vectors[0].size);
const s32 return_value = const s32 return_value =
ExecuteCommand(request, request.in_vectors[0].address, request.in_vectors[0].size, ExecuteCommand(request, request.in_vectors[0].address, request.in_vectors[0].size,

View File

@ -10,6 +10,7 @@
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Core/Core.h" #include "Core/Core.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -17,6 +18,9 @@ static std::unique_ptr<IOCtlRequest> s_event_hook_request;
std::optional<IPCReply> STMImmediateDevice::IOCtl(const IOCtlRequest& request) std::optional<IPCReply> STMImmediateDevice::IOCtl(const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
s32 return_value = IPC_SUCCESS; s32 return_value = IPC_SUCCESS;
switch (request.request) switch (request.request)
{ {
@ -32,7 +36,7 @@ std::optional<IPCReply> STMImmediateDevice::IOCtl(const IOCtlRequest& request)
return_value = IPC_ENOENT; return_value = IPC_ENOENT;
break; break;
} }
Memory::Write_U32(0, s_event_hook_request->buffer_out); memory.Write_U32(0, s_event_hook_request->buffer_out);
m_ios.EnqueueIPCReply(*s_event_hook_request, IPC_SUCCESS); m_ios.EnqueueIPCReply(*s_event_hook_request, IPC_SUCCESS);
s_event_hook_request.reset(); s_event_hook_request.reset();
break; break;
@ -45,7 +49,7 @@ std::optional<IPCReply> STMImmediateDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_STM_VIDIMMING: // (Input: 20 bytes, Output: 20 bytes) case IOCTL_STM_VIDIMMING: // (Input: 20 bytes, Output: 20 bytes)
INFO_LOG_FMT(IOS_STM, "{} - IOCtl:", GetDeviceName()); INFO_LOG_FMT(IOS_STM, "{} - IOCtl:", GetDeviceName());
INFO_LOG_FMT(IOS_STM, " IOCTL_STM_VIDIMMING"); INFO_LOG_FMT(IOS_STM, " IOCTL_STM_VIDIMMING");
// Memory::Write_U32(1, buffer_out); // memory.Write_U32(1, buffer_out);
// return_value = 1; // return_value = 1;
break; break;
@ -101,7 +105,9 @@ void STMEventHookDevice::TriggerEvent(const u32 event) const
if (!m_is_active || !s_event_hook_request) if (!m_is_active || !s_event_hook_request)
return; return;
Memory::Write_U32(event, s_event_hook_request->buffer_out); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(event, s_event_hook_request->buffer_out);
m_ios.EnqueueIPCReply(*s_event_hook_request, IPC_SUCCESS); m_ios.EnqueueIPCReply(*s_event_hook_request, IPC_SUCCESS);
s_event_hook_request.reset(); s_event_hook_request.reset();
} }

View File

@ -164,15 +164,18 @@ std::optional<IPCReply> BluetoothEmuDevice::IOCtlV(const IOCtlVRequest& request)
{ {
case ACL_DATA_OUT: // ACL data is received from the stack case ACL_DATA_OUT: // ACL data is received from the stack
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// This is the ACL datapath from CPU to Wii Remote // This is the ACL datapath from CPU to Wii Remote
const auto* acl_header = const auto* acl_header =
reinterpret_cast<hci_acldata_hdr_t*>(Memory::GetPointer(ctrl.data_address)); reinterpret_cast<hci_acldata_hdr_t*>(memory.GetPointer(ctrl.data_address));
DEBUG_ASSERT(HCI_BC_FLAG(acl_header->con_handle) == HCI_POINT2POINT); DEBUG_ASSERT(HCI_BC_FLAG(acl_header->con_handle) == HCI_POINT2POINT);
DEBUG_ASSERT(HCI_PB_FLAG(acl_header->con_handle) == HCI_PACKET_START); DEBUG_ASSERT(HCI_PB_FLAG(acl_header->con_handle) == HCI_PACKET_START);
SendToDevice(HCI_CON_HANDLE(acl_header->con_handle), SendToDevice(HCI_CON_HANDLE(acl_header->con_handle),
Memory::GetPointer(ctrl.data_address + sizeof(hci_acldata_hdr_t)), memory.GetPointer(ctrl.data_address + sizeof(hci_acldata_hdr_t)),
acl_header->length); acl_header->length);
break; break;
} }
@ -244,8 +247,11 @@ void BluetoothEmuDevice::SendACLPacket(const bdaddr_t& source, const u8* data, u
DEBUG_LOG_FMT(IOS_WIIMOTE, "ACL endpoint valid, sending packet to {:08x}", DEBUG_LOG_FMT(IOS_WIIMOTE, "ACL endpoint valid, sending packet to {:08x}",
m_acl_endpoint->ios_request.address); m_acl_endpoint->ios_request.address);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
hci_acldata_hdr_t* header = hci_acldata_hdr_t* header =
reinterpret_cast<hci_acldata_hdr_t*>(Memory::GetPointer(m_acl_endpoint->data_address)); reinterpret_cast<hci_acldata_hdr_t*>(memory.GetPointer(m_acl_endpoint->data_address));
header->con_handle = HCI_MK_CON_HANDLE(connection_handle, HCI_PACKET_START, HCI_POINT2POINT); header->con_handle = HCI_MK_CON_HANDLE(connection_handle, HCI_PACKET_START, HCI_POINT2POINT);
header->length = size; header->length = size;
@ -421,7 +427,10 @@ void BluetoothEmuDevice::ACLPool::WriteToEndpoint(const USB::V0BulkMessage& endp
DEBUG_LOG_FMT(IOS_WIIMOTE, "ACL packet being written from queue to {:08x}", DEBUG_LOG_FMT(IOS_WIIMOTE, "ACL packet being written from queue to {:08x}",
endpoint.ios_request.address); endpoint.ios_request.address);
hci_acldata_hdr_t* header = (hci_acldata_hdr_t*)Memory::GetPointer(endpoint.data_address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
hci_acldata_hdr_t* header = (hci_acldata_hdr_t*)memory.GetPointer(endpoint.data_address);
header->con_handle = HCI_MK_CON_HANDLE(conn_handle, HCI_PACKET_START, HCI_POINT2POINT); header->con_handle = HCI_MK_CON_HANDLE(conn_handle, HCI_PACKET_START, HCI_POINT2POINT);
header->length = size; header->length = size;
@ -957,10 +966,13 @@ bool BluetoothEmuDevice::SendEventConPacketTypeChange(u16 connection_handle, u16
// This is called from the USB::IOCTLV_USBV0_CTRLMSG Ioctlv // This is called from the USB::IOCTLV_USBV0_CTRLMSG Ioctlv
void BluetoothEmuDevice::ExecuteHCICommandMessage(const USB::V0CtrlMessage& ctrl_message) void BluetoothEmuDevice::ExecuteHCICommandMessage(const USB::V0CtrlMessage& ctrl_message)
{ {
const u8* input = Memory::GetPointer(ctrl_message.data_address + 3); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8* input = memory.GetPointer(ctrl_message.data_address + 3);
SCommandMessage msg; SCommandMessage msg;
std::memcpy(&msg, Memory::GetPointer(ctrl_message.data_address), sizeof(msg)); std::memcpy(&msg, memory.GetPointer(ctrl_message.data_address), sizeof(msg));
const u16 ocf = HCI_OCF(msg.Opcode); const u16 ocf = HCI_OCF(msg.Opcode);
const u16 ogf = HCI_OGF(msg.Opcode); const u16 ogf = HCI_OGF(msg.Opcode);

View File

@ -28,6 +28,7 @@
#include "Core/Core.h" #include "Core/Core.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/System.h"
#include "VideoCommon/OnScreenDisplay.h" #include "VideoCommon/OnScreenDisplay.h"
namespace IOS::HLE namespace IOS::HLE
@ -213,9 +214,12 @@ std::optional<IPCReply> BluetoothRealDevice::IOCtlV(const IOCtlVRequest& request
// HCI commands to the Bluetooth adapter // HCI commands to the Bluetooth adapter
case USB::IOCTLV_USBV0_CTRLMSG: case USB::IOCTLV_USBV0_CTRLMSG:
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::lock_guard lk(m_transfers_mutex); std::lock_guard lk(m_transfers_mutex);
auto cmd = std::make_unique<USB::V0CtrlMessage>(m_ios, request); auto cmd = std::make_unique<USB::V0CtrlMessage>(m_ios, request);
const u16 opcode = Common::swap16(Memory::Read_U16(cmd->data_address)); const u16 opcode = Common::swap16(memory.Read_U16(cmd->data_address));
if (opcode == HCI_CMD_READ_BUFFER_SIZE) if (opcode == HCI_CMD_READ_BUFFER_SIZE)
{ {
m_fake_read_buffer_size_reply.Set(); m_fake_read_buffer_size_reply.Set();
@ -231,7 +235,7 @@ std::optional<IPCReply> BluetoothRealDevice::IOCtlV(const IOCtlVRequest& request
{ {
// Delete link key(s) from our own link key storage when the game tells the adapter to // Delete link key(s) from our own link key storage when the game tells the adapter to
hci_delete_stored_link_key_cp delete_cmd; hci_delete_stored_link_key_cp delete_cmd;
Memory::CopyFromEmu(&delete_cmd, cmd->data_address, sizeof(delete_cmd)); memory.CopyFromEmu(&delete_cmd, cmd->data_address, sizeof(delete_cmd));
if (delete_cmd.delete_all) if (delete_cmd.delete_all)
m_link_keys.clear(); m_link_keys.clear();
else else
@ -240,7 +244,7 @@ std::optional<IPCReply> BluetoothRealDevice::IOCtlV(const IOCtlVRequest& request
auto buffer = std::make_unique<u8[]>(cmd->length + LIBUSB_CONTROL_SETUP_SIZE); auto buffer = std::make_unique<u8[]>(cmd->length + LIBUSB_CONTROL_SETUP_SIZE);
libusb_fill_control_setup(buffer.get(), cmd->request_type, cmd->request, cmd->value, cmd->index, libusb_fill_control_setup(buffer.get(), cmd->request_type, cmd->request, cmd->value, cmd->index,
cmd->length); cmd->length);
Memory::CopyFromEmu(buffer.get() + LIBUSB_CONTROL_SETUP_SIZE, cmd->data_address, cmd->length); memory.CopyFromEmu(buffer.get() + LIBUSB_CONTROL_SETUP_SIZE, cmd->data_address, cmd->length);
libusb_transfer* transfer = libusb_alloc_transfer(0); libusb_transfer* transfer = libusb_alloc_transfer(0);
transfer->flags |= LIBUSB_TRANSFER_FREE_TRANSFER; transfer->flags |= LIBUSB_TRANSFER_FREE_TRANSFER;
libusb_fill_control_transfer(transfer, m_handle, buffer.get(), nullptr, this, 0); libusb_fill_control_transfer(transfer, m_handle, buffer.get(), nullptr, this, 0);
@ -490,13 +494,16 @@ bool BluetoothRealDevice::SendHCIStoreLinkKeyCommand()
void BluetoothRealDevice::FakeVendorCommandReply(USB::V0IntrMessage& ctrl) void BluetoothRealDevice::FakeVendorCommandReply(USB::V0IntrMessage& ctrl)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
SHCIEventCommand hci_event; SHCIEventCommand hci_event;
Memory::CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event)); memory.CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event));
hci_event.EventType = HCI_EVENT_COMMAND_COMPL; hci_event.EventType = HCI_EVENT_COMMAND_COMPL;
hci_event.PayloadLength = sizeof(SHCIEventCommand) - 2; hci_event.PayloadLength = sizeof(SHCIEventCommand) - 2;
hci_event.PacketIndicator = 0x01; hci_event.PacketIndicator = 0x01;
hci_event.Opcode = m_fake_vendor_command_reply_opcode; hci_event.Opcode = m_fake_vendor_command_reply_opcode;
Memory::CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event)); memory.CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event));
m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event))); m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event)));
} }
@ -507,13 +514,16 @@ void BluetoothRealDevice::FakeVendorCommandReply(USB::V0IntrMessage& ctrl)
// (including Wiimote disconnects and "event mismatch" warning messages). // (including Wiimote disconnects and "event mismatch" warning messages).
void BluetoothRealDevice::FakeReadBufferSizeReply(USB::V0IntrMessage& ctrl) void BluetoothRealDevice::FakeReadBufferSizeReply(USB::V0IntrMessage& ctrl)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
SHCIEventCommand hci_event; SHCIEventCommand hci_event;
Memory::CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event)); memory.CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event));
hci_event.EventType = HCI_EVENT_COMMAND_COMPL; hci_event.EventType = HCI_EVENT_COMMAND_COMPL;
hci_event.PayloadLength = sizeof(SHCIEventCommand) - 2 + sizeof(hci_read_buffer_size_rp); hci_event.PayloadLength = sizeof(SHCIEventCommand) - 2 + sizeof(hci_read_buffer_size_rp);
hci_event.PacketIndicator = 0x01; hci_event.PacketIndicator = 0x01;
hci_event.Opcode = HCI_CMD_READ_BUFFER_SIZE; hci_event.Opcode = HCI_CMD_READ_BUFFER_SIZE;
Memory::CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event)); memory.CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event));
hci_read_buffer_size_rp reply; hci_read_buffer_size_rp reply;
reply.status = 0x00; reply.status = 0x00;
@ -521,19 +531,22 @@ void BluetoothRealDevice::FakeReadBufferSizeReply(USB::V0IntrMessage& ctrl)
reply.num_acl_pkts = ACL_PKT_NUM; reply.num_acl_pkts = ACL_PKT_NUM;
reply.max_sco_size = SCO_PKT_SIZE; reply.max_sco_size = SCO_PKT_SIZE;
reply.num_sco_pkts = SCO_PKT_NUM; reply.num_sco_pkts = SCO_PKT_NUM;
Memory::CopyToEmu(ctrl.data_address + sizeof(hci_event), &reply, sizeof(reply)); memory.CopyToEmu(ctrl.data_address + sizeof(hci_event), &reply, sizeof(reply));
m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event) + sizeof(reply))); m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event) + sizeof(reply)));
} }
void BluetoothRealDevice::FakeSyncButtonEvent(USB::V0IntrMessage& ctrl, const u8* payload, void BluetoothRealDevice::FakeSyncButtonEvent(USB::V0IntrMessage& ctrl, const u8* payload,
const u8 size) const u8 size)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
hci_event_hdr_t hci_event; hci_event_hdr_t hci_event;
Memory::CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event)); memory.CopyFromEmu(&hci_event, ctrl.data_address, sizeof(hci_event));
hci_event.event = HCI_EVENT_VENDOR; hci_event.event = HCI_EVENT_VENDOR;
hci_event.length = size; hci_event.length = size;
Memory::CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event)); memory.CopyToEmu(ctrl.data_address, &hci_event, sizeof(hci_event));
Memory::CopyToEmu(ctrl.data_address + sizeof(hci_event), payload, size); memory.CopyToEmu(ctrl.data_address + sizeof(hci_event), payload, size);
m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event) + size)); m_ios.EnqueueIPCReply(ctrl.ios_request, static_cast<s32>(sizeof(hci_event) + size));
} }

View File

@ -11,6 +11,7 @@
#include "Common/CommonTypes.h" #include "Common/CommonTypes.h"
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace IOS::HLE::USB namespace IOS::HLE::USB
{ {
@ -18,14 +19,18 @@ std::unique_ptr<u8[]> TransferCommand::MakeBuffer(const size_t size) const
{ {
ASSERT_MSG(IOS_USB, data_address != 0, "Invalid data_address"); ASSERT_MSG(IOS_USB, data_address != 0, "Invalid data_address");
auto buffer = std::make_unique<u8[]>(size); auto buffer = std::make_unique<u8[]>(size);
Memory::CopyFromEmu(buffer.get(), data_address, size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(buffer.get(), data_address, size);
return buffer; return buffer;
} }
void TransferCommand::FillBuffer(const u8* src, const size_t size) const void TransferCommand::FillBuffer(const u8* src, const size_t size) const
{ {
ASSERT_MSG(IOS_USB, size == 0 || data_address != 0, "Invalid data_address"); ASSERT_MSG(IOS_USB, size == 0 || data_address != 0, "Invalid data_address");
Memory::CopyToEmu(data_address, src, size); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(data_address, src, size);
} }
void TransferCommand::OnTransferComplete(s32 return_value) const void TransferCommand::OnTransferComplete(s32 return_value) const
@ -35,7 +40,9 @@ void TransferCommand::OnTransferComplete(s32 return_value) const
void IsoMessage::SetPacketReturnValue(const size_t packet_num, const u16 return_value) const void IsoMessage::SetPacketReturnValue(const size_t packet_num, const u16 return_value) const
{ {
Memory::Write_U16(return_value, static_cast<u32>(packet_sizes_addr + packet_num * sizeof(u16))); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U16(return_value, static_cast<u32>(packet_sizes_addr + packet_num * sizeof(u16)));
} }
Device::~Device() = default; Device::~Device() = default;

View File

@ -21,6 +21,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/System.h"
namespace IOS::HLE::USB namespace IOS::HLE::USB
{ {
@ -247,7 +248,11 @@ int LibusbDevice::SubmitTransfer(std::unique_ptr<CtrlMessage> cmd)
auto buffer = std::make_unique<u8[]>(size); auto buffer = std::make_unique<u8[]>(size);
libusb_fill_control_setup(buffer.get(), cmd->request_type, cmd->request, cmd->value, cmd->index, libusb_fill_control_setup(buffer.get(), cmd->request_type, cmd->request, cmd->value, cmd->index,
cmd->length); cmd->length);
Memory::CopyFromEmu(buffer.get() + LIBUSB_CONTROL_SETUP_SIZE, cmd->data_address, cmd->length);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(buffer.get() + LIBUSB_CONTROL_SETUP_SIZE, cmd->data_address, cmd->length);
libusb_transfer* transfer = libusb_alloc_transfer(0); libusb_transfer* transfer = libusb_alloc_transfer(0);
transfer->flags |= LIBUSB_TRANSFER_FREE_TRANSFER; transfer->flags |= LIBUSB_TRANSFER_FREE_TRANSFER;
libusb_fill_control_transfer(transfer, m_handle, buffer.release(), CtrlTransferCallback, this, 0); libusb_fill_control_transfer(transfer, m_handle, buffer.release(), CtrlTransferCallback, this, 0);

View File

@ -18,6 +18,7 @@
#include "Core/IOS/USB/Common.h" #include "Core/IOS/USB/Common.h"
#include "Core/IOS/USB/USBV0.h" #include "Core/IOS/USB/USBV0.h"
#include "Core/IOS/VersionInfo.h" #include "Core/IOS/VersionInfo.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -92,9 +93,12 @@ IPCReply OH0::CancelInsertionHook(const IOCtlRequest& request)
if (!request.buffer_in || request.buffer_in_size != 4) if (!request.buffer_in || request.buffer_in_size != 4)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// IOS assigns random IDs, but ours are simply the VID + PID (see RegisterInsertionHookWithID) // IOS assigns random IDs, but ours are simply the VID + PID (see RegisterInsertionHookWithID)
TriggerHook(m_insertion_hooks, TriggerHook(m_insertion_hooks,
{Memory::Read_U16(request.buffer_in), Memory::Read_U16(request.buffer_in + 2)}, {memory.Read_U16(request.buffer_in), memory.Read_U16(request.buffer_in + 2)},
USB_ECANCELED); USB_ECANCELED);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -104,11 +108,14 @@ IPCReply OH0::GetDeviceList(const IOCtlVRequest& request) const
if (!request.HasNumberOfValidVectors(2, 2)) if (!request.HasNumberOfValidVectors(2, 2))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const u8 max_entries_count = Memory::Read_U8(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 max_entries_count = memory.Read_U8(request.in_vectors[0].address);
if (request.io_vectors[1].size != max_entries_count * sizeof(DeviceEntry)) if (request.io_vectors[1].size != max_entries_count * sizeof(DeviceEntry))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const u8 interface_class = Memory::Read_U8(request.in_vectors[1].address); const u8 interface_class = memory.Read_U8(request.in_vectors[1].address);
u8 entries_count = 0; u8 entries_count = 0;
std::lock_guard lk(m_devices_mutex); std::lock_guard lk(m_devices_mutex);
for (const auto& device : m_devices) for (const auto& device : m_devices)
@ -122,9 +129,9 @@ IPCReply OH0::GetDeviceList(const IOCtlVRequest& request) const
entry.unknown = 0; entry.unknown = 0;
entry.vid = Common::swap16(device.second->GetVid()); entry.vid = Common::swap16(device.second->GetVid());
entry.pid = Common::swap16(device.second->GetPid()); entry.pid = Common::swap16(device.second->GetPid());
Memory::CopyToEmu(request.io_vectors[1].address + 8 * entries_count++, &entry, 8); memory.CopyToEmu(request.io_vectors[1].address + 8 * entries_count++, &entry, 8);
} }
Memory::Write_U8(entries_count, request.io_vectors[0].address); memory.Write_U8(entries_count, request.io_vectors[0].address);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -133,8 +140,11 @@ IPCReply OH0::GetRhDesca(const IOCtlRequest& request) const
if (!request.buffer_out || request.buffer_out_size != 4) if (!request.buffer_out || request.buffer_out_size != 4)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Based on a hardware test, this ioctl seems to return a constant value // Based on a hardware test, this ioctl seems to return a constant value
Memory::Write_U32(0x02000302, request.buffer_out); memory.Write_U32(0x02000302, request.buffer_out);
request.Dump(GetDeviceName(), Common::Log::LogType::IOS_USB, Common::Log::LogLevel::LWARNING); request.Dump(GetDeviceName(), Common::Log::LogType::IOS_USB, Common::Log::LogLevel::LWARNING);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -174,8 +184,11 @@ std::optional<IPCReply> OH0::RegisterInsertionHook(const IOCtlVRequest& request)
if (!request.HasNumberOfValidVectors(2, 0)) if (!request.HasNumberOfValidVectors(2, 0))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const u16 vid = Memory::Read_U16(request.in_vectors[0].address); auto& system = Core::System::GetInstance();
const u16 pid = Memory::Read_U16(request.in_vectors[1].address); auto& memory = system.GetMemory();
const u16 vid = memory.Read_U16(request.in_vectors[0].address);
const u16 pid = memory.Read_U16(request.in_vectors[1].address);
if (HasDeviceWithVidPid(vid, pid)) if (HasDeviceWithVidPid(vid, pid))
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
@ -190,16 +203,19 @@ std::optional<IPCReply> OH0::RegisterInsertionHookWithID(const IOCtlVRequest& re
if (!request.HasNumberOfValidVectors(3, 1)) if (!request.HasNumberOfValidVectors(3, 1))
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::lock_guard lock{m_hooks_mutex}; std::lock_guard lock{m_hooks_mutex};
const u16 vid = Memory::Read_U16(request.in_vectors[0].address); const u16 vid = memory.Read_U16(request.in_vectors[0].address);
const u16 pid = Memory::Read_U16(request.in_vectors[1].address); const u16 pid = memory.Read_U16(request.in_vectors[1].address);
const bool trigger_only_for_new_device = Memory::Read_U8(request.in_vectors[2].address) == 1; const bool trigger_only_for_new_device = memory.Read_U8(request.in_vectors[2].address) == 1;
if (!trigger_only_for_new_device && HasDeviceWithVidPid(vid, pid)) if (!trigger_only_for_new_device && HasDeviceWithVidPid(vid, pid))
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
// TODO: figure out whether IOS allows more than one hook. // TODO: figure out whether IOS allows more than one hook.
m_insertion_hooks.insert({{vid, pid}, request.address}); m_insertion_hooks.insert({{vid, pid}, request.address});
// The output vector is overwritten with an ID to use with ioctl 31 for cancelling the hook. // The output vector is overwritten with an ID to use with ioctl 31 for cancelling the hook.
Memory::Write_U32(vid << 16 | pid, request.io_vectors[0].address); memory.Write_U32(vid << 16 | pid, request.io_vectors[0].address);
return std::nullopt; return std::nullopt;
} }
@ -316,25 +332,28 @@ std::optional<IPCReply> OH0::DeviceIOCtlV(const u64 device_id, const IOCtlVReque
s32 OH0::SubmitTransfer(USB::Device& device, const IOCtlVRequest& ioctlv) s32 OH0::SubmitTransfer(USB::Device& device, const IOCtlVRequest& ioctlv)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (ioctlv.request) switch (ioctlv.request)
{ {
case USB::IOCTLV_USBV0_CTRLMSG: case USB::IOCTLV_USBV0_CTRLMSG:
if (!ioctlv.HasNumberOfValidVectors(6, 1) || if (!ioctlv.HasNumberOfValidVectors(6, 1) ||
Common::swap16(Memory::Read_U16(ioctlv.in_vectors[4].address)) != ioctlv.io_vectors[0].size) Common::swap16(memory.Read_U16(ioctlv.in_vectors[4].address)) != ioctlv.io_vectors[0].size)
return IPC_EINVAL; return IPC_EINVAL;
return device.SubmitTransfer(std::make_unique<USB::V0CtrlMessage>(m_ios, ioctlv)); return device.SubmitTransfer(std::make_unique<USB::V0CtrlMessage>(m_ios, ioctlv));
case USB::IOCTLV_USBV0_BLKMSG: case USB::IOCTLV_USBV0_BLKMSG:
case USB::IOCTLV_USBV0_LBLKMSG: case USB::IOCTLV_USBV0_LBLKMSG:
if (!ioctlv.HasNumberOfValidVectors(2, 1) || if (!ioctlv.HasNumberOfValidVectors(2, 1) ||
Memory::Read_U16(ioctlv.in_vectors[1].address) != ioctlv.io_vectors[0].size) memory.Read_U16(ioctlv.in_vectors[1].address) != ioctlv.io_vectors[0].size)
return IPC_EINVAL; return IPC_EINVAL;
return device.SubmitTransfer(std::make_unique<USB::V0BulkMessage>( return device.SubmitTransfer(std::make_unique<USB::V0BulkMessage>(
m_ios, ioctlv, ioctlv.request == USB::IOCTLV_USBV0_LBLKMSG)); m_ios, ioctlv, ioctlv.request == USB::IOCTLV_USBV0_LBLKMSG));
case USB::IOCTLV_USBV0_INTRMSG: case USB::IOCTLV_USBV0_INTRMSG:
if (!ioctlv.HasNumberOfValidVectors(2, 1) || if (!ioctlv.HasNumberOfValidVectors(2, 1) ||
Memory::Read_U16(ioctlv.in_vectors[1].address) != ioctlv.io_vectors[0].size) memory.Read_U16(ioctlv.in_vectors[1].address) != ioctlv.io_vectors[0].size)
return IPC_EINVAL; return IPC_EINVAL;
return device.SubmitTransfer(std::make_unique<USB::V0IntrMessage>(m_ios, ioctlv)); return device.SubmitTransfer(std::make_unique<USB::V0IntrMessage>(m_ios, ioctlv));

View File

@ -9,44 +9,53 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/System.h"
namespace IOS::HLE::USB namespace IOS::HLE::USB
{ {
V0CtrlMessage::V0CtrlMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V0CtrlMessage::V0CtrlMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: CtrlMessage(ios, ioctlv, ioctlv.io_vectors[0].address) : CtrlMessage(ios, ioctlv, ioctlv.io_vectors[0].address)
{ {
request_type = Memory::Read_U8(ioctlv.in_vectors[0].address); auto& system = Core::System::GetInstance();
request = Memory::Read_U8(ioctlv.in_vectors[1].address); auto& memory = system.GetMemory();
value = Common::swap16(Memory::Read_U16(ioctlv.in_vectors[2].address)); request_type = memory.Read_U8(ioctlv.in_vectors[0].address);
index = Common::swap16(Memory::Read_U16(ioctlv.in_vectors[3].address)); request = memory.Read_U8(ioctlv.in_vectors[1].address);
length = Common::swap16(Memory::Read_U16(ioctlv.in_vectors[4].address)); value = Common::swap16(memory.Read_U16(ioctlv.in_vectors[2].address));
index = Common::swap16(memory.Read_U16(ioctlv.in_vectors[3].address));
length = Common::swap16(memory.Read_U16(ioctlv.in_vectors[4].address));
} }
V0BulkMessage::V0BulkMessage(Kernel& ios, const IOCtlVRequest& ioctlv, bool long_length) V0BulkMessage::V0BulkMessage(Kernel& ios, const IOCtlVRequest& ioctlv, bool long_length)
: BulkMessage(ios, ioctlv, ioctlv.io_vectors[0].address) : BulkMessage(ios, ioctlv, ioctlv.io_vectors[0].address)
{ {
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
endpoint = memory.Read_U8(ioctlv.in_vectors[0].address);
if (long_length) if (long_length)
length = Memory::Read_U32(ioctlv.in_vectors[1].address); length = memory.Read_U32(ioctlv.in_vectors[1].address);
else else
length = Memory::Read_U16(ioctlv.in_vectors[1].address); length = memory.Read_U16(ioctlv.in_vectors[1].address);
} }
V0IntrMessage::V0IntrMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V0IntrMessage::V0IntrMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: IntrMessage(ios, ioctlv, ioctlv.io_vectors[0].address) : IntrMessage(ios, ioctlv, ioctlv.io_vectors[0].address)
{ {
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address); auto& system = Core::System::GetInstance();
length = Memory::Read_U16(ioctlv.in_vectors[1].address); auto& memory = system.GetMemory();
endpoint = memory.Read_U8(ioctlv.in_vectors[0].address);
length = memory.Read_U16(ioctlv.in_vectors[1].address);
} }
V0IsoMessage::V0IsoMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V0IsoMessage::V0IsoMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: IsoMessage(ios, ioctlv, ioctlv.io_vectors[1].address) : IsoMessage(ios, ioctlv, ioctlv.io_vectors[1].address)
{ {
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address); auto& system = Core::System::GetInstance();
length = Memory::Read_U16(ioctlv.in_vectors[1].address); auto& memory = system.GetMemory();
num_packets = Memory::Read_U8(ioctlv.in_vectors[2].address); endpoint = memory.Read_U8(ioctlv.in_vectors[0].address);
length = memory.Read_U16(ioctlv.in_vectors[1].address);
num_packets = memory.Read_U8(ioctlv.in_vectors[2].address);
packet_sizes_addr = ioctlv.io_vectors[0].address; packet_sizes_addr = ioctlv.io_vectors[0].address;
for (size_t i = 0; i < num_packets; ++i) for (size_t i = 0; i < num_packets; ++i)
packet_sizes.push_back(Memory::Read_U16(static_cast<u32>(packet_sizes_addr + i * sizeof(u16)))); packet_sizes.push_back(memory.Read_U16(static_cast<u32>(packet_sizes_addr + i * sizeof(u16))));
} }
} // namespace IOS::HLE::USB } // namespace IOS::HLE::USB

View File

@ -12,6 +12,7 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/System.h"
namespace IOS::HLE::USB namespace IOS::HLE::USB
{ {
@ -47,8 +48,11 @@ struct HIDRequest
V4CtrlMessage::V4CtrlMessage(Kernel& ios, const IOCtlRequest& ioctl) : CtrlMessage(ios, ioctl, 0) V4CtrlMessage::V4CtrlMessage(Kernel& ios, const IOCtlRequest& ioctl) : CtrlMessage(ios, ioctl, 0)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
HIDRequest hid_request; HIDRequest hid_request;
Memory::CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request)); memory.CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request));
request_type = hid_request.control.bmRequestType; request_type = hid_request.control.bmRequestType;
request = hid_request.control.bmRequest; request = hid_request.control.bmRequest;
value = Common::swap16(hid_request.control.wValue); value = Common::swap16(hid_request.control.wValue);
@ -63,8 +67,11 @@ V4CtrlMessage::V4CtrlMessage(Kernel& ios, const IOCtlRequest& ioctl) : CtrlMessa
V4GetUSStringMessage::V4GetUSStringMessage(Kernel& ios, const IOCtlRequest& ioctl) V4GetUSStringMessage::V4GetUSStringMessage(Kernel& ios, const IOCtlRequest& ioctl)
: CtrlMessage(ios, ioctl, 0) : CtrlMessage(ios, ioctl, 0)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
HIDRequest hid_request; HIDRequest hid_request;
Memory::CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request)); memory.CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request));
request_type = 0x80; request_type = 0x80;
request = REQUEST_GET_DESCRIPTOR; request = REQUEST_GET_DESCRIPTOR;
value = (0x03 << 8) | hid_request.string.bIndex; value = (0x03 << 8) | hid_request.string.bIndex;
@ -75,16 +82,22 @@ V4GetUSStringMessage::V4GetUSStringMessage(Kernel& ios, const IOCtlRequest& ioct
void V4GetUSStringMessage::OnTransferComplete(s32 return_value) const void V4GetUSStringMessage::OnTransferComplete(s32 return_value) const
{ {
std::string message = Memory::GetString(data_address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::string message = memory.GetString(data_address);
std::replace_if(message.begin(), message.end(), std::not_fn(IsPrintableCharacter), '?'); std::replace_if(message.begin(), message.end(), std::not_fn(IsPrintableCharacter), '?');
Memory::CopyToEmu(data_address, message.c_str(), message.size()); memory.CopyToEmu(data_address, message.c_str(), message.size());
TransferCommand::OnTransferComplete(return_value); TransferCommand::OnTransferComplete(return_value);
} }
V4IntrMessage::V4IntrMessage(Kernel& ios, const IOCtlRequest& ioctl) : IntrMessage(ios, ioctl, 0) V4IntrMessage::V4IntrMessage(Kernel& ios, const IOCtlRequest& ioctl) : IntrMessage(ios, ioctl, 0)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
HIDRequest hid_request; HIDRequest hid_request;
Memory::CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request)); memory.CopyFromEmu(&hid_request, ioctl.buffer_in, sizeof(hid_request));
length = Common::swap32(hid_request.interrupt.length); length = Common::swap32(hid_request.interrupt.length);
endpoint = static_cast<u8>(Common::swap32(hid_request.interrupt.endpoint)); endpoint = static_cast<u8>(Common::swap32(hid_request.interrupt.endpoint));
data_address = Common::swap32(hid_request.data_addr); data_address = Common::swap32(hid_request.data_addr);

View File

@ -13,6 +13,7 @@
#include "Common/Swap.h" #include "Common/Swap.h"
#include "Core/CoreTiming.h" #include "Core/CoreTiming.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -21,37 +22,45 @@ namespace USB
V5CtrlMessage::V5CtrlMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V5CtrlMessage::V5CtrlMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: CtrlMessage(ios, ioctlv, ioctlv.GetVector(1)->address) : CtrlMessage(ios, ioctlv, ioctlv.GetVector(1)->address)
{ {
request_type = Memory::Read_U8(ioctlv.in_vectors[0].address + 8); auto& system = Core::System::GetInstance();
request = Memory::Read_U8(ioctlv.in_vectors[0].address + 9); auto& memory = system.GetMemory();
value = Memory::Read_U16(ioctlv.in_vectors[0].address + 10); request_type = memory.Read_U8(ioctlv.in_vectors[0].address + 8);
index = Memory::Read_U16(ioctlv.in_vectors[0].address + 12); request = memory.Read_U8(ioctlv.in_vectors[0].address + 9);
value = memory.Read_U16(ioctlv.in_vectors[0].address + 10);
index = memory.Read_U16(ioctlv.in_vectors[0].address + 12);
length = static_cast<u16>(ioctlv.GetVector(1)->size); length = static_cast<u16>(ioctlv.GetVector(1)->size);
} }
V5BulkMessage::V5BulkMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V5BulkMessage::V5BulkMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: BulkMessage(ios, ioctlv, ioctlv.GetVector(1)->address) : BulkMessage(ios, ioctlv, ioctlv.GetVector(1)->address)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
length = ioctlv.GetVector(1)->size; length = ioctlv.GetVector(1)->size;
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address + 18); endpoint = memory.Read_U8(ioctlv.in_vectors[0].address + 18);
} }
V5IntrMessage::V5IntrMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V5IntrMessage::V5IntrMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: IntrMessage(ios, ioctlv, ioctlv.GetVector(1)->address) : IntrMessage(ios, ioctlv, ioctlv.GetVector(1)->address)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
length = ioctlv.GetVector(1)->size; length = ioctlv.GetVector(1)->size;
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address + 14); endpoint = memory.Read_U8(ioctlv.in_vectors[0].address + 14);
} }
V5IsoMessage::V5IsoMessage(Kernel& ios, const IOCtlVRequest& ioctlv) V5IsoMessage::V5IsoMessage(Kernel& ios, const IOCtlVRequest& ioctlv)
: IsoMessage(ios, ioctlv, ioctlv.GetVector(2)->address) : IsoMessage(ios, ioctlv, ioctlv.GetVector(2)->address)
{ {
num_packets = Memory::Read_U8(ioctlv.in_vectors[0].address + 16); auto& system = Core::System::GetInstance();
endpoint = Memory::Read_U8(ioctlv.in_vectors[0].address + 17); auto& memory = system.GetMemory();
num_packets = memory.Read_U8(ioctlv.in_vectors[0].address + 16);
endpoint = memory.Read_U8(ioctlv.in_vectors[0].address + 17);
packet_sizes_addr = ioctlv.GetVector(1)->address; packet_sizes_addr = ioctlv.GetVector(1)->address;
u32 total_packet_size = 0; u32 total_packet_size = 0;
for (size_t i = 0; i < num_packets; ++i) for (size_t i = 0; i < num_packets; ++i)
{ {
const u32 packet_size = Memory::Read_U16(static_cast<u32>(packet_sizes_addr + i * sizeof(u16))); const u32 packet_size = memory.Read_U16(static_cast<u32>(packet_sizes_addr + i * sizeof(u16)));
packet_sizes.push_back(packet_size); packet_sizes.push_back(packet_size);
total_packet_size += packet_size; total_packet_size += packet_size;
} }
@ -99,8 +108,10 @@ void USBV5ResourceManager::DoState(PointerWrap& p)
USBV5ResourceManager::USBV5Device* USBV5ResourceManager::GetUSBV5Device(u32 in_buffer) USBV5ResourceManager::USBV5Device* USBV5ResourceManager::GetUSBV5Device(u32 in_buffer)
{ {
const u8 index = Memory::Read_U8(in_buffer + offsetof(DeviceID, index)); auto& system = Core::System::GetInstance();
const u16 number = Memory::Read_U16(in_buffer + offsetof(DeviceID, number)); auto& memory = system.GetMemory();
const u8 index = memory.Read_U8(in_buffer + offsetof(DeviceID, index));
const u16 number = memory.Read_U16(in_buffer + offsetof(DeviceID, number));
if (index >= m_usbv5_devices.size()) if (index >= m_usbv5_devices.size())
return nullptr; return nullptr;
@ -135,7 +146,9 @@ IPCReply USBV5ResourceManager::SetAlternateSetting(USBV5Device& device, const IO
if (!host_device->AttachAndChangeInterface(device.interface_number)) if (!host_device->AttachAndChangeInterface(device.interface_number))
return IPCReply(-1); return IPCReply(-1);
const u8 alt_setting = Memory::Read_U8(request.buffer_in + 2 * sizeof(s32)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 alt_setting = memory.Read_U8(request.buffer_in + 2 * sizeof(s32));
const bool success = host_device->SetAltSetting(alt_setting) == 0; const bool success = host_device->SetAltSetting(alt_setting) == 0;
return IPCReply(success ? IPC_SUCCESS : IPC_EINVAL); return IPCReply(success ? IPC_SUCCESS : IPC_EINVAL);
@ -160,8 +173,11 @@ IPCReply USBV5ResourceManager::Shutdown(const IOCtlRequest& request)
IPCReply USBV5ResourceManager::SuspendResume(USBV5Device& device, const IOCtlRequest& request) IPCReply USBV5ResourceManager::SuspendResume(USBV5Device& device, const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const auto host_device = GetDeviceById(device.host_id); const auto host_device = GetDeviceById(device.host_id);
const s32 resumed = Memory::Read_U32(request.buffer_in + 8); const s32 resumed = memory.Read_U32(request.buffer_in + 8);
// Note: this is unimplemented because there's no easy way to do this in a // Note: this is unimplemented because there's no easy way to do this in a
// platform-independant way (libusb does not support power management). // platform-independant way (libusb does not support power management).
@ -232,6 +248,9 @@ void USBV5ResourceManager::TriggerDeviceChangeReply()
return; return;
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::lock_guard lock{m_usbv5_devices_mutex}; std::lock_guard lock{m_usbv5_devices_mutex};
u8 num_devices = 0; u8 num_devices = 0;
for (auto it = m_usbv5_devices.crbegin(); it != m_usbv5_devices.crend(); ++it) for (auto it = m_usbv5_devices.crbegin(); it != m_usbv5_devices.crend(); ++it)
@ -264,7 +283,7 @@ void USBV5ResourceManager::TriggerDeviceChangeReply()
entry.interface_number = usbv5_device.interface_number; entry.interface_number = usbv5_device.interface_number;
entry.num_altsettings = device->GetNumberOfAltSettings(entry.interface_number); entry.num_altsettings = device->GetNumberOfAltSettings(entry.interface_number);
Memory::CopyToEmu(m_devicechange_hook_request->buffer_out + sizeof(entry) * num_devices, &entry, memory.CopyToEmu(m_devicechange_hook_request->buffer_out + sizeof(entry) * num_devices, &entry,
sizeof(entry)); sizeof(entry));
++num_devices; ++num_devices;
} }

View File

@ -18,6 +18,7 @@
#include "Core/IOS/Device.h" #include "Core/IOS/Device.h"
#include "Core/IOS/USB/Common.h" #include "Core/IOS/USB/Common.h"
#include "Core/IOS/USB/USBV4.h" #include "Core/IOS/USB/USBV4.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -32,6 +33,9 @@ USB_HIDv4::~USB_HIDv4()
std::optional<IPCReply> USB_HIDv4::IOCtl(const IOCtlRequest& request) std::optional<IPCReply> USB_HIDv4::IOCtl(const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB); request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB);
switch (request.request) switch (request.request)
{ {
@ -53,7 +57,7 @@ std::optional<IPCReply> USB_HIDv4::IOCtl(const IOCtlRequest& request)
{ {
if (request.buffer_in == 0 || request.buffer_in_size != 32) if (request.buffer_in == 0 || request.buffer_in_size != 32)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const auto device = GetDeviceByIOSID(Memory::Read_U32(request.buffer_in + 16)); const auto device = GetDeviceByIOSID(memory.Read_U32(request.buffer_in + 16));
if (!device->Attach()) if (!device->Attach())
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
return HandleTransfer(device, request.request, return HandleTransfer(device, request.request,
@ -70,10 +74,13 @@ IPCReply USB_HIDv4::CancelInterrupt(const IOCtlRequest& request)
if (request.buffer_in == 0 || request.buffer_in_size != 8) if (request.buffer_in == 0 || request.buffer_in_size != 8)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
auto device = GetDeviceByIOSID(Memory::Read_U32(request.buffer_in)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
auto device = GetDeviceByIOSID(memory.Read_U32(request.buffer_in));
if (!device) if (!device)
return IPCReply(IPC_ENOENT); return IPCReply(IPC_ENOENT);
device->CancelTransfer(Memory::Read_U8(request.buffer_in + 4)); device->CancelTransfer(memory.Read_U8(request.buffer_in + 4));
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
} }
@ -98,7 +105,9 @@ IPCReply USB_HIDv4::Shutdown(const IOCtlRequest& request)
std::lock_guard lk{m_devicechange_hook_address_mutex}; std::lock_guard lk{m_devicechange_hook_address_mutex};
if (m_devicechange_hook_request != 0) if (m_devicechange_hook_request != 0)
{ {
Memory::Write_U32(0xffffffff, m_devicechange_hook_request->buffer_out); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.Write_U32(0xffffffff, m_devicechange_hook_request->buffer_out);
m_ios.EnqueueIPCReply(*m_devicechange_hook_request, -1); m_ios.EnqueueIPCReply(*m_devicechange_hook_request, -1);
m_devicechange_hook_request.reset(); m_devicechange_hook_request.reset();
} }
@ -182,6 +191,9 @@ void USB_HIDv4::TriggerDeviceChangeReply()
if (!m_devicechange_hook_request) if (!m_devicechange_hook_request)
return; return;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
{ {
std::lock_guard lk(m_devices_mutex); std::lock_guard lk(m_devices_mutex);
const u32 dest = m_devicechange_hook_request->buffer_out; const u32 dest = m_devicechange_hook_request->buffer_out;
@ -194,11 +206,11 @@ void USB_HIDv4::TriggerDeviceChangeReply()
WARN_LOG_FMT(IOS_USB, "Too many devices connected, skipping"); WARN_LOG_FMT(IOS_USB, "Too many devices connected, skipping");
break; break;
} }
Memory::CopyToEmu(dest + offset, device_section.data(), device_section.size()); memory.CopyToEmu(dest + offset, device_section.data(), device_section.size());
offset += Common::AlignUp(static_cast<u32>(device_section.size()), 4); offset += Common::AlignUp(static_cast<u32>(device_section.size()), 4);
} }
// IOS writes 0xffffffff to the buffer when there are no more devices // IOS writes 0xffffffff to the buffer when there are no more devices
Memory::Write_U32(0xffffffff, dest + offset); memory.Write_U32(0xffffffff, dest + offset);
} }
m_ios.EnqueueIPCReply(*m_devicechange_hook_request, IPC_SUCCESS, 0, CoreTiming::FromThread::ANY); m_ios.EnqueueIPCReply(*m_devicechange_hook_request, IPC_SUCCESS, 0, CoreTiming::FromThread::ANY);

View File

@ -12,6 +12,7 @@
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/USB/Common.h" #include "Core/IOS/USB/Common.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -24,11 +25,14 @@ USB_HIDv5::~USB_HIDv5()
std::optional<IPCReply> USB_HIDv5::IOCtl(const IOCtlRequest& request) std::optional<IPCReply> USB_HIDv5::IOCtl(const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB); request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB);
switch (request.request) switch (request.request)
{ {
case USB::IOCTL_USBV5_GETVERSION: case USB::IOCTL_USBV5_GETVERSION:
Memory::Write_U32(USBV5_VERSION, request.buffer_out); memory.Write_U32(USBV5_VERSION, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
case USB::IOCTL_USBV5_GETDEVICECHANGE: case USB::IOCTL_USBV5_GETDEVICECHANGE:
return GetDeviceChange(request); return GetDeviceChange(request);
@ -93,11 +97,14 @@ s32 USB_HIDv5::SubmitTransfer(USBV5Device& device, USB::Device& host_device,
{ {
auto message = std::make_unique<USB::V5IntrMessage>(m_ios, ioctlv); auto message = std::make_unique<USB::V5IntrMessage>(m_ios, ioctlv);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Unlike VEN, the endpoint is determined by the value at 8-12. // Unlike VEN, the endpoint is determined by the value at 8-12.
// If it's non-zero, HID submits the request to the interrupt OUT endpoint. // If it's non-zero, HID submits the request to the interrupt OUT endpoint.
// Otherwise, the request is submitted to the IN endpoint. // Otherwise, the request is submitted to the IN endpoint.
AdditionalDeviceData* data = &m_additional_device_data[&device - m_usbv5_devices.data()]; AdditionalDeviceData* data = &m_additional_device_data[&device - m_usbv5_devices.data()];
if (Memory::Read_U32(ioctlv.in_vectors[0].address + 8) != 0) if (memory.Read_U32(ioctlv.in_vectors[0].address + 8) != 0)
message->endpoint = data->interrupt_out_endpoint; message->endpoint = data->interrupt_out_endpoint;
else else
message->endpoint = data->interrupt_in_endpoint; message->endpoint = data->interrupt_in_endpoint;
@ -111,7 +118,10 @@ s32 USB_HIDv5::SubmitTransfer(USBV5Device& device, USB::Device& host_device,
IPCReply USB_HIDv5::CancelEndpoint(USBV5Device& device, const IOCtlRequest& request) IPCReply USB_HIDv5::CancelEndpoint(USBV5Device& device, const IOCtlRequest& request)
{ {
const u8 value = Memory::Read_U8(request.buffer_in + 8); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 value = memory.Read_U8(request.buffer_in + 8);
u8 endpoint = 0; u8 endpoint = 0;
switch (value) switch (value)
{ {
@ -138,21 +148,24 @@ IPCReply USB_HIDv5::GetDeviceInfo(USBV5Device& device, const IOCtlRequest& reque
if (request.buffer_out == 0 || request.buffer_out_size != 0x60) if (request.buffer_out == 0 || request.buffer_out_size != 0x60)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const std::shared_ptr<USB::Device> host_device = GetDeviceById(device.host_id); auto& system = Core::System::GetInstance();
const u8 alt_setting = Memory::Read_U8(request.buffer_in + 8); auto& memory = system.GetMemory();
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); const std::shared_ptr<USB::Device> host_device = GetDeviceById(device.host_id);
Memory::Write_U32(Memory::Read_U32(request.buffer_in), request.buffer_out); const u8 alt_setting = memory.Read_U8(request.buffer_in + 8);
Memory::Write_U32(1, request.buffer_out + 4);
memory.Memset(request.buffer_out, 0, request.buffer_out_size);
memory.Write_U32(memory.Read_U32(request.buffer_in), request.buffer_out);
memory.Write_U32(1, request.buffer_out + 4);
USB::DeviceDescriptor device_descriptor = host_device->GetDeviceDescriptor(); USB::DeviceDescriptor device_descriptor = host_device->GetDeviceDescriptor();
device_descriptor.Swap(); device_descriptor.Swap();
Memory::CopyToEmu(request.buffer_out + 36, &device_descriptor, sizeof(device_descriptor)); memory.CopyToEmu(request.buffer_out + 36, &device_descriptor, sizeof(device_descriptor));
// Just like VEN, HIDv5 only cares about the first configuration. // Just like VEN, HIDv5 only cares about the first configuration.
USB::ConfigDescriptor config_descriptor = host_device->GetConfigurations()[0]; USB::ConfigDescriptor config_descriptor = host_device->GetConfigurations()[0];
config_descriptor.Swap(); config_descriptor.Swap();
Memory::CopyToEmu(request.buffer_out + 56, &config_descriptor, sizeof(config_descriptor)); memory.CopyToEmu(request.buffer_out + 56, &config_descriptor, sizeof(config_descriptor));
std::vector<USB::InterfaceDescriptor> interfaces = host_device->GetInterfaces(0); std::vector<USB::InterfaceDescriptor> interfaces = host_device->GetInterfaces(0);
auto it = std::find_if(interfaces.begin(), interfaces.end(), auto it = std::find_if(interfaces.begin(), interfaces.end(),
@ -163,7 +176,7 @@ IPCReply USB_HIDv5::GetDeviceInfo(USBV5Device& device, const IOCtlRequest& reque
if (it == interfaces.end()) if (it == interfaces.end())
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
it->Swap(); it->Swap();
Memory::CopyToEmu(request.buffer_out + 68, &*it, sizeof(*it)); memory.CopyToEmu(request.buffer_out + 68, &*it, sizeof(*it));
auto endpoints = host_device->GetEndpoints(0, it->bInterfaceNumber, it->bAlternateSetting); auto endpoints = host_device->GetEndpoints(0, it->bInterfaceNumber, it->bAlternateSetting);
for (auto& endpoint : endpoints) for (auto& endpoint : endpoints)
@ -182,7 +195,7 @@ IPCReply USB_HIDv5::GetDeviceInfo(USBV5Device& device, const IOCtlRequest& reque
const u32 offset = is_in_endpoint ? 80 : 88; const u32 offset = is_in_endpoint ? 80 : 88;
endpoint.Swap(); endpoint.Swap();
Memory::CopyToEmu(request.buffer_out + offset, &endpoint, sizeof(endpoint)); memory.CopyToEmu(request.buffer_out + offset, &endpoint, sizeof(endpoint));
} }
} }

View File

@ -13,6 +13,7 @@
#include "Core/Config/MainSettings.h" #include "Core/Config/MainSettings.h"
#include "Core/Core.h" // Local core functions #include "Core/Core.h" // Local core functions
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "InputCommon/ControlReference/ControlReference.h" // For background input check #include "InputCommon/ControlReference/ControlReference.h" // For background input check
#ifdef _WIN32 #ifdef _WIN32
@ -213,7 +214,9 @@ std::optional<IPCReply> USB_KBD::IOCtl(const IOCtlRequest& request)
if (Config::Get(Config::MAIN_WII_KEYBOARD) && !Core::WantsDeterminism() && if (Config::Get(Config::MAIN_WII_KEYBOARD) && !Core::WantsDeterminism() &&
ControlReference::GetInputGate() && !m_message_queue.empty()) ControlReference::GetInputGate() && !m_message_queue.empty())
{ {
Memory::CopyToEmu(request.buffer_out, &m_message_queue.front(), sizeof(MessageData)); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyToEmu(request.buffer_out, &m_message_queue.front(), sizeof(MessageData));
m_message_queue.pop(); m_message_queue.pop();
} }
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);

View File

@ -12,6 +12,7 @@
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/USB/Common.h" #include "Core/IOS/USB/Common.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -24,11 +25,14 @@ USB_VEN::~USB_VEN()
std::optional<IPCReply> USB_VEN::IOCtl(const IOCtlRequest& request) std::optional<IPCReply> USB_VEN::IOCtl(const IOCtlRequest& request)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB); request.Log(GetDeviceName(), Common::Log::LogType::IOS_USB);
switch (request.request) switch (request.request)
{ {
case USB::IOCTL_USBV5_GETVERSION: case USB::IOCTL_USBV5_GETVERSION:
Memory::Write_U32(USBV5_VERSION, request.buffer_out); memory.Write_U32(USBV5_VERSION, request.buffer_out);
return IPCReply(IPC_SUCCESS); return IPCReply(IPC_SUCCESS);
case USB::IOCTL_USBV5_GETDEVICECHANGE: case USB::IOCTL_USBV5_GETDEVICECHANGE:
return GetDeviceChange(request); return GetDeviceChange(request);
@ -110,7 +114,10 @@ s32 USB_VEN::SubmitTransfer(USB::Device& device, const IOCtlVRequest& ioctlv)
IPCReply USB_VEN::CancelEndpoint(USBV5Device& device, const IOCtlRequest& request) IPCReply USB_VEN::CancelEndpoint(USBV5Device& device, const IOCtlRequest& request)
{ {
const u8 endpoint = Memory::Read_U8(request.buffer_in + 8); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8 endpoint = memory.Read_U8(request.buffer_in + 8);
// IPC_EINVAL (-4) is returned when no transfer was cancelled. // IPC_EINVAL (-4) is returned when no transfer was cancelled.
if (GetDeviceById(device.host_id)->CancelTransfer(endpoint) < 0) if (GetDeviceById(device.host_id)->CancelTransfer(endpoint) < 0)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
@ -122,21 +129,24 @@ IPCReply USB_VEN::GetDeviceInfo(USBV5Device& device, const IOCtlRequest& request
if (request.buffer_out == 0 || request.buffer_out_size != 0xc0) if (request.buffer_out == 0 || request.buffer_out_size != 0xc0)
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
const std::shared_ptr<USB::Device> host_device = GetDeviceById(device.host_id); auto& system = Core::System::GetInstance();
const u8 alt_setting = Memory::Read_U8(request.buffer_in + 8); auto& memory = system.GetMemory();
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); const std::shared_ptr<USB::Device> host_device = GetDeviceById(device.host_id);
Memory::Write_U32(Memory::Read_U32(request.buffer_in), request.buffer_out); const u8 alt_setting = memory.Read_U8(request.buffer_in + 8);
Memory::Write_U32(1, request.buffer_out + 4);
memory.Memset(request.buffer_out, 0, request.buffer_out_size);
memory.Write_U32(memory.Read_U32(request.buffer_in), request.buffer_out);
memory.Write_U32(1, request.buffer_out + 4);
USB::DeviceDescriptor device_descriptor = host_device->GetDeviceDescriptor(); USB::DeviceDescriptor device_descriptor = host_device->GetDeviceDescriptor();
device_descriptor.Swap(); device_descriptor.Swap();
Memory::CopyToEmu(request.buffer_out + 20, &device_descriptor, sizeof(device_descriptor)); memory.CopyToEmu(request.buffer_out + 20, &device_descriptor, sizeof(device_descriptor));
// VEN only cares about the first configuration. // VEN only cares about the first configuration.
USB::ConfigDescriptor config_descriptor = host_device->GetConfigurations()[0]; USB::ConfigDescriptor config_descriptor = host_device->GetConfigurations()[0];
config_descriptor.Swap(); config_descriptor.Swap();
Memory::CopyToEmu(request.buffer_out + 40, &config_descriptor, sizeof(config_descriptor)); memory.CopyToEmu(request.buffer_out + 40, &config_descriptor, sizeof(config_descriptor));
std::vector<USB::InterfaceDescriptor> interfaces = host_device->GetInterfaces(0); std::vector<USB::InterfaceDescriptor> interfaces = host_device->GetInterfaces(0);
auto it = std::find_if(interfaces.begin(), interfaces.end(), auto it = std::find_if(interfaces.begin(), interfaces.end(),
@ -147,13 +157,13 @@ IPCReply USB_VEN::GetDeviceInfo(USBV5Device& device, const IOCtlRequest& request
if (it == interfaces.end()) if (it == interfaces.end())
return IPCReply(IPC_EINVAL); return IPCReply(IPC_EINVAL);
it->Swap(); it->Swap();
Memory::CopyToEmu(request.buffer_out + 52, &*it, sizeof(*it)); memory.CopyToEmu(request.buffer_out + 52, &*it, sizeof(*it));
auto endpoints = host_device->GetEndpoints(0, it->bInterfaceNumber, it->bAlternateSetting); auto endpoints = host_device->GetEndpoints(0, it->bInterfaceNumber, it->bAlternateSetting);
for (size_t i = 0; i < endpoints.size(); ++i) for (size_t i = 0; i < endpoints.size(); ++i)
{ {
endpoints[i].Swap(); endpoints[i].Swap();
Memory::CopyToEmu(request.buffer_out + 64 + 8 * static_cast<u8>(i), &endpoints[i], memory.CopyToEmu(request.buffer_out + 64 + 8 * static_cast<u8>(i), &endpoints[i],
sizeof(endpoints[i])); sizeof(endpoints[i]));
} }

View File

@ -20,6 +20,7 @@
#include "Core/IOS/ES/Formats.h" #include "Core/IOS/ES/Formats.h"
#include "Core/IOS/IOS.h" #include "Core/IOS/IOS.h"
#include "Core/IOS/WFS/WFSSRV.h" #include "Core/IOS/WFS/WFSSRV.h"
#include "Core/System.h"
namespace namespace
{ {
@ -127,15 +128,18 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
{ {
s32 return_error_code = IPC_SUCCESS; s32 return_error_code = IPC_SUCCESS;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (request.request) switch (request.request)
{ {
case IOCTL_WFSI_IMPORT_TITLE_INIT: case IOCTL_WFSI_IMPORT_TITLE_INIT:
{ {
const u32 tmd_addr = Memory::Read_U32(request.buffer_in); const u32 tmd_addr = memory.Read_U32(request.buffer_in);
const u32 tmd_size = Memory::Read_U32(request.buffer_in + 4); const u32 tmd_size = memory.Read_U32(request.buffer_in + 4);
m_patch_type = static_cast<PatchType>(Memory::Read_U32(request.buffer_in + 32)); m_patch_type = static_cast<PatchType>(memory.Read_U32(request.buffer_in + 32));
m_continue_install = Memory::Read_U32(request.buffer_in + 36); m_continue_install = memory.Read_U32(request.buffer_in + 36);
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_IMPORT_TITLE_INIT: patch type {}, continue install: {}", INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_IMPORT_TITLE_INIT: patch type {}, continue install: {}",
static_cast<u32>(m_patch_type), m_continue_install ? "true" : "false"); static_cast<u32>(m_patch_type), m_continue_install ? "true" : "false");
@ -157,7 +161,7 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
} }
std::vector<u8> tmd_bytes; std::vector<u8> tmd_bytes;
tmd_bytes.resize(tmd_size); tmd_bytes.resize(tmd_size);
Memory::CopyFromEmu(tmd_bytes.data(), tmd_addr, tmd_size); memory.CopyFromEmu(tmd_bytes.data(), tmd_addr, tmd_size);
m_tmd.SetBytes(std::move(tmd_bytes)); m_tmd.SetBytes(std::move(tmd_bytes));
const ES::TicketReader ticket = m_ios.GetES()->FindSignedTicket(m_tmd.GetTitleId()); const ES::TicketReader ticket = m_ios.GetES()->FindSignedTicket(m_tmd.GetTitleId());
@ -190,7 +194,7 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
"IOCTL_WFSI_PREPARE_CONTENT"; "IOCTL_WFSI_PREPARE_CONTENT";
// Initializes the IV from the index of the content in the TMD contents. // Initializes the IV from the index of the content in the TMD contents.
const u32 content_id = Memory::Read_U32(request.buffer_in + 8); const u32 content_id = memory.Read_U32(request.buffer_in + 8);
ES::Content content_info; ES::Content content_info;
if (!m_tmd.FindContentById(content_id, &content_info)) if (!m_tmd.FindContentById(content_id, &content_info))
{ {
@ -216,14 +220,14 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
"IOCTL_WFSI_IMPORT_PROFILE" : "IOCTL_WFSI_IMPORT_PROFILE" :
"IOCTL_WFSI_IMPORT_CONTENT"; "IOCTL_WFSI_IMPORT_CONTENT";
const u32 content_id = Memory::Read_U32(request.buffer_in + 0xC); const u32 content_id = memory.Read_U32(request.buffer_in + 0xC);
const u32 input_ptr = Memory::Read_U32(request.buffer_in + 0x10); const u32 input_ptr = memory.Read_U32(request.buffer_in + 0x10);
const u32 input_size = Memory::Read_U32(request.buffer_in + 0x14); const u32 input_size = memory.Read_U32(request.buffer_in + 0x14);
INFO_LOG_FMT(IOS_WFS, "{}: {:08x} bytes of data at {:08x} from content id {}", ioctl_name, INFO_LOG_FMT(IOS_WFS, "{}: {:08x} bytes of data at {:08x} from content id {}", ioctl_name,
input_size, input_ptr, content_id); input_size, input_ptr, content_id);
std::vector<u8> decrypted(input_size); std::vector<u8> decrypted(input_size);
m_aes_ctx->Crypt(m_aes_iv, m_aes_iv, Memory::GetPointer(input_ptr), decrypted.data(), m_aes_ctx->Crypt(m_aes_iv, m_aes_iv, memory.GetPointer(input_ptr), decrypted.data(),
input_size); input_size);
m_arc_unpacker.AddBytes(decrypted); m_arc_unpacker.AddBytes(decrypted);
@ -331,8 +335,8 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFSI_CHANGE_TITLE: case IOCTL_WFSI_CHANGE_TITLE:
{ {
const u64 title_id = Memory::Read_U64(request.buffer_in); const u64 title_id = memory.Read_U64(request.buffer_in);
const u16 group_id = Memory::Read_U16(request.buffer_in + 0x1C); const u16 group_id = memory.Read_U16(request.buffer_in + 0x1C);
// TODO: Handle permissions // TODO: Handle permissions
SetCurrentTitleIdAndGroupId(title_id, group_id); SetCurrentTitleIdAndGroupId(title_id, group_id);
@ -357,14 +361,14 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFSI_GET_VERSION: case IOCTL_WFSI_GET_VERSION:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_GET_VERSION"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_GET_VERSION");
Memory::Write_U32(0x20, request.buffer_out); memory.Write_U32(0x20, request.buffer_out);
break; break;
case IOCTL_WFSI_IMPORT_TITLE_CANCEL: case IOCTL_WFSI_IMPORT_TITLE_CANCEL:
{ {
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_IMPORT_TITLE_CANCEL"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_IMPORT_TITLE_CANCEL");
const bool continue_install = Memory::Read_U32(request.buffer_in) != 0; const bool continue_install = memory.Read_U32(request.buffer_in) != 0;
if (m_patch_type == PatchType::NOT_A_PATCH) if (m_patch_type == PatchType::NOT_A_PATCH)
return_error_code = CancelTitleImport(continue_install); return_error_code = CancelTitleImport(continue_install);
else if (m_patch_type == PatchType::PATCH_TYPE_1 || m_patch_type == PatchType::PATCH_TYPE_2) else if (m_patch_type == PatchType::PATCH_TYPE_1 || m_patch_type == PatchType::PATCH_TYPE_2)
@ -394,7 +398,7 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFSI_SET_DEVICE_NAME: case IOCTL_WFSI_SET_DEVICE_NAME:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_SET_DEVICE_NAME"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_SET_DEVICE_NAME");
m_device_name = Memory::GetString(request.buffer_in); m_device_name = memory.GetString(request.buffer_in);
break; break;
case IOCTL_WFSI_APPLY_TITLE_PROFILE: case IOCTL_WFSI_APPLY_TITLE_PROFILE:
@ -438,30 +442,30 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFSI_GET_TMD: case IOCTL_WFSI_GET_TMD:
{ {
const u64 subtitle_id = Memory::Read_U64(request.buffer_in); const u64 subtitle_id = memory.Read_U64(request.buffer_in);
const u32 address = Memory::Read_U32(request.buffer_in + 24); const u32 address = memory.Read_U32(request.buffer_in + 24);
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_GET_TMD: subtitle ID {:016x}", subtitle_id); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFSI_GET_TMD: subtitle ID {:016x}", subtitle_id);
u32 tmd_size; u32 tmd_size;
return_error_code = return_error_code =
GetTmd(m_current_group_id, m_current_title_id, subtitle_id, address, &tmd_size); GetTmd(m_current_group_id, m_current_title_id, subtitle_id, address, &tmd_size);
Memory::Write_U32(tmd_size, request.buffer_out); memory.Write_U32(tmd_size, request.buffer_out);
break; break;
} }
case IOCTL_WFSI_GET_TMD_ABSOLUTE: case IOCTL_WFSI_GET_TMD_ABSOLUTE:
{ {
const u64 subtitle_id = Memory::Read_U64(request.buffer_in); const u64 subtitle_id = memory.Read_U64(request.buffer_in);
const u32 address = Memory::Read_U32(request.buffer_in + 24); const u32 address = memory.Read_U32(request.buffer_in + 24);
const u16 group_id = Memory::Read_U16(request.buffer_in + 36); const u16 group_id = memory.Read_U16(request.buffer_in + 36);
const u32 title_id = Memory::Read_U32(request.buffer_in + 32); const u32 title_id = memory.Read_U32(request.buffer_in + 32);
INFO_LOG_FMT(IOS_WFS, INFO_LOG_FMT(IOS_WFS,
"IOCTL_WFSI_GET_TMD_ABSOLUTE: tid {:08x}, gid {:04x}, subtitle ID {:016x}", "IOCTL_WFSI_GET_TMD_ABSOLUTE: tid {:08x}, gid {:04x}, subtitle ID {:016x}",
title_id, group_id, subtitle_id); title_id, group_id, subtitle_id);
u32 tmd_size; u32 tmd_size;
return_error_code = GetTmd(group_id, title_id, subtitle_id, address, &tmd_size); return_error_code = GetTmd(group_id, title_id, subtitle_id, address, &tmd_size);
Memory::Write_U32(tmd_size, request.buffer_out); memory.Write_U32(tmd_size, request.buffer_out);
break; break;
} }
@ -480,9 +484,9 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
std::string path = fmt::format("/vol/{}/title/{}/{}/content", m_device_name, std::string path = fmt::format("/vol/{}/title/{}/{}/content", m_device_name,
m_current_group_id_str, m_current_title_id_str); m_current_group_id_str, m_current_title_id_str);
const u32 dol_addr = Memory::Read_U32(request.buffer_in + 0x18); const u32 dol_addr = memory.Read_U32(request.buffer_in + 0x18);
const u32 max_dol_size = Memory::Read_U32(request.buffer_in + 0x14); const u32 max_dol_size = memory.Read_U32(request.buffer_in + 0x14);
const u16 dol_extension_id = Memory::Read_U16(request.buffer_in + 0x1e); const u16 dol_extension_id = memory.Read_U16(request.buffer_in + 0x1e);
if (dol_extension_id == 0) if (dol_extension_id == 0)
{ {
@ -508,13 +512,13 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
if (dol_addr == 0) if (dol_addr == 0)
{ {
// Write the expected size to the size parameter, in the input. // Write the expected size to the size parameter, in the input.
Memory::Write_U32(real_dol_size, request.buffer_in + 0x14); memory.Write_U32(real_dol_size, request.buffer_in + 0x14);
} }
else else
{ {
fp.ReadBytes(Memory::GetPointer(dol_addr), max_dol_size); fp.ReadBytes(memory.GetPointer(dol_addr), max_dol_size);
} }
Memory::Write_U32(real_dol_size, request.buffer_out); memory.Write_U32(real_dol_size, request.buffer_out);
break; break;
} }
@ -541,7 +545,7 @@ std::optional<IPCReply> WFSIDevice::IOCtl(const IOCtlRequest& request)
// succeeding. // succeeding.
request.DumpUnknown(GetDeviceName(), Common::Log::LogType::IOS_WFS, request.DumpUnknown(GetDeviceName(), Common::Log::LogType::IOS_WFS,
Common::Log::LogLevel::LWARNING); Common::Log::LogLevel::LWARNING);
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); memory.Memset(request.buffer_out, 0, request.buffer_out_size);
break; break;
} }
@ -560,7 +564,9 @@ u32 WFSIDevice::GetTmd(u16 group_id, u32 title_id, u64 subtitle_id, u32 address,
} }
if (address) if (address)
{ {
fp.ReadBytes(Memory::GetPointer(address), fp.GetSize()); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
fp.ReadBytes(memory.GetPointer(address), fp.GetSize());
} }
*size = fp.GetSize(); *size = fp.GetSize();
return IPC_SUCCESS; return IPC_SUCCESS;

View File

@ -13,6 +13,7 @@
#include "Common/Logging/Log.h" #include "Common/Logging/Log.h"
#include "Common/NandPaths.h" #include "Common/NandPaths.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
namespace IOS::HLE namespace IOS::HLE
{ {
@ -33,6 +34,9 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
{ {
int return_error_code = IPC_SUCCESS; int return_error_code = IPC_SUCCESS;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (request.request) switch (request.request)
{ {
case IOCTL_WFS_INIT: case IOCTL_WFS_INIT:
@ -43,8 +47,8 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_UNKNOWN_8: case IOCTL_WFS_UNKNOWN_8:
// TODO(wfs): Figure out what this actually does. // TODO(wfs): Figure out what this actually does.
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_UNKNOWN_8"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_UNKNOWN_8");
Memory::Write_U8(7, request.buffer_out); memory.Write_U8(7, request.buffer_out);
Memory::CopyToEmu(request.buffer_out + 1, "msc01\x00\x00\x00", 8); memory.CopyToEmu(request.buffer_out + 1, "msc01\x00\x00\x00", 8);
break; break;
case IOCTL_WFS_SHUTDOWN: case IOCTL_WFS_SHUTDOWN:
@ -54,33 +58,33 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
for (auto address : m_hanging) for (auto address : m_hanging)
{ {
IOCtlRequest hanging_request{address}; IOCtlRequest hanging_request{address};
Memory::Write_U32(0x80000000, hanging_request.buffer_out); memory.Write_U32(0x80000000, hanging_request.buffer_out);
Memory::Write_U32(0, hanging_request.buffer_out + 4); memory.Write_U32(0, hanging_request.buffer_out + 4);
Memory::Write_U32(0, hanging_request.buffer_out + 8); memory.Write_U32(0, hanging_request.buffer_out + 8);
m_ios.EnqueueIPCReply(hanging_request, 0); m_ios.EnqueueIPCReply(hanging_request, 0);
} }
break; break;
case IOCTL_WFS_DEVICE_INFO: case IOCTL_WFS_DEVICE_INFO:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_DEVICE_INFO"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_DEVICE_INFO");
Memory::Write_U64(16ull << 30, request.buffer_out); // 16GB storage. memory.Write_U64(16ull << 30, request.buffer_out); // 16GB storage.
Memory::Write_U8(4, request.buffer_out + 8); memory.Write_U8(4, request.buffer_out + 8);
break; break;
case IOCTL_WFS_GET_DEVICE_NAME: case IOCTL_WFS_GET_DEVICE_NAME:
{ {
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_DEVICE_NAME"); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_DEVICE_NAME");
Memory::Write_U8(static_cast<u8>(m_device_name.size()), request.buffer_out); memory.Write_U8(static_cast<u8>(m_device_name.size()), request.buffer_out);
Memory::CopyToEmu(request.buffer_out + 1, m_device_name.data(), m_device_name.size()); memory.CopyToEmu(request.buffer_out + 1, m_device_name.data(), m_device_name.size());
break; break;
} }
case IOCTL_WFS_ATTACH_DETACH_2: case IOCTL_WFS_ATTACH_DETACH_2:
// TODO(wfs): Implement. // TODO(wfs): Implement.
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_ATTACH_DETACH_2({})", request.request); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_ATTACH_DETACH_2({})", request.request);
Memory::Write_U32(1, request.buffer_out); memory.Write_U32(1, request.buffer_out);
Memory::Write_U32(0, request.buffer_out + 4); // device id? memory.Write_U32(0, request.buffer_out + 4); // device id?
Memory::Write_U32(0, request.buffer_out + 8); memory.Write_U32(0, request.buffer_out + 8);
break; break;
case IOCTL_WFS_ATTACH_DETACH: case IOCTL_WFS_ATTACH_DETACH:
@ -98,7 +102,7 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_MKDIR: case IOCTL_WFS_MKDIR:
{ {
const std::string path = NormalizePath( const std::string path = NormalizePath(
Memory::GetString(request.buffer_in + 34, Memory::Read_U16(request.buffer_in + 32))); memory.GetString(request.buffer_in + 34, memory.Read_U16(request.buffer_in + 32)));
const std::string native_path = WFS::NativePath(path); const std::string native_path = WFS::NativePath(path);
if (File::Exists(native_path)) if (File::Exists(native_path))
@ -122,8 +126,8 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
// (listing /vol/*) which is required to get the installer to work. // (listing /vol/*) which is required to get the installer to work.
case IOCTL_WFS_GLOB_START: case IOCTL_WFS_GLOB_START:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GLOB_START({})", request.request); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GLOB_START({})", request.request);
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); memory.Memset(request.buffer_out, 0, request.buffer_out_size);
Memory::CopyToEmu(request.buffer_out + 0x14, m_device_name.data(), m_device_name.size()); memory.CopyToEmu(request.buffer_out + 0x14, m_device_name.data(), m_device_name.size());
break; break;
case IOCTL_WFS_GLOB_NEXT: case IOCTL_WFS_GLOB_NEXT:
@ -133,34 +137,33 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_GLOB_END: case IOCTL_WFS_GLOB_END:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GLOB_END({})", request.request); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GLOB_END({})", request.request);
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); memory.Memset(request.buffer_out, 0, request.buffer_out_size);
break; break;
case IOCTL_WFS_SET_HOMEDIR: case IOCTL_WFS_SET_HOMEDIR:
m_home_directory = m_home_directory = memory.GetString(request.buffer_in + 2, memory.Read_U16(request.buffer_in));
Memory::GetString(request.buffer_in + 2, Memory::Read_U16(request.buffer_in));
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_SET_HOMEDIR: {}", m_home_directory); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_SET_HOMEDIR: {}", m_home_directory);
break; break;
case IOCTL_WFS_CHDIR: case IOCTL_WFS_CHDIR:
m_current_directory = m_current_directory =
Memory::GetString(request.buffer_in + 2, Memory::Read_U16(request.buffer_in)); memory.GetString(request.buffer_in + 2, memory.Read_U16(request.buffer_in));
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CHDIR: {}", m_current_directory); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CHDIR: {}", m_current_directory);
break; break;
case IOCTL_WFS_GET_HOMEDIR: case IOCTL_WFS_GET_HOMEDIR:
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_HOMEDIR: {}", m_home_directory); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_HOMEDIR: {}", m_home_directory);
Memory::Write_U16(static_cast<u16>(m_home_directory.size()), request.buffer_out); memory.Write_U16(static_cast<u16>(m_home_directory.size()), request.buffer_out);
Memory::CopyToEmu(request.buffer_out + 2, m_home_directory.data(), m_home_directory.size()); memory.CopyToEmu(request.buffer_out + 2, m_home_directory.data(), m_home_directory.size());
break; break;
case IOCTL_WFS_GET_ATTRIBUTES: case IOCTL_WFS_GET_ATTRIBUTES:
{ {
const std::string path = NormalizePath( const std::string path =
Memory::GetString(request.buffer_in + 2, Memory::Read_U16(request.buffer_in))); NormalizePath(memory.GetString(request.buffer_in + 2, memory.Read_U16(request.buffer_in)));
const std::string native_path = WFS::NativePath(path); const std::string native_path = WFS::NativePath(path);
Memory::Memset(0, request.buffer_out, request.buffer_out_size); memory.Memset(0, request.buffer_out, request.buffer_out_size);
if (!File::Exists(native_path)) if (!File::Exists(native_path))
{ {
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): no such file or directory", path); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): no such file or directory", path);
@ -169,13 +172,13 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
else if (File::IsDirectory(native_path)) else if (File::IsDirectory(native_path))
{ {
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): directory", path); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): directory", path);
Memory::Write_U32(0x80000000, request.buffer_out + 4); memory.Write_U32(0x80000000, request.buffer_out + 4);
} }
else else
{ {
const auto size = static_cast<u32>(File::GetSize(native_path)); const auto size = static_cast<u32>(File::GetSize(native_path));
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): file with size {}", path, size); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_ATTRIBUTES({}): file with size {}", path, size);
Memory::Write_U32(size, request.buffer_out); memory.Write_U32(size, request.buffer_out);
} }
break; break;
} }
@ -184,9 +187,9 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_RENAME_2: case IOCTL_WFS_RENAME_2:
{ {
const std::string source_path = const std::string source_path =
Memory::GetString(request.buffer_in + 2, Memory::Read_U16(request.buffer_in)); memory.GetString(request.buffer_in + 2, memory.Read_U16(request.buffer_in));
const std::string dest_path = const std::string dest_path =
Memory::GetString(request.buffer_in + 512 + 2, Memory::Read_U16(request.buffer_in + 512)); memory.GetString(request.buffer_in + 512 + 2, memory.Read_U16(request.buffer_in + 512));
return_error_code = Rename(source_path, dest_path); return_error_code = Rename(source_path, dest_path);
break; break;
} }
@ -196,9 +199,9 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
{ {
const char* ioctl_name = const char* ioctl_name =
request.request == IOCTL_WFS_OPEN ? "IOCTL_WFS_OPEN" : "IOCTL_WFS_CREATE_OPEN"; request.request == IOCTL_WFS_OPEN ? "IOCTL_WFS_OPEN" : "IOCTL_WFS_CREATE_OPEN";
const u32 mode = request.request == IOCTL_WFS_OPEN ? Memory::Read_U32(request.buffer_in) : 2; const u32 mode = request.request == IOCTL_WFS_OPEN ? memory.Read_U32(request.buffer_in) : 2;
const u16 path_len = Memory::Read_U16(request.buffer_in + 0x20); const u16 path_len = memory.Read_U16(request.buffer_in + 0x20);
std::string path = Memory::GetString(request.buffer_in + 0x22, path_len); std::string path = memory.GetString(request.buffer_in + 0x22, path_len);
path = NormalizePath(path); path = NormalizePath(path);
@ -220,18 +223,18 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
INFO_LOG_FMT(IOS_WFS, "{}({}, {}) -> {}", ioctl_name, path, mode, fd); INFO_LOG_FMT(IOS_WFS, "{}({}, {}) -> {}", ioctl_name, path, mode, fd);
if (request.request == IOCTL_WFS_OPEN) if (request.request == IOCTL_WFS_OPEN)
{ {
Memory::Write_U16(fd, request.buffer_out + 0x14); memory.Write_U16(fd, request.buffer_out + 0x14);
} }
else else
{ {
Memory::Write_U16(fd, request.buffer_out); memory.Write_U16(fd, request.buffer_out);
} }
break; break;
} }
case IOCTL_WFS_GET_SIZE: case IOCTL_WFS_GET_SIZE:
{ {
const u16 fd = Memory::Read_U16(request.buffer_in); const u16 fd = memory.Read_U16(request.buffer_in);
FileDescriptor* fd_obj = FindFileDescriptor(fd); FileDescriptor* fd_obj = FindFileDescriptor(fd);
if (fd_obj == nullptr) if (fd_obj == nullptr)
{ {
@ -247,13 +250,13 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
{ {
ERROR_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_SIZE: file {} too large ({})", fd, size); ERROR_LOG_FMT(IOS_WFS, "IOCTL_WFS_GET_SIZE: file {} too large ({})", fd, size);
} }
Memory::Write_U32(truncated_size, request.buffer_out); memory.Write_U32(truncated_size, request.buffer_out);
break; break;
} }
case IOCTL_WFS_CLOSE: case IOCTL_WFS_CLOSE:
{ {
const u16 fd = Memory::Read_U16(request.buffer_in + 0x4); const u16 fd = memory.Read_U16(request.buffer_in + 0x4);
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CLOSE({})", fd); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CLOSE({})", fd);
ReleaseFileDescriptor(fd); ReleaseFileDescriptor(fd);
break; break;
@ -263,7 +266,7 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
{ {
// TODO(wfs): Figure out the exact semantics difference from the other // TODO(wfs): Figure out the exact semantics difference from the other
// close. // close.
const u16 fd = Memory::Read_U16(request.buffer_in + 0x4); const u16 fd = memory.Read_U16(request.buffer_in + 0x4);
INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CLOSE_2({})", fd); INFO_LOG_FMT(IOS_WFS, "IOCTL_WFS_CLOSE_2({})", fd);
ReleaseFileDescriptor(fd); ReleaseFileDescriptor(fd);
break; break;
@ -272,10 +275,10 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_READ: case IOCTL_WFS_READ:
case IOCTL_WFS_READ_ABSOLUTE: case IOCTL_WFS_READ_ABSOLUTE:
{ {
const u32 addr = Memory::Read_U32(request.buffer_in); const u32 addr = memory.Read_U32(request.buffer_in);
const u32 position = Memory::Read_U32(request.buffer_in + 4); // Only for absolute. const u32 position = memory.Read_U32(request.buffer_in + 4); // Only for absolute.
const u16 fd = Memory::Read_U16(request.buffer_in + 0xC); const u16 fd = memory.Read_U16(request.buffer_in + 0xC);
const u32 size = Memory::Read_U32(request.buffer_in + 8); const u32 size = memory.Read_U32(request.buffer_in + 8);
const bool absolute = request.request == IOCTL_WFS_READ_ABSOLUTE; const bool absolute = request.request == IOCTL_WFS_READ_ABSOLUTE;
@ -293,7 +296,7 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
fd_obj->file.Seek(position, File::SeekOrigin::Begin); fd_obj->file.Seek(position, File::SeekOrigin::Begin);
} }
size_t read_bytes; size_t read_bytes;
fd_obj->file.ReadArray(Memory::GetPointer(addr), size, &read_bytes); fd_obj->file.ReadArray(memory.GetPointer(addr), size, &read_bytes);
// TODO(wfs): Handle read errors. // TODO(wfs): Handle read errors.
if (absolute) if (absolute)
{ {
@ -313,10 +316,10 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
case IOCTL_WFS_WRITE: case IOCTL_WFS_WRITE:
case IOCTL_WFS_WRITE_ABSOLUTE: case IOCTL_WFS_WRITE_ABSOLUTE:
{ {
const u32 addr = Memory::Read_U32(request.buffer_in); const u32 addr = memory.Read_U32(request.buffer_in);
const u32 position = Memory::Read_U32(request.buffer_in + 4); // Only for absolute. const u32 position = memory.Read_U32(request.buffer_in + 4); // Only for absolute.
const u16 fd = Memory::Read_U16(request.buffer_in + 0xC); const u16 fd = memory.Read_U16(request.buffer_in + 0xC);
const u32 size = Memory::Read_U32(request.buffer_in + 8); const u32 size = memory.Read_U32(request.buffer_in + 8);
const bool absolute = request.request == IOCTL_WFS_WRITE_ABSOLUTE; const bool absolute = request.request == IOCTL_WFS_WRITE_ABSOLUTE;
@ -333,7 +336,7 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
{ {
fd_obj->file.Seek(position, File::SeekOrigin::Begin); fd_obj->file.Seek(position, File::SeekOrigin::Begin);
} }
fd_obj->file.WriteArray(Memory::GetPointer(addr), size); fd_obj->file.WriteArray(memory.GetPointer(addr), size);
// TODO(wfs): Handle write errors. // TODO(wfs): Handle write errors.
if (absolute) if (absolute)
{ {
@ -354,7 +357,7 @@ std::optional<IPCReply> WFSSRVDevice::IOCtl(const IOCtlRequest& request)
// properly stubbed it's easier to simulate the methods succeeding. // properly stubbed it's easier to simulate the methods succeeding.
request.DumpUnknown(GetDeviceName(), Common::Log::LogType::IOS_WFS, request.DumpUnknown(GetDeviceName(), Common::Log::LogType::IOS_WFS,
Common::Log::LogLevel::LWARNING); Common::Log::LogLevel::LWARNING);
Memory::Memset(request.buffer_out, 0, request.buffer_out_size); memory.Memset(request.buffer_out, 0, request.buffer_out_size);
break; break;
} }

View File

@ -809,7 +809,10 @@ static void ReadMemory()
if (!PowerPC::HostIsRAMAddress(addr)) if (!PowerPC::HostIsRAMAddress(addr))
return SendReply("E00"); return SendReply("E00");
u8* data = Memory::GetPointer(addr);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* data = memory.GetPointer(addr);
Mem2hex(reply, data, len); Mem2hex(reply, data, len);
reply[len * 2] = '\0'; reply[len * 2] = '\0';
SendReply((char*)reply); SendReply((char*)reply);
@ -833,7 +836,10 @@ static void WriteMemory()
if (!PowerPC::HostIsRAMAddress(addr)) if (!PowerPC::HostIsRAMAddress(addr))
return SendReply("E00"); return SendReply("E00");
u8* dst = Memory::GetPointer(addr);
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* dst = memory.GetPointer(addr);
Hex2mem(dst, s_cmd_bfr + i + 1, len); Hex2mem(dst, s_cmd_bfr + i + 1, len);
SendReply("OK"); SendReply("OK");
} }

View File

@ -224,12 +224,15 @@ bool Jit64::HandleFault(uintptr_t access_address, SContext* ctx)
// Only instructions that access I/O will get these, and there won't be that // Only instructions that access I/O will get these, and there won't be that
// many of them in a typical program/game. // many of them in a typical program/game.
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// TODO: do we properly handle off-the-end? // TODO: do we properly handle off-the-end?
const auto base_ptr = reinterpret_cast<uintptr_t>(Memory::physical_base); const auto base_ptr = reinterpret_cast<uintptr_t>(memory.GetPhysicalBase());
if (access_address >= base_ptr && access_address < base_ptr + 0x100010000) if (access_address >= base_ptr && access_address < base_ptr + 0x100010000)
return BackPatch(static_cast<u32>(access_address - base_ptr), ctx); return BackPatch(static_cast<u32>(access_address - base_ptr), ctx);
const auto logical_base_ptr = reinterpret_cast<uintptr_t>(Memory::logical_base); const auto logical_base_ptr = reinterpret_cast<uintptr_t>(memory.GetLogicalBase());
if (access_address >= logical_base_ptr && access_address < logical_base_ptr + 0x100010000) if (access_address >= logical_base_ptr && access_address < logical_base_ptr + 0x100010000)
return BackPatch(static_cast<u32>(access_address - logical_base_ptr), ctx); return BackPatch(static_cast<u32>(access_address - logical_base_ptr), ctx);
@ -330,7 +333,10 @@ void Jit64::Init()
{ {
EnableBlockLink(); EnableBlockLink();
jo.fastmem_arena = m_fastmem_enabled && Memory::InitFastmemArena(); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
jo.fastmem_arena = m_fastmem_enabled && memory.InitFastmemArena();
jo.optimizeGatherPipe = true; jo.optimizeGatherPipe = true;
jo.accurateSinglePrecision = true; jo.accurateSinglePrecision = true;
UpdateMemoryAndExceptionOptions(); UpdateMemoryAndExceptionOptions();
@ -404,7 +410,9 @@ void Jit64::Shutdown()
FreeStack(); FreeStack();
FreeCodeSpace(); FreeCodeSpace();
Memory::ShutdownFastmemArena(); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.ShutdownFastmemArena();
blocks.Shutdown(); blocks.Shutdown();
m_far_code.Shutdown(); m_far_code.Shutdown();

View File

@ -16,6 +16,7 @@
#include "Core/PowerPC/Jit64/Jit.h" #include "Core/PowerPC/Jit64/Jit.h"
#include "Core/PowerPC/Jit64Common/Jit64PowerPCState.h" #include "Core/PowerPC/Jit64Common/Jit64PowerPCState.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
using namespace Gen; using namespace Gen;
@ -102,6 +103,9 @@ void Jit64AsmRoutineManager::Generate()
dispatcher_no_check = GetCodePtr(); dispatcher_no_check = GetCodePtr();
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// The following is a translation of JitBaseBlockCache::Dispatch into assembly. // The following is a translation of JitBaseBlockCache::Dispatch into assembly.
const bool assembly_dispatcher = true; const bool assembly_dispatcher = true;
if (assembly_dispatcher) if (assembly_dispatcher)
@ -141,10 +145,10 @@ void Jit64AsmRoutineManager::Generate()
// Switch to the correct memory base, in case MSR.DR has changed. // Switch to the correct memory base, in case MSR.DR has changed.
TEST(32, PPCSTATE(msr), Imm32(1 << (31 - 27))); TEST(32, PPCSTATE(msr), Imm32(1 << (31 - 27)));
FixupBranch physmem = J_CC(CC_Z); FixupBranch physmem = J_CC(CC_Z);
MOV(64, R(RMEM), ImmPtr(Memory::logical_base)); MOV(64, R(RMEM), ImmPtr(memory.GetLogicalBase()));
JMPptr(MDisp(RSCRATCH, static_cast<s32>(offsetof(JitBlockData, normalEntry)))); JMPptr(MDisp(RSCRATCH, static_cast<s32>(offsetof(JitBlockData, normalEntry))));
SetJumpTarget(physmem); SetJumpTarget(physmem);
MOV(64, R(RMEM), ImmPtr(Memory::physical_base)); MOV(64, R(RMEM), ImmPtr(memory.GetPhysicalBase()));
JMPptr(MDisp(RSCRATCH, static_cast<s32>(offsetof(JitBlockData, normalEntry)))); JMPptr(MDisp(RSCRATCH, static_cast<s32>(offsetof(JitBlockData, normalEntry))));
SetJumpTarget(not_found); SetJumpTarget(not_found);
@ -165,10 +169,10 @@ void Jit64AsmRoutineManager::Generate()
// Switch to the correct memory base, in case MSR.DR has changed. // Switch to the correct memory base, in case MSR.DR has changed.
TEST(32, PPCSTATE(msr), Imm32(1 << (31 - 27))); TEST(32, PPCSTATE(msr), Imm32(1 << (31 - 27)));
FixupBranch physmem = J_CC(CC_Z); FixupBranch physmem = J_CC(CC_Z);
MOV(64, R(RMEM), ImmPtr(Memory::logical_base)); MOV(64, R(RMEM), ImmPtr(memory.GetLogicalBase()));
JMPptr(R(ABI_RETURN)); JMPptr(R(ABI_RETURN));
SetJumpTarget(physmem); SetJumpTarget(physmem);
MOV(64, R(RMEM), ImmPtr(Memory::physical_base)); MOV(64, R(RMEM), ImmPtr(memory.GetPhysicalBase()));
JMPptr(R(ABI_RETURN)); JMPptr(R(ABI_RETURN));
SetJumpTarget(no_block_available); SetJumpTarget(no_block_available);

View File

@ -442,7 +442,9 @@ void EmuCodeBlock::SafeLoadToRegImmediate(X64Reg reg_value, u32 address, int acc
u32 mmioAddress = PowerPC::IsOptimizableMMIOAccess(address, accessSize); u32 mmioAddress = PowerPC::IsOptimizableMMIOAccess(address, accessSize);
if (accessSize != 64 && mmioAddress) if (accessSize != 64 && mmioAddress)
{ {
MMIOLoadToReg(Memory::mmio_mapping.get(), reg_value, registersInUse, mmioAddress, accessSize, auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
MMIOLoadToReg(memory.GetMMIOMapping(), reg_value, registersInUse, mmioAddress, accessSize,
signExtend); signExtend);
return; return;
} }

View File

@ -53,7 +53,10 @@ void JitArm64::Init()
AllocCodeSpace(CODE_SIZE + child_code_size); AllocCodeSpace(CODE_SIZE + child_code_size);
AddChildCodeSpace(&m_far_code, child_code_size); AddChildCodeSpace(&m_far_code, child_code_size);
jo.fastmem_arena = m_fastmem_enabled && Memory::InitFastmemArena(); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
jo.fastmem_arena = m_fastmem_enabled && memory.InitFastmemArena();
jo.enableBlocklink = true; jo.enableBlocklink = true;
jo.optimizeGatherPipe = true; jo.optimizeGatherPipe = true;
UpdateMemoryAndExceptionOptions(); UpdateMemoryAndExceptionOptions();
@ -154,7 +157,9 @@ void JitArm64::ResetFreeMemoryRanges()
void JitArm64::Shutdown() void JitArm64::Shutdown()
{ {
Memory::ShutdownFastmemArena(); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.ShutdownFastmemArena();
FreeCodeSpace(); FreeCodeSpace();
blocks.Shutdown(); blocks.Shutdown();
FreeStack(); FreeStack();

View File

@ -22,6 +22,7 @@
#include "Core/PowerPC/JitArmCommon/BackPatch.h" #include "Core/PowerPC/JitArmCommon/BackPatch.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
using namespace Arm64Gen; using namespace Arm64Gen;
@ -302,14 +303,17 @@ void JitArm64::EmitBackpatchRoutine(u32 flags, MemAccessMode mode, ARM64Reg RS,
bool JitArm64::HandleFastmemFault(uintptr_t access_address, SContext* ctx) bool JitArm64::HandleFastmemFault(uintptr_t access_address, SContext* ctx)
{ {
if (!(access_address >= (uintptr_t)Memory::physical_base && auto& system = Core::System::GetInstance();
access_address < (uintptr_t)Memory::physical_base + 0x100010000) && auto& memory = system.GetMemory();
!(access_address >= (uintptr_t)Memory::logical_base &&
access_address < (uintptr_t)Memory::logical_base + 0x100010000)) if (!(access_address >= (uintptr_t)memory.GetPhysicalBase() &&
access_address < (uintptr_t)memory.GetPhysicalBase() + 0x100010000) &&
!(access_address >= (uintptr_t)memory.GetLogicalBase() &&
access_address < (uintptr_t)memory.GetLogicalBase() + 0x100010000))
{ {
ERROR_LOG_FMT(DYNA_REC, ERROR_LOG_FMT(DYNA_REC,
"Exception handler - access below memory space. PC: {:#018x} {:#018x} < {:#018x}", "Exception handler - access below memory space. PC: {:#018x} {:#018x} < {:#018x}",
ctx->CTX_PC, access_address, (uintptr_t)Memory::physical_base); ctx->CTX_PC, access_address, (uintptr_t)memory.GetPhysicalBase());
return false; return false;
} }

View File

@ -20,6 +20,7 @@
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/PowerPC/PPCTables.h" #include "Core/PowerPC/PPCTables.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
using namespace Arm64Gen; using namespace Arm64Gen;
@ -144,8 +145,10 @@ void JitArm64::SafeLoadToReg(u32 dest, s32 addr, s32 offsetReg, u32 flags, s32 o
regs_in_use[DecodeReg(ARM64Reg::W0)] = 0; regs_in_use[DecodeReg(ARM64Reg::W0)] = 0;
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0; regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
regs_in_use[DecodeReg(dest_reg)] = 0; regs_in_use[DecodeReg(dest_reg)] = 0;
MMIOLoadToReg(Memory::mmio_mapping.get(), this, &m_float_emit, regs_in_use, fprs_in_use, auto& system = Core::System::GetInstance();
dest_reg, mmio_address, flags); auto& memory = system.GetMemory();
MMIOLoadToReg(memory.GetMMIOMapping(), this, &m_float_emit, regs_in_use, fprs_in_use, dest_reg,
mmio_address, flags);
addr_reg_set = false; addr_reg_set = false;
} }
else else
@ -316,8 +319,10 @@ void JitArm64::SafeStoreFromReg(s32 dest, u32 value, s32 regOffset, u32 flags, s
regs_in_use[DecodeReg(ARM64Reg::W1)] = 0; regs_in_use[DecodeReg(ARM64Reg::W1)] = 0;
regs_in_use[DecodeReg(ARM64Reg::W30)] = 0; regs_in_use[DecodeReg(ARM64Reg::W30)] = 0;
regs_in_use[DecodeReg(RS)] = 0; regs_in_use[DecodeReg(RS)] = 0;
MMIOWriteRegToAddr(Memory::mmio_mapping.get(), this, &m_float_emit, regs_in_use, fprs_in_use, auto& system = Core::System::GetInstance();
RS, mmio_address, flags); auto& memory = system.GetMemory();
MMIOWriteRegToAddr(memory.GetMMIOMapping(), this, &m_float_emit, regs_in_use, fprs_in_use, RS,
mmio_address, flags);
addr_reg_set = false; addr_reg_set = false;
} }
else else

View File

@ -20,6 +20,7 @@
#include "Core/PowerPC/JitCommon/JitCache.h" #include "Core/PowerPC/JitCommon/JitCache.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
using namespace Arm64Gen; using namespace Arm64Gen;
@ -88,15 +89,20 @@ void JitArm64::GenerateAsm()
bool assembly_dispatcher = true; bool assembly_dispatcher = true;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (assembly_dispatcher) if (assembly_dispatcher)
{ {
// set the mem_base based on MSR flags // set the mem_base based on MSR flags
LDR(IndexType::Unsigned, ARM64Reg::W28, PPC_REG, PPCSTATE_OFF(msr)); LDR(IndexType::Unsigned, ARM64Reg::W28, PPC_REG, PPCSTATE_OFF(msr));
FixupBranch physmem = TBNZ(ARM64Reg::W28, 31 - 27); FixupBranch physmem = TBNZ(ARM64Reg::W28, 31 - 27);
MOVP2R(MEM_REG, jo.fastmem_arena ? Memory::physical_base : Memory::physical_page_mappings_base); MOVP2R(MEM_REG,
jo.fastmem_arena ? memory.GetPhysicalBase() : memory.GetPhysicalPageMappingsBase());
FixupBranch membaseend = B(); FixupBranch membaseend = B();
SetJumpTarget(physmem); SetJumpTarget(physmem);
MOVP2R(MEM_REG, jo.fastmem_arena ? Memory::logical_base : Memory::logical_page_mappings_base); MOVP2R(MEM_REG,
jo.fastmem_arena ? memory.GetLogicalBase() : memory.GetLogicalPageMappingsBase());
SetJumpTarget(membaseend); SetJumpTarget(membaseend);
// iCache[(address >> 2) & iCache_Mask]; // iCache[(address >> 2) & iCache_Mask];
@ -141,10 +147,11 @@ void JitArm64::GenerateAsm()
// set the mem_base based on MSR flags and jump to next block. // set the mem_base based on MSR flags and jump to next block.
LDR(IndexType::Unsigned, ARM64Reg::W28, PPC_REG, PPCSTATE_OFF(msr)); LDR(IndexType::Unsigned, ARM64Reg::W28, PPC_REG, PPCSTATE_OFF(msr));
FixupBranch physmem = TBNZ(ARM64Reg::W28, 31 - 27); FixupBranch physmem = TBNZ(ARM64Reg::W28, 31 - 27);
MOVP2R(MEM_REG, jo.fastmem_arena ? Memory::physical_base : Memory::physical_page_mappings_base); MOVP2R(MEM_REG,
jo.fastmem_arena ? memory.GetPhysicalBase() : memory.GetPhysicalPageMappingsBase());
BR(ARM64Reg::X0); BR(ARM64Reg::X0);
SetJumpTarget(physmem); SetJumpTarget(physmem);
MOVP2R(MEM_REG, jo.fastmem_arena ? Memory::logical_base : Memory::logical_page_mappings_base); MOVP2R(MEM_REG, jo.fastmem_arena ? memory.GetLogicalBase() : memory.GetLogicalPageMappingsBase());
BR(ARM64Reg::X0); BR(ARM64Reg::X0);
// Call JIT // Call JIT

View File

@ -171,7 +171,7 @@ BatTable dbat_table;
static void GenerateDSIException(u32 effective_address, bool write); static void GenerateDSIException(u32 effective_address, bool write);
template <XCheckTLBFlag flag, typename T, bool never_translate = false> template <XCheckTLBFlag flag, typename T, bool never_translate = false>
static T ReadFromHardware(u32 em_address) static T ReadFromHardware(Memory::MemoryManager& memory, u32 em_address)
{ {
const u32 em_address_start_page = em_address & ~HW_PAGE_MASK; const u32 em_address_start_page = em_address & ~HW_PAGE_MASK;
const u32 em_address_end_page = (em_address + sizeof(T) - 1) & ~HW_PAGE_MASK; const u32 em_address_end_page = (em_address + sizeof(T) - 1) & ~HW_PAGE_MASK;
@ -183,7 +183,7 @@ static T ReadFromHardware(u32 em_address)
// Note that "word" means 32-bit, so paired singles or doubles might still be 32-bit aligned! // Note that "word" means 32-bit, so paired singles or doubles might still be 32-bit aligned!
u64 var = 0; u64 var = 0;
for (u32 i = 0; i < sizeof(T); ++i) for (u32 i = 0; i < sizeof(T); ++i)
var = (var << 8) | ReadFromHardware<flag, u8, never_translate>(em_address + i); var = (var << 8) | ReadFromHardware<flag, u8, never_translate>(memory, em_address + i);
return static_cast<T>(var); return static_cast<T>(var);
} }
@ -204,42 +204,42 @@ static T ReadFromHardware(u32 em_address)
if (em_address < 0x0c000000) if (em_address < 0x0c000000)
return EFB_Read(em_address); return EFB_Read(em_address);
else else
return static_cast<T>(Memory::mmio_mapping->Read<std::make_unsigned_t<T>>(em_address)); return static_cast<T>(memory.GetMMIOMapping()->Read<std::make_unsigned_t<T>>(em_address));
} }
// Locked L1 technically doesn't have a fixed address, but games all use 0xE0000000. // Locked L1 technically doesn't have a fixed address, but games all use 0xE0000000.
if (Memory::m_pL1Cache && (em_address >> 28) == 0xE && if (memory.GetL1Cache() && (em_address >> 28) == 0xE &&
(em_address < (0xE0000000 + Memory::GetL1CacheSize()))) (em_address < (0xE0000000 + memory.GetL1CacheSize())))
{ {
T value; T value;
std::memcpy(&value, &Memory::m_pL1Cache[em_address & 0x0FFFFFFF], sizeof(T)); std::memcpy(&value, &memory.GetL1Cache()[em_address & 0x0FFFFFFF], sizeof(T));
return bswap(value); return bswap(value);
} }
if (Memory::m_pRAM && (em_address & 0xF8000000) == 0x00000000) if (memory.GetRAM() && (em_address & 0xF8000000) == 0x00000000)
{ {
// Handle RAM; the masking intentionally discards bits (essentially creating // Handle RAM; the masking intentionally discards bits (essentially creating
// mirrors of memory). // mirrors of memory).
T value; T value;
std::memcpy(&value, &Memory::m_pRAM[em_address & Memory::GetRamMask()], sizeof(T)); std::memcpy(&value, &memory.GetRAM()[em_address & memory.GetRamMask()], sizeof(T));
return bswap(value); return bswap(value);
} }
if (Memory::m_pEXRAM && (em_address >> 28) == 0x1 && if (memory.GetEXRAM() && (em_address >> 28) == 0x1 &&
(em_address & 0x0FFFFFFF) < Memory::GetExRamSizeReal()) (em_address & 0x0FFFFFFF) < memory.GetExRamSizeReal())
{ {
T value; T value;
std::memcpy(&value, &Memory::m_pEXRAM[em_address & 0x0FFFFFFF], sizeof(T)); std::memcpy(&value, &memory.GetEXRAM()[em_address & 0x0FFFFFFF], sizeof(T));
return bswap(value); return bswap(value);
} }
// In Fake-VMEM mode, we need to map the memory somewhere into // In Fake-VMEM mode, we need to map the memory somewhere into
// physical memory for BAT translation to work; we currently use // physical memory for BAT translation to work; we currently use
// [0x7E000000, 0x80000000). // [0x7E000000, 0x80000000).
if (Memory::m_pFakeVMEM && ((em_address & 0xFE000000) == 0x7E000000)) if (memory.GetFakeVMEM() && ((em_address & 0xFE000000) == 0x7E000000))
{ {
T value; T value;
std::memcpy(&value, &Memory::m_pFakeVMEM[em_address & Memory::GetFakeVMemMask()], sizeof(T)); std::memcpy(&value, &memory.GetFakeVMEM()[em_address & memory.GetFakeVMemMask()], sizeof(T));
return bswap(value); return bswap(value);
} }
@ -253,7 +253,8 @@ static T ReadFromHardware(u32 em_address)
} }
template <XCheckTLBFlag flag, bool never_translate = false> template <XCheckTLBFlag flag, bool never_translate = false>
static void WriteToHardware(u32 em_address, const u32 data, const u32 size) static void WriteToHardware(Memory::MemoryManager& memory, u32 em_address, const u32 data,
const u32 size)
{ {
DEBUG_ASSERT(size <= 4); DEBUG_ASSERT(size <= 4);
@ -267,8 +268,8 @@ static void WriteToHardware(u32 em_address, const u32 data, const u32 size)
const u32 first_half_size = em_address_end_page - em_address; const u32 first_half_size = em_address_end_page - em_address;
const u32 second_half_size = size - first_half_size; const u32 second_half_size = size - first_half_size;
WriteToHardware<flag, never_translate>( WriteToHardware<flag, never_translate>(
em_address, Common::RotateRight(data, second_half_size * 8), first_half_size); memory, em_address, Common::RotateRight(data, second_half_size * 8), first_half_size);
WriteToHardware<flag, never_translate>(em_address_end_page, data, second_half_size); WriteToHardware<flag, never_translate>(memory, em_address_end_page, data, second_half_size);
return; return;
} }
@ -333,20 +334,20 @@ static void WriteToHardware(u32 em_address, const u32 data, const u32 size)
switch (size) switch (size)
{ {
case 1: case 1:
Memory::mmio_mapping->Write<u8>(em_address, static_cast<u8>(data)); memory.GetMMIOMapping()->Write<u8>(em_address, static_cast<u8>(data));
return; return;
case 2: case 2:
Memory::mmio_mapping->Write<u16>(em_address, static_cast<u16>(data)); memory.GetMMIOMapping()->Write<u16>(em_address, static_cast<u16>(data));
return; return;
case 4: case 4:
Memory::mmio_mapping->Write<u32>(em_address, data); memory.GetMMIOMapping()->Write<u32>(em_address, data);
return; return;
default: default:
// Some kind of misaligned write. TODO: Does this match how the actual hardware handles it? // Some kind of misaligned write. TODO: Does this match how the actual hardware handles it?
for (size_t i = size * 8; i > 0; em_address++) for (size_t i = size * 8; i > 0; em_address++)
{ {
i -= 8; i -= 8;
Memory::mmio_mapping->Write<u8>(em_address, static_cast<u8>(data >> i)); memory.GetMMIOMapping()->Write<u8>(em_address, static_cast<u8>(data >> i));
} }
return; return;
} }
@ -355,10 +356,10 @@ static void WriteToHardware(u32 em_address, const u32 data, const u32 size)
const u32 swapped_data = Common::swap32(Common::RotateRight(data, size * 8)); const u32 swapped_data = Common::swap32(Common::RotateRight(data, size * 8));
// Locked L1 technically doesn't have a fixed address, but games all use 0xE0000000. // Locked L1 technically doesn't have a fixed address, but games all use 0xE0000000.
if (Memory::m_pL1Cache && (em_address >> 28 == 0xE) && if (memory.GetL1Cache() && (em_address >> 28 == 0xE) &&
(em_address < (0xE0000000 + Memory::GetL1CacheSize()))) (em_address < (0xE0000000 + memory.GetL1CacheSize())))
{ {
std::memcpy(&Memory::m_pL1Cache[em_address & 0x0FFFFFFF], &swapped_data, size); std::memcpy(&memory.GetL1Cache()[em_address & 0x0FFFFFFF], &swapped_data, size);
return; return;
} }
@ -379,34 +380,34 @@ static void WriteToHardware(u32 em_address, const u32 data, const u32 size)
for (u32 addr = em_address & ~0x7; addr < em_address + size; addr += 8) for (u32 addr = em_address & ~0x7; addr < em_address + size; addr += 8)
{ {
WriteToHardware<flag, true>(addr, rotated_data, 4); WriteToHardware<flag, true>(memory, addr, rotated_data, 4);
WriteToHardware<flag, true>(addr + 4, rotated_data, 4); WriteToHardware<flag, true>(memory, addr + 4, rotated_data, 4);
} }
return; return;
} }
if (Memory::m_pRAM && (em_address & 0xF8000000) == 0x00000000) if (memory.GetRAM() && (em_address & 0xF8000000) == 0x00000000)
{ {
// Handle RAM; the masking intentionally discards bits (essentially creating // Handle RAM; the masking intentionally discards bits (essentially creating
// mirrors of memory). // mirrors of memory).
std::memcpy(&Memory::m_pRAM[em_address & Memory::GetRamMask()], &swapped_data, size); std::memcpy(&memory.GetRAM()[em_address & memory.GetRamMask()], &swapped_data, size);
return; return;
} }
if (Memory::m_pEXRAM && (em_address >> 28) == 0x1 && if (memory.GetEXRAM() && (em_address >> 28) == 0x1 &&
(em_address & 0x0FFFFFFF) < Memory::GetExRamSizeReal()) (em_address & 0x0FFFFFFF) < memory.GetExRamSizeReal())
{ {
std::memcpy(&Memory::m_pEXRAM[em_address & 0x0FFFFFFF], &swapped_data, size); std::memcpy(&memory.GetEXRAM()[em_address & 0x0FFFFFFF], &swapped_data, size);
return; return;
} }
// In Fake-VMEM mode, we need to map the memory somewhere into // In Fake-VMEM mode, we need to map the memory somewhere into
// physical memory for BAT translation to work; we currently use // physical memory for BAT translation to work; we currently use
// [0x7E000000, 0x80000000). // [0x7E000000, 0x80000000).
if (Memory::m_pFakeVMEM && ((em_address & 0xFE000000) == 0x7E000000)) if (memory.GetFakeVMEM() && ((em_address & 0xFE000000) == 0x7E000000))
{ {
std::memcpy(&Memory::m_pFakeVMEM[em_address & Memory::GetFakeVMemMask()], &swapped_data, size); std::memcpy(&memory.GetFakeVMEM()[em_address & memory.GetFakeVMemMask()], &swapped_data, size);
return; return;
} }
@ -454,11 +455,14 @@ TryReadInstResult TryReadInstruction(u32 address)
} }
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 hex; u32 hex;
// TODO: Refactor this. This icache implementation is totally wrong if used with the fake vmem. // TODO: Refactor this. This icache implementation is totally wrong if used with the fake vmem.
if (Memory::m_pFakeVMEM && ((address & 0xFE000000) == 0x7E000000)) if (memory.GetFakeVMEM() && ((address & 0xFE000000) == 0x7E000000))
{ {
hex = Common::swap32(&Memory::m_pFakeVMEM[address & Memory::GetFakeVMemMask()]); hex = Common::swap32(&memory.GetFakeVMEM()[address & memory.GetFakeVMemMask()]);
} }
else else
{ {
@ -469,7 +473,9 @@ TryReadInstResult TryReadInstruction(u32 address)
u32 HostRead_Instruction(const u32 address) u32 HostRead_Instruction(const u32 address)
{ {
return ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(memory, address);
} }
std::optional<ReadResult<u32>> HostTryReadInstruction(const u32 address, std::optional<ReadResult<u32>> HostTryReadInstruction(const u32 address,
@ -478,23 +484,27 @@ std::optional<ReadResult<u32>> HostTryReadInstruction(const u32 address,
if (!HostIsInstructionRAMAddress(address, space)) if (!HostIsInstructionRAMAddress(address, space))
return std::nullopt; return std::nullopt;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (space) switch (space)
{ {
case RequestedAddressSpace::Effective: case RequestedAddressSpace::Effective:
{ {
const u32 value = ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(address); const u32 value = ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(memory, address);
return ReadResult<u32>(!!MSR.DR, value); return ReadResult<u32>(!!MSR.DR, value);
} }
case RequestedAddressSpace::Physical: case RequestedAddressSpace::Physical:
{ {
const u32 value = ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32, true>(address); const u32 value =
ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32, true>(memory, address);
return ReadResult<u32>(false, value); return ReadResult<u32>(false, value);
} }
case RequestedAddressSpace::Virtual: case RequestedAddressSpace::Virtual:
{ {
if (!MSR.DR) if (!MSR.DR)
return std::nullopt; return std::nullopt;
const u32 value = ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(address); const u32 value = ReadFromHardware<XCheckTLBFlag::OpcodeNoException, u32>(memory, address);
return ReadResult<u32>(true, value); return ReadResult<u32>(true, value);
} }
} }
@ -541,28 +551,36 @@ static void Memcheck(u32 address, u64 var, bool write, size_t size)
u8 Read_U8(const u32 address) u8 Read_U8(const u32 address)
{ {
u8 var = ReadFromHardware<XCheckTLBFlag::Read, u8>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8 var = ReadFromHardware<XCheckTLBFlag::Read, u8>(memory, address);
Memcheck(address, var, false, 1); Memcheck(address, var, false, 1);
return var; return var;
} }
u16 Read_U16(const u32 address) u16 Read_U16(const u32 address)
{ {
u16 var = ReadFromHardware<XCheckTLBFlag::Read, u16>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u16 var = ReadFromHardware<XCheckTLBFlag::Read, u16>(memory, address);
Memcheck(address, var, false, 2); Memcheck(address, var, false, 2);
return var; return var;
} }
u32 Read_U32(const u32 address) u32 Read_U32(const u32 address)
{ {
u32 var = ReadFromHardware<XCheckTLBFlag::Read, u32>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u32 var = ReadFromHardware<XCheckTLBFlag::Read, u32>(memory, address);
Memcheck(address, var, false, 4); Memcheck(address, var, false, 4);
return var; return var;
} }
u64 Read_U64(const u32 address) u64 Read_U64(const u32 address)
{ {
u64 var = ReadFromHardware<XCheckTLBFlag::Read, u64>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u64 var = ReadFromHardware<XCheckTLBFlag::Read, u64>(memory, address);
Memcheck(address, var, false, 8); Memcheck(address, var, false, 8);
return var; return var;
} }
@ -587,23 +605,26 @@ static std::optional<ReadResult<T>> HostTryReadUX(const u32 address, RequestedAd
if (!HostIsRAMAddress(address, space)) if (!HostIsRAMAddress(address, space))
return std::nullopt; return std::nullopt;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (space) switch (space)
{ {
case RequestedAddressSpace::Effective: case RequestedAddressSpace::Effective:
{ {
T value = ReadFromHardware<XCheckTLBFlag::NoException, T>(address); T value = ReadFromHardware<XCheckTLBFlag::NoException, T>(memory, address);
return ReadResult<T>(!!MSR.DR, std::move(value)); return ReadResult<T>(!!MSR.DR, std::move(value));
} }
case RequestedAddressSpace::Physical: case RequestedAddressSpace::Physical:
{ {
T value = ReadFromHardware<XCheckTLBFlag::NoException, T, true>(address); T value = ReadFromHardware<XCheckTLBFlag::NoException, T, true>(memory, address);
return ReadResult<T>(false, std::move(value)); return ReadResult<T>(false, std::move(value));
} }
case RequestedAddressSpace::Virtual: case RequestedAddressSpace::Virtual:
{ {
if (!MSR.DR) if (!MSR.DR)
return std::nullopt; return std::nullopt;
T value = ReadFromHardware<XCheckTLBFlag::NoException, T>(address); T value = ReadFromHardware<XCheckTLBFlag::NoException, T>(memory, address);
return ReadResult<T>(true, std::move(value)); return ReadResult<T>(true, std::move(value));
} }
} }
@ -661,13 +682,17 @@ u32 Read_U16_ZX(const u32 address)
void Write_U8(const u32 var, const u32 address) void Write_U8(const u32 var, const u32 address)
{ {
Memcheck(address, var, true, 1); Memcheck(address, var, true, 1);
WriteToHardware<XCheckTLBFlag::Write>(address, var, 1); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::Write>(memory, address, var, 1);
} }
void Write_U16(const u32 var, const u32 address) void Write_U16(const u32 var, const u32 address)
{ {
Memcheck(address, var, true, 2); Memcheck(address, var, true, 2);
WriteToHardware<XCheckTLBFlag::Write>(address, var, 2); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::Write>(memory, address, var, 2);
} }
void Write_U16_Swap(const u32 var, const u32 address) void Write_U16_Swap(const u32 var, const u32 address)
{ {
@ -677,7 +702,9 @@ void Write_U16_Swap(const u32 var, const u32 address)
void Write_U32(const u32 var, const u32 address) void Write_U32(const u32 var, const u32 address)
{ {
Memcheck(address, var, true, 4); Memcheck(address, var, true, 4);
WriteToHardware<XCheckTLBFlag::Write>(address, var, 4); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::Write>(memory, address, var, 4);
} }
void Write_U32_Swap(const u32 var, const u32 address) void Write_U32_Swap(const u32 var, const u32 address)
{ {
@ -687,8 +714,10 @@ void Write_U32_Swap(const u32 var, const u32 address)
void Write_U64(const u64 var, const u32 address) void Write_U64(const u64 var, const u32 address)
{ {
Memcheck(address, var, true, 8); Memcheck(address, var, true, 8);
WriteToHardware<XCheckTLBFlag::Write>(address, static_cast<u32>(var >> 32), 4); auto& system = Core::System::GetInstance();
WriteToHardware<XCheckTLBFlag::Write>(address + sizeof(u32), static_cast<u32>(var), 4); auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::Write>(memory, address, static_cast<u32>(var >> 32), 4);
WriteToHardware<XCheckTLBFlag::Write>(memory, address + sizeof(u32), static_cast<u32>(var), 4);
} }
void Write_U64_Swap(const u64 var, const u32 address) void Write_U64_Swap(const u64 var, const u32 address)
{ {
@ -704,22 +733,30 @@ void Write_F64(const double var, const u32 address)
u8 HostRead_U8(const u32 address) u8 HostRead_U8(const u32 address)
{ {
return ReadFromHardware<XCheckTLBFlag::NoException, u8>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return ReadFromHardware<XCheckTLBFlag::NoException, u8>(memory, address);
} }
u16 HostRead_U16(const u32 address) u16 HostRead_U16(const u32 address)
{ {
return ReadFromHardware<XCheckTLBFlag::NoException, u16>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return ReadFromHardware<XCheckTLBFlag::NoException, u16>(memory, address);
} }
u32 HostRead_U32(const u32 address) u32 HostRead_U32(const u32 address)
{ {
return ReadFromHardware<XCheckTLBFlag::NoException, u32>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return ReadFromHardware<XCheckTLBFlag::NoException, u32>(memory, address);
} }
u64 HostRead_U64(const u32 address) u64 HostRead_U64(const u32 address)
{ {
return ReadFromHardware<XCheckTLBFlag::NoException, u64>(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return ReadFromHardware<XCheckTLBFlag::NoException, u64>(memory, address);
} }
float HostRead_F32(const u32 address) float HostRead_F32(const u32 address)
@ -738,23 +775,32 @@ double HostRead_F64(const u32 address)
void HostWrite_U8(const u32 var, const u32 address) void HostWrite_U8(const u32 var, const u32 address)
{ {
WriteToHardware<XCheckTLBFlag::NoException>(address, var, 1); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::NoException>(memory, address, var, 1);
} }
void HostWrite_U16(const u32 var, const u32 address) void HostWrite_U16(const u32 var, const u32 address)
{ {
WriteToHardware<XCheckTLBFlag::NoException>(address, var, 2); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::NoException>(memory, address, var, 2);
} }
void HostWrite_U32(const u32 var, const u32 address) void HostWrite_U32(const u32 var, const u32 address)
{ {
WriteToHardware<XCheckTLBFlag::NoException>(address, var, 4); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::NoException>(memory, address, var, 4);
} }
void HostWrite_U64(const u64 var, const u32 address) void HostWrite_U64(const u64 var, const u32 address)
{ {
WriteToHardware<XCheckTLBFlag::NoException>(address, static_cast<u32>(var >> 32), 4); auto& system = Core::System::GetInstance();
WriteToHardware<XCheckTLBFlag::NoException>(address + sizeof(u32), static_cast<u32>(var), 4); auto& memory = system.GetMemory();
WriteToHardware<XCheckTLBFlag::NoException>(memory, address, static_cast<u32>(var >> 32), 4);
WriteToHardware<XCheckTLBFlag::NoException>(memory, address + sizeof(u32), static_cast<u32>(var),
4);
} }
void HostWrite_F32(const float var, const u32 address) void HostWrite_F32(const float var, const u32 address)
@ -777,18 +823,21 @@ static std::optional<WriteResult> HostTryWriteUX(const u32 var, const u32 addres
if (!HostIsRAMAddress(address, space)) if (!HostIsRAMAddress(address, space))
return std::nullopt; return std::nullopt;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (space) switch (space)
{ {
case RequestedAddressSpace::Effective: case RequestedAddressSpace::Effective:
WriteToHardware<XCheckTLBFlag::NoException>(address, var, size); WriteToHardware<XCheckTLBFlag::NoException>(memory, address, var, size);
return WriteResult(!!MSR.DR); return WriteResult(!!MSR.DR);
case RequestedAddressSpace::Physical: case RequestedAddressSpace::Physical:
WriteToHardware<XCheckTLBFlag::NoException, true>(address, var, size); WriteToHardware<XCheckTLBFlag::NoException, true>(memory, address, var, size);
return WriteResult(false); return WriteResult(false);
case RequestedAddressSpace::Virtual: case RequestedAddressSpace::Virtual:
if (!MSR.DR) if (!MSR.DR)
return std::nullopt; return std::nullopt;
WriteToHardware<XCheckTLBFlag::NoException>(address, var, size); WriteToHardware<XCheckTLBFlag::NoException>(memory, address, var, size);
return WriteResult(true); return WriteResult(true);
} }
@ -893,7 +942,7 @@ bool IsOptimizableRAMAddress(const u32 address)
} }
template <XCheckTLBFlag flag> template <XCheckTLBFlag flag>
static bool IsRAMAddress(u32 address, bool translate) static bool IsRAMAddress(Memory::MemoryManager& memory, u32 address, bool translate)
{ {
if (translate) if (translate)
{ {
@ -904,21 +953,21 @@ static bool IsRAMAddress(u32 address, bool translate)
} }
u32 segment = address >> 28; u32 segment = address >> 28;
if (Memory::m_pRAM && segment == 0x0 && (address & 0x0FFFFFFF) < Memory::GetRamSizeReal()) if (memory.GetRAM() && segment == 0x0 && (address & 0x0FFFFFFF) < memory.GetRamSizeReal())
{ {
return true; return true;
} }
else if (Memory::m_pEXRAM && segment == 0x1 && else if (memory.GetEXRAM() && segment == 0x1 &&
(address & 0x0FFFFFFF) < Memory::GetExRamSizeReal()) (address & 0x0FFFFFFF) < memory.GetExRamSizeReal())
{ {
return true; return true;
} }
else if (Memory::m_pFakeVMEM && ((address & 0xFE000000) == 0x7E000000)) else if (memory.GetFakeVMEM() && ((address & 0xFE000000) == 0x7E000000))
{ {
return true; return true;
} }
else if (Memory::m_pL1Cache && segment == 0xE && else if (memory.GetL1Cache() && segment == 0xE &&
(address < (0xE0000000 + Memory::GetL1CacheSize()))) (address < (0xE0000000 + memory.GetL1CacheSize())))
{ {
return true; return true;
} }
@ -927,16 +976,19 @@ static bool IsRAMAddress(u32 address, bool translate)
bool HostIsRAMAddress(u32 address, RequestedAddressSpace space) bool HostIsRAMAddress(u32 address, RequestedAddressSpace space)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (space) switch (space)
{ {
case RequestedAddressSpace::Effective: case RequestedAddressSpace::Effective:
return IsRAMAddress<XCheckTLBFlag::NoException>(address, MSR.DR); return IsRAMAddress<XCheckTLBFlag::NoException>(memory, address, MSR.DR);
case RequestedAddressSpace::Physical: case RequestedAddressSpace::Physical:
return IsRAMAddress<XCheckTLBFlag::NoException>(address, false); return IsRAMAddress<XCheckTLBFlag::NoException>(memory, address, false);
case RequestedAddressSpace::Virtual: case RequestedAddressSpace::Virtual:
if (!MSR.DR) if (!MSR.DR)
return false; return false;
return IsRAMAddress<XCheckTLBFlag::NoException>(address, true); return IsRAMAddress<XCheckTLBFlag::NoException>(memory, address, true);
} }
ASSERT(0); ASSERT(0);
@ -949,16 +1001,19 @@ bool HostIsInstructionRAMAddress(u32 address, RequestedAddressSpace space)
if (address & 3) if (address & 3)
return false; return false;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
switch (space) switch (space)
{ {
case RequestedAddressSpace::Effective: case RequestedAddressSpace::Effective:
return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(address, MSR.IR); return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(memory, address, MSR.IR);
case RequestedAddressSpace::Physical: case RequestedAddressSpace::Physical:
return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(address, false); return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(memory, address, false);
case RequestedAddressSpace::Virtual: case RequestedAddressSpace::Virtual:
if (!MSR.IR) if (!MSR.IR)
return false; return false;
return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(address, true); return IsRAMAddress<XCheckTLBFlag::OpcodeNoException>(memory, address, true);
} }
ASSERT(0); ASSERT(0);
@ -967,6 +1022,9 @@ bool HostIsInstructionRAMAddress(u32 address, RequestedAddressSpace space)
void DMA_LCToMemory(const u32 mem_address, const u32 cache_address, const u32 num_blocks) void DMA_LCToMemory(const u32 mem_address, const u32 cache_address, const u32 num_blocks)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// TODO: It's not completely clear this is the right spot for this code; // TODO: It's not completely clear this is the right spot for this code;
// what would happen if, for example, the DVD drive tried to write to the EFB? // what would happen if, for example, the DVD drive tried to write to the EFB?
// TODO: This is terribly slow. // TODO: This is terribly slow.
@ -976,7 +1034,7 @@ void DMA_LCToMemory(const u32 mem_address, const u32 cache_address, const u32 nu
{ {
for (u32 i = 0; i < 32 * num_blocks; i += 4) for (u32 i = 0; i < 32 * num_blocks; i += 4)
{ {
const u32 data = Common::swap32(Memory::m_pL1Cache + ((cache_address + i) & 0x3FFFF)); const u32 data = Common::swap32(memory.GetL1Cache() + ((cache_address + i) & 0x3FFFF));
EFB_Write(data, mem_address + i); EFB_Write(data, mem_address + i);
} }
return; return;
@ -988,14 +1046,14 @@ void DMA_LCToMemory(const u32 mem_address, const u32 cache_address, const u32 nu
{ {
for (u32 i = 0; i < 32 * num_blocks; i += 4) for (u32 i = 0; i < 32 * num_blocks; i += 4)
{ {
const u32 data = Common::swap32(Memory::m_pL1Cache + ((cache_address + i) & 0x3FFFF)); const u32 data = Common::swap32(memory.GetL1Cache() + ((cache_address + i) & 0x3FFFF));
Memory::mmio_mapping->Write(mem_address + i, data); memory.GetMMIOMapping()->Write(mem_address + i, data);
} }
return; return;
} }
const u8* src = Memory::m_pL1Cache + (cache_address & 0x3FFFF); const u8* src = memory.GetL1Cache() + (cache_address & 0x3FFFF);
u8* dst = Memory::GetPointer(mem_address); u8* dst = memory.GetPointer(mem_address);
if (dst == nullptr) if (dst == nullptr)
return; return;
@ -1004,8 +1062,11 @@ void DMA_LCToMemory(const u32 mem_address, const u32 cache_address, const u32 nu
void DMA_MemoryToLC(const u32 cache_address, const u32 mem_address, const u32 num_blocks) void DMA_MemoryToLC(const u32 cache_address, const u32 mem_address, const u32 num_blocks)
{ {
const u8* src = Memory::GetPointer(mem_address); auto& system = Core::System::GetInstance();
u8* dst = Memory::m_pL1Cache + (cache_address & 0x3FFFF); auto& memory = system.GetMemory();
const u8* src = memory.GetPointer(mem_address);
u8* dst = memory.GetL1Cache() + (cache_address & 0x3FFFF);
// No known game uses this; here for completeness. // No known game uses this; here for completeness.
// TODO: Refactor. // TODO: Refactor.
@ -1014,7 +1075,7 @@ void DMA_MemoryToLC(const u32 cache_address, const u32 mem_address, const u32 nu
for (u32 i = 0; i < 32 * num_blocks; i += 4) for (u32 i = 0; i < 32 * num_blocks; i += 4)
{ {
const u32 data = Common::swap32(EFB_Read(mem_address + i)); const u32 data = Common::swap32(EFB_Read(mem_address + i));
std::memcpy(Memory::m_pL1Cache + ((cache_address + i) & 0x3FFFF), &data, sizeof(u32)); std::memcpy(memory.GetL1Cache() + ((cache_address + i) & 0x3FFFF), &data, sizeof(u32));
} }
return; return;
} }
@ -1025,8 +1086,8 @@ void DMA_MemoryToLC(const u32 cache_address, const u32 mem_address, const u32 nu
{ {
for (u32 i = 0; i < 32 * num_blocks; i += 4) for (u32 i = 0; i < 32 * num_blocks; i += 4)
{ {
const u32 data = Common::swap32(Memory::mmio_mapping->Read<u32>(mem_address + i)); const u32 data = Common::swap32(memory.GetMMIOMapping()->Read<u32>(mem_address + i));
std::memcpy(Memory::m_pL1Cache + ((cache_address + i) & 0x3FFFF), &data, sizeof(u32)); std::memcpy(memory.GetL1Cache() + ((cache_address + i) & 0x3FFFF), &data, sizeof(u32));
} }
return; return;
} }
@ -1059,10 +1120,13 @@ void ClearCacheLine(u32 address)
address = translated_address.address; address = translated_address.address;
} }
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// TODO: This isn't precisely correct for non-RAM regions, but the difference // TODO: This isn't precisely correct for non-RAM regions, but the difference
// is unlikely to matter. // is unlikely to matter.
for (u32 i = 0; i < 32; i += 4) for (u32 i = 0; i < 32; i += 4)
WriteToHardware<XCheckTLBFlag::Write, true>(address + i, 0, 4); WriteToHardware<XCheckTLBFlag::Write, true>(memory, address + i, 0, 4);
} }
u32 IsOptimizableMMIOAccess(u32 address, u32 access_size) u32 IsOptimizableMMIOAccess(u32 address, u32 access_size)
@ -1346,6 +1410,9 @@ static TranslateAddressResult TranslatePageAddress(const EffectiveAddress addres
pte1.API = api; pte1.API = api;
pte1.V = 1; pte1.V = 1;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
for (int hash_func = 0; hash_func < 2; hash_func++) for (int hash_func = 0; hash_func < 2; hash_func++)
{ {
// hash function no 2 "not" .360 // hash function no 2 "not" .360
@ -1360,11 +1427,11 @@ static TranslateAddressResult TranslatePageAddress(const EffectiveAddress addres
for (int i = 0; i < 8; i++, pteg_addr += 8) for (int i = 0; i < 8; i++, pteg_addr += 8)
{ {
const u32 pteg = Memory::Read_U32(pteg_addr); const u32 pteg = memory.Read_U32(pteg_addr);
if (pte1.Hex == pteg) if (pte1.Hex == pteg)
{ {
UPTE_Hi pte2(Memory::Read_U32(pteg_addr + 4)); UPTE_Hi pte2(memory.Read_U32(pteg_addr + 4));
// set the access bits // set the access bits
switch (flag) switch (flag)
@ -1386,7 +1453,7 @@ static TranslateAddressResult TranslatePageAddress(const EffectiveAddress addres
if (!IsNoExceptionFlag(flag)) if (!IsNoExceptionFlag(flag))
{ {
Memory::Write_U32(pte2.Hex, pteg_addr + 4); memory.Write_U32(pte2.Hex, pteg_addr + 4);
} }
// We already updated the TLB entry if this was caused by a C bit. // We already updated the TLB entry if this was caused by a C bit.
@ -1405,6 +1472,9 @@ static TranslateAddressResult TranslatePageAddress(const EffectiveAddress addres
static void UpdateBATs(BatTable& bat_table, u32 base_spr) static void UpdateBATs(BatTable& bat_table, u32 base_spr)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// TODO: Separate BATs for MSR.PR==0 and MSR.PR==1 // TODO: Separate BATs for MSR.PR==0 and MSR.PR==1
// TODO: Handle PP settings. // TODO: Handle PP settings.
// TODO: Check how hardware reacts to overlapping BATs (including // TODO: Check how hardware reacts to overlapping BATs (including
@ -1464,21 +1534,21 @@ static void UpdateBATs(BatTable& bat_table, u32 base_spr)
// that fastmem doesn't emulate properly (though no normal games are known to rely on them). // that fastmem doesn't emulate properly (though no normal games are known to rely on them).
if (!wi) if (!wi)
{ {
if (Memory::m_pFakeVMEM && (physical_address & 0xFE000000) == 0x7E000000) if (memory.GetFakeVMEM() && (physical_address & 0xFE000000) == 0x7E000000)
{ {
valid_bit |= BAT_PHYSICAL_BIT; valid_bit |= BAT_PHYSICAL_BIT;
} }
else if (physical_address < Memory::GetRamSizeReal()) else if (physical_address < memory.GetRamSizeReal())
{ {
valid_bit |= BAT_PHYSICAL_BIT; valid_bit |= BAT_PHYSICAL_BIT;
} }
else if (Memory::m_pEXRAM && physical_address >> 28 == 0x1 && else if (memory.GetEXRAM() && physical_address >> 28 == 0x1 &&
(physical_address & 0x0FFFFFFF) < Memory::GetExRamSizeReal()) (physical_address & 0x0FFFFFFF) < memory.GetExRamSizeReal())
{ {
valid_bit |= BAT_PHYSICAL_BIT; valid_bit |= BAT_PHYSICAL_BIT;
} }
else if (physical_address >> 28 == 0xE && else if (physical_address >> 28 == 0xE &&
physical_address < 0xE0000000 + Memory::GetL1CacheSize()) physical_address < 0xE0000000 + memory.GetL1CacheSize())
{ {
valid_bit |= BAT_PHYSICAL_BIT; valid_bit |= BAT_PHYSICAL_BIT;
} }
@ -1497,12 +1567,15 @@ static void UpdateBATs(BatTable& bat_table, u32 base_spr)
static void UpdateFakeMMUBat(BatTable& bat_table, u32 start_addr) static void UpdateFakeMMUBat(BatTable& bat_table, u32 start_addr)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
for (u32 i = 0; i < (0x10000000 >> BAT_INDEX_SHIFT); ++i) for (u32 i = 0; i < (0x10000000 >> BAT_INDEX_SHIFT); ++i)
{ {
// Map from 0x4XXXXXXX or 0x7XXXXXXX to the range // Map from 0x4XXXXXXX or 0x7XXXXXXX to the range
// [0x7E000000,0x80000000). // [0x7E000000,0x80000000).
u32 e_address = i + (start_addr >> BAT_INDEX_SHIFT); u32 e_address = i + (start_addr >> BAT_INDEX_SHIFT);
u32 p_address = 0x7E000000 | (i << BAT_INDEX_SHIFT & Memory::GetFakeVMemMask()); u32 p_address = 0x7E000000 | (i << BAT_INDEX_SHIFT & memory.GetFakeVMemMask());
u32 flags = BAT_MAPPED_BIT | BAT_PHYSICAL_BIT; u32 flags = BAT_MAPPED_BIT | BAT_PHYSICAL_BIT;
if (PowerPC::memchecks.OverlapsMemcheck(e_address << BAT_INDEX_SHIFT, BAT_PAGE_SIZE)) if (PowerPC::memchecks.OverlapsMemcheck(e_address << BAT_INDEX_SHIFT, BAT_PAGE_SIZE))
@ -1514,12 +1587,15 @@ static void UpdateFakeMMUBat(BatTable& bat_table, u32 start_addr)
void DBATUpdated() void DBATUpdated()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
dbat_table = {}; dbat_table = {};
UpdateBATs(dbat_table, SPR_DBAT0U); UpdateBATs(dbat_table, SPR_DBAT0U);
bool extended_bats = SConfig::GetInstance().bWii && HID4.SBE; bool extended_bats = SConfig::GetInstance().bWii && HID4.SBE;
if (extended_bats) if (extended_bats)
UpdateBATs(dbat_table, SPR_DBAT4U); UpdateBATs(dbat_table, SPR_DBAT4U);
if (Memory::m_pFakeVMEM) if (memory.GetFakeVMEM())
{ {
// In Fake-MMU mode, insert some extra entries into the BAT tables. // In Fake-MMU mode, insert some extra entries into the BAT tables.
UpdateFakeMMUBat(dbat_table, 0x40000000); UpdateFakeMMUBat(dbat_table, 0x40000000);
@ -1527,7 +1603,7 @@ void DBATUpdated()
} }
#ifndef _ARCH_32 #ifndef _ARCH_32
Memory::UpdateLogicalMemory(dbat_table); memory.UpdateLogicalMemory(dbat_table);
#endif #endif
// IsOptimizable*Address and dcbz depends on the BAT mapping, so we need a flush here. // IsOptimizable*Address and dcbz depends on the BAT mapping, so we need a flush here.
@ -1536,12 +1612,15 @@ void DBATUpdated()
void IBATUpdated() void IBATUpdated()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
ibat_table = {}; ibat_table = {};
UpdateBATs(ibat_table, SPR_IBAT0U); UpdateBATs(ibat_table, SPR_IBAT0U);
bool extended_bats = SConfig::GetInstance().bWii && HID4.SBE; bool extended_bats = SConfig::GetInstance().bWii && HID4.SBE;
if (extended_bats) if (extended_bats)
UpdateBATs(ibat_table, SPR_IBAT4U); UpdateBATs(ibat_table, SPR_IBAT4U);
if (Memory::m_pFakeVMEM) if (memory.GetFakeVMEM())
{ {
// In Fake-MMU mode, insert some extra entries into the BAT tables. // In Fake-MMU mode, insert some extra entries into the BAT tables.
UpdateFakeMMUBat(ibat_table, 0x40000000); UpdateFakeMMUBat(ibat_table, 0x40000000);

View File

@ -12,6 +12,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/PowerPC/JitInterface.h" #include "Core/PowerPC/JitInterface.h"
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/System.h"
namespace PowerPC namespace PowerPC
{ {
@ -139,8 +140,11 @@ void InstructionCache::Invalidate(u32 addr)
u32 InstructionCache::ReadInstruction(u32 addr) u32 InstructionCache::ReadInstruction(u32 addr)
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
if (!HID0.ICE || m_disable_icache) // instruction cache is disabled if (!HID0.ICE || m_disable_icache) // instruction cache is disabled
return Memory::Read_U32(addr); return memory.Read_U32(addr);
u32 set = (addr >> 5) & 0x7f; u32 set = (addr >> 5) & 0x7f;
u32 tag = addr >> 12; u32 tag = addr >> 12;
@ -161,14 +165,14 @@ u32 InstructionCache::ReadInstruction(u32 addr)
if (t == 0xff) // load to the cache if (t == 0xff) // load to the cache
{ {
if (HID0.ILOCK) // instruction cache is locked if (HID0.ILOCK) // instruction cache is locked
return Memory::Read_U32(addr); return memory.Read_U32(addr);
// select a way // select a way
if (valid[set] != 0xff) if (valid[set] != 0xff)
t = s_way_from_valid[valid[set]]; t = s_way_from_valid[valid[set]];
else else
t = s_way_from_plru[plru[set]]; t = s_way_from_plru[plru[set]];
// load // load
Memory::CopyFromEmu(reinterpret_cast<u8*>(data[set][t].data()), (addr & ~0x1f), 32); memory.CopyFromEmu(reinterpret_cast<u8*>(data[set][t].data()), (addr & ~0x1f), 32);
if (valid[set] & (1 << t)) if (valid[set] & (1 << t))
{ {
if (tags[set][t] & (ICACHE_VMEM_BIT >> 12)) if (tags[set][t] & (ICACHE_VMEM_BIT >> 12))
@ -191,7 +195,7 @@ u32 InstructionCache::ReadInstruction(u32 addr)
// update plru // update plru
plru[set] = (plru[set] & ~s_plru_mask[t]) | s_plru_value[t]; plru[set] = (plru[set] & ~s_plru_mask[t]) | s_plru_value[t];
const u32 res = Common::swap32(data[set][t][(addr >> 2) & 7]); const u32 res = Common::swap32(data[set][t][(addr >> 2) & 7]);
const u32 inmem = Memory::Read_U32(addr); const u32 inmem = memory.Read_U32(addr);
if (res != inmem) if (res != inmem)
{ {
INFO_LOG_FMT(POWERPC, INFO_LOG_FMT(POWERPC,

View File

@ -194,17 +194,19 @@ static void DoState(PointerWrap& p)
} }
// Check to make sure the emulated memory sizes are the same as the savestate // Check to make sure the emulated memory sizes are the same as the savestate
u32 state_mem1_size = Memory::GetRamSizeReal(); auto& system = Core::System::GetInstance();
u32 state_mem2_size = Memory::GetExRamSizeReal(); auto& memory = system.GetMemory();
u32 state_mem1_size = memory.GetRamSizeReal();
u32 state_mem2_size = memory.GetExRamSizeReal();
p.Do(state_mem1_size); p.Do(state_mem1_size);
p.Do(state_mem2_size); p.Do(state_mem2_size);
if (state_mem1_size != Memory::GetRamSizeReal() || state_mem2_size != Memory::GetExRamSizeReal()) if (state_mem1_size != memory.GetRamSizeReal() || state_mem2_size != memory.GetExRamSizeReal())
{ {
OSD::AddMessage(fmt::format("Memory size mismatch!\n" OSD::AddMessage(fmt::format("Memory size mismatch!\n"
"Current | MEM1 {:08X} ({:3}MB) MEM2 {:08X} ({:3}MB)\n" "Current | MEM1 {:08X} ({:3}MB) MEM2 {:08X} ({:3}MB)\n"
"State | MEM1 {:08X} ({:3}MB) MEM2 {:08X} ({:3}MB)", "State | MEM1 {:08X} ({:3}MB) MEM2 {:08X} ({:3}MB)",
Memory::GetRamSizeReal(), Memory::GetRamSizeReal() / 0x100000U, memory.GetRamSizeReal(), memory.GetRamSizeReal() / 0x100000U,
Memory::GetExRamSizeReal(), Memory::GetExRamSizeReal() / 0x100000U, memory.GetExRamSizeReal(), memory.GetExRamSizeReal() / 0x100000U,
state_mem1_size, state_mem1_size / 0x100000U, state_mem2_size, state_mem1_size, state_mem1_size / 0x100000U, state_mem2_size,
state_mem2_size / 0x100000U)); state_mem2_size / 0x100000U));
p.SetMeasureMode(); p.SetMeasureMode();
@ -225,7 +227,7 @@ static void DoState(PointerWrap& p)
p.DoMarker("PowerPC"); p.DoMarker("PowerPC");
// CoreTiming needs to be restored before restoring Hardware because // CoreTiming needs to be restored before restoring Hardware because
// the controller code might need to schedule an event if the controller has changed. // the controller code might need to schedule an event if the controller has changed.
Core::System::GetInstance().GetCoreTiming().DoState(p); system.GetCoreTiming().DoState(p);
p.DoMarker("CoreTiming"); p.DoMarker("CoreTiming");
HW::DoState(p); HW::DoState(p);
p.DoMarker("HW"); p.DoMarker("HW");

View File

@ -13,6 +13,7 @@
#include "Core/HW/DVD/DVDInterface.h" #include "Core/HW/DVD/DVDInterface.h"
#include "Core/HW/DVD/DVDThread.h" #include "Core/HW/DVD/DVDThread.h"
#include "Core/HW/EXI/EXI.h" #include "Core/HW/EXI/EXI.h"
#include "Core/HW/Memmap.h"
#include "Core/HW/MemoryInterface.h" #include "Core/HW/MemoryInterface.h"
#include "Core/HW/SI/SI.h" #include "Core/HW/SI/SI.h"
#include "Core/HW/Sram.h" #include "Core/HW/Sram.h"
@ -34,6 +35,7 @@ struct System::Impl
DVDInterface::DVDInterfaceState m_dvd_interface_state; DVDInterface::DVDInterfaceState m_dvd_interface_state;
DVDThread::DVDThreadState m_dvd_thread_state; DVDThread::DVDThreadState m_dvd_thread_state;
ExpansionInterface::ExpansionInterfaceState m_expansion_interface_state; ExpansionInterface::ExpansionInterfaceState m_expansion_interface_state;
Memory::MemoryManager m_memory;
MemoryInterface::MemoryInterfaceState m_memory_interface_state; MemoryInterface::MemoryInterfaceState m_memory_interface_state;
SerialInterface::SerialInterfaceState m_serial_interface_state; SerialInterface::SerialInterfaceState m_serial_interface_state;
Sram m_sram; Sram m_sram;
@ -118,6 +120,11 @@ ExpansionInterface::ExpansionInterfaceState& System::GetExpansionInterfaceState(
return m_impl->m_expansion_interface_state; return m_impl->m_expansion_interface_state;
} }
Memory::MemoryManager& System::GetMemory() const
{
return m_impl->m_memory;
}
MemoryInterface::MemoryInterfaceState& System::GetMemoryInterfaceState() const MemoryInterface::MemoryInterfaceState& System::GetMemoryInterfaceState() const
{ {
return m_impl->m_memory_interface_state; return m_impl->m_memory_interface_state;

View File

@ -36,6 +36,10 @@ namespace ExpansionInterface
{ {
class ExpansionInterfaceState; class ExpansionInterfaceState;
}; };
namespace Memory
{
class MemoryManager;
};
namespace MemoryInterface namespace MemoryInterface
{ {
class MemoryInterfaceState; class MemoryInterfaceState;
@ -90,6 +94,7 @@ public:
DVDInterface::DVDInterfaceState& GetDVDInterfaceState() const; DVDInterface::DVDInterfaceState& GetDVDInterfaceState() const;
DVDThread::DVDThreadState& GetDVDThreadState() const; DVDThread::DVDThreadState& GetDVDThreadState() const;
ExpansionInterface::ExpansionInterfaceState& GetExpansionInterfaceState() const; ExpansionInterface::ExpansionInterfaceState& GetExpansionInterfaceState() const;
Memory::MemoryManager& GetMemory() const;
MemoryInterface::MemoryInterfaceState& GetMemoryInterfaceState() const; MemoryInterface::MemoryInterfaceState& GetMemoryInterfaceState() const;
SerialInterface::SerialInterfaceState& GetSerialInterfaceState() const; SerialInterface::SerialInterfaceState& GetSerialInterfaceState() const;
Sram& GetSRAM() const; Sram& GetSRAM() const;

View File

@ -18,6 +18,7 @@
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/IOS/FS/FileSystem.h" #include "Core/IOS/FS/FileSystem.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/System.h"
#include "DiscIO/DirectoryBlob.h" #include "DiscIO/DirectoryBlob.h"
#include "DiscIO/RiivolutionParser.h" #include "DiscIO/RiivolutionParser.h"
@ -582,6 +583,9 @@ static void ApplyOcarinaMemoryPatch(const Patch& patch, const Memory& memory_pat
void ApplyGeneralMemoryPatches(const std::vector<Patch>& patches) void ApplyGeneralMemoryPatches(const std::vector<Patch>& patches)
{ {
auto& system = Core::System::GetInstance();
auto& system_memory = system.GetMemory();
for (const auto& patch : patches) for (const auto& patch : patches)
{ {
for (const auto& memory : patch.m_memory_patches) for (const auto& memory : patch.m_memory_patches)
@ -590,7 +594,7 @@ void ApplyGeneralMemoryPatches(const std::vector<Patch>& patches)
continue; continue;
if (memory.m_search) if (memory.m_search)
ApplySearchMemoryPatch(patch, memory, 0x80000000, ::Memory::GetRamSize()); ApplySearchMemoryPatch(patch, memory, 0x80000000, system_memory.GetRamSize());
else else
ApplyMemoryPatch(patch, memory); ApplyMemoryPatch(patch, memory);
} }

View File

@ -23,6 +23,7 @@
#include "Core/Core.h" #include "Core/Core.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/PowerPC/MMU.h" #include "Core/PowerPC/MMU.h"
#include "Core/System.h"
#include "DolphinQt/QtUtils/ModalMessageBox.h" #include "DolphinQt/QtUtils/ModalMessageBox.h"
#include "DolphinQt/QtUtils/NonDefaultQPushButton.h" #include "DolphinQt/QtUtils/NonDefaultQPushButton.h"
@ -164,9 +165,11 @@ void CheatSearchFactoryWidget::OnNewSearchClicked()
return; return;
} }
memory_ranges.emplace_back(0x80000000, Memory::GetRamSizeReal()); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory_ranges.emplace_back(0x80000000, memory.GetRamSizeReal());
if (SConfig::GetInstance().bWii) if (SConfig::GetInstance().bWii)
memory_ranges.emplace_back(0x90000000, Memory::GetExRamSizeReal()); memory_ranges.emplace_back(0x90000000, memory.GetExRamSizeReal());
address_space = PowerPC::RequestedAddressSpace::Virtual; address_space = PowerPC::RequestedAddressSpace::Virtual;
} }
else else

View File

@ -43,6 +43,7 @@
#include "Core/PowerPC/PowerPC.h" #include "Core/PowerPC/PowerPC.h"
#include "Core/PowerPC/SignatureDB/SignatureDB.h" #include "Core/PowerPC/SignatureDB/SignatureDB.h"
#include "Core/State.h" #include "Core/State.h"
#include "Core/System.h"
#include "Core/TitleDatabase.h" #include "Core/TitleDatabase.h"
#include "Core/WiiUtils.h" #include "Core/WiiUtils.h"
@ -1194,15 +1195,21 @@ void MenuBar::ClearSymbols()
void MenuBar::GenerateSymbolsFromAddress() void MenuBar::GenerateSymbolsFromAddress()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR, PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR,
Memory::MEM1_BASE_ADDR + Memory::GetRamSizeReal(), &g_symbolDB); Memory::MEM1_BASE_ADDR + memory.GetRamSizeReal(), &g_symbolDB);
emit NotifySymbolsUpdated(); emit NotifySymbolsUpdated();
} }
void MenuBar::GenerateSymbolsFromSignatureDB() void MenuBar::GenerateSymbolsFromSignatureDB()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR, PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR,
Memory::MEM1_BASE_ADDR + Memory::GetRamSizeReal(), &g_symbolDB); Memory::MEM1_BASE_ADDR + memory.GetRamSizeReal(), &g_symbolDB);
SignatureDB db(SignatureDB::HandlerType::DSY); SignatureDB db(SignatureDB::HandlerType::DSY);
if (db.Load(File::GetSysDirectory() + TOTALDB)) if (db.Load(File::GetSysDirectory() + TOTALDB))
{ {
@ -1409,6 +1416,9 @@ RSOVector MenuBar::DetectRSOModules(ParallelProgressDialog& progress)
void MenuBar::LoadSymbolMap() void MenuBar::LoadSymbolMap()
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
std::string existing_map_file, writable_map_file; std::string existing_map_file, writable_map_file;
bool map_exists = CBoot::FindMapFile(&existing_map_file, &writable_map_file); bool map_exists = CBoot::FindMapFile(&existing_map_file, &writable_map_file);
@ -1416,7 +1426,7 @@ void MenuBar::LoadSymbolMap()
{ {
g_symbolDB.Clear(); g_symbolDB.Clear();
PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR + 0x1300000, PPCAnalyst::FindFunctions(Memory::MEM1_BASE_ADDR + 0x1300000,
Memory::MEM1_BASE_ADDR + Memory::GetRamSizeReal(), &g_symbolDB); Memory::MEM1_BASE_ADDR + memory.GetRamSizeReal(), &g_symbolDB);
SignatureDB db(SignatureDB::HandlerType::DSY); SignatureDB db(SignatureDB::HandlerType::DSY);
if (db.Load(File::GetSysDirectory() + TOTALDB)) if (db.Load(File::GetSysDirectory() + TOTALDB))
db.Apply(&g_symbolDB); db.Apply(&g_symbolDB);
@ -1657,8 +1667,11 @@ void MenuBar::SearchInstruction()
if (!good) if (!good)
return; return;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
bool found = false; bool found = false;
for (u32 addr = Memory::MEM1_BASE_ADDR; addr < Memory::MEM1_BASE_ADDR + Memory::GetRamSizeReal(); for (u32 addr = Memory::MEM1_BASE_ADDR; addr < Memory::MEM1_BASE_ADDR + memory.GetRamSizeReal();
addr += 4) addr += 4)
{ {
const auto ins_name = const auto ins_name =

View File

@ -9,6 +9,7 @@
#include "Common/CommonTypes.h" #include "Common/CommonTypes.h"
#include "Common/MsgHandler.h" #include "Common/MsgHandler.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/BPMemory.h" #include "VideoCommon/BPMemory.h"
#include "VideoCommon/TextureDecoder.h" #include "VideoCommon/TextureDecoder.h"
@ -132,8 +133,11 @@ void SampleMip(s32 s, s32 t, s32 mip, bool linear, u8 texmap, u8* sample)
} }
else else
{ {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u32 imageBase = texUnit.texImage3.image_base << 5; const u32 imageBase = texUnit.texImage3.image_base << 5;
imageSrc = Memory::GetPointer(imageBase); imageSrc = memory.GetPointer(imageBase);
} }
int image_width_minus_1 = ti0.width; int image_width_minus_1 = ti0.width;

View File

@ -357,7 +357,9 @@ static void BPWritten(const BPCmd& bp, int cycles_into_future)
if (!SConfig::GetInstance().bWii) if (!SConfig::GetInstance().bWii)
addr = addr & 0x01FFFFFF; addr = addr & 0x01FFFFFF;
Memory::CopyFromEmu(texMem + tlutTMemAddr, addr, tlutXferCount); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(texMem + tlutTMemAddr, addr, tlutXferCount);
if (OpcodeDecoder::g_record_fifo_data) if (OpcodeDecoder::g_record_fifo_data)
FifoRecorder::GetInstance().UseMemory(addr, tlutXferCount, MemoryUpdate::TMEM); FifoRecorder::GetInstance().UseMemory(addr, tlutXferCount, MemoryUpdate::TMEM);
@ -549,11 +551,15 @@ static void BPWritten(const BPCmd& bp, int cycles_into_future)
if (tmem_addr_even + bytes_read > TMEM_SIZE) if (tmem_addr_even + bytes_read > TMEM_SIZE)
bytes_read = TMEM_SIZE - tmem_addr_even; bytes_read = TMEM_SIZE - tmem_addr_even;
Memory::CopyFromEmu(texMem + tmem_addr_even, src_addr, bytes_read); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(texMem + tmem_addr_even, src_addr, bytes_read);
} }
else // RGBA8 tiles (and CI14, but that might just be stupid libogc!) else // RGBA8 tiles (and CI14, but that might just be stupid libogc!)
{ {
u8* src_ptr = Memory::GetPointer(src_addr); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* src_ptr = memory.GetPointer(src_addr);
// AR and GB tiles are stored in separate TMEM banks => can't use a single memcpy for // AR and GB tiles are stored in separate TMEM banks => can't use a single memcpy for
// everything // everything

View File

@ -270,7 +270,9 @@ static void ReadDataFromFifo(u32 readPtr)
s_video_buffer_read_ptr = s_video_buffer; s_video_buffer_read_ptr = s_video_buffer;
} }
// Copy new video instructions to s_video_buffer for future use in rendering the new picture // Copy new video instructions to s_video_buffer for future use in rendering the new picture
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE);
s_video_buffer_write_ptr += GPFifo::GATHER_PIPE_SIZE; s_video_buffer_write_ptr += GPFifo::GATHER_PIPE_SIZE;
} }
@ -303,7 +305,9 @@ static void ReadDataFromFifoOnCPU(u32 readPtr)
return; return;
} }
} }
Memory::CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
memory.CopyFromEmu(s_video_buffer_write_ptr, readPtr, GPFifo::GATHER_PIPE_SIZE);
s_video_buffer_pp_read_ptr = OpcodeDecoder::RunFifo<true>( s_video_buffer_pp_read_ptr = OpcodeDecoder::RunFifo<true>(
DataReader(s_video_buffer_pp_read_ptr, write_ptr + GPFifo::GATHER_PIPE_SIZE), nullptr); DataReader(s_video_buffer_pp_read_ptr, write_ptr + GPFifo::GATHER_PIPE_SIZE), nullptr);
// This would have to be locked if the GPU thread didn't spin. // This would have to be locked if the GPU thread didn't spin.

View File

@ -153,7 +153,9 @@ public:
if constexpr (is_preprocess) if constexpr (is_preprocess)
{ {
const u8* const start_address = Memory::GetPointer(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8* const start_address = memory.GetPointer(address);
Fifo::PushFifoAuxBuffer(start_address, size); Fifo::PushFifoAuxBuffer(start_address, size);
@ -167,11 +169,17 @@ public:
const u8* start_address; const u8* start_address;
if (Fifo::UseDeterministicGPUThread()) if (Fifo::UseDeterministicGPUThread())
{
start_address = static_cast<u8*>(Fifo::PopFifoAuxBuffer(size)); start_address = static_cast<u8*>(Fifo::PopFifoAuxBuffer(size));
}
else else
start_address = Memory::GetPointer(address); {
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
start_address = memory.GetPointer(address);
}
// Avoid the crash if Memory::GetPointer failed .. // Avoid the crash if memory.GetPointer failed ..
if (start_address != nullptr) if (start_address != nullptr)
{ {
// temporarily swap dl and non-dl (small "hack" for the stats) // temporarily swap dl and non-dl (small "hack" for the stats)

View File

@ -31,6 +31,7 @@
#include "Core/FifoPlayer/FifoPlayer.h" #include "Core/FifoPlayer/FifoPlayer.h"
#include "Core/FifoPlayer/FifoRecorder.h" #include "Core/FifoPlayer/FifoRecorder.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/AbstractFramebuffer.h" #include "VideoCommon/AbstractFramebuffer.h"
#include "VideoCommon/AbstractStagingTexture.h" #include "VideoCommon/AbstractStagingTexture.h"
@ -1732,7 +1733,9 @@ TextureCacheBase::TCacheEntry*
TextureCacheBase::GetXFBTexture(u32 address, u32 width, u32 height, u32 stride, TextureCacheBase::GetXFBTexture(u32 address, u32 width, u32 height, u32 stride,
MathUtil::Rectangle<int>* display_rect) MathUtil::Rectangle<int>* display_rect)
{ {
const u8* src_data = Memory::GetPointer(address); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8* src_data = memory.GetPointer(address);
if (!src_data) if (!src_data)
{ {
ERROR_LOG_FMT(VIDEO, "Trying to load XFB texture from invalid address {:#010x}", address); ERROR_LOG_FMT(VIDEO, "Trying to load XFB texture from invalid address {:#010x}", address);
@ -2107,7 +2110,9 @@ void TextureCacheBase::CopyRenderTargetToTexture(
!(is_xfb_copy ? g_ActiveConfig.bSkipXFBCopyToRam : g_ActiveConfig.bSkipEFBCopyToRam) || !(is_xfb_copy ? g_ActiveConfig.bSkipXFBCopyToRam : g_ActiveConfig.bSkipEFBCopyToRam) ||
!copy_to_vram; !copy_to_vram;
u8* dst = Memory::GetPointer(dstAddr); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* dst = memory.GetPointer(dstAddr);
if (dst == nullptr) if (dst == nullptr)
{ {
ERROR_LOG_FMT(VIDEO, "Trying to copy from EFB to invalid address {:#010x}", dstAddr); ERROR_LOG_FMT(VIDEO, "Trying to copy from EFB to invalid address {:#010x}", dstAddr);
@ -2422,7 +2427,9 @@ void TextureCacheBase::WriteEFBCopyToRAM(u8* dst_ptr, u32 width, u32 height, u32
void TextureCacheBase::FlushEFBCopy(TCacheEntry* entry) void TextureCacheBase::FlushEFBCopy(TCacheEntry* entry)
{ {
// Copy from texture -> guest memory. // Copy from texture -> guest memory.
u8* const dst = Memory::GetPointer(entry->addr); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* const dst = memory.GetPointer(entry->addr);
WriteEFBCopyToRAM(dst, entry->pending_efb_copy_width, entry->pending_efb_copy_height, WriteEFBCopyToRAM(dst, entry->pending_efb_copy_width, entry->pending_efb_copy_height,
entry->memory_stride, std::move(entry->pending_efb_copy)); entry->memory_stride, std::move(entry->pending_efb_copy));
@ -3024,7 +3031,9 @@ u64 TextureCacheBase::TCacheEntry::CalculateHash() const
const u32 hash_sample_size = HashSampleSize(); const u32 hash_sample_size = HashSampleSize();
// FIXME: textures from tmem won't get the correct hash. // FIXME: textures from tmem won't get the correct hash.
u8* ptr = Memory::GetPointer(addr); auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
u8* ptr = memory.GetPointer(addr);
if (memory_stride == bytes_per_row) if (memory_stride == bytes_per_row)
{ {
return Common::GetHash64(ptr, size_in_bytes, hash_sample_size); return Common::GetHash64(ptr, size_in_bytes, hash_sample_size);

View File

@ -8,6 +8,7 @@
#include "Common/Align.h" #include "Common/Align.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/BPMemory.h" #include "VideoCommon/BPMemory.h"
#include "VideoCommon/TextureDecoder.h" #include "VideoCommon/TextureDecoder.h"
@ -44,7 +45,9 @@ TextureInfo TextureInfo::FromStage(u32 stage)
&texMem[tmem_address_even], mip_count); &texMem[tmem_address_even], mip_count);
} }
return TextureInfo(stage, Memory::GetPointer(address), tlut_ptr, address, texture_format, auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
return TextureInfo(stage, memory.GetPointer(address), tlut_ptr, address, texture_format,
tlut_format, width, height, false, nullptr, nullptr, mip_count); tlut_format, width, height, false, nullptr, nullptr, mip_count);
} }

View File

@ -18,6 +18,7 @@
#include "Core/DolphinAnalytics.h" #include "Core/DolphinAnalytics.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/BPMemory.h" #include "VideoCommon/BPMemory.h"
#include "VideoCommon/CPMemory.h" #include "VideoCommon/CPMemory.h"
@ -81,6 +82,9 @@ void UpdateVertexArrayPointers()
if (!g_bases_dirty) [[likely]] if (!g_bases_dirty) [[likely]]
return; return;
auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
// Some games such as Burnout 2 can put invalid addresses into // Some games such as Burnout 2 can put invalid addresses into
// the array base registers. (see issue 8591) // the array base registers. (see issue 8591)
// But the vertex arrays with invalid addresses aren't actually enabled. // But the vertex arrays with invalid addresses aren't actually enabled.
@ -89,24 +93,24 @@ void UpdateVertexArrayPointers()
// We also only update the array base if the vertex description states we are going to use it. // We also only update the array base if the vertex description states we are going to use it.
if (IsIndexed(g_main_cp_state.vtx_desc.low.Position)) if (IsIndexed(g_main_cp_state.vtx_desc.low.Position))
cached_arraybases[CPArray::Position] = cached_arraybases[CPArray::Position] =
Memory::GetPointer(g_main_cp_state.array_bases[CPArray::Position]); memory.GetPointer(g_main_cp_state.array_bases[CPArray::Position]);
if (IsIndexed(g_main_cp_state.vtx_desc.low.Normal)) if (IsIndexed(g_main_cp_state.vtx_desc.low.Normal))
cached_arraybases[CPArray::Normal] = cached_arraybases[CPArray::Normal] =
Memory::GetPointer(g_main_cp_state.array_bases[CPArray::Normal]); memory.GetPointer(g_main_cp_state.array_bases[CPArray::Normal]);
for (u8 i = 0; i < g_main_cp_state.vtx_desc.low.Color.Size(); i++) for (u8 i = 0; i < g_main_cp_state.vtx_desc.low.Color.Size(); i++)
{ {
if (IsIndexed(g_main_cp_state.vtx_desc.low.Color[i])) if (IsIndexed(g_main_cp_state.vtx_desc.low.Color[i]))
cached_arraybases[CPArray::Color0 + i] = cached_arraybases[CPArray::Color0 + i] =
Memory::GetPointer(g_main_cp_state.array_bases[CPArray::Color0 + i]); memory.GetPointer(g_main_cp_state.array_bases[CPArray::Color0 + i]);
} }
for (u8 i = 0; i < g_main_cp_state.vtx_desc.high.TexCoord.Size(); i++) for (u8 i = 0; i < g_main_cp_state.vtx_desc.high.TexCoord.Size(); i++)
{ {
if (IsIndexed(g_main_cp_state.vtx_desc.high.TexCoord[i])) if (IsIndexed(g_main_cp_state.vtx_desc.high.TexCoord[i]))
cached_arraybases[CPArray::TexCoord0 + i] = cached_arraybases[CPArray::TexCoord0 + i] =
Memory::GetPointer(g_main_cp_state.array_bases[CPArray::TexCoord0 + i]); memory.GetPointer(g_main_cp_state.array_bases[CPArray::TexCoord0 + i]);
} }
g_bases_dirty = false; g_bases_dirty = false;

View File

@ -10,6 +10,7 @@
#include "Core/DolphinAnalytics.h" #include "Core/DolphinAnalytics.h"
#include "Core/HW/Memmap.h" #include "Core/HW/Memmap.h"
#include "Core/System.h"
#include "VideoCommon/CPMemory.h" #include "VideoCommon/CPMemory.h"
#include "VideoCommon/Fifo.h" #include "VideoCommon/Fifo.h"
@ -262,7 +263,9 @@ void LoadIndexedXF(CPArray array, u32 index, u16 address, u8 size)
} }
else else
{ {
newData = (u32*)Memory::GetPointer(g_main_cp_state.array_bases[array] + auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
newData = (u32*)memory.GetPointer(g_main_cp_state.array_bases[array] +
g_main_cp_state.array_strides[array] * index); g_main_cp_state.array_strides[array] * index);
} }
bool changed = false; bool changed = false;
@ -284,7 +287,9 @@ void LoadIndexedXF(CPArray array, u32 index, u16 address, u8 size)
void PreprocessIndexedXF(CPArray array, u32 index, u16 address, u8 size) void PreprocessIndexedXF(CPArray array, u32 index, u16 address, u8 size)
{ {
const u8* new_data = Memory::GetPointer(g_preprocess_cp_state.array_bases[array] + auto& system = Core::System::GetInstance();
auto& memory = system.GetMemory();
const u8* new_data = memory.GetPointer(g_preprocess_cp_state.array_bases[array] +
g_preprocess_cp_state.array_strides[array] * index); g_preprocess_cp_state.array_strides[array] * index);
const size_t buf_size = size * sizeof(u32); const size_t buf_size = size * sizeof(u32);