2018-02-05 00:08:20 +01:00
|
|
|
using ChocolArm64.State;
|
2018-04-24 20:57:39 +02:00
|
|
|
using Ryujinx.Core.Logging;
|
2018-02-20 21:09:23 +01:00
|
|
|
using Ryujinx.Core.OsHle.Handles;
|
2018-04-19 04:52:23 +02:00
|
|
|
using System.Threading;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-04 21:07:44 +02:00
|
|
|
using static Ryujinx.Core.OsHle.ErrorCode;
|
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
namespace Ryujinx.Core.OsHle.Kernel
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
|
|
|
partial class SvcHandler
|
|
|
|
{
|
2018-02-18 20:28:07 +01:00
|
|
|
private void SvcCreateThread(AThreadState ThreadState)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-18 20:28:07 +01:00
|
|
|
long EntryPoint = (long)ThreadState.X1;
|
|
|
|
long ArgsPtr = (long)ThreadState.X2;
|
|
|
|
long StackTop = (long)ThreadState.X3;
|
|
|
|
int Priority = (int)ThreadState.X4;
|
|
|
|
int ProcessorId = (int)ThreadState.X5;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
if (ProcessorId == -2)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-04-19 04:52:23 +02:00
|
|
|
//TODO: Get this value from the NPDM file.
|
|
|
|
ProcessorId = 0;
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
int Handle = Process.MakeThread(
|
|
|
|
EntryPoint,
|
|
|
|
StackTop,
|
|
|
|
ArgsPtr,
|
|
|
|
Priority,
|
|
|
|
ProcessorId);
|
|
|
|
|
|
|
|
ThreadState.X0 = 0;
|
|
|
|
ThreadState.X1 = (ulong)Handle;
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-02-18 20:28:07 +01:00
|
|
|
private void SvcStartThread(AThreadState ThreadState)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-18 20:28:07 +01:00
|
|
|
int Handle = (int)ThreadState.X0;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
KThread CurrThread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
if (CurrThread != null)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-04-19 04:52:23 +02:00
|
|
|
Process.Scheduler.StartThread(CurrThread);
|
|
|
|
|
|
|
|
Process.Scheduler.Yield(Process.GetThread(ThreadState.Tpidr));
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-03-05 20:18:37 +01:00
|
|
|
ThreadState.X0 = 0;
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
2018-04-19 04:52:23 +02:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 20:57:39 +02:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
private void SvcExitThread(AThreadState ThreadState)
|
|
|
|
{
|
2018-03-19 19:58:46 +01:00
|
|
|
KThread CurrThread = Process.GetThread(ThreadState.Tpidr);
|
|
|
|
|
2018-03-12 05:04:52 +01:00
|
|
|
CurrThread.Thread.StopExecution();
|
|
|
|
}
|
|
|
|
|
2018-02-18 20:28:07 +01:00
|
|
|
private void SvcSleepThread(AThreadState ThreadState)
|
2018-04-04 21:07:44 +02:00
|
|
|
{
|
2018-04-19 09:06:23 +02:00
|
|
|
ulong Ns = ThreadState.X0;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-03-19 19:58:46 +01:00
|
|
|
KThread CurrThread = Process.GetThread(ThreadState.Tpidr);
|
2018-04-04 21:07:44 +02:00
|
|
|
|
2018-04-19 09:06:23 +02:00
|
|
|
if (Ns == 0)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-14 03:43:08 +01:00
|
|
|
Process.Scheduler.Yield(CurrThread);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-04-19 04:52:23 +02:00
|
|
|
Process.Scheduler.Suspend(CurrThread.ProcessorId);
|
|
|
|
|
2018-04-19 09:06:23 +02:00
|
|
|
Thread.Sleep(NsTimeConverter.GetTimeMs(Ns));
|
2018-04-19 04:52:23 +02:00
|
|
|
|
|
|
|
Process.Scheduler.Resume(CurrThread);
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 20:28:07 +01:00
|
|
|
private void SvcGetThreadPriority(AThreadState ThreadState)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-18 20:28:07 +01:00
|
|
|
int Handle = (int)ThreadState.X1;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
KThread CurrThread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
if (CurrThread != null)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-05 20:18:37 +01:00
|
|
|
ThreadState.X0 = 0;
|
2018-04-21 21:07:16 +02:00
|
|
|
ThreadState.X1 = (ulong)CurrThread.ActualPriority;
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
2018-04-19 04:52:23 +02:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 20:57:39 +02:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
2018-02-25 19:58:16 +01:00
|
|
|
|
|
|
|
private void SvcSetThreadPriority(AThreadState ThreadState)
|
|
|
|
{
|
2018-04-19 09:06:23 +02:00
|
|
|
int Handle = (int)ThreadState.X0;
|
|
|
|
int Priority = (int)ThreadState.X1;
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
KThread CurrThread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
if (CurrThread != null)
|
2018-02-25 19:58:16 +01:00
|
|
|
{
|
2018-04-21 21:07:16 +02:00
|
|
|
CurrThread.SetPriority(Priority);
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-03-05 20:18:37 +01:00
|
|
|
ThreadState.X0 = 0;
|
2018-02-25 19:58:16 +01:00
|
|
|
}
|
2018-04-19 04:52:23 +02:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 20:57:39 +02:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-25 19:58:16 +01:00
|
|
|
}
|
|
|
|
|
|
|
|
private void SvcSetThreadCoreMask(AThreadState ThreadState)
|
|
|
|
{
|
2018-03-05 20:18:37 +01:00
|
|
|
ThreadState.X0 = 0;
|
2018-02-25 19:58:16 +01:00
|
|
|
|
|
|
|
//TODO: Error codes.
|
|
|
|
}
|
|
|
|
|
2018-04-05 00:16:59 +02:00
|
|
|
private void SvcGetCurrentProcessorNumber(AThreadState ThreadState)
|
|
|
|
{
|
2018-04-19 04:52:23 +02:00
|
|
|
ThreadState.X0 = (ulong)Process.GetThread(ThreadState.Tpidr).ProcessorId;
|
2018-04-05 00:16:59 +02:00
|
|
|
}
|
|
|
|
|
2018-02-25 19:58:16 +01:00
|
|
|
private void SvcGetThreadId(AThreadState ThreadState)
|
|
|
|
{
|
2018-04-04 21:07:44 +02:00
|
|
|
int Handle = (int)ThreadState.X1;
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
KThread CurrThread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
if (CurrThread != null)
|
2018-02-25 19:58:16 +01:00
|
|
|
{
|
2018-03-05 20:18:37 +01:00
|
|
|
ThreadState.X0 = 0;
|
2018-04-19 04:52:23 +02:00
|
|
|
ThreadState.X1 = (ulong)CurrThread.ThreadId;
|
2018-02-25 19:58:16 +01:00
|
|
|
}
|
2018-04-04 21:07:44 +02:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 20:57:39 +02:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-04-19 21:18:30 +02:00
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void SvcSetThreadActivity(AThreadState ThreadState)
|
|
|
|
{
|
|
|
|
int Handle = (int)ThreadState.X0;
|
2018-04-22 06:21:49 +02:00
|
|
|
bool Active = (int)ThreadState.X1 == 0;
|
2018-04-19 21:18:30 +02:00
|
|
|
|
2018-04-22 06:21:49 +02:00
|
|
|
KThread Thread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-04-19 21:18:30 +02:00
|
|
|
|
2018-04-22 06:21:49 +02:00
|
|
|
if (Thread != null)
|
2018-04-19 21:18:30 +02:00
|
|
|
{
|
2018-04-22 06:21:49 +02:00
|
|
|
Process.Scheduler.SetThreadActivity(Thread, Active);
|
2018-04-19 21:18:30 +02:00
|
|
|
|
|
|
|
ThreadState.X0 = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-04-24 20:57:39 +02:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-25 19:58:16 +01:00
|
|
|
|
2018-04-04 21:07:44 +02:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-25 19:58:16 +01:00
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
}
|