mirror of
https://github.com/Ryujinx/Ryujinx.git
synced 2024-11-10 14:21:43 +00:00
95017b8c66
* Back to the origins: Make memory manager take guest PA rather than host address once again * Direct mapping with alias support on Windows * Fixes and remove more of the emulated shared memory * Linux support * Make shared and transfer memory not depend on SharedMemoryStorage * More efficient view mapping on Windows (no more restricted to 4KB pages at a time) * Handle potential access violations caused by partial unmap * Implement host mapping using shared memory on Linux * Add new GetPhysicalAddressChecked method, used to ensure the virtual address is mapped before address translation Also align GetRef behaviour with software memory manager * We don't need a mirrorable memory block for software memory manager mode * Disable memory aliasing tests while we don't have shared memory support on Mac * Shared memory & SIGBUS handler for macOS * Fix typo + nits + re-enable memory tests * Set MAP_JIT_DARWIN on x86 Mac too * Add back the address space mirror * Only set MAP_JIT_DARWIN if we are mapping as executable * Disable aliasing tests again (still fails on Mac) * Fix UnmapView4KB (by not casting size to int) * Use ref counting on memory blocks to delay closing the shared memory handle until all blocks using it are disposed * Address PR feedback * Make RO hold a reference to the guest process memory manager to avoid early disposal Co-authored-by: nastys <nastys@users.noreply.github.com>
84 lines
No EOL
2.6 KiB
C#
84 lines
No EOL
2.6 KiB
C#
using NUnit.Framework;
|
|
using System;
|
|
using System.Runtime.InteropServices;
|
|
|
|
namespace Ryujinx.Memory.Tests
|
|
{
|
|
public class Tests
|
|
{
|
|
private const ulong MemorySize = 0x8000;
|
|
|
|
private MemoryBlock _memoryBlock;
|
|
|
|
[SetUp]
|
|
public void Setup()
|
|
{
|
|
_memoryBlock = new MemoryBlock(MemorySize);
|
|
}
|
|
|
|
[TearDown]
|
|
public void Teardown()
|
|
{
|
|
_memoryBlock.Dispose();
|
|
}
|
|
|
|
[Test]
|
|
public void Test_Read()
|
|
{
|
|
Marshal.WriteInt32(_memoryBlock.Pointer, 0x2020, 0x1234abcd);
|
|
|
|
Assert.AreEqual(_memoryBlock.Read<int>(0x2020), 0x1234abcd);
|
|
}
|
|
|
|
[Test]
|
|
public void Test_Write()
|
|
{
|
|
_memoryBlock.Write(0x2040, 0xbadc0de);
|
|
|
|
Assert.AreEqual(Marshal.ReadInt32(_memoryBlock.Pointer, 0x2040), 0xbadc0de);
|
|
}
|
|
|
|
[Test, Explicit]
|
|
public void Test_Alias()
|
|
{
|
|
using MemoryBlock backing = new MemoryBlock(0x10000, MemoryAllocationFlags.Mirrorable);
|
|
using MemoryBlock toAlias = new MemoryBlock(0x10000, MemoryAllocationFlags.Reserve | MemoryAllocationFlags.ViewCompatible);
|
|
|
|
toAlias.MapView(backing, 0x1000, 0, 0x4000);
|
|
toAlias.UnmapView(backing, 0x3000, 0x1000);
|
|
|
|
toAlias.Write(0, 0xbadc0de);
|
|
Assert.AreEqual(Marshal.ReadInt32(backing.Pointer, 0x1000), 0xbadc0de);
|
|
}
|
|
|
|
[Test, Explicit]
|
|
public void Test_AliasRandom()
|
|
{
|
|
using MemoryBlock backing = new MemoryBlock(0x80000, MemoryAllocationFlags.Mirrorable);
|
|
using MemoryBlock toAlias = new MemoryBlock(0x80000, MemoryAllocationFlags.Reserve | MemoryAllocationFlags.ViewCompatible);
|
|
|
|
Random rng = new Random(123);
|
|
|
|
for (int i = 0; i < 20000; i++)
|
|
{
|
|
int srcPage = rng.Next(0, 64);
|
|
int dstPage = rng.Next(0, 64);
|
|
int pages = rng.Next(1, 65);
|
|
|
|
if ((rng.Next() & 1) != 0)
|
|
{
|
|
toAlias.MapView(backing, (ulong)srcPage << 12, (ulong)dstPage << 12, (ulong)pages << 12);
|
|
|
|
int offset = rng.Next(0, 0x1000 - sizeof(int));
|
|
|
|
toAlias.Write((ulong)((dstPage << 12) + offset), 0xbadc0de);
|
|
Assert.AreEqual(Marshal.ReadInt32(backing.Pointer, (srcPage << 12) + offset), 0xbadc0de);
|
|
}
|
|
else
|
|
{
|
|
toAlias.UnmapView(backing, (ulong)dstPage << 12, (ulong)pages << 12);
|
|
}
|
|
}
|
|
}
|
|
}
|
|
} |