2018-02-08 21:26:20 -06:00
|
|
|
using ChocolArm64.Exceptions;
|
2018-02-04 17:08:20 -06:00
|
|
|
using ChocolArm64.State;
|
|
|
|
using System;
|
|
|
|
using System.Collections.Generic;
|
2018-03-09 20:12:57 -06:00
|
|
|
using System.Runtime.InteropServices;
|
2018-02-04 17:08:20 -06:00
|
|
|
|
|
|
|
namespace ChocolArm64.Memory
|
|
|
|
{
|
NvServices refactoring (#120)
* Initial implementation of NvMap/NvHostCtrl
* More work on NvHostCtrl
* Refactoring of nvservices, move GPU Vmm, make Vmm per-process, refactor most gpu devices, move Gpu to Core, fix CbBind
* Implement GetGpuTime, support CancelSynchronization, fix issue on InsertWaitingMutex, proper double buffering support (again, not working properly for commercial games, only hb)
* Try to fix perf regression reading/writing textures, moved syncpts and events to a UserCtx class, delete global state when the process exits, other minor tweaks
* Remove now unused code, add comment about probably wrong result codes
2018-05-07 13:53:23 -05:00
|
|
|
public unsafe class AMemory : IAMemory, IDisposable
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-18 13:28:07 -06:00
|
|
|
private const long ErgMask = (4 << AThreadState.ErgSizeLog2) - 1;
|
2018-02-06 09:15:08 -06:00
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
public AMemoryMgr Manager { get; private set; }
|
|
|
|
|
|
|
|
private struct ExMonitor
|
|
|
|
{
|
|
|
|
public long Position { get; private set; }
|
|
|
|
|
|
|
|
private bool ExState;
|
|
|
|
|
|
|
|
public ExMonitor(long Position, bool ExState)
|
|
|
|
{
|
|
|
|
this.Position = Position;
|
|
|
|
this.ExState = ExState;
|
|
|
|
}
|
|
|
|
|
|
|
|
public bool HasExclusiveAccess(long Position)
|
|
|
|
{
|
|
|
|
return this.Position == Position && ExState;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void Reset()
|
|
|
|
{
|
|
|
|
ExState = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private Dictionary<int, ExMonitor> Monitors;
|
|
|
|
|
|
|
|
private HashSet<long> ExAddrs;
|
|
|
|
|
2018-03-09 20:12:57 -06:00
|
|
|
public IntPtr Ram { get; private set; }
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
private byte* RamPtr;
|
|
|
|
|
2018-03-09 20:12:57 -06:00
|
|
|
public AMemory()
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-27 17:45:07 -06:00
|
|
|
Manager = new AMemoryMgr();
|
2018-02-04 17:08:20 -06:00
|
|
|
|
|
|
|
Monitors = new Dictionary<int, ExMonitor>();
|
|
|
|
|
|
|
|
ExAddrs = new HashSet<long>();
|
|
|
|
|
2018-03-09 20:12:57 -06:00
|
|
|
Ram = Marshal.AllocHGlobal((IntPtr)AMemoryMgr.RamSize + AMemoryMgr.PageSize);
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
RamPtr = (byte*)Ram;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void RemoveMonitor(int ThreadId)
|
|
|
|
{
|
|
|
|
lock (Monitors)
|
|
|
|
{
|
2018-02-06 09:15:08 -06:00
|
|
|
if (Monitors.TryGetValue(ThreadId, out ExMonitor Monitor))
|
|
|
|
{
|
|
|
|
ExAddrs.Remove(Monitor.Position);
|
|
|
|
}
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
Monitors.Remove(ThreadId);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 13:28:07 -06:00
|
|
|
public void SetExclusive(AThreadState ThreadState, long Position)
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-13 20:43:08 -06:00
|
|
|
Position &= ~ErgMask;
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
lock (Monitors)
|
|
|
|
{
|
2018-02-18 13:28:07 -06:00
|
|
|
if (Monitors.TryGetValue(ThreadState.ThreadId, out ExMonitor Monitor))
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-06 09:15:08 -06:00
|
|
|
ExAddrs.Remove(Monitor.Position);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
2018-02-06 09:15:08 -06:00
|
|
|
bool ExState = ExAddrs.Add(Position);
|
|
|
|
|
|
|
|
Monitor = new ExMonitor(Position, ExState);
|
2018-02-04 17:08:20 -06:00
|
|
|
|
2018-02-18 13:28:07 -06:00
|
|
|
if (!Monitors.TryAdd(ThreadState.ThreadId, Monitor))
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-18 13:28:07 -06:00
|
|
|
Monitors[ThreadState.ThreadId] = Monitor;
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 13:28:07 -06:00
|
|
|
public bool TestExclusive(AThreadState ThreadState, long Position)
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-02-13 20:43:08 -06:00
|
|
|
Position &= ~ErgMask;
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
lock (Monitors)
|
|
|
|
{
|
2018-02-18 13:28:07 -06:00
|
|
|
if (!Monitors.TryGetValue(ThreadState.ThreadId, out ExMonitor Monitor))
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return Monitor.HasExclusiveAccess(Position);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 13:28:07 -06:00
|
|
|
public void ClearExclusive(AThreadState ThreadState)
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
|
|
|
lock (Monitors)
|
|
|
|
{
|
2018-02-18 13:28:07 -06:00
|
|
|
if (Monitors.TryGetValue(ThreadState.ThreadId, out ExMonitor Monitor))
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
|
|
|
Monitor.Reset();
|
|
|
|
ExAddrs.Remove(Monitor.Position);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-13 20:43:08 -06:00
|
|
|
public bool AcquireAddress(long Position)
|
|
|
|
{
|
|
|
|
Position &= ~ErgMask;
|
|
|
|
|
|
|
|
lock (Monitors)
|
|
|
|
{
|
|
|
|
return ExAddrs.Add(Position);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
public void ReleaseAddress(long Position)
|
|
|
|
{
|
|
|
|
Position &= ~ErgMask;
|
|
|
|
|
|
|
|
lock (Monitors)
|
|
|
|
{
|
|
|
|
ExAddrs.Remove(Position);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
public sbyte ReadSByte(long Position)
|
|
|
|
{
|
|
|
|
return (sbyte)ReadByte(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public short ReadInt16(long Position)
|
|
|
|
{
|
|
|
|
return (short)ReadUInt16(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int ReadInt32(long Position)
|
|
|
|
{
|
|
|
|
return (int)ReadUInt32(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public long ReadInt64(long Position)
|
|
|
|
{
|
|
|
|
return (long)ReadUInt64(Position);
|
|
|
|
}
|
2018-02-04 17:08:20 -06:00
|
|
|
|
|
|
|
public byte ReadByte(long Position)
|
|
|
|
{
|
2018-02-08 21:26:20 -06:00
|
|
|
EnsureAccessIsValid(Position, AMemoryPerm.Read);
|
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
return ReadByteUnchecked(Position);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public ushort ReadUInt16(long Position)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Read);
|
|
|
|
EnsureAccessIsValid(Position + 1, AMemoryPerm.Read);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
return ReadUInt16Unchecked(Position);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public uint ReadUInt32(long Position)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Read);
|
|
|
|
EnsureAccessIsValid(Position + 3, AMemoryPerm.Read);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
return ReadUInt32Unchecked(Position);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public ulong ReadUInt64(long Position)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Read);
|
|
|
|
EnsureAccessIsValid(Position + 7, AMemoryPerm.Read);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
return ReadUInt64Unchecked(Position);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
2018-02-17 15:06:11 -06:00
|
|
|
public AVec ReadVector8(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { B0 = ReadByte(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector16(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { H0 = ReadUInt16(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector32(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { W0 = ReadUInt32(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector64(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { X0 = ReadUInt64(Position) };
|
|
|
|
}
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
public AVec ReadVector128(long Position)
|
|
|
|
{
|
|
|
|
return new AVec()
|
|
|
|
{
|
|
|
|
X0 = ReadUInt64(Position + 0),
|
|
|
|
X1 = ReadUInt64(Position + 8)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
public sbyte ReadSByteUnchecked(long Position)
|
|
|
|
{
|
|
|
|
return (sbyte)ReadByteUnchecked(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public short ReadInt16Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return (short)ReadUInt16Unchecked(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public int ReadInt32Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return (int)ReadUInt32Unchecked(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public long ReadInt64Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return (long)ReadUInt64Unchecked(Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
public byte ReadByteUnchecked(long Position)
|
|
|
|
{
|
|
|
|
return *((byte*)(RamPtr + (uint)Position));
|
|
|
|
}
|
|
|
|
|
|
|
|
public ushort ReadUInt16Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return *((ushort*)(RamPtr + (uint)Position));
|
|
|
|
}
|
|
|
|
|
|
|
|
public uint ReadUInt32Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return *((uint*)(RamPtr + (uint)Position));
|
|
|
|
}
|
|
|
|
|
|
|
|
public ulong ReadUInt64Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return *((ulong*)(RamPtr + (uint)Position));
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector8Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { B0 = ReadByteUnchecked(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector16Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { H0 = ReadUInt16Unchecked(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector32Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { W0 = ReadUInt32Unchecked(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector64Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return new AVec() { X0 = ReadUInt64Unchecked(Position) };
|
|
|
|
}
|
|
|
|
|
|
|
|
public AVec ReadVector128Unchecked(long Position)
|
|
|
|
{
|
|
|
|
return new AVec()
|
|
|
|
{
|
|
|
|
X0 = ReadUInt64Unchecked(Position + 0),
|
|
|
|
X1 = ReadUInt64Unchecked(Position + 8)
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteSByte(long Position, sbyte Value)
|
|
|
|
{
|
|
|
|
WriteByte(Position, (byte)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt16(long Position, short Value)
|
|
|
|
{
|
|
|
|
WriteUInt16(Position, (ushort)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt32(long Position, int Value)
|
|
|
|
{
|
|
|
|
WriteUInt32(Position, (uint)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt64(long Position, long Value)
|
|
|
|
{
|
|
|
|
WriteUInt64(Position, (ulong)Value);
|
|
|
|
}
|
2018-02-04 17:08:20 -06:00
|
|
|
|
|
|
|
public void WriteByte(long Position, byte Value)
|
|
|
|
{
|
2018-02-08 21:26:20 -06:00
|
|
|
EnsureAccessIsValid(Position, AMemoryPerm.Write);
|
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
WriteByteUnchecked(Position, Value);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt16(long Position, ushort Value)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Write);
|
|
|
|
EnsureAccessIsValid(Position + 1, AMemoryPerm.Write);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
WriteUInt16Unchecked(Position, Value);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt32(long Position, uint Value)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Write);
|
|
|
|
EnsureAccessIsValid(Position + 3, AMemoryPerm.Write);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
WriteUInt32Unchecked(Position, Value);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt64(long Position, ulong Value)
|
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
EnsureAccessIsValid(Position + 0, AMemoryPerm.Write);
|
|
|
|
EnsureAccessIsValid(Position + 7, AMemoryPerm.Write);
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
WriteUInt64Unchecked(Position, Value);
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
|
2018-02-17 15:06:11 -06:00
|
|
|
public void WriteVector8(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteByte(Position, Value.B0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector16(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt16(Position, Value.H0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector32(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt32(Position, Value.W0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector64(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt64(Position, Value.X0);
|
|
|
|
}
|
|
|
|
|
2018-02-04 17:08:20 -06:00
|
|
|
public void WriteVector128(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt64(Position + 0, Value.X0);
|
|
|
|
WriteUInt64(Position + 8, Value.X1);
|
|
|
|
}
|
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
public void WriteSByteUnchecked(long Position, sbyte Value)
|
|
|
|
{
|
|
|
|
WriteByteUnchecked(Position, (byte)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt16Unchecked(long Position, short Value)
|
|
|
|
{
|
|
|
|
WriteUInt16Unchecked(Position, (ushort)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt32Unchecked(long Position, int Value)
|
|
|
|
{
|
|
|
|
WriteUInt32Unchecked(Position, (uint)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteInt64Unchecked(long Position, long Value)
|
|
|
|
{
|
|
|
|
WriteUInt64Unchecked(Position, (ulong)Value);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteByteUnchecked(long Position, byte Value)
|
|
|
|
{
|
|
|
|
*((byte*)(RamPtr + (uint)Position)) = Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt16Unchecked(long Position, ushort Value)
|
|
|
|
{
|
|
|
|
*((ushort*)(RamPtr + (uint)Position)) = Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt32Unchecked(long Position, uint Value)
|
|
|
|
{
|
|
|
|
*((uint*)(RamPtr + (uint)Position)) = Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteUInt64Unchecked(long Position, ulong Value)
|
|
|
|
{
|
|
|
|
*((ulong*)(RamPtr + (uint)Position)) = Value;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector8Unchecked(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteByteUnchecked(Position, Value.B0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector16Unchecked(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt16Unchecked(Position, Value.H0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector32Unchecked(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt32Unchecked(Position, Value.W0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector64Unchecked(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt64Unchecked(Position, Value.X0);
|
|
|
|
}
|
|
|
|
|
|
|
|
public void WriteVector128Unchecked(long Position, AVec Value)
|
|
|
|
{
|
|
|
|
WriteUInt64Unchecked(Position + 0, Value.X0);
|
|
|
|
WriteUInt64Unchecked(Position + 8, Value.X1);
|
|
|
|
}
|
|
|
|
|
2018-02-08 21:26:20 -06:00
|
|
|
private void EnsureAccessIsValid(long Position, AMemoryPerm Perm)
|
2018-02-04 17:08:20 -06:00
|
|
|
{
|
2018-03-10 17:39:16 -06:00
|
|
|
if (!Manager.IsMapped(Position))
|
2018-02-08 21:26:20 -06:00
|
|
|
{
|
2018-03-10 17:39:16 -06:00
|
|
|
throw new VmmPageFaultException(Position);
|
|
|
|
}
|
2018-03-09 20:12:57 -06:00
|
|
|
|
2018-03-10 17:39:16 -06:00
|
|
|
if (!Manager.HasPermission(Position, Perm))
|
|
|
|
{
|
|
|
|
throw new VmmAccessViolationException(Position, Perm);
|
2018-02-08 21:26:20 -06:00
|
|
|
}
|
2018-03-09 20:12:57 -06:00
|
|
|
}
|
2018-02-08 21:26:20 -06:00
|
|
|
|
2018-03-09 20:12:57 -06:00
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual void Dispose(bool disposing)
|
|
|
|
{
|
|
|
|
if (Ram != IntPtr.Zero)
|
2018-02-08 21:26:20 -06:00
|
|
|
{
|
2018-03-09 20:12:57 -06:00
|
|
|
Marshal.FreeHGlobal(Ram);
|
|
|
|
|
|
|
|
Ram = IntPtr.Zero;
|
2018-02-08 21:26:20 -06:00
|
|
|
}
|
2018-02-04 17:08:20 -06:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|