261 lines
8.0 KiB
C#
261 lines
8.0 KiB
C#
|
|
|||
|
using System;
|
|||
|
using UnityEngine;
|
|||
|
|
|||
|
namespace Cysharp.Threading.Tasks.Internal
|
|||
|
{
|
|||
|
internal sealed class PlayerLoopRunner
|
|||
|
{
|
|||
|
const int InitialSize = 16;
|
|||
|
|
|||
|
readonly PlayerLoopTiming timing;
|
|||
|
readonly object runningAndQueueLock = new object();
|
|||
|
readonly object arrayLock = new object();
|
|||
|
readonly Action<Exception> unhandledExceptionCallback;
|
|||
|
|
|||
|
int tail = 0;
|
|||
|
bool running = false;
|
|||
|
IPlayerLoopItem[] loopItems = new IPlayerLoopItem[InitialSize];
|
|||
|
MinimumQueue<IPlayerLoopItem> waitQueue = new MinimumQueue<IPlayerLoopItem>(InitialSize);
|
|||
|
|
|||
|
|
|||
|
|
|||
|
public PlayerLoopRunner(PlayerLoopTiming timing)
|
|||
|
{
|
|||
|
this.unhandledExceptionCallback = ex => Debug.LogException(ex);
|
|||
|
this.timing = timing;
|
|||
|
}
|
|||
|
|
|||
|
public void AddAction(IPlayerLoopItem item)
|
|||
|
{
|
|||
|
lock (runningAndQueueLock)
|
|||
|
{
|
|||
|
if (running)
|
|||
|
{
|
|||
|
waitQueue.Enqueue(item);
|
|||
|
return;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
lock (arrayLock)
|
|||
|
{
|
|||
|
// Ensure Capacity
|
|||
|
if (loopItems.Length == tail)
|
|||
|
{
|
|||
|
Array.Resize(ref loopItems, checked(tail * 2));
|
|||
|
}
|
|||
|
loopItems[tail++] = item;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
public int Clear()
|
|||
|
{
|
|||
|
lock (arrayLock)
|
|||
|
{
|
|||
|
var rest = 0;
|
|||
|
|
|||
|
for (var index = 0; index < loopItems.Length; index++)
|
|||
|
{
|
|||
|
if (loopItems[index] != null)
|
|||
|
{
|
|||
|
rest++;
|
|||
|
}
|
|||
|
|
|||
|
loopItems[index] = null;
|
|||
|
}
|
|||
|
|
|||
|
tail = 0;
|
|||
|
return rest;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// delegate entrypoint.
|
|||
|
public void Run()
|
|||
|
{
|
|||
|
// for debugging, create named stacktrace.
|
|||
|
#if DEBUG
|
|||
|
switch (timing)
|
|||
|
{
|
|||
|
case PlayerLoopTiming.Initialization:
|
|||
|
Initialization();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastInitialization:
|
|||
|
LastInitialization();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.EarlyUpdate:
|
|||
|
EarlyUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastEarlyUpdate:
|
|||
|
LastEarlyUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.FixedUpdate:
|
|||
|
FixedUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastFixedUpdate:
|
|||
|
LastFixedUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.PreUpdate:
|
|||
|
PreUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastPreUpdate:
|
|||
|
LastPreUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.Update:
|
|||
|
Update();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastUpdate:
|
|||
|
LastUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.PreLateUpdate:
|
|||
|
PreLateUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastPreLateUpdate:
|
|||
|
LastPreLateUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.PostLateUpdate:
|
|||
|
PostLateUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastPostLateUpdate:
|
|||
|
LastPostLateUpdate();
|
|||
|
break;
|
|||
|
#if UNITY_2020_2_OR_NEWER
|
|||
|
case PlayerLoopTiming.TimeUpdate:
|
|||
|
TimeUpdate();
|
|||
|
break;
|
|||
|
case PlayerLoopTiming.LastTimeUpdate:
|
|||
|
LastTimeUpdate();
|
|||
|
break;
|
|||
|
#endif
|
|||
|
default:
|
|||
|
break;
|
|||
|
}
|
|||
|
#else
|
|||
|
RunCore();
|
|||
|
#endif
|
|||
|
}
|
|||
|
|
|||
|
void Initialization() => RunCore();
|
|||
|
void LastInitialization() => RunCore();
|
|||
|
void EarlyUpdate() => RunCore();
|
|||
|
void LastEarlyUpdate() => RunCore();
|
|||
|
void FixedUpdate() => RunCore();
|
|||
|
void LastFixedUpdate() => RunCore();
|
|||
|
void PreUpdate() => RunCore();
|
|||
|
void LastPreUpdate() => RunCore();
|
|||
|
void Update() => RunCore();
|
|||
|
void LastUpdate() => RunCore();
|
|||
|
void PreLateUpdate() => RunCore();
|
|||
|
void LastPreLateUpdate() => RunCore();
|
|||
|
void PostLateUpdate() => RunCore();
|
|||
|
void LastPostLateUpdate() => RunCore();
|
|||
|
#if UNITY_2020_2_OR_NEWER
|
|||
|
void TimeUpdate() => RunCore();
|
|||
|
void LastTimeUpdate() => RunCore();
|
|||
|
#endif
|
|||
|
|
|||
|
[System.Diagnostics.DebuggerHidden]
|
|||
|
void RunCore()
|
|||
|
{
|
|||
|
lock (runningAndQueueLock)
|
|||
|
{
|
|||
|
running = true;
|
|||
|
}
|
|||
|
|
|||
|
lock (arrayLock)
|
|||
|
{
|
|||
|
var j = tail - 1;
|
|||
|
|
|||
|
for (int i = 0; i < loopItems.Length; i++)
|
|||
|
{
|
|||
|
var action = loopItems[i];
|
|||
|
if (action != null)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
if (!action.MoveNext())
|
|||
|
{
|
|||
|
loopItems[i] = null;
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
continue; // next i
|
|||
|
}
|
|||
|
}
|
|||
|
catch (Exception ex)
|
|||
|
{
|
|||
|
loopItems[i] = null;
|
|||
|
try
|
|||
|
{
|
|||
|
unhandledExceptionCallback(ex);
|
|||
|
}
|
|||
|
catch { }
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
// find null, loop from tail
|
|||
|
while (i < j)
|
|||
|
{
|
|||
|
var fromTail = loopItems[j];
|
|||
|
if (fromTail != null)
|
|||
|
{
|
|||
|
try
|
|||
|
{
|
|||
|
if (!fromTail.MoveNext())
|
|||
|
{
|
|||
|
loopItems[j] = null;
|
|||
|
j--;
|
|||
|
continue; // next j
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
// swap
|
|||
|
loopItems[i] = fromTail;
|
|||
|
loopItems[j] = null;
|
|||
|
j--;
|
|||
|
goto NEXT_LOOP; // next i
|
|||
|
}
|
|||
|
}
|
|||
|
catch (Exception ex)
|
|||
|
{
|
|||
|
loopItems[j] = null;
|
|||
|
j--;
|
|||
|
try
|
|||
|
{
|
|||
|
unhandledExceptionCallback(ex);
|
|||
|
}
|
|||
|
catch { }
|
|||
|
continue; // next j
|
|||
|
}
|
|||
|
}
|
|||
|
else
|
|||
|
{
|
|||
|
j--;
|
|||
|
}
|
|||
|
}
|
|||
|
|
|||
|
tail = i; // loop end
|
|||
|
break; // LOOP END
|
|||
|
|
|||
|
NEXT_LOOP:
|
|||
|
continue;
|
|||
|
}
|
|||
|
|
|||
|
|
|||
|
lock (runningAndQueueLock)
|
|||
|
{
|
|||
|
running = false;
|
|||
|
while (waitQueue.Count != 0)
|
|||
|
{
|
|||
|
if (loopItems.Length == tail)
|
|||
|
{
|
|||
|
Array.Resize(ref loopItems, checked(tail * 2));
|
|||
|
}
|
|||
|
loopItems[tail++] = waitQueue.Dequeue();
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
}
|
|||
|
|