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>
/// The amount of seconds to keep a lag compensation position history
/// </summary>
public float SecondsHistory = 5;
public int SecondsHistory = 5;
/// <summary>
/// Wheter or not to make the library handle object spawning
/// </summary>

View File

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

View File

@ -212,9 +212,10 @@ namespace MLAPI.MonoBehaviours.Core
private ConnectionConfig Init(bool server)
{
networkTime = 0f;
lastSendTickTime = 0;
lastEventTickTime = 0;
lastReceiveTickTime = 0;
lastSendTickTime = 0f;
lastEventTickTime = 0f;
lastReceiveTickTime = 0f;
eventOvershootCounter = 0f;
pendingClients = new HashSet<uint>();
connectedClients = new Dictionary<uint, NetworkedClient>();
messageBuffer = new byte[NetworkConfig.MessageBufferSize];
@ -679,6 +680,7 @@ namespace MLAPI.MonoBehaviours.Core
private float lastReceiveTickTime;
private float lastSendTickTime;
private float lastEventTickTime;
private float eventOvershootCounter;
private float lastTimeSyncTime;
private void Update()
{
@ -781,12 +783,19 @@ namespace MLAPI.MonoBehaviours.Core
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();
NetworkedObject.InvokeSyncvarUpdate();
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)
{

View File

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