From f5a7a83ccb544765e617e02fc3da1713146a3ebc Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?Albin=20Cor=C3=A9n?= <2108U9@gmail.com> Date: Fri, 20 Apr 2018 16:28:24 +0200 Subject: [PATCH] Reworked the internal event loop & Lag compensator --- MLAPI/Data/FixedQueue.cs | 40 ++++++++++++ MLAPI/Data/NetworkConfig.cs | 2 +- MLAPI/MLAPI.csproj | 1 + .../MonoBehaviours/Core/NetworkingManager.cs | 17 +++-- MLAPI/MonoBehaviours/Core/TrackedObject.cs | 64 ++++++++++--------- 5 files changed, 89 insertions(+), 35 deletions(-) create mode 100644 MLAPI/Data/FixedQueue.cs diff --git a/MLAPI/Data/FixedQueue.cs b/MLAPI/Data/FixedQueue.cs new file mode 100644 index 0000000..f27279b --- /dev/null +++ b/MLAPI/Data/FixedQueue.cs @@ -0,0 +1,40 @@ +using System; + +namespace MLAPI.Data +{ + public class FixedQueue + { + 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]; + } +} diff --git a/MLAPI/Data/NetworkConfig.cs b/MLAPI/Data/NetworkConfig.cs index dd4d22c..c4fea53 100644 --- a/MLAPI/Data/NetworkConfig.cs +++ b/MLAPI/Data/NetworkConfig.cs @@ -104,7 +104,7 @@ namespace MLAPI.Data /// /// The amount of seconds to keep a lag compensation position history /// - public float SecondsHistory = 5; + public int SecondsHistory = 5; /// /// Wheter or not to make the library handle object spawning /// diff --git a/MLAPI/MLAPI.csproj b/MLAPI/MLAPI.csproj index 73dadbd..8ee2128 100644 --- a/MLAPI/MLAPI.csproj +++ b/MLAPI/MLAPI.csproj @@ -74,6 +74,7 @@ + diff --git a/MLAPI/MonoBehaviours/Core/NetworkingManager.cs b/MLAPI/MonoBehaviours/Core/NetworkingManager.cs index d5c2bc4..cdc2228 100644 --- a/MLAPI/MonoBehaviours/Core/NetworkingManager.cs +++ b/MLAPI/MonoBehaviours/Core/NetworkingManager.cs @@ -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(); connectedClients = new Dictionary(); 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) { diff --git a/MLAPI/MonoBehaviours/Core/TrackedObject.cs b/MLAPI/MonoBehaviours/Core/TrackedObject.cs index 716c2d5..07df96a 100644 --- a/MLAPI/MonoBehaviours/Core/TrackedObject.cs +++ b/MLAPI/MonoBehaviours/Core/TrackedObject.cs @@ -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 FrameData = new Dictionary(); - internal LinkedList Framekeys = new LinkedList(); + internal FixedQueue 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 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(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 node = Framekeys.First; - LinkedListNode 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); } } }