2018-02-26 01:14:58 +00:00
|
|
|
using ChocolArm64.Events;
|
2018-02-04 23:08:20 +00:00
|
|
|
using ChocolArm64.Memory;
|
|
|
|
using ChocolArm64.State;
|
2018-03-12 04:04:52 +00:00
|
|
|
using Ryujinx.Core.OsHle.Handles;
|
2018-02-04 23:08:20 +00:00
|
|
|
using System;
|
2018-04-19 02:52:23 +00:00
|
|
|
using System.Collections.Concurrent;
|
2018-02-04 23:08:20 +00:00
|
|
|
using System.Collections.Generic;
|
|
|
|
|
2018-04-19 02:52:23 +00:00
|
|
|
namespace Ryujinx.Core.OsHle.Kernel
|
2018-02-04 23:08:20 +00:00
|
|
|
{
|
2018-03-12 04:04:52 +00:00
|
|
|
partial class SvcHandler : IDisposable
|
2018-02-04 23:08:20 +00:00
|
|
|
{
|
2018-02-18 19:28:07 +00:00
|
|
|
private delegate void SvcFunc(AThreadState ThreadState);
|
2018-02-17 21:36:08 +00:00
|
|
|
|
2018-02-14 02:43:08 +00:00
|
|
|
private Dictionary<int, SvcFunc> SvcFuncs;
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
|
|
private Switch Ns;
|
2018-02-14 02:43:08 +00:00
|
|
|
private Process Process;
|
2018-02-04 23:08:20 +00:00
|
|
|
private AMemory Memory;
|
|
|
|
|
2018-04-19 03:00:29 +00:00
|
|
|
private ConcurrentDictionary<long, MutualExclusion> Mutexes;
|
2018-04-19 02:52:23 +00:00
|
|
|
private ConcurrentDictionary<long, ConditionVariable> CondVars;
|
|
|
|
|
2018-03-12 04:04:52 +00:00
|
|
|
private HashSet<(HSharedMem, long)> MappedSharedMems;
|
|
|
|
|
2018-02-27 23:45:07 +00:00
|
|
|
private ulong CurrentHeapSize;
|
2018-02-04 23:08:20 +00:00
|
|
|
|
2018-03-12 04:04:52 +00:00
|
|
|
private static Random Rng;
|
|
|
|
|
2018-02-14 02:43:08 +00:00
|
|
|
public SvcHandler(Switch Ns, Process Process)
|
2018-02-04 23:08:20 +00:00
|
|
|
{
|
2018-02-14 02:43:08 +00:00
|
|
|
SvcFuncs = new Dictionary<int, SvcFunc>()
|
|
|
|
{
|
|
|
|
{ 0x01, SvcSetHeapSize },
|
|
|
|
{ 0x03, SvcSetMemoryAttribute },
|
|
|
|
{ 0x04, SvcMapMemory },
|
2018-02-27 23:45:07 +00:00
|
|
|
{ 0x05, SvcUnmapMemory },
|
2018-02-14 02:43:08 +00:00
|
|
|
{ 0x06, SvcQueryMemory },
|
2018-02-15 12:16:16 +00:00
|
|
|
{ 0x07, SvcExitProcess },
|
2018-02-14 02:43:08 +00:00
|
|
|
{ 0x08, SvcCreateThread },
|
|
|
|
{ 0x09, SvcStartThread },
|
2018-03-12 04:04:52 +00:00
|
|
|
{ 0x0a, SvcExitThread },
|
2018-02-14 02:43:08 +00:00
|
|
|
{ 0x0b, SvcSleepThread },
|
|
|
|
{ 0x0c, SvcGetThreadPriority },
|
2018-02-25 18:58:16 +00:00
|
|
|
{ 0x0d, SvcSetThreadPriority },
|
|
|
|
{ 0x0f, SvcSetThreadCoreMask },
|
2018-04-04 22:16:59 +00:00
|
|
|
{ 0x10, SvcGetCurrentProcessorNumber },
|
2018-03-05 15:58:19 +00:00
|
|
|
{ 0x12, SvcClearEvent },
|
2018-02-14 02:43:08 +00:00
|
|
|
{ 0x13, SvcMapSharedMemory },
|
|
|
|
{ 0x14, SvcUnmapSharedMemory },
|
|
|
|
{ 0x15, SvcCreateTransferMemory },
|
|
|
|
{ 0x16, SvcCloseHandle },
|
|
|
|
{ 0x17, SvcResetSignal },
|
|
|
|
{ 0x18, SvcWaitSynchronization },
|
|
|
|
{ 0x1a, SvcArbitrateLock },
|
|
|
|
{ 0x1b, SvcArbitrateUnlock },
|
|
|
|
{ 0x1c, SvcWaitProcessWideKeyAtomic },
|
|
|
|
{ 0x1d, SvcSignalProcessWideKey },
|
|
|
|
{ 0x1e, SvcGetSystemTick },
|
|
|
|
{ 0x1f, SvcConnectToNamedPort },
|
|
|
|
{ 0x21, SvcSendSyncRequest },
|
|
|
|
{ 0x22, SvcSendSyncRequestWithUserBuffer },
|
2018-02-25 18:58:16 +00:00
|
|
|
{ 0x25, SvcGetThreadId },
|
2018-02-14 02:43:08 +00:00
|
|
|
{ 0x26, SvcBreak },
|
|
|
|
{ 0x27, SvcOutputDebugString },
|
|
|
|
{ 0x29, SvcGetInfo }
|
|
|
|
};
|
|
|
|
|
|
|
|
this.Ns = Ns;
|
|
|
|
this.Process = Process;
|
|
|
|
this.Memory = Process.Memory;
|
2018-03-12 04:04:52 +00:00
|
|
|
|
2018-04-19 02:52:23 +00:00
|
|
|
Mutexes = new ConcurrentDictionary<long, MutualExclusion>();
|
|
|
|
CondVars = new ConcurrentDictionary<long, ConditionVariable>();
|
|
|
|
|
2018-03-12 04:04:52 +00:00
|
|
|
MappedSharedMems = new HashSet<(HSharedMem, long)>();
|
2018-02-04 23:08:20 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static SvcHandler()
|
|
|
|
{
|
|
|
|
Rng = new Random();
|
|
|
|
}
|
|
|
|
|
2018-02-26 01:14:58 +00:00
|
|
|
public void SvcCall(object sender, AInstExceptionEventArgs e)
|
2018-02-04 23:08:20 +00:00
|
|
|
{
|
2018-02-18 19:28:07 +00:00
|
|
|
AThreadState ThreadState = (AThreadState)sender;
|
2018-02-04 23:08:20 +00:00
|
|
|
|
|
|
|
if (SvcFuncs.TryGetValue(e.Id, out SvcFunc Func))
|
|
|
|
{
|
2018-04-14 01:02:24 +00:00
|
|
|
Logging.Trace(LogClass.KernelSvc, $"(Thread {ThreadState.ThreadId}) {Func.Method.Name} called.");
|
2018-02-14 02:43:08 +00:00
|
|
|
|
2018-02-18 19:28:07 +00:00
|
|
|
Func(ThreadState);
|
2018-02-14 02:43:08 +00:00
|
|
|
|
2018-04-14 01:02:24 +00:00
|
|
|
Logging.Trace(LogClass.KernelSvc, $"(Thread {ThreadState.ThreadId}) {Func.Method.Name} ended.");
|
2018-02-04 23:08:20 +00:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-02-10 13:24:16 +00:00
|
|
|
throw new NotImplementedException(e.Id.ToString("x4"));
|
2018-02-04 23:08:20 +00:00
|
|
|
}
|
|
|
|
}
|
2018-03-12 04:04:52 +00:00
|
|
|
|
|
|
|
public void Dispose()
|
|
|
|
{
|
|
|
|
Dispose(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
protected virtual void Dispose(bool Disposing)
|
|
|
|
{
|
|
|
|
if (Disposing)
|
|
|
|
{
|
|
|
|
lock (MappedSharedMems)
|
|
|
|
{
|
|
|
|
foreach ((HSharedMem SharedMem, long Position) in MappedSharedMems)
|
|
|
|
{
|
|
|
|
SharedMem.RemoveVirtualPosition(Memory, Position);
|
|
|
|
}
|
|
|
|
|
|
|
|
MappedSharedMems.Clear();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2018-02-04 23:08:20 +00:00
|
|
|
}
|
|
|
|
}
|