2018-02-05 00:08:20 +01:00
|
|
|
using ChocolArm64;
|
2018-04-21 21:07:16 +02:00
|
|
|
using System;
|
2018-02-05 00:08:20 +01:00
|
|
|
|
2018-02-20 21:09:23 +01:00
|
|
|
namespace Ryujinx.Core.OsHle.Handles
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-03-19 19:58:46 +01:00
|
|
|
class KThread : KSynchronizationObject
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
|
|
|
public AThread Thread { get; private set; }
|
|
|
|
|
2018-04-21 21:07:16 +02:00
|
|
|
public KThread MutexOwner { get; set; }
|
|
|
|
|
|
|
|
public KThread NextMutexThread { get; set; }
|
|
|
|
public KThread NextCondVarThread { get; set; }
|
|
|
|
|
|
|
|
public long MutexAddress { get; set; }
|
|
|
|
public long CondVarAddress { get; set; }
|
|
|
|
|
|
|
|
public int ActualPriority { get; private set; }
|
|
|
|
public int WantedPriority { get; private set; }
|
|
|
|
|
2018-04-19 04:52:23 +02:00
|
|
|
public int ProcessorId { get; private set; }
|
|
|
|
|
2018-04-21 21:07:16 +02:00
|
|
|
public int WaitHandle { get; set; }
|
2018-02-14 03:43:08 +01:00
|
|
|
|
|
|
|
public int ThreadId => Thread.ThreadId;
|
|
|
|
|
2018-03-19 19:58:46 +01:00
|
|
|
public KThread(AThread Thread, int ProcessorId, int Priority)
|
2018-02-05 00:08:20 +01:00
|
|
|
{
|
2018-02-14 03:43:08 +01:00
|
|
|
this.Thread = Thread;
|
|
|
|
this.ProcessorId = ProcessorId;
|
2018-04-21 21:07:16 +02:00
|
|
|
|
|
|
|
ActualPriority = WantedPriority = Priority;
|
|
|
|
}
|
|
|
|
|
|
|
|
public void SetPriority(int Priority)
|
|
|
|
{
|
|
|
|
WantedPriority = Priority;
|
|
|
|
|
|
|
|
UpdatePriority();
|
|
|
|
}
|
|
|
|
|
|
|
|
public void UpdatePriority()
|
|
|
|
{
|
|
|
|
int OldPriority = ActualPriority;
|
|
|
|
|
|
|
|
int CurrPriority = WantedPriority;
|
|
|
|
|
|
|
|
if (NextMutexThread != null && CurrPriority > NextMutexThread.WantedPriority)
|
|
|
|
{
|
|
|
|
CurrPriority = NextMutexThread.WantedPriority;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (CurrPriority != OldPriority)
|
|
|
|
{
|
|
|
|
ActualPriority = CurrPriority;
|
|
|
|
|
|
|
|
UpdateWaitList();
|
|
|
|
|
|
|
|
MutexOwner?.UpdatePriority();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
private void UpdateWaitList()
|
|
|
|
{
|
|
|
|
KThread OwnerThread = MutexOwner;
|
|
|
|
|
|
|
|
if (OwnerThread != null)
|
|
|
|
{
|
|
|
|
//The MutexOwner field should only be non null when the thread is
|
|
|
|
//waiting for the lock, and the lock belongs to another thread.
|
|
|
|
if (OwnerThread == this)
|
|
|
|
{
|
|
|
|
throw new InvalidOperationException();
|
|
|
|
}
|
|
|
|
|
|
|
|
lock (OwnerThread)
|
|
|
|
{
|
|
|
|
//Remove itself from the list.
|
|
|
|
KThread CurrThread = OwnerThread;
|
|
|
|
|
|
|
|
while (CurrThread.NextMutexThread != null)
|
|
|
|
{
|
|
|
|
if (CurrThread.NextMutexThread == this)
|
|
|
|
{
|
|
|
|
CurrThread.NextMutexThread = NextMutexThread;
|
|
|
|
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
CurrThread = CurrThread.NextMutexThread;
|
|
|
|
}
|
|
|
|
|
|
|
|
//Re-add taking new priority into account.
|
|
|
|
CurrThread = OwnerThread;
|
|
|
|
|
|
|
|
while (CurrThread.NextMutexThread != null)
|
|
|
|
{
|
|
|
|
if (CurrThread.NextMutexThread.ActualPriority < ActualPriority)
|
|
|
|
{
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
CurrThread = CurrThread.NextMutexThread;
|
|
|
|
}
|
|
|
|
|
|
|
|
NextMutexThread = CurrThread.NextMutexThread;
|
|
|
|
|
|
|
|
CurrThread.NextMutexThread = this;
|
|
|
|
}
|
|
|
|
}
|
2018-02-05 00:08:20 +01:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|