Reworked the internal event loop & Lag compensator

This commit is contained in:
Albin Corén 2018-04-20 16:28:24 +02:00
parent 80b167ef48
commit f5a7a83ccb
5 changed files with 89 additions and 35 deletions

40
MLAPI/Data/FixedQueue.cs Normal file
View File

@ -0,0 +1,40 @@
using System;
namespace MLAPI.Data
{
public class FixedQueue<T>
{
protected readonly T[] queue;
protected int queueCount = 0;
protected int queueStart;
public int Count { get => queueCount; }
public FixedQueue(int maxSize)
{
queue = new T[maxSize];
queueStart = 0;
}
public bool Enqueue(T t)
{
queue[(queueStart + queueCount) % queue.Length] = t;
if (++queueCount > queue.Length)
{
--queueCount;
return true;
}
return false;
}
public T Dequeue()
{
if (--queueCount == -1) throw new IndexOutOfRangeException("Cannot dequeue empty queue!");
T res = queue[queueStart];
queueStart = (queueStart + 1) % queue.Length;
return res;
}
public T ElementAt(int index) => queue[(queueStart + index) % queue.Length];
}
}

View File

@ -104,7 +104,7 @@ namespace MLAPI.Data
/// <summary> /// <summary>
/// The amount of seconds to keep a lag compensation position history /// The amount of seconds to keep a lag compensation position history
/// </summary> /// </summary>
public float SecondsHistory = 5; public int SecondsHistory = 5;
/// <summary> /// <summary>
/// Wheter or not to make the library handle object spawning /// Wheter or not to make the library handle object spawning
/// </summary> /// </summary>

View File

@ -74,6 +74,7 @@
<Compile Include="Data\Channel.cs" /> <Compile Include="Data\Channel.cs" />
<Compile Include="Data\FieldType.cs" /> <Compile Include="Data\FieldType.cs" />
<Compile Include="Attributes\SyncedVar.cs" /> <Compile Include="Attributes\SyncedVar.cs" />
<Compile Include="Data\FixedQueue.cs" />
<Compile Include="Data\NetworkConfig.cs" /> <Compile Include="Data\NetworkConfig.cs" />
<Compile Include="Data\NetworkedPrefab.cs" /> <Compile Include="Data\NetworkedPrefab.cs" />
<Compile Include="Data\NetworkPool.cs" /> <Compile Include="Data\NetworkPool.cs" />

View File

