2018-02-04 15:08:20 -08:00
|
|
|
using ChocolArm64.State;
|
2018-06-10 17:46:42 -07:00
|
|
|
using Ryujinx.HLE.Logging;
|
|
|
|
using Ryujinx.HLE.OsHle.Handles;
|
2018-04-18 19:52:23 -07:00
|
|
|
using System.Threading;
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-06-10 17:46:42 -07:00
|
|
|
using static Ryujinx.HLE.OsHle.ErrorCode;
|
2018-04-04 12:07:44 -07:00
|
|
|
|
2018-06-10 17:46:42 -07:00
|
|
|
namespace Ryujinx.HLE.OsHle.Kernel
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
|
|
|
partial class SvcHandler
|
|
|
|
{
|
2018-02-18 11:28:07 -08:00
|
|
|
private void SvcCreateThread(AThreadState ThreadState)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-02-18 11:28:07 -08: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-04 15:08:20 -08:00
|
|
|
|
2018-05-13 18:00:29 -07:00
|
|
|
if ((uint)Priority > 0x3f)
|
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid priority 0x{Priority:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidPriority);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-04-18 19:52:23 -07:00
|
|
|
if (ProcessorId == -2)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-04-18 19:52:23 -07:00
|
|
|
//TODO: Get this value from the NPDM file.
|
|
|
|
ProcessorId = 0;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
2018-05-13 18:00:29 -07:00
|
|
|
else if ((uint)ProcessorId > 3)
|
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid core id 0x{ProcessorId:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidCoreId);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-04-18 19:52:23 -07:00
|
|
|
int Handle = Process.MakeThread(
|
|
|
|
EntryPoint,
|
|
|
|
StackTop,
|
|
|
|
ArgsPtr,
|
|
|
|
Priority,
|
|
|
|
ProcessorId);
|
|
|
|
|
|
|
|
ThreadState.X0 = 0;
|
|
|
|
ThreadState.X1 = (ulong)Handle;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2018-02-18 11:28:07 -08:00
|
|
|
private void SvcStartThread(AThreadState ThreadState)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-02-18 11:28:07 -08:00
|
|
|
int Handle = (int)ThreadState.X0;
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-05-15 18:36:08 -07:00
|
|
|
KThread NewThread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-05-15 18:36:08 -07:00
|
|
|
if (NewThread != null)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-05-15 18:36:08 -07:00
|
|
|
Process.Scheduler.StartThread(NewThread);
|
|
|
|
Process.Scheduler.SetReschedule(NewThread.ProcessorId);
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-03-05 11:18:37 -08:00
|
|
|
ThreadState.X0 = 0;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
2018-04-18 19:52:23 -07:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 11:57:39 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-04-18 19:52:23 -07:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
|
2018-03-11 21:04:52 -07:00
|
|
|
private void SvcExitThread(AThreadState ThreadState)
|
|
|
|
{
|
2018-03-19 11:58:46 -07:00
|
|
|
KThread CurrThread = Process.GetThread(ThreadState.Tpidr);
|
|
|
|
|
2018-03-11 21:04:52 -07:00
|
|
|
CurrThread.Thread.StopExecution();
|
|
|
|
}
|
|
|
|
|
2018-02-18 11:28:07 -08:00
|
|
|
private void SvcSleepThread(AThreadState ThreadState)
|
2018-04-04 12:07:44 -07:00
|
|
|
{
|
2018-05-15 18:36:08 -07:00
|
|
|
ulong TimeoutNs = ThreadState.X0;
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-06-21 19:05:42 -07:00
|
|
|
Ns.Log.PrintDebug(LogClass.KernelSvc, "Timeout = " + TimeoutNs.ToString("x16"));
|
|
|
|
|
2018-03-19 11:58:46 -07:00
|
|
|
KThread CurrThread = Process.GetThread(ThreadState.Tpidr);
|
2018-04-04 12:07:44 -07:00
|
|
|
|
2018-05-15 18:36:08 -07:00
|
|
|
if (TimeoutNs == 0)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-06-09 14:19:14 -07:00
|
|
|
Process.Scheduler.Yield(CurrThread);
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-05-12 20:22:42 -07:00
|
|
|
Process.Scheduler.Suspend(CurrThread);
|
2018-04-18 19:52:23 -07:00
|
|
|
|
2018-05-15 18:36:08 -07:00
|
|
|
Thread.Sleep(NsTimeConverter.GetTimeMs(TimeoutNs));
|
2018-04-18 19:52:23 -07:00
|
|
|
|
|
|
|
Process.Scheduler.Resume(CurrThread);
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-02-18 11:28:07 -08:00
|
|
|
private void SvcGetThreadPriority(AThreadState ThreadState)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-02-18 11:28:07 -08:00
|
|
|
int Handle = (int)ThreadState.X1;
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
KThread Thread = GetThread(ThreadState.Tpidr, Handle);
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
if (Thread != null)
|
2018-02-04 15:08:20 -08:00
|
|
|
{
|
2018-03-05 11:18:37 -08:00
|
|
|
ThreadState.X0 = 0;
|
2018-05-11 19:05:06 -07:00
|
|
|
ThreadState.X1 = (ulong)Thread.ActualPriority;
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
2018-04-18 19:52:23 -07:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 11:57:39 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-04 15:08:20 -08:00
|
|
|
|
2018-04-18 19:52:23 -07:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
2018-02-25 10:58:16 -08:00
|
|
|
|
|
|
|
private void SvcSetThreadPriority(AThreadState ThreadState)
|
|
|
|
{
|
2018-04-19 00:06:23 -07:00
|
|
|
int Handle = (int)ThreadState.X0;
|
|
|
|
int Priority = (int)ThreadState.X1;
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-06-21 19:05:42 -07:00
|
|
|
Ns.Log.PrintDebug(LogClass.KernelSvc,
|
|
|
|
"Handle = " + Handle .ToString("x8") + ", " +
|
|
|
|
"Priority = " + Priority.ToString("x8"));
|
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
KThread Thread = GetThread(ThreadState.Tpidr, Handle);
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
if (Thread != null)
|
2018-02-25 10:58:16 -08:00
|
|
|
{
|
2018-05-11 19:05:06 -07:00
|
|
|
Thread.SetPriority(Priority);
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-03-05 11:18:37 -08:00
|
|
|
ThreadState.X0 = 0;
|
2018-02-25 10:58:16 -08:00
|
|
|
}
|
2018-04-18 19:52:23 -07:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 11:57:39 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-04-18 19:52:23 -07:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-25 10:58:16 -08:00
|
|
|
}
|
|
|
|
|
2018-06-09 21:36:07 -07:00
|
|
|
private void SvcGetThreadCoreMask(AThreadState ThreadState)
|
|
|
|
{
|
|
|
|
int Handle = (int)ThreadState.X2;
|
|
|
|
|
|
|
|
Ns.Log.PrintDebug(LogClass.KernelSvc, "Handle = " + Handle.ToString("x8"));
|
|
|
|
|
|
|
|
KThread Thread = GetThread(ThreadState.Tpidr, Handle);
|
|
|
|
|
|
|
|
if (Thread != null)
|
|
|
|
{
|
|
|
|
ThreadState.X0 = 0;
|
|
|
|
ThreadState.X1 = (ulong)Thread.IdealCore;
|
|
|
|
ThreadState.X2 = (ulong)Thread.CoreMask;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
|
|
|
}
|
2018-06-10 17:46:42 -07:00
|
|
|
|
2018-02-25 10:58:16 -08:00
|
|
|
private void SvcSetThreadCoreMask(AThreadState ThreadState)
|
|
|
|
{
|
2018-05-13 18:00:29 -07:00
|
|
|
int Handle = (int)ThreadState.X0;
|
|
|
|
int IdealCore = (int)ThreadState.X1;
|
|
|
|
long CoreMask = (long)ThreadState.X2;
|
|
|
|
|
2018-05-13 23:01:10 -07:00
|
|
|
Ns.Log.PrintDebug(LogClass.KernelSvc,
|
|
|
|
"Handle = " + Handle .ToString("x8") + ", " +
|
|
|
|
"IdealCore = " + IdealCore.ToString("x8") + ", " +
|
|
|
|
"CoreMask = " + CoreMask .ToString("x16"));
|
|
|
|
|
2018-05-13 18:00:29 -07:00
|
|
|
KThread Thread = GetThread(ThreadState.Tpidr, Handle);
|
|
|
|
|
|
|
|
if (IdealCore == -2)
|
|
|
|
{
|
2018-05-13 23:09:08 -07:00
|
|
|
//TODO: Get this value from the NPDM file.
|
2018-05-13 18:00:29 -07:00
|
|
|
IdealCore = 0;
|
|
|
|
|
|
|
|
CoreMask = 1 << IdealCore;
|
|
|
|
}
|
2018-06-21 19:05:42 -07:00
|
|
|
else
|
2018-05-13 18:00:29 -07:00
|
|
|
{
|
|
|
|
if ((uint)IdealCore > 3)
|
|
|
|
{
|
2018-05-13 23:01:10 -07:00
|
|
|
if ((IdealCore | 2) != -1)
|
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid core id 0x{IdealCore:x8}!");
|
2018-05-13 18:00:29 -07:00
|
|
|
|
2018-05-13 23:01:10 -07:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidCoreId);
|
2018-05-13 18:00:29 -07:00
|
|
|
|
2018-05-13 23:01:10 -07:00
|
|
|
return;
|
|
|
|
}
|
2018-05-13 18:00:29 -07:00
|
|
|
}
|
2018-05-13 23:01:10 -07:00
|
|
|
else if ((CoreMask & (1 << IdealCore)) == 0)
|
2018-05-13 18:00:29 -07:00
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid core mask 0x{CoreMask:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidCoreMask);
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-05-13 18:00:29 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (Thread == null)
|
|
|
|
{
|
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2018-05-13 23:01:10 -07:00
|
|
|
//-1 is used as "don't care", so the IdealCore value is ignored.
|
|
|
|
//-2 is used as "use NPDM default core id" (handled above).
|
|
|
|
//-3 is used as "don't update", the old IdealCore value is kept.
|
2018-06-21 19:05:42 -07:00
|
|
|
if (IdealCore == -3 && (CoreMask & (1 << Thread.IdealCore)) == 0)
|
2018-05-13 18:00:29 -07:00
|
|
|
{
|
2018-05-13 23:01:10 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid core mask 0x{CoreMask:x8}!");
|
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidCoreMask);
|
|
|
|
|
|
|
|
return;
|
2018-05-13 18:00:29 -07:00
|
|
|
}
|
|
|
|
|
2018-06-21 19:05:42 -07:00
|
|
|
Process.Scheduler.ChangeCore(Thread, IdealCore, (int)CoreMask);
|
2018-05-13 18:00:29 -07:00
|
|
|
|
|
|
|
ThreadState.X0 = 0;
|
2018-02-25 10:58:16 -08:00
|
|
|
}
|
|
|
|
|
2018-04-04 15:16:59 -07:00
|
|
|
private void SvcGetCurrentProcessorNumber(AThreadState ThreadState)
|
|
|
|
{
|
2018-05-12 20:22:42 -07:00
|
|
|
ThreadState.X0 = (ulong)Process.GetThread(ThreadState.Tpidr).ActualCore;
|
2018-04-04 15:16:59 -07:00
|
|
|
}
|
|
|
|
|
2018-02-25 10:58:16 -08:00
|
|
|
private void SvcGetThreadId(AThreadState ThreadState)
|
|
|
|
{
|
2018-04-04 12:07:44 -07:00
|
|
|
int Handle = (int)ThreadState.X1;
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
KThread Thread = GetThread(ThreadState.Tpidr, Handle);
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-05-11 19:05:06 -07:00
|
|
|
if (Thread != null)
|
2018-02-25 10:58:16 -08:00
|
|
|
{
|
2018-03-05 11:18:37 -08:00
|
|
|
ThreadState.X0 = 0;
|
2018-05-11 19:05:06 -07:00
|
|
|
ThreadState.X1 = (ulong)Thread.ThreadId;
|
2018-02-25 10:58:16 -08:00
|
|
|
}
|
2018-04-04 12:07:44 -07:00
|
|
|
else
|
|
|
|
{
|
2018-04-24 11:57:39 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-04-19 12:18:30 -07:00
|
|
|
|
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void SvcSetThreadActivity(AThreadState ThreadState)
|
|
|
|
{
|
|
|
|
int Handle = (int)ThreadState.X0;
|
2018-04-21 21:21:49 -07:00
|
|
|
bool Active = (int)ThreadState.X1 == 0;
|
2018-04-19 12:18:30 -07:00
|
|
|
|
2018-04-21 21:21:49 -07:00
|
|
|
KThread Thread = Process.HandleTable.GetData<KThread>(Handle);
|
2018-04-19 12:18:30 -07:00
|
|
|
|
2018-04-21 21:21:49 -07:00
|
|
|
if (Thread != null)
|
2018-04-19 12:18:30 -07:00
|
|
|
{
|
2018-04-21 21:21:49 -07:00
|
|
|
Process.Scheduler.SetThreadActivity(Thread, Active);
|
2018-04-19 12:18:30 -07:00
|
|
|
|
|
|
|
ThreadState.X0 = 0;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2018-04-24 11:57:39 -07:00
|
|
|
Ns.Log.PrintWarning(LogClass.KernelSvc, $"Invalid thread handle 0x{Handle:x8}!");
|
2018-02-25 10:58:16 -08:00
|
|
|
|
2018-04-04 12:07:44 -07:00
|
|
|
ThreadState.X0 = MakeError(ErrorModule.Kernel, KernelErr.InvalidHandle);
|
|
|
|
}
|
2018-02-25 10:58:16 -08:00
|
|
|
}
|
2018-02-04 15:08:20 -08:00
|
|
|
}
|
|
|
|
}
|