diff --git a/.github/workflows/release.yml b/.github/workflows/release.yml
index 9843487a3a..f2bebc77fc 100644
--- a/.github/workflows/release.yml
+++ b/.github/workflows/release.yml
@@ -71,7 +71,7 @@ jobs:
- uses: actions/setup-dotnet@v4
with:
global-json-file: global.json
-
+
- name: Overwrite csc problem matcher
run: echo "::add-matcher::.github/csc.json"
@@ -104,37 +104,30 @@ jobs:
if: matrix.platform.os == 'windows-latest'
run: |
pushd publish_ava
+ cp publish/Ryujinx.exe publish/Ryujinx.Ava.exe
7z a ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.zip publish
+ 7z a ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.zip publish
popd
pushd publish_sdl2_headless
7z a ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.zip publish
popd
-
- pushd publish_ava
- mv publish/Ryujinx.exe publish/Ryujinx.Ava.exe
- 7z a ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.zip publish
- popd
shell: bash
- name: Packing Linux builds
if: matrix.platform.os == 'ubuntu-latest'
run: |
pushd publish_ava
- chmod +x publish/Ryujinx.sh publish/Ryujinx
+ cp publish/Ryujinx publish/Ryujinx.Ava
+ chmod +x publish/Ryujinx.sh publish/Ryujinx.Ava publish/Ryujinx
tar -czvf ../release_output/ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.tar.gz publish
+ tar -czvf ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.tar.gz publish
popd
pushd publish_sdl2_headless
chmod +x publish/Ryujinx.sh publish/Ryujinx.Headless.SDL2
tar -czvf ../release_output/sdl2-ryujinx-headless-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.tar.gz publish
popd
-
- pushd publish_ava
- mv publish/Ryujinx publish/Ryujinx.Ava
- chmod +x publish/Ryujinx.sh publish/Ryujinx.Ava
- tar -czvf ../release_output/test-ava-ryujinx-${{ steps.version_info.outputs.build_version }}-${{ matrix.platform.zip_os_name }}.tar.gz publish
- popd
shell: bash
- name: Pushing new release
diff --git a/Directory.Packages.props b/Directory.Packages.props
index 47904c3fae..609915cb4c 100644
--- a/Directory.Packages.props
+++ b/Directory.Packages.props
@@ -3,13 +3,13 @@
true
-
-
-
-
-
-
-
+
+
+
+
+
+
+
@@ -19,9 +19,9 @@
-
+
-
+
@@ -43,10 +43,9 @@
-
-
+
+
-
diff --git a/src/ARMeilleure/Instructions/NativeInterface.cs b/src/ARMeilleure/Instructions/NativeInterface.cs
index d1b2e353c5..0cd3754f7f 100644
--- a/src/ARMeilleure/Instructions/NativeInterface.cs
+++ b/src/ARMeilleure/Instructions/NativeInterface.cs
@@ -91,54 +91,54 @@ namespace ARMeilleure.Instructions
#region "Read"
public static byte ReadByte(ulong address)
{
- return GetMemoryManager().ReadTracked(address);
+ return GetMemoryManager().ReadGuest(address);
}
public static ushort ReadUInt16(ulong address)
{
- return GetMemoryManager().ReadTracked(address);
+ return GetMemoryManager().ReadGuest(address);
}
public static uint ReadUInt32(ulong address)
{
- return GetMemoryManager().ReadTracked(address);
+ return GetMemoryManager().ReadGuest(address);
}
public static ulong ReadUInt64(ulong address)
{
- return GetMemoryManager().ReadTracked(address);
+ return GetMemoryManager().ReadGuest(address);
}
public static V128 ReadVector128(ulong address)
{
- return GetMemoryManager().ReadTracked(address);
+ return GetMemoryManager().ReadGuest(address);
}
#endregion
#region "Write"
public static void WriteByte(ulong address, byte value)
{
- GetMemoryManager().Write(address, value);
+ GetMemoryManager().WriteGuest(address, value);
}
public static void WriteUInt16(ulong address, ushort value)
{
- GetMemoryManager().Write(address, value);
+ GetMemoryManager().WriteGuest(address, value);
}
public static void WriteUInt32(ulong address, uint value)
{
- GetMemoryManager().Write(address, value);
+ GetMemoryManager().WriteGuest(address, value);
}
public static void WriteUInt64(ulong address, ulong value)
{
- GetMemoryManager().Write(address, value);
+ GetMemoryManager().WriteGuest(address, value);
}
public static void WriteVector128(ulong address, V128 value)
{
- GetMemoryManager().Write(address, value);
+ GetMemoryManager().WriteGuest(address, value);
}
#endregion
diff --git a/src/ARMeilleure/Memory/IMemoryManager.cs b/src/ARMeilleure/Memory/IMemoryManager.cs
index 952cd2b4f0..46d4426559 100644
--- a/src/ARMeilleure/Memory/IMemoryManager.cs
+++ b/src/ARMeilleure/Memory/IMemoryManager.cs
@@ -28,6 +28,17 @@ namespace ARMeilleure.Memory
/// The data
T ReadTracked(ulong va) where T : unmanaged;
+ ///
+ /// Reads data from CPU mapped memory, from guest code. (with read tracking)
+ ///
+ /// Type of the data being read
+ /// Virtual address of the data in memory
+ /// The data
+ T ReadGuest(ulong va) where T : unmanaged
+ {
+ return ReadTracked(va);
+ }
+
///
/// Writes data to CPU mapped memory.
///
@@ -36,6 +47,17 @@ namespace ARMeilleure.Memory
/// Data to be written
void Write(ulong va, T value) where T : unmanaged;
+ ///
+ /// Writes data to CPU mapped memory, from guest code.
+ ///
+ /// Type of the data being written
+ /// Virtual address to write the data into
+ /// Data to be written
+ void WriteGuest(ulong va, T value) where T : unmanaged
+ {
+ Write(va, value);
+ }
+
///
/// Gets a read-only span of data from CPU mapped memory.
///
diff --git a/src/Ryujinx.Common/Utilities/FileSystemUtils.cs b/src/Ryujinx.Common/Utilities/FileSystemUtils.cs
index e76c2b60bf..a57fa8a788 100644
--- a/src/Ryujinx.Common/Utilities/FileSystemUtils.cs
+++ b/src/Ryujinx.Common/Utilities/FileSystemUtils.cs
@@ -1,4 +1,6 @@
+using System.Collections.Generic;
using System.IO;
+using System.Linq;
namespace Ryujinx.Common.Utilities
{
@@ -44,5 +46,11 @@ namespace Ryujinx.Common.Utilities
CopyDirectory(sourceDir, destinationDir, true);
Directory.Delete(sourceDir, true);
}
+
+ public static string SanitizeFileName(string fileName)
+ {
+ var reservedChars = new HashSet(Path.GetInvalidFileNameChars());
+ return string.Concat(fileName.Select(c => reservedChars.Contains(c) ? '_' : c));
+ }
}
}
diff --git a/src/Ryujinx.Cpu/AppleHv/HvMemoryManager.cs b/src/Ryujinx.Cpu/AppleHv/HvMemoryManager.cs
index 2f9743ab45..80f7c8a1f8 100644
--- a/src/Ryujinx.Cpu/AppleHv/HvMemoryManager.cs
+++ b/src/Ryujinx.Cpu/AppleHv/HvMemoryManager.cs
@@ -8,7 +8,6 @@ using System.Linq;
using System.Runtime.CompilerServices;
using System.Runtime.InteropServices;
using System.Runtime.Versioning;
-using System.Threading;
namespace Ryujinx.Cpu.AppleHv
{
@@ -16,31 +15,10 @@ namespace Ryujinx.Cpu.AppleHv
/// Represents a CPU memory manager which maps guest virtual memory directly onto the Hypervisor page table.
///
[SupportedOSPlatform("macos")]
- public class HvMemoryManager : MemoryManagerBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
+ public class HvMemoryManager : VirtualMemoryManagerRefCountedBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
{
- public const int PageBits = 12;
- public const int PageSize = 1 << PageBits;
- public const int PageMask = PageSize - 1;
-
- public const int PageToPteShift = 5; // 32 pages (2 bits each) in one ulong page table entry.
- public const ulong BlockMappedMask = 0x5555555555555555; // First bit of each table entry set.
-
- private enum HostMappedPtBits : ulong
- {
- Unmapped = 0,
- Mapped,
- WriteTracked,
- ReadWriteTracked,
-
- MappedReplicated = 0x5555555555555555,
- WriteTrackedReplicated = 0xaaaaaaaaaaaaaaaa,
- ReadWriteTrackedReplicated = ulong.MaxValue,
- }
-
private readonly InvalidAccessHandler _invalidAccessHandler;
- private readonly ulong _addressSpaceSize;
-
private readonly HvAddressSpace _addressSpace;
internal HvAddressSpace AddressSpace => _addressSpace;
@@ -48,7 +26,7 @@ namespace Ryujinx.Cpu.AppleHv
private readonly MemoryBlock _backingMemory;
private readonly PageTable _pageTable;
- private readonly ulong[] _pageBitmap;
+ private readonly ManagedPageFlags _pages;
public bool Supports4KBPages => true;
@@ -62,6 +40,8 @@ namespace Ryujinx.Cpu.AppleHv
public event Action UnmapEvent;
+ protected override ulong AddressSpaceSize { get; }
+
///
/// Creates a new instance of the Hypervisor memory manager.
///
@@ -73,7 +53,7 @@ namespace Ryujinx.Cpu.AppleHv
_backingMemory = backingMemory;
_pageTable = new PageTable();
_invalidAccessHandler = invalidAccessHandler;
- _addressSpaceSize = addressSpaceSize;
+ AddressSpaceSize = addressSpaceSize;
ulong asSize = PageSize;
int asBits = PageBits;
@@ -88,46 +68,10 @@ namespace Ryujinx.Cpu.AppleHv
AddressSpaceBits = asBits;
- _pageBitmap = new ulong[1 << (AddressSpaceBits - (PageBits + PageToPteShift))];
+ _pages = new ManagedPageFlags(AddressSpaceBits);
Tracking = new MemoryTracking(this, PageSize, invalidAccessHandler);
}
- ///
- /// Checks if the virtual address is part of the addressable space.
- ///
- /// Virtual address
- /// True if the virtual address is part of the addressable space
- private bool ValidateAddress(ulong va)
- {
- return va < _addressSpaceSize;
- }
-
- ///
- /// Checks if the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// True if the combination of virtual address and size is part of the addressable space
- private bool ValidateAddressAndSize(ulong va, ulong size)
- {
- ulong endVa = va + size;
- return endVa >= va && endVa >= size && endVa <= _addressSpaceSize;
- }
-
- ///
- /// Ensures the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// Throw when the memory region specified outside the addressable space
- private void AssertValidAddressAndSize(ulong va, ulong size)
- {
- if (!ValidateAddressAndSize(va, size))
- {
- throw new InvalidMemoryRegionException($"va=0x{va:X16}, size=0x{size:X16}");
- }
- }
-
///
public void Map(ulong va, ulong pa, ulong size, MemoryMapFlags flags)
{
@@ -135,7 +79,7 @@ namespace Ryujinx.Cpu.AppleHv
PtMap(va, pa, size);
_addressSpace.MapUser(va, pa, size, MemoryPermission.ReadWriteExecute);
- AddMapping(va, size);
+ _pages.AddMapping(va, size);
Tracking.Map(va, size);
}
@@ -166,7 +110,7 @@ namespace Ryujinx.Cpu.AppleHv
UnmapEvent?.Invoke(va, size);
Tracking.Unmap(va, size);
- RemoveMapping(va, size);
+ _pages.RemoveMapping(va, size);
_addressSpace.UnmapUser(va, size);
PtUnmap(va, size);
}
@@ -209,9 +153,19 @@ namespace Ryujinx.Cpu.AppleHv
}
///
- public void Read(ulong va, Span data)
+ public override void Read(ulong va, Span data)
{
- ReadImpl(va, data);
+ try
+ {
+ base.Read(va, data);
+ }
+ catch (InvalidMemoryRegionException)
+ {
+ if (_invalidAccessHandler == null || !_invalidAccessHandler(va))
+ {
+ throw;
+ }
+ }
}
///
@@ -340,7 +294,7 @@ namespace Ryujinx.Cpu.AppleHv
{
Span data = new byte[size];
- ReadImpl(va, data);
+ base.Read(va, data);
return data;
}
@@ -367,7 +321,7 @@ namespace Ryujinx.Cpu.AppleHv
{
Memory memory = new byte[size];
- ReadImpl(va, memory.Span);
+ base.Read(va, memory.Span);
return new WritableRegion(this, va, memory);
}
@@ -390,22 +344,7 @@ namespace Ryujinx.Cpu.AppleHv
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool IsMapped(ulong va)
{
- return ValidateAddress(va) && IsMappedImpl(va);
- }
-
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- private bool IsMappedImpl(ulong va)
- {
- ulong page = va >> PageBits;
-
- int bit = (int)((page & 31) << 1);
-
- int pageIndex = (int)(page >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte = Volatile.Read(ref pageRef);
-
- return ((pte >> bit) & 3) != 0;
+ return ValidateAddress(va) && _pages.IsMapped(va);
}
///
@@ -413,58 +352,7 @@ namespace Ryujinx.Cpu.AppleHv
{
AssertValidAddressAndSize(va, size);
- return IsRangeMappedImpl(va, size);
- }
-
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- private static void GetPageBlockRange(ulong pageStart, ulong pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex)
- {
- startMask = ulong.MaxValue << ((int)(pageStart & 31) << 1);
- endMask = ulong.MaxValue >> (64 - ((int)(pageEnd & 31) << 1));
-
- pageIndex = (int)(pageStart >> PageToPteShift);
- pageEndIndex = (int)((pageEnd - 1) >> PageToPteShift);
- }
-
- private bool IsRangeMappedImpl(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
-
- if (pages == 1)
- {
- return IsMappedImpl(va);
- }
-
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- // Check if either bit in each 2 bit page entry is set.
- // OR the block with itself shifted down by 1, and check the first bit of each entry.
-
- ulong mask = BlockMappedMask & startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
- ulong pte = Volatile.Read(ref pageRef);
-
- pte |= pte >> 1;
- if ((pte & mask) != mask)
- {
- return false;
- }
-
- mask = BlockMappedMask;
- }
-
- return true;
+ return _pages.IsRangeMapped(va, size);
}
private static void ThrowMemoryNotContiguous() => throw new MemoryNotContiguousException();
@@ -576,48 +464,6 @@ namespace Ryujinx.Cpu.AppleHv
return regions;
}
- private void ReadImpl(ulong va, Span data)
- {
- if (data.Length == 0)
- {
- return;
- }
-
- try
- {
- AssertValidAddressAndSize(va, (ulong)data.Length);
-
- int offset = 0, size;
-
- if ((va & PageMask) != 0)
- {
- ulong pa = GetPhysicalAddressChecked(va);
-
- size = Math.Min(data.Length, PageSize - (int)(va & PageMask));
-
- _backingMemory.GetSpan(pa, size).CopyTo(data[..size]);
-
- offset += size;
- }
-
- for (; offset < data.Length; offset += size)
- {
- ulong pa = GetPhysicalAddressChecked(va + (ulong)offset);
-
- size = Math.Min(data.Length - offset, PageSize);
-
- _backingMemory.GetSpan(pa, size).CopyTo(data.Slice(offset, size));
- }
- }
- catch (InvalidMemoryRegionException)
- {
- if (_invalidAccessHandler == null || !_invalidAccessHandler(va))
- {
- throw;
- }
- }
- }
-
///
///
/// This function also validates that the given range is both valid and mapped, and will throw if it is not.
@@ -632,76 +478,7 @@ namespace Ryujinx.Cpu.AppleHv
return;
}
- // Software table, used for managed memory tracking.
-
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
-
- if (pages == 1)
- {
- ulong tag = (ulong)(write ? HostMappedPtBits.WriteTracked : HostMappedPtBits.ReadWriteTracked);
-
- int bit = (int)((pageStart & 31) << 1);
-
- int pageIndex = (int)(pageStart >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte = Volatile.Read(ref pageRef);
- ulong state = ((pte >> bit) & 3);
-
- if (state >= tag)
- {
- Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
- return;
- }
- else if (state == 0)
- {
- ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
- }
- }
- else
- {
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- ulong anyTrackingTag = (ulong)HostMappedPtBits.WriteTrackedReplicated;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte = Volatile.Read(ref pageRef);
- ulong mappedMask = mask & BlockMappedMask;
-
- ulong mappedPte = pte | (pte >> 1);
- if ((mappedPte & mappedMask) != mappedMask)
- {
- ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
- }
-
- pte &= mask;
- if ((pte & anyTrackingTag) != 0) // Search for any tracking.
- {
- // Writes trigger any tracking.
- // Only trigger tracking from reads if both bits are set on any page.
- if (write || (pte & (pte >> 1) & BlockMappedMask) != 0)
- {
- Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
- break;
- }
- }
-
- mask = ulong.MaxValue;
- }
- }
+ _pages.SignalMemoryTracking(Tracking, va, size, write, exemptId);
}
///
@@ -728,103 +505,28 @@ namespace Ryujinx.Cpu.AppleHv
}
///
- public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest)
{
- // Protection is inverted on software pages, since the default value is 0.
- protection = (~protection) & MemoryPermission.ReadAndWrite;
-
- int pages = GetPagesCount(va, size, out va);
- ulong pageStart = va >> PageBits;
-
- if (pages == 1)
+ if (guest)
{
- ulong protTag = protection switch
- {
- MemoryPermission.None => (ulong)HostMappedPtBits.Mapped,
- MemoryPermission.Write => (ulong)HostMappedPtBits.WriteTracked,
- _ => (ulong)HostMappedPtBits.ReadWriteTracked,
- };
-
- int bit = (int)((pageStart & 31) << 1);
-
- ulong tagMask = 3UL << bit;
- ulong invTagMask = ~tagMask;
-
- ulong tag = protTag << bit;
-
- int pageIndex = (int)(pageStart >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte;
-
- do
- {
- pte = Volatile.Read(ref pageRef);
- }
- while ((pte & tagMask) != 0 && Interlocked.CompareExchange(ref pageRef, (pte & invTagMask) | tag, pte) != pte);
+ _addressSpace.ReprotectUser(va, size, protection);
}
else
{
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- ulong protTag = protection switch
- {
- MemoryPermission.None => (ulong)HostMappedPtBits.MappedReplicated,
- MemoryPermission.Write => (ulong)HostMappedPtBits.WriteTrackedReplicated,
- _ => (ulong)HostMappedPtBits.ReadWriteTrackedReplicated,
- };
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte;
- ulong mappedMask;
-
- // Change the protection of all 2 bit entries that are mapped.
- do
- {
- pte = Volatile.Read(ref pageRef);
-
- mappedMask = pte | (pte >> 1);
- mappedMask |= (mappedMask & BlockMappedMask) << 1;
- mappedMask &= mask; // Only update mapped pages within the given range.
- }
- while (Interlocked.CompareExchange(ref pageRef, (pte & (~mappedMask)) | (protTag & mappedMask), pte) != pte);
-
- mask = ulong.MaxValue;
- }
+ _pages.TrackingReprotect(va, size, protection);
}
-
- protection = protection switch
- {
- MemoryPermission.None => MemoryPermission.ReadAndWrite,
- MemoryPermission.Write => MemoryPermission.Read,
- _ => MemoryPermission.None,
- };
-
- _addressSpace.ReprotectUser(va, size, protection);
}
///
- public RegionHandle BeginTracking(ulong address, ulong size, int id)
+ public RegionHandle BeginTracking(ulong address, ulong size, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginTracking(address, size, id);
+ return Tracking.BeginTracking(address, size, id, flags);
}
///
- public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id)
+ public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginGranularTracking(address, size, handles, granularity, id);
+ return Tracking.BeginGranularTracking(address, size, handles, granularity, id, flags);
}
///
@@ -833,86 +535,6 @@ namespace Ryujinx.Cpu.AppleHv
return Tracking.BeginSmartGranularTracking(address, size, granularity, id);
}
- ///
- /// Adds the given address mapping to the page table.
- ///
- /// Virtual memory address
- /// Size to be mapped
- private void AddMapping(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte;
- ulong mappedMask;
-
- // Map all 2-bit entries that are unmapped.
- do
- {
- pte = Volatile.Read(ref pageRef);
-
- mappedMask = pte | (pte >> 1);
- mappedMask |= (mappedMask & BlockMappedMask) << 1;
- mappedMask |= ~mask; // Treat everything outside the range as mapped, thus unchanged.
- }
- while (Interlocked.CompareExchange(ref pageRef, (pte & mappedMask) | (BlockMappedMask & (~mappedMask)), pte) != pte);
-
- mask = ulong.MaxValue;
- }
- }
-
- ///
- /// Removes the given address mapping from the page table.
- ///
- /// Virtual memory address
- /// Size to be unmapped
- private void RemoveMapping(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- startMask = ~startMask;
- endMask = ~endMask;
-
- ulong mask = startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask |= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
- ulong pte;
-
- do
- {
- pte = Volatile.Read(ref pageRef);
- }
- while (Interlocked.CompareExchange(ref pageRef, pte & mask, pte) != pte);
-
- mask = 0;
- }
- }
-
private ulong GetPhysicalAddressChecked(ulong va)
{
if (!IsMapped(va))
@@ -936,6 +558,10 @@ namespace Ryujinx.Cpu.AppleHv
_addressSpace.Dispose();
}
- private static void ThrowInvalidMemoryRegionException(string message) => throw new InvalidMemoryRegionException(message);
+ protected override Span GetPhysicalAddressSpan(ulong pa, int size)
+ => _backingMemory.GetSpan(pa, size);
+
+ protected override ulong TranslateVirtualAddressForRead(ulong va)
+ => GetPhysicalAddressChecked(va);
}
}
diff --git a/src/Ryujinx.Cpu/IVirtualMemoryManagerTracked.cs b/src/Ryujinx.Cpu/IVirtualMemoryManagerTracked.cs
index 199bff240e..e8d11ede5e 100644
--- a/src/Ryujinx.Cpu/IVirtualMemoryManagerTracked.cs
+++ b/src/Ryujinx.Cpu/IVirtualMemoryManagerTracked.cs
@@ -28,8 +28,9 @@ namespace Ryujinx.Cpu
/// CPU virtual address of the region
/// Size of the region
/// Handle ID
+ /// Region flags
/// The memory tracking handle
- RegionHandle BeginTracking(ulong address, ulong size, int id);
+ RegionHandle BeginTracking(ulong address, ulong size, int id, RegionFlags flags = RegionFlags.None);
///
/// Obtains a memory tracking handle for the given virtual region, with a specified granularity. This should be disposed when finished with.
@@ -39,8 +40,9 @@ namespace Ryujinx.Cpu
/// Handles to inherit state from or reuse. When none are present, provide null
/// Desired granularity of write tracking
/// Handle ID
+ /// Region flags
/// The memory tracking handle
- MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id);
+ MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id, RegionFlags flags = RegionFlags.None);
///
/// Obtains a smart memory tracking handle for the given virtual region, with a specified granularity. This should be disposed when finished with.
diff --git a/src/Ryujinx.Cpu/Jit/MemoryManager.cs b/src/Ryujinx.Cpu/Jit/MemoryManager.cs
index b9a547025c..c87c8b8cc5 100644
--- a/src/Ryujinx.Cpu/Jit/MemoryManager.cs
+++ b/src/Ryujinx.Cpu/Jit/MemoryManager.cs
@@ -14,12 +14,8 @@ namespace Ryujinx.Cpu.Jit
///
/// Represents a CPU memory manager.
///
- public sealed class MemoryManager : MemoryManagerBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
+ public sealed class MemoryManager : VirtualMemoryManagerRefCountedBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
{
- public const int PageBits = 12;
- public const int PageSize = 1 << PageBits;
- public const int PageMask = PageSize - 1;
-
private const int PteSize = 8;
private const int PointerTagBit = 62;
@@ -35,10 +31,10 @@ namespace Ryujinx.Cpu.Jit
///
public int AddressSpaceBits { get; }
- private readonly ulong _addressSpaceSize;
-
private readonly MemoryBlock _pageTable;
+ private readonly ManagedPageFlags _pages;
+
///
/// Page table base pointer.
///
@@ -50,6 +46,8 @@ namespace Ryujinx.Cpu.Jit
public event Action UnmapEvent;
+ protected override ulong AddressSpaceSize { get; }
+
///
/// Creates a new instance of the memory manager.
///
@@ -71,9 +69,11 @@ namespace Ryujinx.Cpu.Jit
}
AddressSpaceBits = asBits;
- _addressSpaceSize = asSize;
+ AddressSpaceSize = asSize;
_pageTable = new MemoryBlock((asSize / PageSize) * PteSize);
+ _pages = new ManagedPageFlags(AddressSpaceBits);
+
Tracking = new MemoryTracking(this, PageSize);
}
@@ -93,6 +93,7 @@ namespace Ryujinx.Cpu.Jit
remainingSize -= PageSize;
}
+ _pages.AddMapping(oVa, size);
Tracking.Map(oVa, size);
}
@@ -115,6 +116,7 @@ namespace Ryujinx.Cpu.Jit
UnmapEvent?.Invoke(va, size);
Tracking.Unmap(va, size);
+ _pages.RemoveMapping(va, size);
ulong remainingSize = size;
while (remainingSize != 0)
@@ -153,9 +155,39 @@ namespace Ryujinx.Cpu.Jit
}
///
- public void Read(ulong va, Span data)
+ public T ReadGuest(ulong va) where T : unmanaged
{
- ReadImpl(va, data);
+ try
+ {
+ SignalMemoryTrackingImpl(va, (ulong)Unsafe.SizeOf(), false, true);
+
+ return Read(va);
+ }
+ catch (InvalidMemoryRegionException)
+ {
+ if (_invalidAccessHandler == null || !_invalidAccessHandler(va))
+ {
+ throw;
+ }
+
+ return default;
+ }
+ }
+
+ ///
+ public override void Read(ulong va, Span data)
+ {
+ try
+ {
+ base.Read(va, data);
+ }
+ catch (InvalidMemoryRegionException)
+ {
+ if (_invalidAccessHandler == null || !_invalidAccessHandler(va))
+ {
+ throw;
+ }
+ }
}
///
@@ -177,6 +209,16 @@ namespace Ryujinx.Cpu.Jit
WriteImpl(va, data);
}
+ ///
+ public void WriteGuest(ulong va, T value) where T : unmanaged
+ {
+ Span data = MemoryMarshal.Cast(MemoryMarshal.CreateSpan(ref value, 1));
+
+ SignalMemoryTrackingImpl(va, (ulong)data.Length, true, true);
+
+ WriteImpl(va, data);
+ }
+
///
public void WriteUntracked(ulong va, ReadOnlySpan data)
{
@@ -290,7 +332,7 @@ namespace Ryujinx.Cpu.Jit
{
Span data = new byte[size];
- ReadImpl(va, data);
+ base.Read(va, data);
return data;
}
@@ -462,48 +504,6 @@ namespace Ryujinx.Cpu.Jit
return regions;
}
- private void ReadImpl(ulong va, Span data)
- {
- if (data.Length == 0)
- {
- return;
- }
-
- try
- {
- AssertValidAddressAndSize(va, (ulong)data.Length);
-
- int offset = 0, size;
-
- if ((va & PageMask) != 0)
- {
- ulong pa = GetPhysicalAddressInternal(va);
-
- size = Math.Min(data.Length, PageSize - (int)(va & PageMask));
-
- _backingMemory.GetSpan(pa, size).CopyTo(data[..size]);
-
- offset += size;
- }
-
- for (; offset < data.Length; offset += size)
- {
- ulong pa = GetPhysicalAddressInternal(va + (ulong)offset);
-
- size = Math.Min(data.Length - offset, PageSize);
-
- _backingMemory.GetSpan(pa, size).CopyTo(data.Slice(offset, size));
- }
- }
- catch (InvalidMemoryRegionException)
- {
- if (_invalidAccessHandler == null || !_invalidAccessHandler(va))
- {
- throw;
- }
- }
- }
-
///
public bool IsRangeMapped(ulong va, ulong size)
{
@@ -544,37 +544,6 @@ namespace Ryujinx.Cpu.Jit
return _pageTable.Read((va / PageSize) * PteSize) != 0;
}
- private bool ValidateAddress(ulong va)
- {
- return va < _addressSpaceSize;
- }
-
- ///
- /// Checks if the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// True if the combination of virtual address and size is part of the addressable space
- private bool ValidateAddressAndSize(ulong va, ulong size)
- {
- ulong endVa = va + size;
- return endVa >= va && endVa >= size && endVa <= _addressSpaceSize;
- }
-
- ///
- /// Ensures the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// Throw when the memory region specified outside the addressable space
- private void AssertValidAddressAndSize(ulong va, ulong size)
- {
- if (!ValidateAddressAndSize(va, size))
- {
- throw new InvalidMemoryRegionException($"va=0x{va:X16}, size=0x{size:X16}");
- }
- }
-
private ulong GetPhysicalAddressInternal(ulong va)
{
return PteToPa(_pageTable.Read((va / PageSize) * PteSize) & ~(0xffffUL << 48)) + (va & PageMask);
@@ -587,50 +556,57 @@ namespace Ryujinx.Cpu.Jit
}
///
- public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest)
{
AssertValidAddressAndSize(va, size);
- // Protection is inverted on software pages, since the default value is 0.
- protection = (~protection) & MemoryPermission.ReadAndWrite;
-
- long tag = protection switch
+ if (guest)
{
- MemoryPermission.None => 0L,
- MemoryPermission.Write => 2L << PointerTagBit,
- _ => 3L << PointerTagBit,
- };
+ // Protection is inverted on software pages, since the default value is 0.
+ protection = (~protection) & MemoryPermission.ReadAndWrite;
- int pages = GetPagesCount(va, (uint)size, out va);
- ulong pageStart = va >> PageBits;
- long invTagMask = ~(0xffffL << 48);
-
- for (int page = 0; page < pages; page++)
- {
- ref long pageRef = ref _pageTable.GetRef(pageStart * PteSize);
-
- long pte;
-
- do
+ long tag = protection switch
{
- pte = Volatile.Read(ref pageRef);
- }
- while (pte != 0 && Interlocked.CompareExchange(ref pageRef, (pte & invTagMask) | tag, pte) != pte);
+ MemoryPermission.None => 0L,
+ MemoryPermission.Write => 2L << PointerTagBit,
+ _ => 3L << PointerTagBit,
+ };
- pageStart++;
+ int pages = GetPagesCount(va, (uint)size, out va);
+ ulong pageStart = va >> PageBits;
+ long invTagMask = ~(0xffffL << 48);
+
+ for (int page = 0; page < pages; page++)
+ {
+ ref long pageRef = ref _pageTable.GetRef(pageStart * PteSize);
+
+ long pte;
+
+ do
+ {
+ pte = Volatile.Read(ref pageRef);
+ }
+ while (pte != 0 && Interlocked.CompareExchange(ref pageRef, (pte & invTagMask) | tag, pte) != pte);
+
+ pageStart++;
+ }
+ }
+ else
+ {
+ _pages.TrackingReprotect(va, size, protection);
}
}
///
- public RegionHandle BeginTracking(ulong address, ulong size, int id)
+ public RegionHandle BeginTracking(ulong address, ulong size, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginTracking(address, size, id);
+ return Tracking.BeginTracking(address, size, id, flags);
}
///
- public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id)
+ public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginGranularTracking(address, size, handles, granularity, id);
+ return Tracking.BeginGranularTracking(address, size, handles, granularity, id, flags);
}
///
@@ -639,8 +615,7 @@ namespace Ryujinx.Cpu.Jit
return Tracking.BeginSmartGranularTracking(address, size, granularity, id);
}
- ///
- public void SignalMemoryTracking(ulong va, ulong size, bool write, bool precise = false, int? exemptId = null)
+ private void SignalMemoryTrackingImpl(ulong va, ulong size, bool write, bool guest, bool precise = false, int? exemptId = null)
{
AssertValidAddressAndSize(va, size);
@@ -650,31 +625,47 @@ namespace Ryujinx.Cpu.Jit
return;
}
- // We emulate guard pages for software memory access. This makes for an easy transition to
- // tracking using host guard pages in future, but also supporting platforms where this is not possible.
+ // If the memory tracking is coming from the guest, use the tag bits in the page table entry.
+ // Otherwise, use the managed page flags.
- // Write tag includes read protection, since we don't have any read actions that aren't performed before write too.
- long tag = (write ? 3L : 1L) << PointerTagBit;
-
- int pages = GetPagesCount(va, (uint)size, out _);
- ulong pageStart = va >> PageBits;
-
- for (int page = 0; page < pages; page++)
+ if (guest)
{
- ref long pageRef = ref _pageTable.GetRef(pageStart * PteSize);
+ // We emulate guard pages for software memory access. This makes for an easy transition to
+ // tracking using host guard pages in future, but also supporting platforms where this is not possible.
- long pte;
+ // Write tag includes read protection, since we don't have any read actions that aren't performed before write too.
+ long tag = (write ? 3L : 1L) << PointerTagBit;
- pte = Volatile.Read(ref pageRef);
+ int pages = GetPagesCount(va, (uint)size, out _);
+ ulong pageStart = va >> PageBits;
- if ((pte & tag) != 0)
+ for (int page = 0; page < pages; page++)
{
- Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
- break;
- }
+ ref long pageRef = ref _pageTable.GetRef(pageStart * PteSize);
- pageStart++;
+ long pte;
+
+ pte = Volatile.Read(ref pageRef);
+
+ if ((pte & tag) != 0)
+ {
+ Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId, true);
+ break;
+ }
+
+ pageStart++;
+ }
}
+ else
+ {
+ _pages.SignalMemoryTracking(Tracking, va, size, write, exemptId);
+ }
+ }
+
+ ///
+ public void SignalMemoryTracking(ulong va, ulong size, bool write, bool precise = false, int? exemptId = null)
+ {
+ SignalMemoryTrackingImpl(va, size, write, false, precise, exemptId);
}
private ulong PaToPte(ulong pa)
@@ -691,5 +682,11 @@ namespace Ryujinx.Cpu.Jit
/// Disposes of resources used by the memory manager.
///
protected override void Destroy() => _pageTable.Dispose();
+
+ protected override Span GetPhysicalAddressSpan(ulong pa, int size)
+ => _backingMemory.GetSpan(pa, size);
+
+ protected override ulong TranslateVirtualAddressForRead(ulong va)
+ => GetPhysicalAddressInternal(va);
}
}
diff --git a/src/Ryujinx.Cpu/Jit/MemoryManagerHostMapped.cs b/src/Ryujinx.Cpu/Jit/MemoryManagerHostMapped.cs
index 2b315e8413..f410d02e96 100644
--- a/src/Ryujinx.Cpu/Jit/MemoryManagerHostMapped.cs
+++ b/src/Ryujinx.Cpu/Jit/MemoryManagerHostMapped.cs
@@ -6,46 +6,24 @@ using System;
using System.Collections.Generic;
using System.Linq;
using System.Runtime.CompilerServices;
-using System.Threading;
namespace Ryujinx.Cpu.Jit
{
///
/// Represents a CPU memory manager which maps guest virtual memory directly onto a host virtual region.
///
- public sealed class MemoryManagerHostMapped : MemoryManagerBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
+ public sealed class MemoryManagerHostMapped : VirtualMemoryManagerRefCountedBase, IMemoryManager, IVirtualMemoryManagerTracked, IWritableBlock
{
- public const int PageBits = 12;
- public const int PageSize = 1 << PageBits;
- public const int PageMask = PageSize - 1;
-
- public const int PageToPteShift = 5; // 32 pages (2 bits each) in one ulong page table entry.
- public const ulong BlockMappedMask = 0x5555555555555555; // First bit of each table entry set.
-
- private enum HostMappedPtBits : ulong
- {
- Unmapped = 0,
- Mapped,
- WriteTracked,
- ReadWriteTracked,
-
- MappedReplicated = 0x5555555555555555,
- WriteTrackedReplicated = 0xaaaaaaaaaaaaaaaa,
- ReadWriteTrackedReplicated = ulong.MaxValue,
- }
-
private readonly InvalidAccessHandler _invalidAccessHandler;
private readonly bool _unsafeMode;
private readonly AddressSpace _addressSpace;
- public ulong AddressSpaceSize { get; }
-
private readonly PageTable _pageTable;
private readonly MemoryEhMeilleure _memoryEh;
- private readonly ulong[] _pageBitmap;
+ private readonly ManagedPageFlags _pages;
///
public bool Supports4KBPages => MemoryBlock.GetPageSize() == PageSize;
@@ -60,6 +38,8 @@ namespace Ryujinx.Cpu.Jit
public event Action UnmapEvent;
+ protected override ulong AddressSpaceSize { get; }
+
///
/// Creates a new instance of the host mapped memory manager.
///
@@ -85,48 +65,12 @@ namespace Ryujinx.Cpu.Jit
AddressSpaceBits = asBits;
- _pageBitmap = new ulong[1 << (AddressSpaceBits - (PageBits + PageToPteShift))];
+ _pages = new ManagedPageFlags(AddressSpaceBits);
Tracking = new MemoryTracking(this, (int)MemoryBlock.GetPageSize(), invalidAccessHandler);
_memoryEh = new MemoryEhMeilleure(_addressSpace.Base, _addressSpace.Mirror, Tracking);
}
- ///
- /// Checks if the virtual address is part of the addressable space.
- ///
- /// Virtual address
- /// True if the virtual address is part of the addressable space
- private bool ValidateAddress(ulong va)
- {
- return va < AddressSpaceSize;
- }
-
- ///
- /// Checks if the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// True if the combination of virtual address and size is part of the addressable space
- private bool ValidateAddressAndSize(ulong va, ulong size)
- {
- ulong endVa = va + size;
- return endVa >= va && endVa >= size && endVa <= AddressSpaceSize;
- }
-
- ///
- /// Ensures the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// Throw when the memory region specified outside the addressable space
- private void AssertValidAddressAndSize(ulong va, ulong size)
- {
- if (!ValidateAddressAndSize(va, size))
- {
- throw new InvalidMemoryRegionException($"va=0x{va:X16}, size=0x{size:X16}");
- }
- }
-
///
/// Ensures the combination of virtual address and size is part of the addressable space and fully mapped.
///
@@ -134,7 +78,7 @@ namespace Ryujinx.Cpu.Jit
/// Size of the range in bytes
private void AssertMapped(ulong va, ulong size)
{
- if (!ValidateAddressAndSize(va, size) || !IsRangeMappedImpl(va, size))
+ if (!ValidateAddressAndSize(va, size) || !_pages.IsRangeMapped(va, size))
{
throw new InvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
}
@@ -146,7 +90,7 @@ namespace Ryujinx.Cpu.Jit
AssertValidAddressAndSize(va, size);
_addressSpace.Map(va, pa, size, flags);
- AddMapping(va, size);
+ _pages.AddMapping(va, size);
PtMap(va, pa, size);
Tracking.Map(va, size);
@@ -166,7 +110,7 @@ namespace Ryujinx.Cpu.Jit
UnmapEvent?.Invoke(va, size);
Tracking.Unmap(va, size);
- RemoveMapping(va, size);
+ _pages.RemoveMapping(va, size);
PtUnmap(va, size);
_addressSpace.Unmap(va, size);
}
@@ -235,7 +179,7 @@ namespace Ryujinx.Cpu.Jit
}
///
- public void Read(ulong va, Span data)
+ public override void Read(ulong va, Span data)
{
try
{
@@ -377,22 +321,7 @@ namespace Ryujinx.Cpu.Jit
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool IsMapped(ulong va)
{
- return ValidateAddress(va) && IsMappedImpl(va);
- }
-
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- private bool IsMappedImpl(ulong va)
- {
- ulong page = va >> PageBits;
-
- int bit = (int)((page & 31) << 1);
-
- int pageIndex = (int)(page >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte = Volatile.Read(ref pageRef);
-
- return ((pte >> bit) & 3) != 0;
+ return ValidateAddress(va) && _pages.IsMapped(va);
}
///
@@ -400,58 +329,7 @@ namespace Ryujinx.Cpu.Jit
{
AssertValidAddressAndSize(va, size);
- return IsRangeMappedImpl(va, size);
- }
-
- [MethodImpl(MethodImplOptions.AggressiveInlining)]
- private static void GetPageBlockRange(ulong pageStart, ulong pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex)
- {
- startMask = ulong.MaxValue << ((int)(pageStart & 31) << 1);
- endMask = ulong.MaxValue >> (64 - ((int)(pageEnd & 31) << 1));
-
- pageIndex = (int)(pageStart >> PageToPteShift);
- pageEndIndex = (int)((pageEnd - 1) >> PageToPteShift);
- }
-
- private bool IsRangeMappedImpl(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
-
- if (pages == 1)
- {
- return IsMappedImpl(va);
- }
-
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- // Check if either bit in each 2 bit page entry is set.
- // OR the block with itself shifted down by 1, and check the first bit of each entry.
-
- ulong mask = BlockMappedMask & startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
- ulong pte = Volatile.Read(ref pageRef);
-
- pte |= pte >> 1;
- if ((pte & mask) != mask)
- {
- return false;
- }
-
- mask = BlockMappedMask;
- }
-
- return true;
+ return _pages.IsRangeMapped(va, size);
}
///
@@ -526,76 +404,7 @@ namespace Ryujinx.Cpu.Jit
return;
}
- // Software table, used for managed memory tracking.
-
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
-
- if (pages == 1)
- {
- ulong tag = (ulong)(write ? HostMappedPtBits.WriteTracked : HostMappedPtBits.ReadWriteTracked);
-
- int bit = (int)((pageStart & 31) << 1);
-
- int pageIndex = (int)(pageStart >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte = Volatile.Read(ref pageRef);
- ulong state = ((pte >> bit) & 3);
-
- if (state >= tag)
- {
- Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
- return;
- }
- else if (state == 0)
- {
- ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
- }
- }
- else
- {
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- ulong anyTrackingTag = (ulong)HostMappedPtBits.WriteTrackedReplicated;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte = Volatile.Read(ref pageRef);
- ulong mappedMask = mask & BlockMappedMask;
-
- ulong mappedPte = pte | (pte >> 1);
- if ((mappedPte & mappedMask) != mappedMask)
- {
- ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
- }
-
- pte &= mask;
- if ((pte & anyTrackingTag) != 0) // Search for any tracking.
- {
- // Writes trigger any tracking.
- // Only trigger tracking from reads if both bits are set on any page.
- if (write || (pte & (pte >> 1) & BlockMappedMask) != 0)
- {
- Tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
- break;
- }
- }
-
- mask = ulong.MaxValue;
- }
- }
+ _pages.SignalMemoryTracking(Tracking, va, size, write, exemptId);
}
///
@@ -622,103 +431,28 @@ namespace Ryujinx.Cpu.Jit
}
///
- public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest)
{
- // Protection is inverted on software pages, since the default value is 0.
- protection = (~protection) & MemoryPermission.ReadAndWrite;
-
- int pages = GetPagesCount(va, size, out va);
- ulong pageStart = va >> PageBits;
-
- if (pages == 1)
+ if (guest)
{
- ulong protTag = protection switch
- {
- MemoryPermission.None => (ulong)HostMappedPtBits.Mapped,
- MemoryPermission.Write => (ulong)HostMappedPtBits.WriteTracked,
- _ => (ulong)HostMappedPtBits.ReadWriteTracked,
- };
-
- int bit = (int)((pageStart & 31) << 1);
-
- ulong tagMask = 3UL << bit;
- ulong invTagMask = ~tagMask;
-
- ulong tag = protTag << bit;
-
- int pageIndex = (int)(pageStart >> PageToPteShift);
- ref ulong pageRef = ref _pageBitmap[pageIndex];
-
- ulong pte;
-
- do
- {
- pte = Volatile.Read(ref pageRef);
- }
- while ((pte & tagMask) != 0 && Interlocked.CompareExchange(ref pageRef, (pte & invTagMask) | tag, pte) != pte);
+ _addressSpace.Base.Reprotect(va, size, protection, false);
}
else
{
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- ulong protTag = protection switch
- {
- MemoryPermission.None => (ulong)HostMappedPtBits.MappedReplicated,
- MemoryPermission.Write => (ulong)HostMappedPtBits.WriteTrackedReplicated,
- _ => (ulong)HostMappedPtBits.ReadWriteTrackedReplicated,
- };
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte;
- ulong mappedMask;
-
- // Change the protection of all 2 bit entries that are mapped.
- do
- {
- pte = Volatile.Read(ref pageRef);
-
- mappedMask = pte | (pte >> 1);
- mappedMask |= (mappedMask & BlockMappedMask) << 1;
- mappedMask &= mask; // Only update mapped pages within the given range.
- }
- while (Interlocked.CompareExchange(ref pageRef, (pte & (~mappedMask)) | (protTag & mappedMask), pte) != pte);
-
- mask = ulong.MaxValue;
- }
+ _pages.TrackingReprotect(va, size, protection);
}
-
- protection = protection switch
- {
- MemoryPermission.None => MemoryPermission.ReadAndWrite,
- MemoryPermission.Write => MemoryPermission.Read,
- _ => MemoryPermission.None,
- };
-
- _addressSpace.Base.Reprotect(va, size, protection, false);
}
///
- public RegionHandle BeginTracking(ulong address, ulong size, int id)
+ public RegionHandle BeginTracking(ulong address, ulong size, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginTracking(address, size, id);
+ return Tracking.BeginTracking(address, size, id, flags);
}
///
- public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id)
+ public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id, RegionFlags flags = RegionFlags.None)
{
- return Tracking.BeginGranularTracking(address, size, handles, granularity, id);
+ return Tracking.BeginGranularTracking(address, size, handles, granularity, id, flags);
}
///
@@ -727,86 +461,6 @@ namespace Ryujinx.Cpu.Jit
return Tracking.BeginSmartGranularTracking(address, size, granularity, id);
}
- ///
- /// Adds the given address mapping to the page table.
- ///
- /// Virtual memory address
- /// Size to be mapped
- private void AddMapping(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- ulong mask = startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask &= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
-
- ulong pte;
- ulong mappedMask;
-
- // Map all 2-bit entries that are unmapped.
- do
- {
- pte = Volatile.Read(ref pageRef);
-
- mappedMask = pte | (pte >> 1);
- mappedMask |= (mappedMask & BlockMappedMask) << 1;
- mappedMask |= ~mask; // Treat everything outside the range as mapped, thus unchanged.
- }
- while (Interlocked.CompareExchange(ref pageRef, (pte & mappedMask) | (BlockMappedMask & (~mappedMask)), pte) != pte);
-
- mask = ulong.MaxValue;
- }
- }
-
- ///
- /// Removes the given address mapping from the page table.
- ///
- /// Virtual memory address
- /// Size to be unmapped
- private void RemoveMapping(ulong va, ulong size)
- {
- int pages = GetPagesCount(va, size, out _);
- ulong pageStart = va >> PageBits;
- ulong pageEnd = pageStart + (ulong)pages;
-
- GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
-
- startMask = ~startMask;
- endMask = ~endMask;
-
- ulong mask = startMask;
-
- while (pageIndex <= pageEndIndex)
- {
- if (pageIndex == pageEndIndex)
- {
- mask |= endMask;
- }
-
- ref ulong pageRef = ref _pageBitmap[pageIndex++];
- ulong pte;
-
- do
- {
- pte = Volatile.Read(ref pageRef);
- }
- while (Interlocked.CompareExchange(ref pageRef, pte & mask, pte) != pte);
-
- mask = 0;
- }
- }
-
///
/// Disposes of resources used by the memory manager.
///
@@ -816,6 +470,10 @@ namespace Ryujinx.Cpu.Jit
_memoryEh.Dispose();
}
- private static void ThrowInvalidMemoryRegionException(string message) => throw new InvalidMemoryRegionException(message);
+ protected override Span GetPhysicalAddressSpan(ulong pa, int size)
+ => _addressSpace.Mirror.GetSpan(pa, size);
+
+ protected override ulong TranslateVirtualAddressForRead(ulong va)
+ => va;
}
}
diff --git a/src/Ryujinx.Cpu/LightningJit/Arm64/InstName.cs b/src/Ryujinx.Cpu/LightningJit/Arm64/InstName.cs
index 58d78ae6e3..3656406453 100644
--- a/src/Ryujinx.Cpu/LightningJit/Arm64/InstName.cs
+++ b/src/Ryujinx.Cpu/LightningJit/Arm64/InstName.cs
@@ -1106,6 +1106,7 @@ namespace Ryujinx.Cpu.LightningJit.Arm64
case InstName.Mrs:
case InstName.MsrImm:
case InstName.MsrReg:
+ case InstName.Sysl:
return true;
}
diff --git a/src/Ryujinx.Cpu/LightningJit/Arm64/SysUtils.cs b/src/Ryujinx.Cpu/LightningJit/Arm64/SysUtils.cs
new file mode 100644
index 0000000000..69689a3910
--- /dev/null
+++ b/src/Ryujinx.Cpu/LightningJit/Arm64/SysUtils.cs
@@ -0,0 +1,48 @@
+using System.Diagnostics;
+
+namespace Ryujinx.Cpu.LightningJit.Arm64
+{
+ static class SysUtils
+ {
+ public static (uint, uint, uint, uint) UnpackOp1CRnCRmOp2(uint encoding)
+ {
+ uint op1 = (encoding >> 16) & 7;
+ uint crn = (encoding >> 12) & 0xf;
+ uint crm = (encoding >> 8) & 0xf;
+ uint op2 = (encoding >> 5) & 7;
+
+ return (op1, crn, crm, op2);
+ }
+
+ public static bool IsCacheInstEl0(uint encoding)
+ {
+ (uint op1, uint crn, uint crm, uint op2) = UnpackOp1CRnCRmOp2(encoding);
+
+ return ((op1 << 11) | (crn << 7) | (crm << 3) | op2) switch
+ {
+ 0b011_0111_0100_001 => true, // DC ZVA
+ 0b011_0111_1010_001 => true, // DC CVAC
+ 0b011_0111_1100_001 => true, // DC CVAP
+ 0b011_0111_1011_001 => true, // DC CVAU
+ 0b011_0111_1110_001 => true, // DC CIVAC
+ 0b011_0111_0101_001 => true, // IC IVAU
+ _ => false,
+ };
+ }
+
+ public static bool IsCacheInstUciTrapped(uint encoding)
+ {
+ (uint op1, uint crn, uint crm, uint op2) = UnpackOp1CRnCRmOp2(encoding);
+
+ return ((op1 << 11) | (crn << 7) | (crm << 3) | op2) switch
+ {
+ 0b011_0111_1010_001 => true, // DC CVAC
+ 0b011_0111_1100_001 => true, // DC CVAP
+ 0b011_0111_1011_001 => true, // DC CVAU
+ 0b011_0111_1110_001 => true, // DC CIVAC
+ 0b011_0111_0101_001 => true, // IC IVAU
+ _ => false,
+ };
+ }
+ }
+}
diff --git a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/Decoder.cs b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/Decoder.cs
index e9ba8ba215..00a1758f29 100644
--- a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/Decoder.cs
+++ b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/Decoder.cs
@@ -257,7 +257,7 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
(name, flags, AddressForm addressForm) = InstTable.GetInstNameAndFlags(encoding, cpuPreset.Version, cpuPreset.Features);
- if (name.IsPrivileged())
+ if (name.IsPrivileged() || (name == InstName.Sys && IsPrivilegedSys(encoding)))
{
name = InstName.UdfPermUndef;
flags = InstFlags.None;
@@ -341,6 +341,11 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
return new(startAddress, address, insts, !isTruncated && !name.IsException(), isTruncated, isLoopEnd);
}
+ private static bool IsPrivilegedSys(uint encoding)
+ {
+ return !SysUtils.IsCacheInstEl0(encoding);
+ }
+
private static bool IsMrsNzcv(uint encoding)
{
return (encoding & ~0x1fu) == 0xd53b4200u;
diff --git a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitMemory.cs b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitMemory.cs
index ece1520fda..e03d9373a1 100644
--- a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitMemory.cs
+++ b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitMemory.cs
@@ -13,6 +13,14 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
public static void RewriteSysInstruction(int asBits, MemoryManagerType mmType, CodeWriter writer, RegisterAllocator regAlloc, uint encoding)
{
+ // TODO: Handle IC instruction, it should invalidate the JIT cache.
+
+ if (InstEmitSystem.IsCacheInstForbidden(encoding))
+ {
+ // Current OS does not allow cache maintenance instructions from user mode, just do nothing.
+ return;
+ }
+
int rtIndex = RegisterUtils.ExtractRt(encoding);
if (rtIndex == RegisterUtils.ZrIndex)
{
diff --git a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitSystem.cs b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitSystem.cs
index 3d4204fc13..82cb29d731 100644
--- a/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitSystem.cs
+++ b/src/Ryujinx.Cpu/LightningJit/Arm64/Target/Arm64/InstEmitSystem.cs
@@ -69,7 +69,7 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
asm.LdrRiUn(Register((int)rd), Register(regAlloc.FixedContextRegister), NativeContextOffsets.TpidrEl0Offset);
}
}
- else if ((encoding & ~0x1f) == 0xd53b0020 && IsAppleOS()) // mrs x0, ctr_el0
+ else if ((encoding & ~0x1f) == 0xd53b0020 && IsCtrEl0AccessForbidden()) // mrs x0, ctr_el0
{
uint rd = encoding & 0x1f;
@@ -115,7 +115,7 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
{
return true;
}
- else if ((encoding & ~0x1f) == 0xd53b0020 && IsAppleOS()) // mrs x0, ctr_el0
+ else if ((encoding & ~0x1f) == 0xd53b0020 && IsCtrEl0AccessForbidden()) // mrs x0, ctr_el0
{
return true;
}
@@ -127,9 +127,16 @@ namespace Ryujinx.Cpu.LightningJit.Arm64.Target.Arm64
return false;
}
- private static bool IsAppleOS()
+ private static bool IsCtrEl0AccessForbidden()
{
- return OperatingSystem.IsMacOS() || OperatingSystem.IsIOS();
+ // Only Linux allows accessing CTR_EL0 from user mode.
+ return OperatingSystem.IsWindows() || OperatingSystem.IsMacOS() || OperatingSystem.IsIOS();
+ }
+
+ public static bool IsCacheInstForbidden(uint encoding)
+ {
+ // Windows does not allow the cache maintenance instructions to be used from user mode.
+ return OperatingSystem.IsWindows() && SysUtils.IsCacheInstUciTrapped(encoding);
}
public static bool NeedsContextStoreLoad(InstName name)
diff --git a/src/Ryujinx.Cpu/ManagedPageFlags.cs b/src/Ryujinx.Cpu/ManagedPageFlags.cs
new file mode 100644
index 0000000000..a839dae676
--- /dev/null
+++ b/src/Ryujinx.Cpu/ManagedPageFlags.cs
@@ -0,0 +1,389 @@
+using Ryujinx.Memory;
+using Ryujinx.Memory.Tracking;
+using System;
+using System.Runtime.CompilerServices;
+using System.Threading;
+
+namespace Ryujinx.Cpu
+{
+ ///
+ /// A page bitmap that keeps track of mapped state and tracking protection
+ /// for managed memory accesses (not using host page protection).
+ ///
+ internal readonly struct ManagedPageFlags
+ {
+ public const int PageBits = 12;
+ public const int PageSize = 1 << PageBits;
+ public const int PageMask = PageSize - 1;
+
+ private readonly ulong[] _pageBitmap;
+
+ public const int PageToPteShift = 5; // 32 pages (2 bits each) in one ulong page table entry.
+ public const ulong BlockMappedMask = 0x5555555555555555; // First bit of each table entry set.
+
+ private enum ManagedPtBits : ulong
+ {
+ Unmapped = 0,
+ Mapped,
+ WriteTracked,
+ ReadWriteTracked,
+
+ MappedReplicated = 0x5555555555555555,
+ WriteTrackedReplicated = 0xaaaaaaaaaaaaaaaa,
+ ReadWriteTrackedReplicated = ulong.MaxValue,
+ }
+
+ public ManagedPageFlags(int addressSpaceBits)
+ {
+ int bits = Math.Max(0, addressSpaceBits - (PageBits + PageToPteShift));
+ _pageBitmap = new ulong[1 << bits];
+ }
+
+ ///
+ /// Computes the number of pages in a virtual address range.
+ ///
+ /// Virtual address of the range
+ /// Size of the range
+ /// The virtual address of the beginning of the first page
+ /// This function does not differentiate between allocated and unallocated pages.
+ [MethodImpl(MethodImplOptions.AggressiveInlining)]
+ private static int GetPagesCount(ulong va, ulong size, out ulong startVa)
+ {
+ // WARNING: Always check if ulong does not overflow during the operations.
+ startVa = va & ~(ulong)PageMask;
+ ulong vaSpan = (va - startVa + size + PageMask) & ~(ulong)PageMask;
+
+ return (int)(vaSpan / PageSize);
+ }
+
+ ///
+ /// Checks if the page at a given CPU virtual address is mapped.
+ ///
+ /// Virtual address to check
+ /// True if the address is mapped, false otherwise
+ [MethodImpl(MethodImplOptions.AggressiveInlining)]
+ public readonly bool IsMapped(ulong va)
+ {
+ ulong page = va >> PageBits;
+
+ int bit = (int)((page & 31) << 1);
+
+ int pageIndex = (int)(page >> PageToPteShift);
+ ref ulong pageRef = ref _pageBitmap[pageIndex];
+
+ ulong pte = Volatile.Read(ref pageRef);
+
+ return ((pte >> bit) & 3) != 0;
+ }
+
+ [MethodImpl(MethodImplOptions.AggressiveInlining)]
+ private static void GetPageBlockRange(ulong pageStart, ulong pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex)
+ {
+ startMask = ulong.MaxValue << ((int)(pageStart & 31) << 1);
+ endMask = ulong.MaxValue >> (64 - ((int)(pageEnd & 31) << 1));
+
+ pageIndex = (int)(pageStart >> PageToPteShift);
+ pageEndIndex = (int)((pageEnd - 1) >> PageToPteShift);
+ }
+
+ ///
+ /// Checks if a memory range is mapped.
+ ///
+ /// Virtual address of the range
+ /// Size of the range in bytes
+ /// True if the entire range is mapped, false otherwise
+ public readonly bool IsRangeMapped(ulong va, ulong size)
+ {
+ int pages = GetPagesCount(va, size, out _);
+
+ if (pages == 1)
+ {
+ return IsMapped(va);
+ }
+
+ ulong pageStart = va >> PageBits;
+ ulong pageEnd = pageStart + (ulong)pages;
+
+ GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
+
+ // Check if either bit in each 2 bit page entry is set.
+ // OR the block with itself shifted down by 1, and check the first bit of each entry.
+
+ ulong mask = BlockMappedMask & startMask;
+
+ while (pageIndex <= pageEndIndex)
+ {
+ if (pageIndex == pageEndIndex)
+ {
+ mask &= endMask;
+ }
+
+ ref ulong pageRef = ref _pageBitmap[pageIndex++];
+ ulong pte = Volatile.Read(ref pageRef);
+
+ pte |= pte >> 1;
+ if ((pte & mask) != mask)
+ {
+ return false;
+ }
+
+ mask = BlockMappedMask;
+ }
+
+ return true;
+ }
+
+ ///
+ /// Reprotect a region of virtual memory for tracking.
+ ///
+ /// Virtual address base
+ /// Size of the region to protect
+ /// Memory protection to set
+ public readonly void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ {
+ // Protection is inverted on software pages, since the default value is 0.
+ protection = (~protection) & MemoryPermission.ReadAndWrite;
+
+ int pages = GetPagesCount(va, size, out va);
+ ulong pageStart = va >> PageBits;
+
+ if (pages == 1)
+ {
+ ulong protTag = protection switch
+ {
+ MemoryPermission.None => (ulong)ManagedPtBits.Mapped,
+ MemoryPermission.Write => (ulong)ManagedPtBits.WriteTracked,
+ _ => (ulong)ManagedPtBits.ReadWriteTracked,
+ };
+
+ int bit = (int)((pageStart & 31) << 1);
+
+ ulong tagMask = 3UL << bit;
+ ulong invTagMask = ~tagMask;
+
+ ulong tag = protTag << bit;
+
+ int pageIndex = (int)(pageStart >> PageToPteShift);
+ ref ulong pageRef = ref _pageBitmap[pageIndex];
+
+ ulong pte;
+
+ do
+ {
+ pte = Volatile.Read(ref pageRef);
+ }
+ while ((pte & tagMask) != 0 && Interlocked.CompareExchange(ref pageRef, (pte & invTagMask) | tag, pte) != pte);
+ }
+ else
+ {
+ ulong pageEnd = pageStart + (ulong)pages;
+
+ GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
+
+ ulong mask = startMask;
+
+ ulong protTag = protection switch
+ {
+ MemoryPermission.None => (ulong)ManagedPtBits.MappedReplicated,
+ MemoryPermission.Write => (ulong)ManagedPtBits.WriteTrackedReplicated,
+ _ => (ulong)ManagedPtBits.ReadWriteTrackedReplicated,
+ };
+
+ while (pageIndex <= pageEndIndex)
+ {
+ if (pageIndex == pageEndIndex)
+ {
+ mask &= endMask;
+ }
+
+ ref ulong pageRef = ref _pageBitmap[pageIndex++];
+
+ ulong pte;
+ ulong mappedMask;
+
+ // Change the protection of all 2 bit entries that are mapped.
+ do
+ {
+ pte = Volatile.Read(ref pageRef);
+
+ mappedMask = pte | (pte >> 1);
+ mappedMask |= (mappedMask & BlockMappedMask) << 1;
+ mappedMask &= mask; // Only update mapped pages within the given range.
+ }
+ while (Interlocked.CompareExchange(ref pageRef, (pte & (~mappedMask)) | (protTag & mappedMask), pte) != pte);
+
+ mask = ulong.MaxValue;
+ }
+ }
+ }
+
+ ///
+ /// Alerts the memory tracking that a given region has been read from or written to.
+ /// This should be called before read/write is performed.
+ ///
+ /// Memory tracking structure to call when pages are protected
+ /// Virtual address of the region
+ /// Size of the region
+ /// True if the region was written, false if read
+ /// Optional ID of the handles that should not be signalled
+ ///
+ /// This function also validates that the given range is both valid and mapped, and will throw if it is not.
+ ///
+ [MethodImpl(MethodImplOptions.AggressiveInlining)]
+ public readonly void SignalMemoryTracking(MemoryTracking tracking, ulong va, ulong size, bool write, int? exemptId = null)
+ {
+ // Software table, used for managed memory tracking.
+
+ int pages = GetPagesCount(va, size, out _);
+ ulong pageStart = va >> PageBits;
+
+ if (pages == 1)
+ {
+ ulong tag = (ulong)(write ? ManagedPtBits.WriteTracked : ManagedPtBits.ReadWriteTracked);
+
+ int bit = (int)((pageStart & 31) << 1);
+
+ int pageIndex = (int)(pageStart >> PageToPteShift);
+ ref ulong pageRef = ref _pageBitmap[pageIndex];
+
+ ulong pte = Volatile.Read(ref pageRef);
+ ulong state = ((pte >> bit) & 3);
+
+ if (state >= tag)
+ {
+ tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
+ return;
+ }
+ else if (state == 0)
+ {
+ ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
+ }
+ }
+ else
+ {
+ ulong pageEnd = pageStart + (ulong)pages;
+
+ GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
+
+ ulong mask = startMask;
+
+ ulong anyTrackingTag = (ulong)ManagedPtBits.WriteTrackedReplicated;
+
+ while (pageIndex <= pageEndIndex)
+ {
+ if (pageIndex == pageEndIndex)
+ {
+ mask &= endMask;
+ }
+
+ ref ulong pageRef = ref _pageBitmap[pageIndex++];
+
+ ulong pte = Volatile.Read(ref pageRef);
+ ulong mappedMask = mask & BlockMappedMask;
+
+ ulong mappedPte = pte | (pte >> 1);
+ if ((mappedPte & mappedMask) != mappedMask)
+ {
+ ThrowInvalidMemoryRegionException($"Not mapped: va=0x{va:X16}, size=0x{size:X16}");
+ }
+
+ pte &= mask;
+ if ((pte & anyTrackingTag) != 0) // Search for any tracking.
+ {
+ // Writes trigger any tracking.
+ // Only trigger tracking from reads if both bits are set on any page.
+ if (write || (pte & (pte >> 1) & BlockMappedMask) != 0)
+ {
+ tracking.VirtualMemoryEvent(va, size, write, precise: false, exemptId);
+ break;
+ }
+ }
+
+ mask = ulong.MaxValue;
+ }
+ }
+ }
+
+ ///
+ /// Adds the given address mapping to the page table.
+ ///
+ /// Virtual memory address
+ /// Size to be mapped
+ public readonly void AddMapping(ulong va, ulong size)
+ {
+ int pages = GetPagesCount(va, size, out _);
+ ulong pageStart = va >> PageBits;
+ ulong pageEnd = pageStart + (ulong)pages;
+
+ GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
+
+ ulong mask = startMask;
+
+ while (pageIndex <= pageEndIndex)
+ {
+ if (pageIndex == pageEndIndex)
+ {
+ mask &= endMask;
+ }
+
+ ref ulong pageRef = ref _pageBitmap[pageIndex++];
+
+ ulong pte;
+ ulong mappedMask;
+
+ // Map all 2-bit entries that are unmapped.
+ do
+ {
+ pte = Volatile.Read(ref pageRef);
+
+ mappedMask = pte | (pte >> 1);
+ mappedMask |= (mappedMask & BlockMappedMask) << 1;
+ mappedMask |= ~mask; // Treat everything outside the range as mapped, thus unchanged.
+ }
+ while (Interlocked.CompareExchange(ref pageRef, (pte & mappedMask) | (BlockMappedMask & (~mappedMask)), pte) != pte);
+
+ mask = ulong.MaxValue;
+ }
+ }
+
+ ///
+ /// Removes the given address mapping from the page table.
+ ///
+ /// Virtual memory address
+ /// Size to be unmapped
+ public readonly void RemoveMapping(ulong va, ulong size)
+ {
+ int pages = GetPagesCount(va, size, out _);
+ ulong pageStart = va >> PageBits;
+ ulong pageEnd = pageStart + (ulong)pages;
+
+ GetPageBlockRange(pageStart, pageEnd, out ulong startMask, out ulong endMask, out int pageIndex, out int pageEndIndex);
+
+ startMask = ~startMask;
+ endMask = ~endMask;
+
+ ulong mask = startMask;
+
+ while (pageIndex <= pageEndIndex)
+ {
+ if (pageIndex == pageEndIndex)
+ {
+ mask |= endMask;
+ }
+
+ ref ulong pageRef = ref _pageBitmap[pageIndex++];
+ ulong pte;
+
+ do
+ {
+ pte = Volatile.Read(ref pageRef);
+ }
+ while (Interlocked.CompareExchange(ref pageRef, pte & mask, pte) != pte);
+
+ mask = 0;
+ }
+ }
+
+ private static void ThrowInvalidMemoryRegionException(string message) => throw new InvalidMemoryRegionException(message);
+ }
+}
diff --git a/src/Ryujinx.Cpu/MemoryManagerBase.cs b/src/Ryujinx.Cpu/VirtualMemoryManagerRefCountedBase.cs
similarity index 70%
rename from src/Ryujinx.Cpu/MemoryManagerBase.cs
rename to src/Ryujinx.Cpu/VirtualMemoryManagerRefCountedBase.cs
index 3288e3a498..c2d8cfb1a0 100644
--- a/src/Ryujinx.Cpu/MemoryManagerBase.cs
+++ b/src/Ryujinx.Cpu/VirtualMemoryManagerRefCountedBase.cs
@@ -1,10 +1,13 @@
using Ryujinx.Memory;
using System.Diagnostics;
+using System.Numerics;
using System.Threading;
namespace Ryujinx.Cpu
{
- public abstract class MemoryManagerBase : IRefCounted
+ public abstract class VirtualMemoryManagerRefCountedBase : VirtualMemoryManagerBase, IRefCounted
+ where TVirtual : IBinaryInteger
+ where TPhysical : IBinaryInteger
{
private int _referenceCount;
diff --git a/src/Ryujinx.Graphics.Gpu/Engine/Threed/StateUpdater.cs b/src/Ryujinx.Graphics.Gpu/Engine/Threed/StateUpdater.cs
index 6b4ea89f39..b3eb62185a 100644
--- a/src/Ryujinx.Graphics.Gpu/Engine/Threed/StateUpdater.cs
+++ b/src/Ryujinx.Graphics.Gpu/Engine/Threed/StateUpdater.cs
@@ -343,11 +343,22 @@ namespace Ryujinx.Graphics.Gpu.Engine.Threed
bool unalignedChanged = _currentSpecState.SetHasUnalignedStorageBuffer(_channel.BufferManager.HasUnalignedStorageBuffers);
- if (!_channel.TextureManager.CommitGraphicsBindings(_shaderSpecState) || unalignedChanged)
+ bool scaleMismatch;
+ do
{
- // Shader must be reloaded. _vtgWritesRtLayer should not change.
- UpdateShaderState();
+ if (!_channel.TextureManager.CommitGraphicsBindings(_shaderSpecState, out scaleMismatch) || unalignedChanged)
+ {
+ // Shader must be reloaded. _vtgWritesRtLayer should not change.
+ UpdateShaderState();
+ }
+
+ if (scaleMismatch)
+ {
+ // Binding textures changed scale of the bound render targets, correct the render target scale and rebind.
+ UpdateRenderTargetState();
+ }
}
+ while (scaleMismatch);
_channel.BufferManager.CommitGraphicsBindings(_drawState.DrawIndexed);
}
diff --git a/src/Ryujinx.Graphics.Gpu/Image/AutoDeleteCache.cs b/src/Ryujinx.Graphics.Gpu/Image/AutoDeleteCache.cs
index 05782605b1..732ec5d4c8 100644
--- a/src/Ryujinx.Graphics.Gpu/Image/AutoDeleteCache.cs
+++ b/src/Ryujinx.Graphics.Gpu/Image/AutoDeleteCache.cs
@@ -46,7 +46,7 @@ namespace Ryujinx.Graphics.Gpu.Image
{
private const int MinCountForDeletion = 32;
private const int MaxCapacity = 2048;
- private const ulong MaxTextureSizeCapacity = 512 * 1024 * 1024; // MB;
+ private const ulong MaxTextureSizeCapacity = 1024 * 1024 * 1024; // MB;
private readonly LinkedList _textures;
private ulong _totalSize;
diff --git a/src/Ryujinx.Graphics.Gpu/Image/Pool.cs b/src/Ryujinx.Graphics.Gpu/Image/Pool.cs
index 0c3a219de0..6ede019710 100644
--- a/src/Ryujinx.Graphics.Gpu/Image/Pool.cs
+++ b/src/Ryujinx.Graphics.Gpu/Image/Pool.cs
@@ -69,7 +69,7 @@ namespace Ryujinx.Graphics.Gpu.Image
Address = address;
Size = size;
- _memoryTracking = physicalMemory.BeginGranularTracking(address, size, ResourceKind.Pool);
+ _memoryTracking = physicalMemory.BeginGranularTracking(address, size, ResourceKind.Pool, RegionFlags.None);
_memoryTracking.RegisterPreciseAction(address, size, PreciseAction);
_modifiedDelegate = RegionModified;
}
diff --git a/src/Ryujinx.Graphics.Gpu/Image/TextureManager.cs b/src/Ryujinx.Graphics.Gpu/Image/TextureManager.cs
index 3bf0beefdb..8c2a887271 100644
--- a/src/Ryujinx.Graphics.Gpu/Image/TextureManager.cs
+++ b/src/Ryujinx.Graphics.Gpu/Image/TextureManager.cs
@@ -360,15 +360,16 @@ namespace Ryujinx.Graphics.Gpu.Image
/// Commits bindings on the graphics pipeline.
///
/// Specialization state for the bound shader
+ /// True if there is a scale mismatch in the render targets, indicating they must be re-evaluated
/// True if all bound textures match the current shader specialization state, false otherwise
- public bool CommitGraphicsBindings(ShaderSpecializationState specState)
+ public bool CommitGraphicsBindings(ShaderSpecializationState specState, out bool scaleMismatch)
{
_texturePoolCache.Tick();
_samplerPoolCache.Tick();
bool result = _gpBindingsManager.CommitBindings(specState);
- UpdateRenderTargets();
+ scaleMismatch = UpdateRenderTargets();
return result;
}
@@ -426,9 +427,12 @@ namespace Ryujinx.Graphics.Gpu.Image
///
/// Update host framebuffer attachments based on currently bound render target buffers.
///
- public void UpdateRenderTargets()
+ /// True if there is a scale mismatch in the render targets, indicating they must be re-evaluated
+ public bool UpdateRenderTargets()
{
bool anyChanged = false;
+ float expectedScale = RenderTargetScale;
+ bool scaleMismatch = false;
Texture dsTexture = _rtDepthStencil;
ITexture hostDsTexture = null;
@@ -448,6 +452,11 @@ namespace Ryujinx.Graphics.Gpu.Image
{
_rtHostDs = hostDsTexture;
anyChanged = true;
+
+ if (dsTexture != null && dsTexture.ScaleFactor != expectedScale)
+ {
+ scaleMismatch = true;
+ }
}
for (int index = 0; index < _rtColors.Length; index++)
@@ -470,6 +479,11 @@ namespace Ryujinx.Graphics.Gpu.Image
{
_rtHostColors[index] = hostTexture;
anyChanged = true;
+
+ if (texture != null && texture.ScaleFactor != expectedScale)
+ {
+ scaleMismatch = true;
+ }
}
}
@@ -477,6 +491,8 @@ namespace Ryujinx.Graphics.Gpu.Image
{
_context.Renderer.Pipeline.SetRenderTargets(_rtHostColors, _rtHostDs);
}
+
+ return scaleMismatch;
}
///
diff --git a/src/Ryujinx.Graphics.Gpu/Memory/Buffer.cs b/src/Ryujinx.Graphics.Gpu/Memory/Buffer.cs
index e01e5142c8..d293060b5a 100644
--- a/src/Ryujinx.Graphics.Gpu/Memory/Buffer.cs
+++ b/src/Ryujinx.Graphics.Gpu/Memory/Buffer.cs
@@ -128,13 +128,13 @@ namespace Ryujinx.Graphics.Gpu.Memory
if (_useGranular)
{
- _memoryTrackingGranular = physicalMemory.BeginGranularTracking(address, size, ResourceKind.Buffer, baseHandles);
+ _memoryTrackingGranular = physicalMemory.BeginGranularTracking(address, size, ResourceKind.Buffer, RegionFlags.UnalignedAccess, baseHandles);
_memoryTrackingGranular.RegisterPreciseAction(address, size, PreciseAction);
}
else
{
- _memoryTracking = physicalMemory.BeginTracking(address, size, ResourceKind.Buffer);
+ _memoryTracking = physicalMemory.BeginTracking(address, size, ResourceKind.Buffer, RegionFlags.UnalignedAccess);
if (baseHandles != null)
{
diff --git a/src/Ryujinx.Graphics.Gpu/Memory/PhysicalMemory.cs b/src/Ryujinx.Graphics.Gpu/Memory/PhysicalMemory.cs
index 69a3054a49..cca02bb156 100644
--- a/src/Ryujinx.Graphics.Gpu/Memory/PhysicalMemory.cs
+++ b/src/Ryujinx.Graphics.Gpu/Memory/PhysicalMemory.cs
@@ -368,10 +368,11 @@ namespace Ryujinx.Graphics.Gpu.Memory
/// CPU virtual address of the region
/// Size of the region
/// Kind of the resource being tracked
+ /// Region flags
/// The memory tracking handle
- public RegionHandle BeginTracking(ulong address, ulong size, ResourceKind kind)
+ public RegionHandle BeginTracking(ulong address, ulong size, ResourceKind kind, RegionFlags flags = RegionFlags.None)
{
- return _cpuMemory.BeginTracking(address, size, (int)kind);
+ return _cpuMemory.BeginTracking(address, size, (int)kind, flags);
}
///
@@ -408,12 +409,19 @@ namespace Ryujinx.Graphics.Gpu.Memory
/// CPU virtual address of the region
/// Size of the region
/// Kind of the resource being tracked
+ /// Region flags
/// Handles to inherit state from or reuse
/// Desired granularity of write tracking
/// The memory tracking handle
- public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, ResourceKind kind, IEnumerable handles = null, ulong granularity = 4096)
+ public MultiRegionHandle BeginGranularTracking(
+ ulong address,
+ ulong size,
+ ResourceKind kind,
+ RegionFlags flags = RegionFlags.None,
+ IEnumerable handles = null,
+ ulong granularity = 4096)
{
- return _cpuMemory.BeginGranularTracking(address, size, handles, granularity, (int)kind);
+ return _cpuMemory.BeginGranularTracking(address, size, handles, granularity, (int)kind, flags);
}
///
diff --git a/src/Ryujinx.Graphics.Gpu/Memory/VirtualRangeCache.cs b/src/Ryujinx.Graphics.Gpu/Memory/VirtualRangeCache.cs
index 889f5c9caa..964507a218 100644
--- a/src/Ryujinx.Graphics.Gpu/Memory/VirtualRangeCache.cs
+++ b/src/Ryujinx.Graphics.Gpu/Memory/VirtualRangeCache.cs
@@ -173,7 +173,7 @@ namespace Ryujinx.Graphics.Gpu.Memory
ShrinkOverlapsBufferIfNeeded();
- // If the the range is not properly aligned for sparse mapping,
+ // If the range is not properly aligned for sparse mapping,
// let's just force it to a single range.
// This might cause issues in some applications that uses sparse
// mappings.
diff --git a/src/Ryujinx.Graphics.Gpu/Shader/DiskCache/DiskCacheHostStorage.cs b/src/Ryujinx.Graphics.Gpu/Shader/DiskCache/DiskCacheHostStorage.cs
index c5763b0258..5036186ba1 100644
--- a/src/Ryujinx.Graphics.Gpu/Shader/DiskCache/DiskCacheHostStorage.cs
+++ b/src/Ryujinx.Graphics.Gpu/Shader/DiskCache/DiskCacheHostStorage.cs
@@ -22,7 +22,7 @@ namespace Ryujinx.Graphics.Gpu.Shader.DiskCache
private const ushort FileFormatVersionMajor = 1;
private const ushort FileFormatVersionMinor = 2;
private const uint FileFormatVersionPacked = ((uint)FileFormatVersionMajor << 16) | FileFormatVersionMinor;
- private const uint CodeGenVersion = 6253;
+ private const uint CodeGenVersion = 6462;
private const string SharedTocFileName = "shared.toc";
private const string SharedDataFileName = "shared.data";
diff --git a/src/Ryujinx.Graphics.Shader/CodeGen/Spirv/Declarations.cs b/src/Ryujinx.Graphics.Shader/CodeGen/Spirv/Declarations.cs
index 4ff61d9f25..b748242558 100644
--- a/src/Ryujinx.Graphics.Shader/CodeGen/Spirv/Declarations.cs
+++ b/src/Ryujinx.Graphics.Shader/CodeGen/Spirv/Declarations.cs
@@ -356,6 +356,16 @@ namespace Ryujinx.Graphics.Shader.CodeGen.Spirv
context.AddGlobalVariable(perVertexInputVariable);
context.Inputs.Add(new IoDefinition(StorageKind.Input, IoVariable.Position), perVertexInputVariable);
+
+ if (context.Definitions.Stage == ShaderStage.Geometry &&
+ context.Definitions.GpPassthrough &&
+ context.HostCapabilities.SupportsGeometryShaderPassthrough)
+ {
+ context.MemberDecorate(perVertexInputStructType, 0, Decoration.PassthroughNV);
+ context.MemberDecorate(perVertexInputStructType, 1, Decoration.PassthroughNV);
+ context.MemberDecorate(perVertexInputStructType, 2, Decoration.PassthroughNV);
+ context.MemberDecorate(perVertexInputStructType, 3, Decoration.PassthroughNV);
+ }
}
var perVertexOutputStructType = CreatePerVertexStructType(context);
diff --git a/src/Ryujinx.Graphics.Shader/StructuredIr/PhiFunctions.cs b/src/Ryujinx.Graphics.Shader/StructuredIr/PhiFunctions.cs
index 8b1cb9c566..90f1f2f6d8 100644
--- a/src/Ryujinx.Graphics.Shader/StructuredIr/PhiFunctions.cs
+++ b/src/Ryujinx.Graphics.Shader/StructuredIr/PhiFunctions.cs
@@ -24,17 +24,21 @@ namespace Ryujinx.Graphics.Shader.StructuredIr
continue;
}
+ Operand temp = OperandHelper.Local();
+
for (int index = 0; index < phi.SourcesCount; index++)
{
Operand src = phi.GetSource(index);
-
BasicBlock srcBlock = phi.GetBlock(index);
- Operation copyOp = new(Instruction.Copy, phi.Dest, src);
+ Operation copyOp = new(Instruction.Copy, temp, src);
srcBlock.Append(copyOp);
}
+ Operation copyOp2 = new(Instruction.Copy, phi.Dest, temp);
+
+ nextNode = block.Operations.AddAfter(node, copyOp2).Next;
block.Operations.Remove(node);
node = nextNode;
diff --git a/src/Ryujinx.Graphics.Shader/Translation/Optimizations/Optimizer.cs b/src/Ryujinx.Graphics.Shader/Translation/Optimizations/Optimizer.cs
index 17427a5f90..ea06691ba9 100644
--- a/src/Ryujinx.Graphics.Shader/Translation/Optimizations/Optimizer.cs
+++ b/src/Ryujinx.Graphics.Shader/Translation/Optimizations/Optimizer.cs
@@ -322,7 +322,7 @@ namespace Ryujinx.Graphics.Shader.Translation.Optimizations
Operand lhs = operation.GetSource(0);
Operand rhs = operation.GetSource(1);
- // Check LHS of the the main multiplication operation. We expect an input being multiplied by gl_FragCoord.w.
+ // Check LHS of the main multiplication operation. We expect an input being multiplied by gl_FragCoord.w.
if (lhs.AsgOp is not Operation attrMulOp || attrMulOp.Inst != (Instruction.FP32 | Instruction.Multiply))
{
return;
diff --git a/src/Ryujinx.Graphics.Vulkan/VulkanInitialization.cs b/src/Ryujinx.Graphics.Vulkan/VulkanInitialization.cs
index 3b1321c58e..d59ca7e0ec 100644
--- a/src/Ryujinx.Graphics.Vulkan/VulkanInitialization.cs
+++ b/src/Ryujinx.Graphics.Vulkan/VulkanInitialization.cs
@@ -136,7 +136,7 @@ namespace Ryujinx.Graphics.Vulkan
{
instance.EnumeratePhysicalDevices(out var physicalDevices).ThrowOnError();
- // First we try to pick the the user preferred GPU.
+ // First we try to pick the user preferred GPU.
for (int i = 0; i < physicalDevices.Length; i++)
{
if (IsPreferredAndSuitableDevice(api, physicalDevices[i], surface, preferredGpuId))
diff --git a/src/Ryujinx.Graphics.Vulkan/VulkanRenderer.cs b/src/Ryujinx.Graphics.Vulkan/VulkanRenderer.cs
index 434545fe01..7d7c109525 100644
--- a/src/Ryujinx.Graphics.Vulkan/VulkanRenderer.cs
+++ b/src/Ryujinx.Graphics.Vulkan/VulkanRenderer.cs
@@ -781,7 +781,9 @@ namespace Ryujinx.Graphics.Vulkan
{
PrimitiveTopology.Quads => PrimitiveTopology.Triangles,
PrimitiveTopology.QuadStrip => PrimitiveTopology.TriangleStrip,
- PrimitiveTopology.TriangleFan => Capabilities.PortabilitySubset.HasFlag(PortabilitySubsetFlags.NoTriangleFans) ? PrimitiveTopology.Triangles : topology,
+ PrimitiveTopology.TriangleFan or PrimitiveTopology.Polygon => Capabilities.PortabilitySubset.HasFlag(PortabilitySubsetFlags.NoTriangleFans)
+ ? PrimitiveTopology.Triangles
+ : topology,
_ => topology,
};
}
@@ -791,7 +793,7 @@ namespace Ryujinx.Graphics.Vulkan
return topology switch
{
PrimitiveTopology.Quads => true,
- PrimitiveTopology.TriangleFan => Capabilities.PortabilitySubset.HasFlag(PortabilitySubsetFlags.NoTriangleFans),
+ PrimitiveTopology.TriangleFan or PrimitiveTopology.Polygon => Capabilities.PortabilitySubset.HasFlag(PortabilitySubsetFlags.NoTriangleFans),
_ => false,
};
}
diff --git a/src/Ryujinx.Gtk3/UI/RendererWidgetBase.cs b/src/Ryujinx.Gtk3/UI/RendererWidgetBase.cs
index e27d060440..0e636792db 100644
--- a/src/Ryujinx.Gtk3/UI/RendererWidgetBase.cs
+++ b/src/Ryujinx.Gtk3/UI/RendererWidgetBase.cs
@@ -3,6 +3,7 @@ using Gtk;
using Ryujinx.Common;
using Ryujinx.Common.Configuration;
using Ryujinx.Common.Logging;
+using Ryujinx.Common.Utilities;
using Ryujinx.Graphics.GAL;
using Ryujinx.Graphics.GAL.Multithreading;
using Ryujinx.Graphics.Gpu;
@@ -378,8 +379,12 @@ namespace Ryujinx.UI
{
lock (this)
{
- var currentTime = DateTime.Now;
- string filename = $"ryujinx_capture_{currentTime.Year}-{currentTime.Month:D2}-{currentTime.Day:D2}_{currentTime.Hour:D2}-{currentTime.Minute:D2}-{currentTime.Second:D2}.png";
+ string applicationName = Device.Processes.ActiveApplication.Name;
+ string sanitizedApplicationName = FileSystemUtils.SanitizeFileName(applicationName);
+ DateTime currentTime = DateTime.Now;
+
+ string filename = $"{sanitizedApplicationName}_{currentTime.Year}-{currentTime.Month:D2}-{currentTime.Day:D2}_{currentTime.Hour:D2}-{currentTime.Minute:D2}-{currentTime.Second:D2}.png";
+
string directory = AppDataManager.Mode switch
{
AppDataManager.LaunchMode.Portable or AppDataManager.LaunchMode.Custom => System.IO.Path.Combine(AppDataManager.BaseDirPath, "screenshots"),
diff --git a/src/Ryujinx.HLE/HOS/Applets/SoftwareKeyboard/SoftwareKeyboardRendererBase.cs b/src/Ryujinx.HLE/HOS/Applets/SoftwareKeyboard/SoftwareKeyboardRendererBase.cs
index 75c648ff15..9e48568e13 100644
--- a/src/Ryujinx.HLE/HOS/Applets/SoftwareKeyboard/SoftwareKeyboardRendererBase.cs
+++ b/src/Ryujinx.HLE/HOS/Applets/SoftwareKeyboard/SoftwareKeyboardRendererBase.cs
@@ -44,10 +44,10 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
private readonly Color _textSelectedColor;
private readonly Color _textOverCursorColor;
- private readonly IBrush _panelBrush;
- private readonly IBrush _disabledBrush;
- private readonly IBrush _cursorBrush;
- private readonly IBrush _selectionBoxBrush;
+ private readonly Brush _panelBrush;
+ private readonly Brush _disabledBrush;
+ private readonly Brush _cursorBrush;
+ private readonly Brush _selectionBoxBrush;
private readonly Pen _textBoxOutlinePen;
private readonly Pen _cursorPen;
@@ -97,10 +97,10 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
_cursorBrush = new SolidBrush(_textNormalColor);
_selectionBoxBrush = new SolidBrush(selectionBackgroundColor);
- _textBoxOutlinePen = new Pen(borderColor, _textBoxOutlineWidth);
- _cursorPen = new Pen(_textNormalColor, cursorWidth);
- _selectionBoxPen = new Pen(selectionBackgroundColor, cursorWidth);
- _padPressedPen = new Pen(borderColor, _padPressedPenWidth);
+ _textBoxOutlinePen = Pens.Solid(borderColor, _textBoxOutlineWidth);
+ _cursorPen = Pens.Solid(_textNormalColor, cursorWidth);
+ _selectionBoxPen = Pens.Solid(selectionBackgroundColor, cursorWidth);
+ _padPressedPen = Pens.Solid(borderColor, _padPressedPenWidth);
_inputTextFontSize = 20;
@@ -178,7 +178,7 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
private static void SetGraphicsOptions(IImageProcessingContext context)
{
context.GetGraphicsOptions().Antialias = true;
- context.GetShapeGraphicsOptions().GraphicsOptions.Antialias = true;
+ context.GetDrawingOptions().GraphicsOptions.Antialias = true;
}
private void DrawImmutableElements()
@@ -293,31 +293,31 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
}
private static RectangleF MeasureString(string text, Font font)
{
- RendererOptions options = new(font);
+ TextOptions options = new(font);
if (text == "")
{
- FontRectangle emptyRectangle = TextMeasurer.Measure(" ", options);
+ FontRectangle emptyRectangle = TextMeasurer.MeasureSize(" ", options);
return new RectangleF(0, emptyRectangle.Y, 0, emptyRectangle.Height);
}
- FontRectangle rectangle = TextMeasurer.Measure(text, options);
+ FontRectangle rectangle = TextMeasurer.MeasureSize(text, options);
return new RectangleF(rectangle.X, rectangle.Y, rectangle.Width, rectangle.Height);
}
private static RectangleF MeasureString(ReadOnlySpan text, Font font)
{
- RendererOptions options = new(font);
+ TextOptions options = new(font);
if (text == "")
{
- FontRectangle emptyRectangle = TextMeasurer.Measure(" ", options);
+ FontRectangle emptyRectangle = TextMeasurer.MeasureSize(" ", options);
return new RectangleF(0, emptyRectangle.Y, 0, emptyRectangle.Height);
}
- FontRectangle rectangle = TextMeasurer.Measure(text, options);
+ FontRectangle rectangle = TextMeasurer.MeasureSize(text, options);
return new RectangleF(rectangle.X, rectangle.Y, rectangle.Width, rectangle.Height);
}
@@ -350,7 +350,7 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
// Draw the cursor on top of the text and redraw the text with a different color if necessary.
Color cursorTextColor;
- IBrush cursorBrush;
+ Brush cursorBrush;
Pen cursorPen;
float cursorPositionYTop = inputTextY + 1;
@@ -435,7 +435,7 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
new PointF(cursorPositionXLeft, cursorPositionYBottom),
};
- context.DrawLines(cursorPen, points);
+ context.DrawLine(cursorPen, points);
}
else
{
@@ -466,7 +466,7 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
private void DrawPadButton(IImageProcessingContext context, PointF point, Image icon, string label, bool pressed, bool enabled)
{
- // Use relative positions so we can center the the entire drawing later.
+ // Use relative positions so we can center the entire drawing later.
float iconX = 0;
float iconY = 0;
@@ -522,7 +522,7 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
{
var labelRectangle = MeasureString(ControllerToggleText, _labelsTextFont);
- // Use relative positions so we can center the the entire drawing later.
+ // Use relative positions so we can center the entire drawing later.
float keyWidth = _keyModeIcon.Width;
float keyHeight = _keyModeIcon.Height;
@@ -562,12 +562,12 @@ namespace Ryujinx.HLE.HOS.Applets.SoftwareKeyboard
// Convert the pixel format used in the image to the one used in the Switch surface.
- if (!_surface.TryGetSinglePixelSpan(out Span pixels))
+ if (!_surface.DangerousTryGetSinglePixelMemory(out Memory pixels))
{
return;
}
- _bufferData = MemoryMarshal.AsBytes(pixels).ToArray();
+ _bufferData = MemoryMarshal.AsBytes(pixels.Span).ToArray();
Span dataConvert = MemoryMarshal.Cast(_bufferData);
Debug.Assert(_bufferData.Length == _surfaceInfo.Size);
diff --git a/src/Ryujinx.Memory/AddressSpaceManager.cs b/src/Ryujinx.Memory/AddressSpaceManager.cs
index 021d336636..0a4a951439 100644
--- a/src/Ryujinx.Memory/AddressSpaceManager.cs
+++ b/src/Ryujinx.Memory/AddressSpaceManager.cs
@@ -11,12 +11,8 @@ namespace Ryujinx.Memory
/// Represents a address space manager.
/// Supports virtual memory region mapping, address translation and read/write access to mapped regions.
///
- public sealed class AddressSpaceManager : IVirtualMemoryManager, IWritableBlock
+ public sealed class AddressSpaceManager : VirtualMemoryManagerBase, IVirtualMemoryManager, IWritableBlock
{
- public const int PageBits = PageTable.PageBits;
- public const int PageSize = PageTable.PageSize;
- public const int PageMask = PageTable.PageMask;
-
///
public bool Supports4KBPages => true;
@@ -25,11 +21,11 @@ namespace Ryujinx.Memory
///
public int AddressSpaceBits { get; }
- private readonly ulong _addressSpaceSize;
-
private readonly MemoryBlock _backingMemory;
private readonly PageTable _pageTable;
+ protected override ulong AddressSpaceSize { get; }
+
///
/// Creates a new instance of the memory manager.
///
@@ -47,7 +43,7 @@ namespace Ryujinx.Memory
}
AddressSpaceBits = asBits;
- _addressSpaceSize = asSize;
+ AddressSpaceSize = asSize;
_backingMemory = backingMemory;
_pageTable = new PageTable();
}
@@ -102,12 +98,6 @@ namespace Ryujinx.Memory
return MemoryMarshal.Cast(GetSpan(va, Unsafe.SizeOf()))[0];
}
- ///
- public void Read(ulong va, Span data)
- {
- ReadImpl(va, data);
- }
-
///
public void Write(ulong va, T value) where T : unmanaged
{
@@ -174,7 +164,7 @@ namespace Ryujinx.Memory
{
Span data = new byte[size];
- ReadImpl(va, data);
+ Read(va, data);
return data;
}
@@ -346,34 +336,6 @@ namespace Ryujinx.Memory
return regions;
}
- private void ReadImpl(ulong va, Span data)
- {
- if (data.Length == 0)
- {
- return;
- }
-
- AssertValidAddressAndSize(va, (ulong)data.Length);
-
- int offset = 0, size;
-
- if ((va & PageMask) != 0)
- {
- size = Math.Min(data.Length, PageSize - (int)(va & PageMask));
-
- GetHostSpanContiguous(va, size).CopyTo(data[..size]);
-
- offset += size;
- }
-
- for (; offset < data.Length; offset += size)
- {
- size = Math.Min(data.Length - offset, PageSize);
-
- GetHostSpanContiguous(va + (ulong)offset, size).CopyTo(data.Slice(offset, size));
- }
- }
-
///
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public bool IsMapped(ulong va)
@@ -414,37 +376,6 @@ namespace Ryujinx.Memory
return true;
}
- private bool ValidateAddress(ulong va)
- {
- return va < _addressSpaceSize;
- }
-
- ///
- /// Checks if the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// True if the combination of virtual address and size is part of the addressable space
- private bool ValidateAddressAndSize(ulong va, ulong size)
- {
- ulong endVa = va + size;
- return endVa >= va && endVa >= size && endVa <= _addressSpaceSize;
- }
-
- ///
- /// Ensures the combination of virtual address and size is part of the addressable space.
- ///
- /// Virtual address of the range
- /// Size of the range in bytes
- /// Throw when the memory region specified outside the addressable space
- private void AssertValidAddressAndSize(ulong va, ulong size)
- {
- if (!ValidateAddressAndSize(va, size))
- {
- throw new InvalidMemoryRegionException($"va=0x{va:X16}, size=0x{size:X16}");
- }
- }
-
private unsafe Span GetHostSpanContiguous(ulong va, int size)
{
return new Span((void*)GetHostAddress(va), size);
@@ -461,7 +392,7 @@ namespace Ryujinx.Memory
}
///
- public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest = false)
{
throw new NotImplementedException();
}
@@ -471,5 +402,11 @@ namespace Ryujinx.Memory
{
// Only the ARM Memory Manager has tracking for now.
}
+
+ protected override unsafe Span GetPhysicalAddressSpan(nuint pa, int size)
+ => new((void*)pa, size);
+
+ protected override nuint TranslateVirtualAddressForRead(ulong va)
+ => GetHostAddress(va);
}
}
diff --git a/src/Ryujinx.Memory/IVirtualMemoryManager.cs b/src/Ryujinx.Memory/IVirtualMemoryManager.cs
index 9cf3663cfc..557da2f261 100644
--- a/src/Ryujinx.Memory/IVirtualMemoryManager.cs
+++ b/src/Ryujinx.Memory/IVirtualMemoryManager.cs
@@ -214,6 +214,7 @@ namespace Ryujinx.Memory
/// Virtual address base
/// Size of the region to protect
/// Memory protection to set
- void TrackingReprotect(ulong va, ulong size, MemoryPermission protection);
+ /// True if the protection is for guest access, false otherwise
+ void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest);
}
}
diff --git a/src/Ryujinx.Memory/MemoryBlock.cs b/src/Ryujinx.Memory/MemoryBlock.cs
index 7fe7862a91..59ee269bb0 100644
--- a/src/Ryujinx.Memory/MemoryBlock.cs
+++ b/src/Ryujinx.Memory/MemoryBlock.cs
@@ -174,7 +174,7 @@ namespace Ryujinx.Memory
/// Starting offset of the range being read
/// Span where the bytes being read will be copied to
/// Throw when the memory block has already been disposed
- /// Throw when the memory region specified for the the data is out of range
+ /// Throw when the memory region specified for the data is out of range
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Read(ulong offset, Span data)
{
@@ -188,7 +188,7 @@ namespace Ryujinx.Memory
/// Offset where the data is located
/// Data at the specified address
/// Throw when the memory block has already been disposed
- /// Throw when the memory region specified for the the data is out of range
+ /// Throw when the memory region specified for the data is out of range
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public T Read(ulong offset) where T : unmanaged
{
@@ -201,7 +201,7 @@ namespace Ryujinx.Memory
/// Starting offset of the range being written
/// Span where the bytes being written will be copied from
/// Throw when the memory block has already been disposed
- /// Throw when the memory region specified for the the data is out of range
+ /// Throw when the memory region specified for the data is out of range
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Write(ulong offset, ReadOnlySpan data)
{
@@ -215,7 +215,7 @@ namespace Ryujinx.Memory
/// Offset to write the data into
/// Data to be written
/// Throw when the memory block has already been disposed
- /// Throw when the memory region specified for the the data is out of range
+ /// Throw when the memory region specified for the data is out of range
[MethodImpl(MethodImplOptions.AggressiveInlining)]
public void Write(ulong offset, T data) where T : unmanaged
{
diff --git a/src/Ryujinx.Memory/Tracking/MemoryTracking.cs b/src/Ryujinx.Memory/Tracking/MemoryTracking.cs
index 6febcbbb3f..96cb2c5f52 100644
--- a/src/Ryujinx.Memory/Tracking/MemoryTracking.cs
+++ b/src/Ryujinx.Memory/Tracking/MemoryTracking.cs
@@ -14,9 +14,14 @@ namespace Ryujinx.Memory.Tracking
// Only use these from within the lock.
private readonly NonOverlappingRangeList _virtualRegions;
+ // Guest virtual regions are a subset of the normal virtual regions, with potentially different protection
+ // and expanded area of effect on platforms that don't support misaligned page protection.
+ private readonly NonOverlappingRangeList _guestVirtualRegions;
private readonly int _pageSize;
+ private readonly bool _singleByteGuestTracking;
+
///
/// This lock must be obtained when traversing or updating the region-handle hierarchy.
/// It is not required when reading dirty flags.
@@ -27,16 +32,27 @@ namespace Ryujinx.Memory.Tracking
/// Create a new tracking structure for the given "physical" memory block,
/// with a given "virtual" memory manager that will provide mappings and virtual memory protection.
///
+ ///
+ /// If is true, the memory manager must also support protection on partially
+ /// unmapped regions without throwing exceptions or dropping protection on the mapped portion.
+ ///
/// Virtual memory manager
- /// Physical memory block
/// Page size of the virtual memory space
- public MemoryTracking(IVirtualMemoryManager memoryManager, int pageSize, InvalidAccessHandler invalidAccessHandler = null)
+ /// Method to call for invalid memory accesses
+ /// True if the guest only signals writes for the first byte
+ public MemoryTracking(
+ IVirtualMemoryManager memoryManager,
+ int pageSize,
+ InvalidAccessHandler invalidAccessHandler = null,
+ bool singleByteGuestTracking = false)
{
_memoryManager = memoryManager;
_pageSize = pageSize;
_invalidAccessHandler = invalidAccessHandler;
+ _singleByteGuestTracking = singleByteGuestTracking;
_virtualRegions = new NonOverlappingRangeList();
+ _guestVirtualRegions = new NonOverlappingRangeList();
}
private (ulong address, ulong size) PageAlign(ulong address, ulong size)
@@ -62,20 +78,25 @@ namespace Ryujinx.Memory.Tracking
{
ref var overlaps = ref ThreadStaticArray.Get();
- int count = _virtualRegions.FindOverlapsNonOverlapping(va, size, ref overlaps);
-
- for (int i = 0; i < count; i++)
+ for (int type = 0; type < 2; type++)
{
- VirtualRegion region = overlaps[i];
+ NonOverlappingRangeList regions = type == 0 ? _virtualRegions : _guestVirtualRegions;
- // If the region has been fully remapped, signal that it has been mapped again.
- bool remapped = _memoryManager.IsRangeMapped(region.Address, region.Size);
- if (remapped)
+ int count = regions.FindOverlapsNonOverlapping(va, size, ref overlaps);
+
+ for (int i = 0; i < count; i++)
{
- region.SignalMappingChanged(true);
- }
+ VirtualRegion region = overlaps[i];
- region.UpdateProtection();
+ // If the region has been fully remapped, signal that it has been mapped again.
+ bool remapped = _memoryManager.IsRangeMapped(region.Address, region.Size);
+ if (remapped)
+ {
+ region.SignalMappingChanged(true);
+ }
+
+ region.UpdateProtection();
+ }
}
}
}
@@ -95,27 +116,58 @@ namespace Ryujinx.Memory.Tracking
{
ref var overlaps = ref ThreadStaticArray.Get();
- int count = _virtualRegions.FindOverlapsNonOverlapping(va, size, ref overlaps);
-
- for (int i = 0; i < count; i++)
+ for (int type = 0; type < 2; type++)
{
- VirtualRegion region = overlaps[i];
+ NonOverlappingRangeList regions = type == 0 ? _virtualRegions : _guestVirtualRegions;
- region.SignalMappingChanged(false);
+ int count = regions.FindOverlapsNonOverlapping(va, size, ref overlaps);
+
+ for (int i = 0; i < count; i++)
+ {
+ VirtualRegion region = overlaps[i];
+
+ region.SignalMappingChanged(false);
+ }
}
}
}
+ ///
+ /// Alter a tracked memory region to properly capture unaligned accesses.
+ /// For most memory manager modes, this does nothing.
+ ///
+ /// Original region address
+ /// Original region size
+ /// A new address and size for tracking unaligned accesses
+ internal (ulong newAddress, ulong newSize) GetUnalignedSafeRegion(ulong address, ulong size)
+ {
+ if (_singleByteGuestTracking)
+ {
+ // The guest only signals the first byte of each memory access with the current memory manager.
+ // To catch unaligned access properly, we need to also protect the page before the address.
+
+ // Assume that the address and size are already aligned.
+
+ return (address - (ulong)_pageSize, size + (ulong)_pageSize);
+ }
+ else
+ {
+ return (address, size);
+ }
+ }
+
///
/// Get a list of virtual regions that a handle covers.
///
/// Starting virtual memory address of the handle
/// Size of the handle's memory region
+ /// True if getting handles for guest protection, false otherwise
/// A list of virtual regions within the given range
- internal List GetVirtualRegionsForHandle(ulong va, ulong size)
+ internal List GetVirtualRegionsForHandle(ulong va, ulong size, bool guest)
{
List result = new();
- _virtualRegions.GetOrAddRegions(result, va, size, (va, size) => new VirtualRegion(this, va, size));
+ NonOverlappingRangeList regions = guest ? _guestVirtualRegions : _virtualRegions;
+ regions.GetOrAddRegions(result, va, size, (va, size) => new VirtualRegion(this, va, size, guest));
return result;
}
@@ -126,7 +178,14 @@ namespace Ryujinx.Memory.Tracking
/// Region to remove
internal void RemoveVirtual(VirtualRegion region)
{
- _virtualRegions.Remove(region);
+ if (region.Guest)
+ {
+ _guestVirtualRegions.Remove(region);
+ }
+ else
+ {
+ _virtualRegions.Remove(region);
+ }
}
///
@@ -137,10 +196,11 @@ namespace Ryujinx.Memory.Tracking
/// Handles to inherit state from or reuse. When none are present, provide null
/// Desired granularity of write tracking
/// Handle ID
+ /// Region flags
/// The memory tracking handle
- public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id)
+ public MultiRegionHandle BeginGranularTracking(ulong address, ulong size, IEnumerable handles, ulong granularity, int id, RegionFlags flags = RegionFlags.None)
{
- return new MultiRegionHandle(this, address, size, handles, granularity, id);
+ return new MultiRegionHandle(this, address, size, handles, granularity, id, flags);
}
///
@@ -164,15 +224,16 @@ namespace Ryujinx.Memory.Tracking
/// CPU virtual address of the region
/// Size of the region
/// Handle ID
+ /// Region flags
/// The memory tracking handle
- public RegionHandle BeginTracking(ulong address, ulong size, int id)
+ public RegionHandle BeginTracking(ulong address, ulong size, int id, RegionFlags flags = RegionFlags.None)
{
var (paAddress, paSize) = PageAlign(address, size);
lock (TrackingLock)
{
bool mapped = _memoryManager.IsRangeMapped(address, size);
- RegionHandle handle = new(this, paAddress, paSize, address, size, id, mapped);
+ RegionHandle handle = new(this, paAddress, paSize, address, size, id, flags, mapped);
return handle;
}
@@ -186,15 +247,16 @@ namespace Ryujinx.Memory.Tracking
/// The bitmap owning the dirty flag for this handle
/// The bit of this handle within the dirty flag
/// Handle ID
+ /// Region flags
/// The memory tracking handle
- internal RegionHandle BeginTrackingBitmap(ulong address, ulong size, ConcurrentBitmap bitmap, int bit, int id)
+ internal RegionHandle BeginTrackingBitmap(ulong address, ulong size, ConcurrentBitmap bitmap, int bit, int id, RegionFlags flags = RegionFlags.None)
{
var (paAddress, paSize) = PageAlign(address, size);
lock (TrackingLock)
{
bool mapped = _memoryManager.IsRangeMapped(address, size);
- RegionHandle handle = new(this, paAddress, paSize, address, size, bitmap, bit, id, mapped);
+ RegionHandle handle = new(this, paAddress, paSize, address, size, bitmap, bit, id, flags, mapped);
return handle;
}
@@ -202,6 +264,7 @@ namespace Ryujinx.Memory.Tracking
///
/// Signal that a virtual memory event happened at the given location.
+ /// The memory event is assumed to be triggered by guest code.
///
/// Virtual address accessed
/// Size of the region affected in bytes
@@ -209,7 +272,7 @@ namespace Ryujinx.Memory.Tracking
/// True if the event triggered any tracking regions, false otherwise
public bool VirtualMemoryEvent(ulong address, ulong size, bool write)
{
- return VirtualMemoryEvent(address, size, write, precise: false, null);
+ return VirtualMemoryEvent(address, size, write, precise: false, exemptId: null, guest: true);
}
///
@@ -222,8 +285,9 @@ namespace Ryujinx.Memory.Tracking
/// Whether the region was written to or read
/// True if the access is precise, false otherwise
/// Optional ID that of the handles that should not be signalled
+ /// True if the access is from the guest, false otherwise
/// True if the event triggered any tracking regions, false otherwise
- public bool VirtualMemoryEvent(ulong address, ulong size, bool write, bool precise, int? exemptId = null)
+ public bool VirtualMemoryEvent(ulong address, ulong size, bool write, bool precise, int? exemptId = null, bool guest = false)
{
// Look up the virtual region using the region list.
// Signal up the chain to relevant handles.
@@ -234,7 +298,9 @@ namespace Ryujinx.Memory.Tracking
{
ref var overlaps = ref ThreadStaticArray.Get();
- int count = _virtualRegions.FindOverlapsNonOverlapping(address, size, ref overlaps);
+ NonOverlappingRangeList regions = guest ? _guestVirtualRegions : _virtualRegions;
+
+ int count = regions.FindOverlapsNonOverlapping(address, size, ref overlaps);
if (count == 0 && !precise)
{
@@ -242,7 +308,7 @@ namespace Ryujinx.Memory.Tracking
{
// TODO: There is currently the possibility that a page can be protected after its virtual region is removed.
// This code handles that case when it happens, but it would be better to find out how this happens.
- _memoryManager.TrackingReprotect(address & ~(ulong)(_pageSize - 1), (ulong)_pageSize, MemoryPermission.ReadAndWrite);
+ _memoryManager.TrackingReprotect(address & ~(ulong)(_pageSize - 1), (ulong)_pageSize, MemoryPermission.ReadAndWrite, guest);
return true; // This memory _should_ be mapped, so we need to try again.
}
else
@@ -252,6 +318,12 @@ namespace Ryujinx.Memory.Tracking
}
else
{
+ if (guest && _singleByteGuestTracking)
+ {
+ // Increase the access size to trigger handles with misaligned accesses.
+ size += (ulong)_pageSize;
+ }
+
for (int i = 0; i < count; i++)
{
VirtualRegion region = overlaps[i];
@@ -285,9 +357,10 @@ namespace Ryujinx.Memory.Tracking
///
/// Region to reprotect
/// Memory permission to protect with
- internal void ProtectVirtualRegion(VirtualRegion region, MemoryPermission permission)
+ /// True if the protection is for guest access, false otherwise
+ internal void ProtectVirtualRegion(VirtualRegion region, MemoryPermission permission, bool guest)
{
- _memoryManager.TrackingReprotect(region.Address, region.Size, permission);
+ _memoryManager.TrackingReprotect(region.Address, region.Size, permission, guest);
}
///
diff --git a/src/Ryujinx.Memory/Tracking/MultiRegionHandle.cs b/src/Ryujinx.Memory/Tracking/MultiRegionHandle.cs
index 1c1b48b3c9..6fdca69f5f 100644
--- a/src/Ryujinx.Memory/Tracking/MultiRegionHandle.cs
+++ b/src/Ryujinx.Memory/Tracking/MultiRegionHandle.cs
@@ -37,7 +37,8 @@ namespace Ryujinx.Memory.Tracking
ulong size,
IEnumerable handles,
ulong granularity,
- int id)
+ int id,
+ RegionFlags flags)
{
_handles = new RegionHandle[(size + granularity - 1) / granularity];
Granularity = granularity;
@@ -62,7 +63,7 @@ namespace Ryujinx.Memory.Tracking
// Fill any gap left before this handle.
while (i < startIndex)
{
- RegionHandle fillHandle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id);
+ RegionHandle fillHandle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id, flags);
fillHandle.Parent = this;
_handles[i++] = fillHandle;
}
@@ -83,7 +84,7 @@ namespace Ryujinx.Memory.Tracking
while (i < endIndex)
{
- RegionHandle splitHandle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id);
+ RegionHandle splitHandle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id, flags);
splitHandle.Parent = this;
splitHandle.Reprotect(handle.Dirty);
@@ -106,7 +107,7 @@ namespace Ryujinx.Memory.Tracking
// Fill any remaining space with new handles.
while (i < _handles.Length)
{
- RegionHandle handle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id);
+ RegionHandle handle = tracking.BeginTrackingBitmap(address + (ulong)i * granularity, granularity, _dirtyBitmap, i, id, flags);
handle.Parent = this;
_handles[i++] = handle;
}
diff --git a/src/Ryujinx.Memory/Tracking/RegionFlags.cs b/src/Ryujinx.Memory/Tracking/RegionFlags.cs
new file mode 100644
index 0000000000..ceb8e56a63
--- /dev/null
+++ b/src/Ryujinx.Memory/Tracking/RegionFlags.cs
@@ -0,0 +1,21 @@
+using System;
+using System.Collections.Generic;
+using System.Linq;
+using System.Text;
+using System.Threading.Tasks;
+
+namespace Ryujinx.Memory.Tracking
+{
+ [Flags]
+ public enum RegionFlags
+ {
+ None = 0,
+
+ ///
+ /// Access to the resource is expected to occasionally be unaligned.
+ /// With some memory managers, guest protection must extend into the previous page to cover unaligned access.
+ /// If this is not expected, protection is not altered, which can avoid unintended resource dirty/flush.
+ ///
+ UnalignedAccess = 1,
+ }
+}
diff --git a/src/Ryujinx.Memory/Tracking/RegionHandle.cs b/src/Ryujinx.Memory/Tracking/RegionHandle.cs
index df3d9c311a..a94ffa43c8 100644
--- a/src/Ryujinx.Memory/Tracking/RegionHandle.cs
+++ b/src/Ryujinx.Memory/Tracking/RegionHandle.cs
@@ -55,6 +55,8 @@ namespace Ryujinx.Memory.Tracking
private RegionSignal _preAction; // Action to perform before a read or write. This will block the memory access.
private PreciseRegionSignal _preciseAction; // Action to perform on a precise read or write.
private readonly List _regions;
+ private readonly List _guestRegions;
+ private readonly List _allRegions;
private readonly MemoryTracking _tracking;
private bool _disposed;
@@ -99,6 +101,7 @@ namespace Ryujinx.Memory.Tracking
/// The bitmap the dirty flag for this handle is stored in
/// The bit index representing the dirty flag for this handle
/// Handle ID
+ /// Region flags
/// True if the region handle starts mapped
internal RegionHandle(
MemoryTracking tracking,
@@ -109,6 +112,7 @@ namespace Ryujinx.Memory.Tracking
ConcurrentBitmap bitmap,
int bit,
int id,
+ RegionFlags flags,
bool mapped = true)
{
Bitmap = bitmap;
@@ -128,11 +132,12 @@ namespace Ryujinx.Memory.Tracking
RealEndAddress = realAddress + realSize;
_tracking = tracking;
- _regions = tracking.GetVirtualRegionsForHandle(address, size);
- foreach (var region in _regions)
- {
- region.Handles.Add(this);
- }
+
+ _regions = tracking.GetVirtualRegionsForHandle(address, size, false);
+ _guestRegions = GetGuestRegions(tracking, address, size, flags);
+ _allRegions = new List(_regions.Count + _guestRegions.Count);
+
+ InitializeRegions();
}
///
@@ -145,8 +150,9 @@ namespace Ryujinx.Memory.Tracking
/// The real, unaligned address of the handle
/// The real, unaligned size of the handle
/// Handle ID
+ /// Region flags
/// True if the region handle starts mapped
- internal RegionHandle(MemoryTracking tracking, ulong address, ulong size, ulong realAddress, ulong realSize, int id, bool mapped = true)
+ internal RegionHandle(MemoryTracking tracking, ulong address, ulong size, ulong realAddress, ulong realSize, int id, RegionFlags flags, bool mapped = true)
{
Bitmap = new ConcurrentBitmap(1, mapped);
@@ -163,8 +169,37 @@ namespace Ryujinx.Memory.Tracking
RealEndAddress = realAddress + realSize;
_tracking = tracking;
- _regions = tracking.GetVirtualRegionsForHandle(address, size);
- foreach (var region in _regions)
+
+ _regions = tracking.GetVirtualRegionsForHandle(address, size, false);
+ _guestRegions = GetGuestRegions(tracking, address, size, flags);
+ _allRegions = new List(_regions.Count + _guestRegions.Count);
+
+ InitializeRegions();
+ }
+
+ private List GetGuestRegions(MemoryTracking tracking, ulong address, ulong size, RegionFlags flags)
+ {
+ ulong guestAddress;
+ ulong guestSize;
+
+ if (flags.HasFlag(RegionFlags.UnalignedAccess))
+ {
+ (guestAddress, guestSize) = tracking.GetUnalignedSafeRegion(address, size);
+ }
+ else
+ {
+ (guestAddress, guestSize) = (address, size);
+ }
+
+ return tracking.GetVirtualRegionsForHandle(guestAddress, guestSize, true);
+ }
+
+ private void InitializeRegions()
+ {
+ _allRegions.AddRange(_regions);
+ _allRegions.AddRange(_guestRegions);
+
+ foreach (var region in _allRegions)
{
region.Handles.Add(this);
}
@@ -321,7 +356,7 @@ namespace Ryujinx.Memory.Tracking
lock (_tracking.TrackingLock)
{
- foreach (VirtualRegion region in _regions)
+ foreach (VirtualRegion region in _allRegions)
{
protectionChanged |= region.UpdateProtection();
}
@@ -379,7 +414,7 @@ namespace Ryujinx.Memory.Tracking
{
lock (_tracking.TrackingLock)
{
- foreach (VirtualRegion region in _regions)
+ foreach (VirtualRegion region in _allRegions)
{
region.UpdateProtection();
}
@@ -414,7 +449,16 @@ namespace Ryujinx.Memory.Tracking
/// Virtual region to add as a child
internal void AddChild(VirtualRegion region)
{
- _regions.Add(region);
+ if (region.Guest)
+ {
+ _guestRegions.Add(region);
+ }
+ else
+ {
+ _regions.Add(region);
+ }
+
+ _allRegions.Add(region);
}
///
@@ -469,7 +513,7 @@ namespace Ryujinx.Memory.Tracking
lock (_tracking.TrackingLock)
{
- foreach (VirtualRegion region in _regions)
+ foreach (VirtualRegion region in _allRegions)
{
region.RemoveHandle(this);
}
diff --git a/src/Ryujinx.Memory/Tracking/VirtualRegion.cs b/src/Ryujinx.Memory/Tracking/VirtualRegion.cs
index 538e94fef5..bb087e9af0 100644
--- a/src/Ryujinx.Memory/Tracking/VirtualRegion.cs
+++ b/src/Ryujinx.Memory/Tracking/VirtualRegion.cs
@@ -13,10 +13,14 @@ namespace Ryujinx.Memory.Tracking
private readonly MemoryTracking _tracking;
private MemoryPermission _lastPermission;
- public VirtualRegion(MemoryTracking tracking, ulong address, ulong size, MemoryPermission lastPermission = MemoryPermission.Invalid) : base(address, size)
+ public bool Guest { get; }
+
+ public VirtualRegion(MemoryTracking tracking, ulong address, ulong size, bool guest, MemoryPermission lastPermission = MemoryPermission.Invalid) : base(address, size)
{
_lastPermission = lastPermission;
_tracking = tracking;
+
+ Guest = guest;
}
///
@@ -103,7 +107,7 @@ namespace Ryujinx.Memory.Tracking
if (_lastPermission != permission)
{
- _tracking.ProtectVirtualRegion(this, permission);
+ _tracking.ProtectVirtualRegion(this, permission, Guest);
_lastPermission = permission;
return true;
@@ -131,7 +135,7 @@ namespace Ryujinx.Memory.Tracking
public override INonOverlappingRange Split(ulong splitAddress)
{
- VirtualRegion newRegion = new(_tracking, splitAddress, EndAddress - splitAddress, _lastPermission);
+ VirtualRegion newRegion = new(_tracking, splitAddress, EndAddress - splitAddress, Guest, _lastPermission);
Size = splitAddress - Address;
// The new region inherits all of our parents.
diff --git a/src/Ryujinx.Memory/VirtualMemoryManagerBase.cs b/src/Ryujinx.Memory/VirtualMemoryManagerBase.cs
new file mode 100644
index 0000000000..cbec88cc56
--- /dev/null
+++ b/src/Ryujinx.Memory/VirtualMemoryManagerBase.cs
@@ -0,0 +1,91 @@
+using System;
+using System.Numerics;
+
+namespace Ryujinx.Memory
+{
+ public abstract class VirtualMemoryManagerBase
+ where TVirtual : IBinaryInteger
+ where TPhysical : IBinaryInteger
+ {
+ public const int PageBits = 12;
+ public const int PageSize = 1 << PageBits;
+ public const int PageMask = PageSize - 1;
+
+ protected abstract TVirtual AddressSpaceSize { get; }
+
+ public virtual void Read(TVirtual va, Span data)
+ {
+ if (data.Length == 0)
+ {
+ return;
+ }
+
+ AssertValidAddressAndSize(va, TVirtual.CreateChecked(data.Length));
+
+ int offset = 0, size;
+
+ if ((int.CreateTruncating(va) & PageMask) != 0)
+ {
+ TPhysical pa = TranslateVirtualAddressForRead(va);
+
+ size = Math.Min(data.Length, PageSize - ((int.CreateTruncating(va) & PageMask)));
+
+ GetPhysicalAddressSpan(pa, size).CopyTo(data[..size]);
+
+ offset += size;
+ }
+
+ for (; offset < data.Length; offset += size)
+ {
+ TPhysical pa = TranslateVirtualAddressForRead(va + TVirtual.CreateChecked(offset));
+
+ size = Math.Min(data.Length - offset, PageSize);
+
+ GetPhysicalAddressSpan(pa, size).CopyTo(data.Slice(offset, size));
+ }
+ }
+
+ ///
+ /// Ensures the combination of virtual address and size is part of the addressable space.
+ ///
+ /// Virtual address of the range
+ /// Size of the range in bytes
+ /// Throw when the memory region specified outside the addressable space
+ protected void AssertValidAddressAndSize(TVirtual va, TVirtual size)
+ {
+ if (!ValidateAddressAndSize(va, size))
+ {
+ throw new InvalidMemoryRegionException($"va=0x{va:X16}, size=0x{size:X16}");
+ }
+ }
+
+ protected abstract Span GetPhysicalAddressSpan(TPhysical pa, int size);
+
+ protected abstract TPhysical TranslateVirtualAddressForRead(TVirtual va);
+
+ ///
+ /// Checks if the virtual address is part of the addressable space.
+ ///
+ /// Virtual address
+ /// True if the virtual address is part of the addressable space
+ protected bool ValidateAddress(TVirtual va)
+ {
+ return va < AddressSpaceSize;
+ }
+
+ ///
+ /// Checks if the combination of virtual address and size is part of the addressable space.
+ ///
+ /// Virtual address of the range
+ /// Size of the range in bytes
+ /// True if the combination of virtual address and size is part of the addressable space
+ protected bool ValidateAddressAndSize(TVirtual va, TVirtual size)
+ {
+ TVirtual endVa = va + size;
+ return endVa >= va && endVa >= size && endVa <= AddressSpaceSize;
+ }
+
+ protected static void ThrowInvalidMemoryRegionException(string message)
+ => throw new InvalidMemoryRegionException(message);
+ }
+}
diff --git a/src/Ryujinx.Tests.Memory/MockVirtualMemoryManager.cs b/src/Ryujinx.Tests.Memory/MockVirtualMemoryManager.cs
index 5c8396e6d4..85a1ac02bc 100644
--- a/src/Ryujinx.Tests.Memory/MockVirtualMemoryManager.cs
+++ b/src/Ryujinx.Tests.Memory/MockVirtualMemoryManager.cs
@@ -107,7 +107,7 @@ namespace Ryujinx.Tests.Memory
throw new NotImplementedException();
}
- public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection)
+ public void TrackingReprotect(ulong va, ulong size, MemoryPermission protection, bool guest)
{
OnProtect?.Invoke(va, size, protection);
}
diff --git a/src/Ryujinx.UI.Common/DiscordIntegrationModule.cs b/src/Ryujinx.UI.Common/DiscordIntegrationModule.cs
index 0b9439eaa9..bbece1e1d5 100644
--- a/src/Ryujinx.UI.Common/DiscordIntegrationModule.cs
+++ b/src/Ryujinx.UI.Common/DiscordIntegrationModule.cs
@@ -7,7 +7,7 @@ namespace Ryujinx.UI.Common
public static class DiscordIntegrationModule
{
private const string Description = "A simple, experimental Nintendo Switch emulator.";
- private const string CliendId = "568815339807309834";
+ private const string ApplicationId = "1216775165866807456";
private static DiscordRpcClient _discordClient;
private static RichPresence _discordPresenceMain;
@@ -24,14 +24,14 @@ namespace Ryujinx.UI.Common
Details = "Main Menu",
State = "Idling",
Timestamps = Timestamps.Now,
- Buttons = new[]
- {
+ Buttons =
+ [
new Button
{
Label = "Website",
- Url = "https://ryujinx.org/",
+ Url = "https://ryujinx.org/",
},
- },
+ ],
};
ConfigurationState.Instance.EnableDiscordIntegration.Event += Update;
@@ -52,7 +52,7 @@ namespace Ryujinx.UI.Common
// If we need to activate it and the client isn't active, initialize it
if (evnt.NewValue && _discordClient == null)
{
- _discordClient = new DiscordRpcClient(CliendId);
+ _discordClient = new DiscordRpcClient(ApplicationId);
_discordClient.Initialize();
_discordClient.SetPresence(_discordPresenceMain);
@@ -74,14 +74,14 @@ namespace Ryujinx.UI.Common
Details = $"Playing {titleName}",
State = (titleId == "0000000000000000") ? "Homebrew" : titleId.ToUpper(),
Timestamps = Timestamps.Now,
- Buttons = new[]
- {
+ Buttons =
+ [
new Button
{
Label = "Website",
Url = "https://ryujinx.org/",
},
- },
+ ],
});
}
diff --git a/src/Ryujinx/AppHost.cs b/src/Ryujinx/AppHost.cs
index 04cec95798..2620ea68c6 100644
--- a/src/Ryujinx/AppHost.cs
+++ b/src/Ryujinx/AppHost.cs
@@ -22,6 +22,7 @@ using Ryujinx.Common.Configuration;
using Ryujinx.Common.Configuration.Multiplayer;
using Ryujinx.Common.Logging;
using Ryujinx.Common.SystemInterop;
+using Ryujinx.Common.Utilities;
using Ryujinx.Graphics.GAL;
using Ryujinx.Graphics.GAL.Multithreading;
using Ryujinx.Graphics.Gpu;
@@ -279,8 +280,11 @@ namespace Ryujinx.Ava
{
lock (_lockObject)
{
+ string applicationName = Device.Processes.ActiveApplication.Name;
+ string sanitizedApplicationName = FileSystemUtils.SanitizeFileName(applicationName);
DateTime currentTime = DateTime.Now;
- string filename = $"ryujinx_capture_{currentTime.Year}-{currentTime.Month:D2}-{currentTime.Day:D2}_{currentTime.Hour:D2}-{currentTime.Minute:D2}-{currentTime.Second:D2}.png";
+
+ string filename = $"{sanitizedApplicationName}_{currentTime.Year}-{currentTime.Month:D2}-{currentTime.Day:D2}_{currentTime.Hour:D2}-{currentTime.Minute:D2}-{currentTime.Second:D2}.png";
string directory = AppDataManager.Mode switch
{
diff --git a/src/Ryujinx/Common/Locale/LocaleManager.cs b/src/Ryujinx/Common/Locale/LocaleManager.cs
index e973fcc067..2d3deeaa38 100644
--- a/src/Ryujinx/Common/Locale/LocaleManager.cs
+++ b/src/Ryujinx/Common/Locale/LocaleManager.cs
@@ -30,28 +30,22 @@ namespace Ryujinx.Ava.Common.Locale
Load();
}
- public void Load()
+ private void Load()
{
- // Load the system Language Code.
- string localeLanguageCode = CultureInfo.CurrentCulture.Name.Replace('-', '_');
+ var localeLanguageCode = !string.IsNullOrEmpty(ConfigurationState.Instance.UI.LanguageCode.Value) ?
+ ConfigurationState.Instance.UI.LanguageCode.Value : CultureInfo.CurrentCulture.Name.Replace('-', '_');
- // If the view is loaded with the UI Previewer detached, then override it with the saved one or default.
+ // Load en_US as default, if the target language translation is missing or incomplete.
+ LoadDefaultLanguage();
+ LoadLanguage(localeLanguageCode);
+
+ // Save whatever we ended up with.
if (Program.PreviewerDetached)
{
- if (!string.IsNullOrEmpty(ConfigurationState.Instance.UI.LanguageCode.Value))
- {
- localeLanguageCode = ConfigurationState.Instance.UI.LanguageCode.Value;
- }
- else
- {
- localeLanguageCode = DefaultLanguageCode;
- }
+ ConfigurationState.Instance.UI.LanguageCode.Value = _localeLanguageCode;
+
+ ConfigurationState.Instance.ToFileFormat().SaveConfig(Program.ConfigurationPath);
}
-
- // Load en_US as default, if the target language translation is incomplete.
- LoadDefaultLanguage();
-
- LoadLanguage(localeLanguageCode);
}
public string this[LocaleKeys key]
@@ -126,24 +120,42 @@ namespace Ryujinx.Ava.Common.Locale
private void LoadDefaultLanguage()
{
- _localeDefaultStrings = LoadJsonLanguage();
+ _localeDefaultStrings = LoadJsonLanguage(DefaultLanguageCode);
}
public void LoadLanguage(string languageCode)
{
- foreach (var item in LoadJsonLanguage(languageCode))
+ var locale = LoadJsonLanguage(languageCode);
+
+ if (locale == null)
+ {
+ _localeLanguageCode = DefaultLanguageCode;
+ locale = _localeDefaultStrings;
+ }
+ else
+ {
+ _localeLanguageCode = languageCode;
+ }
+
+ foreach (var item in locale)
{
this[item.Key] = item.Value;
}
- _localeLanguageCode = languageCode;
LocaleChanged?.Invoke();
}
- private static Dictionary LoadJsonLanguage(string languageCode = DefaultLanguageCode)
+ private static Dictionary LoadJsonLanguage(string languageCode)
{
var localeStrings = new Dictionary();
string languageJson = EmbeddedResources.ReadAllText($"Ryujinx/Assets/Locales/{languageCode}.json");
+
+ if (languageJson == null)
+ {
+ // We were unable to find file for that language code.
+ return null;
+ }
+
var strings = JsonHelper.Deserialize(languageJson, CommonJsonContext.Default.StringDictionary);
foreach (var item in strings)
diff --git a/src/Ryujinx/Modules/Updater/Updater.cs b/src/Ryujinx/Modules/Updater/Updater.cs
index d8346c8eb0..9f186f2b38 100644
--- a/src/Ryujinx/Modules/Updater/Updater.cs
+++ b/src/Ryujinx/Modules/Updater/Updater.cs
@@ -298,7 +298,14 @@ namespace Ryujinx.Modules
else
{
// Find the process name.
- string ryuName = Path.GetFileName(Environment.ProcessPath);
+ string ryuName = Path.GetFileName(Environment.ProcessPath) ?? string.Empty;
+
+ // Migration: Start the updated binary.
+ // TODO: Remove this in a future update.
+ if (ryuName.StartsWith("Ryujinx.Ava"))
+ {
+ ryuName = ryuName.Replace(".Ava", "");
+ }
// Some operating systems can see the renamed executable, so strip off the .ryuold if found.
if (ryuName.EndsWith(".ryuold"))
@@ -307,7 +314,7 @@ namespace Ryujinx.Modules
}
// Fallback if the executable could not be found.
- if (!Path.Exists(Path.Combine(executableDirectory, ryuName)))
+ if (ryuName.Length == 0 || !Path.Exists(Path.Combine(executableDirectory, ryuName)))
{
ryuName = OperatingSystem.IsWindows() ? "Ryujinx.exe" : "Ryujinx";
}
@@ -759,6 +766,43 @@ namespace Ryujinx.Modules
{
File.Delete(file);
}
+
+ // Migration: Delete old Ryujinx binary.
+ // TODO: Remove this in a future update.
+ if (!OperatingSystem.IsMacOS())
+ {
+ string[] oldRyuFiles = Directory.GetFiles(_homeDir, "Ryujinx.Ava*", SearchOption.TopDirectoryOnly);
+ // Assume we are running the new one if the process path is not available.
+ // This helps to prevent an infinite loop of restarts.
+ string currentRyuName = Path.GetFileName(Environment.ProcessPath) ?? (OperatingSystem.IsWindows() ? "Ryujinx.exe" : "Ryujinx");
+
+ string newRyuName = Path.Combine(_homeDir, currentRyuName.Replace(".Ava", ""));
+ if (!currentRyuName.Contains("Ryujinx.Ava"))
+ {
+ foreach (string oldRyuFile in oldRyuFiles)
+ {
+ File.Delete(oldRyuFile);
+ }
+ }
+ // Should we be running the old binary, start the new one if possible.
+ else if (File.Exists(newRyuName))
+ {
+ ProcessStartInfo processStart = new(newRyuName)
+ {
+ UseShellExecute = true,
+ WorkingDirectory = _homeDir,
+ };
+
+ foreach (string argument in CommandLineState.Arguments)
+ {
+ processStart.ArgumentList.Add(argument);
+ }
+
+ Process.Start(processStart);
+
+ Environment.Exit(0);
+ }
+ }
}
}
}
diff --git a/src/Ryujinx/Ryujinx.csproj b/src/Ryujinx/Ryujinx.csproj
index d79153cdb2..5ba8be1ebd 100644
--- a/src/Ryujinx/Ryujinx.csproj
+++ b/src/Ryujinx/Ryujinx.csproj
@@ -56,9 +56,6 @@
-
-
-
diff --git a/src/Ryujinx/UI/Helpers/Win32NativeInterop.cs b/src/Ryujinx/UI/Helpers/Win32NativeInterop.cs
index 4834df8029..fce2d518ac 100644
--- a/src/Ryujinx/UI/Helpers/Win32NativeInterop.cs
+++ b/src/Ryujinx/UI/Helpers/Win32NativeInterop.cs
@@ -8,6 +8,8 @@ namespace Ryujinx.Ava.UI.Helpers
[SupportedOSPlatform("windows")]
internal partial class Win32NativeInterop
{
+ internal const int GWLP_WNDPROC = -4;
+
[Flags]
public enum ClassStyles : uint
{
@@ -29,22 +31,7 @@ namespace Ryujinx.Ava.UI.Helpers
[SuppressMessage("Design", "CA1069: Enums values should not be duplicated")]
public enum WindowsMessages : uint
{
- Mousemove = 0x0200,
- Lbuttondown = 0x0201,
- Lbuttonup = 0x0202,
- Lbuttondblclk = 0x0203,
- Rbuttondown = 0x0204,
- Rbuttonup = 0x0205,
- Rbuttondblclk = 0x0206,
- Mbuttondown = 0x0207,
- Mbuttonup = 0x0208,
- Mbuttondblclk = 0x0209,
- Mousewheel = 0x020A,
- Xbuttondown = 0x020B,
- Xbuttonup = 0x020C,
- Xbuttondblclk = 0x020D,
- Mousehwheel = 0x020E,
- Mouselast = 0x020E,
+ NcHitTest = 0x0084,
}
[UnmanagedFunctionPointer(CallingConvention.Winapi)]
@@ -121,5 +108,11 @@ namespace Ryujinx.Ava.UI.Helpers
IntPtr hMenu,
IntPtr hInstance,
IntPtr lpParam);
+
+ [LibraryImport("user32.dll", SetLastError = true)]
+ public static partial IntPtr SetWindowLongPtrW(IntPtr hWnd, int nIndex, IntPtr value);
+
+ [LibraryImport("user32.dll", SetLastError = true)]
+ public static partial IntPtr SetWindowLongW(IntPtr hWnd, int nIndex, int value);
}
}
diff --git a/src/Ryujinx/UI/Renderer/EmbeddedWindow.cs b/src/Ryujinx/UI/Renderer/EmbeddedWindow.cs
index 3bf19b43e3..8c5e31fff4 100644
--- a/src/Ryujinx/UI/Renderer/EmbeddedWindow.cs
+++ b/src/Ryujinx/UI/Renderer/EmbeddedWindow.cs
@@ -141,80 +141,10 @@ namespace Ryujinx.Ava.UI.Renderer
_wndProcDelegate = delegate (IntPtr hWnd, WindowsMessages msg, IntPtr wParam, IntPtr lParam)
{
- if (VisualRoot != null)
+ switch (msg)
{
- if (msg == WindowsMessages.Lbuttondown ||
- msg == WindowsMessages.Rbuttondown ||
- msg == WindowsMessages.Lbuttonup ||
- msg == WindowsMessages.Rbuttonup ||
- msg == WindowsMessages.Mousemove)
- {
- Point rootVisualPosition = this.TranslatePoint(new Point((long)lParam & 0xFFFF, (long)lParam >> 16 & 0xFFFF), this).Value;
- Pointer pointer = new(0, PointerType.Mouse, true);
-
-#pragma warning disable CS0618 // Type or member is obsolete (As of Avalonia 11, the constructors for PointerPressedEventArgs & PointerEventArgs are marked as obsolete)
- switch (msg)
- {
- case WindowsMessages.Lbuttondown:
- case WindowsMessages.Rbuttondown:
- {
- bool isLeft = msg == WindowsMessages.Lbuttondown;
- RawInputModifiers pointerPointModifier = isLeft ? RawInputModifiers.LeftMouseButton : RawInputModifiers.RightMouseButton;
- PointerPointProperties properties = new(pointerPointModifier, isLeft ? PointerUpdateKind.LeftButtonPressed : PointerUpdateKind.RightButtonPressed);
-
- var evnt = new PointerPressedEventArgs(
- this,
- pointer,
- this,
- rootVisualPosition,
- (ulong)Environment.TickCount64,
- properties,
- KeyModifiers.None);
-
- RaiseEvent(evnt);
-
- break;
- }
- case WindowsMessages.Lbuttonup:
- case WindowsMessages.Rbuttonup:
- {
- bool isLeft = msg == WindowsMessages.Lbuttonup;
- RawInputModifiers pointerPointModifier = isLeft ? RawInputModifiers.LeftMouseButton : RawInputModifiers.RightMouseButton;
- PointerPointProperties properties = new(pointerPointModifier, isLeft ? PointerUpdateKind.LeftButtonReleased : PointerUpdateKind.RightButtonReleased);
-
- var evnt = new PointerReleasedEventArgs(
- this,
- pointer,
- this,
- rootVisualPosition,
- (ulong)Environment.TickCount64,
- properties,
- KeyModifiers.None,
- isLeft ? MouseButton.Left : MouseButton.Right);
-
- RaiseEvent(evnt);
-
- break;
- }
- case WindowsMessages.Mousemove:
- {
- var evnt = new PointerEventArgs(
- PointerMovedEvent,
- this,
- pointer,
- this,
- rootVisualPosition,
- (ulong)Environment.TickCount64,
- new PointerPointProperties(RawInputModifiers.None, PointerUpdateKind.Other),
- KeyModifiers.None);
-
- RaiseEvent(evnt);
-
- break;
- }
- }
-#pragma warning restore CS0618
- }
+ case WindowsMessages.NcHitTest:
+ return -1;
}
return DefWindowProc(hWnd, msg, wParam, lParam);
@@ -234,6 +164,8 @@ namespace Ryujinx.Ava.UI.Renderer
WindowHandle = CreateWindowEx(0, _className, "NativeWindow", WindowStyles.WsChild, 0, 0, 640, 480, control.Handle, IntPtr.Zero, IntPtr.Zero, IntPtr.Zero);
+ SetWindowLongPtrW(control.Handle, GWLP_WNDPROC, wndClassEx.lpfnWndProc);
+
Marshal.FreeHGlobal(wndClassEx.lpszClassName);
return new PlatformHandle(WindowHandle, "HWND");
diff --git a/src/Ryujinx/UI/Windows/ContentDialogOverlayWindow.axaml.cs b/src/Ryujinx/UI/Windows/ContentDialogOverlayWindow.axaml.cs
index 2b12d72f1d..0e24102478 100644
--- a/src/Ryujinx/UI/Windows/ContentDialogOverlayWindow.axaml.cs
+++ b/src/Ryujinx/UI/Windows/ContentDialogOverlayWindow.axaml.cs
@@ -9,7 +9,6 @@ namespace Ryujinx.Ava.UI.Windows
{
InitializeComponent();
- ExtendClientAreaToDecorationsHint = true;
TransparencyLevelHint = new[] { WindowTransparencyLevel.Transparent };
WindowStartupLocation = WindowStartupLocation.Manual;
SystemDecorations = SystemDecorations.None;
diff --git a/src/Ryujinx/UI/Windows/DownloadableContentManagerWindow.axaml.cs b/src/Ryujinx/UI/Windows/DownloadableContentManagerWindow.axaml.cs
index 0c02fa0f53..b9e2c7be34 100644
--- a/src/Ryujinx/UI/Windows/DownloadableContentManagerWindow.axaml.cs
+++ b/src/Ryujinx/UI/Windows/DownloadableContentManagerWindow.axaml.cs
@@ -47,7 +47,7 @@ namespace Ryujinx.Ava.UI.Windows
contentDialog.Styles.Add(bottomBorder);
- await ContentDialogHelper.ShowAsync(contentDialog);
+ await contentDialog.ShowAsync();
}
private void SaveAndClose(object sender, RoutedEventArgs routedEventArgs)
diff --git a/src/Ryujinx/UI/Windows/IconColorPicker.cs b/src/Ryujinx/UI/Windows/IconColorPicker.cs
index 4c75a5ff9e..72660351a8 100644
--- a/src/Ryujinx/UI/Windows/IconColorPicker.cs
+++ b/src/Ryujinx/UI/Windows/IconColorPicker.cs
@@ -127,7 +127,7 @@ namespace Ryujinx.Ava.UI.Windows
public static Bgra32[] GetBuffer(Image image)
{
- return image.TryGetSinglePixelSpan(out var data) ? data.ToArray() : Array.Empty();
+ return image.DangerousTryGetSinglePixelMemory(out var data) ? data.ToArray() : Array.Empty();
}
private static int GetColorScore(Dictionary dominantColorBin, int maxHitCount, PaletteColor color)
diff --git a/src/Ryujinx/UI/Windows/MainWindow.axaml b/src/Ryujinx/UI/Windows/MainWindow.axaml
index 4def7c281b..6c2042f93c 100644
--- a/src/Ryujinx/UI/Windows/MainWindow.axaml
+++ b/src/Ryujinx/UI/Windows/MainWindow.axaml
@@ -14,8 +14,8 @@
WindowState="{Binding WindowState}"
Width="{Binding WindowWidth}"
Height="{Binding WindowHeight}"
- MinWidth="1092"
- MinHeight="672"
+ MinWidth="800"
+ MinHeight="500"
d:DesignHeight="720"
d:DesignWidth="1280"
x:DataType="viewModels:MainWindowViewModel"
diff --git a/src/Ryujinx/UI/Windows/TitleUpdateWindow.axaml.cs b/src/Ryujinx/UI/Windows/TitleUpdateWindow.axaml.cs
index f3ac69600e..f5e2503231 100644
--- a/src/Ryujinx/UI/Windows/TitleUpdateWindow.axaml.cs
+++ b/src/Ryujinx/UI/Windows/TitleUpdateWindow.axaml.cs
@@ -1,4 +1,6 @@
+using Avalonia;
using Avalonia.Controls;
+using Avalonia.Controls.ApplicationLifetimes;
using Avalonia.Interactivity;
using Avalonia.Styling;
using FluentAvalonia.UI.Controls;
@@ -47,7 +49,7 @@ namespace Ryujinx.Ava.UI.Windows
contentDialog.Styles.Add(bottomBorder);
- await ContentDialogHelper.ShowAsync(contentDialog);
+ await contentDialog.ShowAsync();
}
private void Close(object sender, RoutedEventArgs e)
@@ -59,9 +61,15 @@ namespace Ryujinx.Ava.UI.Windows
{
ViewModel.Save();
- if (VisualRoot is MainWindow window)
+ if (Application.Current?.ApplicationLifetime is IClassicDesktopStyleApplicationLifetime al)
{
- window.LoadApplications();
+ foreach (Window window in al.Windows)
+ {
+ if (window is MainWindow mainWindow)
+ {
+ mainWindow.LoadApplications();
+ }
+ }
}
((ContentDialog)Parent).Hide();