@ -212,9 +212,10 @@ namespace MLAPI.MonoBehaviours.Core
private ConnectionConfig Init(bool server) private ConnectionConfig Init(bool server)
{ {
networkTime = 0f; networkTime = 0f;
lastSendTickTime = 0; lastSendTickTime = 0f;
lastEventTickTime = 0; lastEventTickTime = 0f;
lastReceiveTickTime = 0; lastReceiveTickTime = 0f;
eventOvershootCounter = 0f;
pendingClients = new HashSet<uint>(); pendingClients = new HashSet<uint>();
connectedClients = new Dictionary<uint, NetworkedClient>(); connectedClients = new Dictionary<uint, NetworkedClient>();
messageBuffer = new byte[NetworkConfig.MessageBufferSize]; messageBuffer = new byte[NetworkConfig.MessageBufferSize];
@ -679,6 +680,7 @@ namespace MLAPI.MonoBehaviours.Core
private float lastReceiveTickTime; private float lastReceiveTickTime;
private float lastSendTickTime; private float lastSendTickTime;
private float lastEventTickTime; private float lastEventTickTime;
private float eventOvershootCounter;
private float lastTimeSyncTime; private float lastTimeSyncTime;
private void Update() private void Update()
{ {
@ -781,12 +783,19 @@ namespace MLAPI.MonoBehaviours.Core
lastReceiveTickTime = NetworkTime; lastReceiveTickTime = NetworkTime;
} }
if (isServer && ((NetworkTime - lastEventTickTime >= (1f / NetworkConfig.EventTickrate)) || NetworkConfig.EventTickrate <= 0)) if (isServer && ((NetworkTime - lastEventTickTime >= (1f / NetworkConfig.EventTickrate))))
{ {
eventOvershootCounter += ((NetworkTime - lastEventTickTime) - (1f / NetworkConfig.EventTickrate));
LagCompensationManager.AddFrames(); LagCompensationManager.AddFrames();
NetworkedObject.InvokeSyncvarUpdate(); NetworkedObject.InvokeSyncvarUpdate();
lastEventTickTime = NetworkTime; lastEventTickTime = NetworkTime;
} }
else if (isServer && eventOvershootCounter >= ((1f / NetworkConfig.EventTickrate)))
{
//We run this one to compensate for previous update overshoots.
eventOvershootCounter -= (1f / NetworkConfig.EventTickrate);
LagCompensationManager.AddFrames();
}
if (NetworkConfig.EnableTimeResync && NetworkTime - lastTimeSyncTime >= 30) if (NetworkConfig.EnableTimeResync && NetworkTime - lastTimeSyncTime >= 30)
{ {

View File

@ -1,5 +1,6 @@
using MLAPI.Data; using MLAPI.Data;
using MLAPI.NetworkingManagerComponents.Core; using MLAPI.NetworkingManagerComponents.Core;
using System;
using System.Collections.Generic; using System.Collections.Generic;
using UnityEngine; using UnityEngine;
@ -15,7 +16,7 @@ namespace MLAPI.MonoBehaviours.Core
public class TrackedObject : MonoBehaviour public class TrackedObject : MonoBehaviour
{ {
internal Dictionary<float, TrackedPointData> FrameData = new Dictionary<float, TrackedPointData>(); internal Dictionary<float, TrackedPointData> FrameData = new Dictionary<float, TrackedPointData>();
internal LinkedList<float> Framekeys = new LinkedList<float>(); internal FixedQueue<float> Framekeys;
private Vector3 savedPosition; private Vector3 savedPosition;
private Quaternion savedRotation; private Quaternion savedRotation;
@ -37,38 +38,49 @@ namespace MLAPI.MonoBehaviours.Core
{ {
get get
{ {
if (Framekeys.First == null || Framekeys.Last == null) if (Framekeys.Count < 2)
return 0; return 0;
float totalSpan = Framekeys.Last.Value - Framekeys.First.Value; float totalSpan = Framekeys.ElementAt(Framekeys.Count - 1) - Framekeys.ElementAt(0);
return (totalSpan / Framekeys.Count) * 1000f; return (totalSpan / Framekeys.Count) * 1000f;
} }
} }
public float TotalTimeHistory
{
get
{
return Framekeys.ElementAt(Framekeys.Count - 1) - Framekeys.ElementAt(0);
}
}
private int maxPoints
{
get
{
return (int)(NetworkingManager.singleton.NetworkConfig.SecondsHistory / (1f / NetworkingManager.singleton.NetworkConfig.EventTickrate));
}
}
internal void ReverseTransform(float secondsAgo) internal void ReverseTransform(float secondsAgo)
{ {
savedPosition = transform.position; savedPosition = transform.position;
savedRotation = transform.rotation; savedRotation = transform.rotation;
float currentTime = NetworkingManager.singleton.NetworkTime; float currentTime = NetworkingManager.singleton.NetworkTime;
float targetTime = currentTime - secondsAgo; float targetTime = currentTime - secondsAgo;
float previousTime = 0;
float nextTime = 0; float previousTime = 0f;
LinkedListNode<float> node = Framekeys.First; float nextTime = 0f;
float previousValue = 0f; for (int i = 0; i < Framekeys.Count; i++)
while(node != null)
{ {
if(previousValue <= targetTime && node.Value >= targetTime) if (previousTime <= targetTime && Framekeys.ElementAt(i) >= targetTime)
{ {
previousTime = previousValue; nextTime = Framekeys.ElementAt(i);
nextTime = node.Value;
break; break;
} }
else else
{ previousTime = Framekeys.ElementAt(i);
previousValue = node.Value;
node = node.Next;
}
} }
float timeBetweenFrames = nextTime - previousTime; float timeBetweenFrames = nextTime - previousTime;
float timeAwayFromPrevious = currentTime - previousTime; float timeAwayFromPrevious = currentTime - previousTime;
float lerpProgress = timeAwayFromPrevious / timeBetweenFrames; float lerpProgress = timeAwayFromPrevious / timeBetweenFrames;
@ -84,35 +96,27 @@ namespace MLAPI.MonoBehaviours.Core
void Start() void Start()
{ {
Framekeys.AddFirst(0); Framekeys = new FixedQueue<float>(maxPoints);
Framekeys.Enqueue(0);
LagCompensationManager.simulationObjects.Add(this); LagCompensationManager.simulationObjects.Add(this);
} }
void OnDestroy() void OnDestroy()
{ {
Framekeys.Clear();
FrameData.Clear();
LagCompensationManager.simulationObjects.Remove(this); LagCompensationManager.simulationObjects.Remove(this);
} }
internal void AddFrame() internal void AddFrame()
{ {
float currentTime = NetworkingManager.singleton.NetworkTime; if (Framekeys.Count == maxPoints)
LinkedListNode<float> node = Framekeys.First; FrameData.Remove(Framekeys.Dequeue());
LinkedListNode<float> nextNode = node.Next;
while (node != null && currentTime - node.Value >= NetworkingManager.singleton.NetworkConfig.SecondsHistory)
{
nextNode = node.Next;
FrameData.Remove(node.Value);
Framekeys.RemoveFirst();
node = nextNode;
}
FrameData.Add(NetworkingManager.singleton.NetworkTime, new TrackedPointData() FrameData.Add(NetworkingManager.singleton.NetworkTime, new TrackedPointData()
{ {
position = transform.position, position = transform.position,
rotation = transform.rotation rotation = transform.rotation
}); });
Framekeys.AddLast(NetworkingManager.singleton.NetworkTime); Framekeys.Enqueue(NetworkingManager.singleton.NetworkTime);
} }
} }
} }