mirror of
				https://github.com/Ryujinx/Ryujinx.git
				synced 2025-10-25 14:02:26 -07:00 
			
		
		
		
	* Refactor SVC handler * Get rid of KernelErr * Split kernel code files into multiple folders
		
			
				
	
	
		
			207 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
			
		
		
	
	
			207 lines
		
	
	
		
			6.0 KiB
		
	
	
	
		
			C#
		
	
	
	
	
	
| using System.Collections.Generic;
 | |
| 
 | |
| namespace Ryujinx.HLE.HOS.Kernel.Threading
 | |
| {
 | |
|     class KSchedulingData
 | |
|     {
 | |
|         private LinkedList<KThread>[][] _scheduledThreadsPerPrioPerCore;
 | |
|         private LinkedList<KThread>[][] _suggestedThreadsPerPrioPerCore;
 | |
| 
 | |
|         private long[] _scheduledPrioritiesPerCore;
 | |
|         private long[] _suggestedPrioritiesPerCore;
 | |
| 
 | |
|         public KSchedulingData()
 | |
|         {
 | |
|             _suggestedThreadsPerPrioPerCore = new LinkedList<KThread>[KScheduler.PrioritiesCount][];
 | |
|             _scheduledThreadsPerPrioPerCore = new LinkedList<KThread>[KScheduler.PrioritiesCount][];
 | |
| 
 | |
|             for (int prio = 0; prio < KScheduler.PrioritiesCount; prio++)
 | |
|             {
 | |
|                 _suggestedThreadsPerPrioPerCore[prio] = new LinkedList<KThread>[KScheduler.CpuCoresCount];
 | |
|                 _scheduledThreadsPerPrioPerCore[prio] = new LinkedList<KThread>[KScheduler.CpuCoresCount];
 | |
| 
 | |
|                 for (int core = 0; core < KScheduler.CpuCoresCount; core++)
 | |
|                 {
 | |
|                     _suggestedThreadsPerPrioPerCore[prio][core] = new LinkedList<KThread>();
 | |
|                     _scheduledThreadsPerPrioPerCore[prio][core] = new LinkedList<KThread>();
 | |
|                 }
 | |
|             }
 | |
| 
 | |
|             _scheduledPrioritiesPerCore = new long[KScheduler.CpuCoresCount];
 | |
|             _suggestedPrioritiesPerCore = new long[KScheduler.CpuCoresCount];
 | |
|         }
 | |
| 
 | |
|         public IEnumerable<KThread> SuggestedThreads(int core)
 | |
|         {
 | |
|             return Iterate(_suggestedThreadsPerPrioPerCore, _suggestedPrioritiesPerCore, core);
 | |
|         }
 | |
| 
 | |
|         public IEnumerable<KThread> ScheduledThreads(int core)
 | |
|         {
 | |
|             return Iterate(_scheduledThreadsPerPrioPerCore, _scheduledPrioritiesPerCore, core);
 | |
|         }
 | |
| 
 | |
|         private IEnumerable<KThread> Iterate(LinkedList<KThread>[][] listPerPrioPerCore, long[] prios, int core)
 | |
|         {
 | |
|             long prioMask = prios[core];
 | |
| 
 | |
|             int prio = CountTrailingZeros(prioMask);
 | |
| 
 | |
|             prioMask &= ~(1L << prio);
 | |
| 
 | |
|             while (prio < KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 LinkedList<KThread> list = listPerPrioPerCore[prio][core];
 | |
| 
 | |
|                 LinkedListNode<KThread> node = list.First;
 | |
| 
 | |
|                 while (node != null)
 | |
|                 {
 | |
|                     yield return node.Value;
 | |
| 
 | |
|                     node = node.Next;
 | |
|                 }
 | |
| 
 | |
|                 prio = CountTrailingZeros(prioMask);
 | |
| 
 | |
|                 prioMask &= ~(1L << prio);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private int CountTrailingZeros(long value)
 | |
|         {
 | |
|             int count = 0;
 | |
| 
 | |
|             while (((value >> count) & 0xf) == 0 && count < 64)
 | |
|             {
 | |
|                 count += 4;
 | |
|             }
 | |
| 
 | |
|             while (((value >> count) & 1) == 0 && count < 64)
 | |
|             {
 | |
|                 count++;
 | |
|             }
 | |
| 
 | |
|             return count;
 | |
|         }
 | |
| 
 | |
|         public void TransferToCore(int prio, int dstCore, KThread thread)
 | |
|         {
 | |
|             bool schedulable = thread.DynamicPriority < KScheduler.PrioritiesCount;
 | |
| 
 | |
|             int srcCore = thread.CurrentCore;
 | |
| 
 | |
|             thread.CurrentCore = dstCore;
 | |
| 
 | |
|             if (srcCore == dstCore || !schedulable)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             if (srcCore >= 0)
 | |
|             {
 | |
|                 Unschedule(prio, srcCore, thread);
 | |
|             }
 | |
| 
 | |
|             if (dstCore >= 0)
 | |
|             {
 | |
|                 Unsuggest(prio, dstCore, thread);
 | |
|                 Schedule(prio, dstCore, thread);
 | |
|             }
 | |
| 
 | |
|             if (srcCore >= 0)
 | |
|             {
 | |
|                 Suggest(prio, srcCore, thread);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Suggest(int prio, int core, KThread thread)
 | |
|         {
 | |
|             if (prio >= KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             thread.SiblingsPerCore[core] = SuggestedQueue(prio, core).AddFirst(thread);
 | |
| 
 | |
|             _suggestedPrioritiesPerCore[core] |= 1L << prio;
 | |
|         }
 | |
| 
 | |
|         public void Unsuggest(int prio, int core, KThread thread)
 | |
|         {
 | |
|             if (prio >= KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             LinkedList<KThread> queue = SuggestedQueue(prio, core);
 | |
| 
 | |
|             queue.Remove(thread.SiblingsPerCore[core]);
 | |
| 
 | |
|             if (queue.First == null)
 | |
|             {
 | |
|                 _suggestedPrioritiesPerCore[core] &= ~(1L << prio);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         public void Schedule(int prio, int core, KThread thread)
 | |
|         {
 | |
|             if (prio >= KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             thread.SiblingsPerCore[core] = ScheduledQueue(prio, core).AddLast(thread);
 | |
| 
 | |
|             _scheduledPrioritiesPerCore[core] |= 1L << prio;
 | |
|         }
 | |
| 
 | |
|         public void SchedulePrepend(int prio, int core, KThread thread)
 | |
|         {
 | |
|             if (prio >= KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             thread.SiblingsPerCore[core] = ScheduledQueue(prio, core).AddFirst(thread);
 | |
| 
 | |
|             _scheduledPrioritiesPerCore[core] |= 1L << prio;
 | |
|         }
 | |
| 
 | |
|         public void Reschedule(int prio, int core, KThread thread)
 | |
|         {
 | |
|             LinkedList<KThread> queue = ScheduledQueue(prio, core);
 | |
| 
 | |
|             queue.Remove(thread.SiblingsPerCore[core]);
 | |
| 
 | |
|             thread.SiblingsPerCore[core] = queue.AddLast(thread);
 | |
|         }
 | |
| 
 | |
|         public void Unschedule(int prio, int core, KThread thread)
 | |
|         {
 | |
|             if (prio >= KScheduler.PrioritiesCount)
 | |
|             {
 | |
|                 return;
 | |
|             }
 | |
| 
 | |
|             LinkedList<KThread> queue = ScheduledQueue(prio, core);
 | |
| 
 | |
|             queue.Remove(thread.SiblingsPerCore[core]);
 | |
| 
 | |
|             if (queue.First == null)
 | |
|             {
 | |
|                 _scheduledPrioritiesPerCore[core] &= ~(1L << prio);
 | |
|             }
 | |
|         }
 | |
| 
 | |
|         private LinkedList<KThread> SuggestedQueue(int prio, int core)
 | |
|         {
 | |
|             return _suggestedThreadsPerPrioPerCore[prio][core];
 | |
|         }
 | |
| 
 | |
|         private LinkedList<KThread> ScheduledQueue(int prio, int core)
 | |
|         {
 | |
|             return _scheduledThreadsPerPrioPerCore[prio][core];
 | |
|         }
 | |
|     }
 | |
| } |