UpdateManager comments + synchronization fix

This commit is contained in:
themode 2020-10-22 15:57:27 +02:00
parent 3cb880cf80
commit 419541b878
3 changed files with 60 additions and 33 deletions

View File

@ -13,11 +13,11 @@ import net.minestom.server.thread.ThreadProvider;
import net.minestom.server.utils.thread.MinestomThread; import net.minestom.server.utils.thread.MinestomThread;
import net.minestom.server.utils.validate.Check; import net.minestom.server.utils.validate.Check;
import java.util.ArrayList;
import java.util.List; import java.util.List;
import java.util.concurrent.ConcurrentLinkedQueue;
import java.util.concurrent.ExecutorService; import java.util.concurrent.ExecutorService;
import java.util.concurrent.Future; import java.util.concurrent.Future;
import java.util.function.Consumer; import java.util.function.DoubleConsumer;
/** /**
* Manager responsible for the server ticks. * Manager responsible for the server ticks.
@ -36,8 +36,8 @@ public final class UpdateManager {
private ThreadProvider threadProvider; private ThreadProvider threadProvider;
private final ArrayList<Runnable> tickStartCallbacks = new ArrayList<>(); private final ConcurrentLinkedQueue<Runnable> tickStartCallbacks = new ConcurrentLinkedQueue<>();
private final ArrayList<Consumer<Double>> tickEndCallbacks = new ArrayList<>(); private final ConcurrentLinkedQueue<DoubleConsumer> tickEndCallbacks = new ConcurrentLinkedQueue<>();
{ {
//threadProvider = new PerInstanceThreadProvider(); //threadProvider = new PerInstanceThreadProvider();
@ -63,28 +63,33 @@ public final class UpdateManager {
long currentTime; long currentTime;
while (!stopRequested) { while (!stopRequested) {
currentTime = System.nanoTime(); currentTime = System.nanoTime();
final long time = System.currentTimeMillis(); final long tickStart = System.currentTimeMillis();
//Tick Callbacks // Tick start callbacks
tickStartCallbacks.forEach(Runnable::run); if (!tickStartCallbacks.isEmpty()) {
Runnable callback;
while ((callback = tickStartCallbacks.poll()) != null) {
callback.run();
}
}
List<Future<?>> futures; List<Future<?>> futures;
// Server tick (instance/chunk/entity) // Server tick (instance/chunk/entity)
// Synchronize with the update manager instance, like the signal for chunk load/unload // Synchronize with the update manager instance, like the signal for chunk load/unload
synchronized (this) { synchronized (this) {
futures = threadProvider.update(time); futures = threadProvider.update(tickStart);
} }
// Waiting players update (newly connected clients waiting to get into the server) // Waiting players update (newly connected clients waiting to get into the server)
entityManager.updateWaitingPlayers(); entityManager.updateWaitingPlayers();
// Keep Alive Handling // Keep Alive Handling
final KeepAlivePacket keepAlivePacket = new KeepAlivePacket(time); final KeepAlivePacket keepAlivePacket = new KeepAlivePacket(tickStart);
for (Player player : connectionManager.getOnlinePlayers()) { for (Player player : connectionManager.getOnlinePlayers()) {
final long lastKeepAlive = time - player.getLastKeepAlive(); final long lastKeepAlive = tickStart - player.getLastKeepAlive();
if (lastKeepAlive > KEEP_ALIVE_DELAY && player.didAnswerKeepAlive()) { if (lastKeepAlive > KEEP_ALIVE_DELAY && player.didAnswerKeepAlive()) {
player.refreshKeepAlive(time); player.refreshKeepAlive(tickStart);
player.getPlayerConnection().sendPacket(keepAlivePacket); player.getPlayerConnection().sendPacket(keepAlivePacket);
} else if (lastKeepAlive >= KEEP_ALIVE_KICK) { } else if (lastKeepAlive >= KEEP_ALIVE_KICK) {
player.kick(TIMEOUT_TEXT); player.kick(TIMEOUT_TEXT);
@ -100,13 +105,17 @@ public final class UpdateManager {
} }
//Tick Callbacks // Tick end callbacks
double tickTime = (System.nanoTime() - currentTime) / 1000000D; if (!tickEndCallbacks.isEmpty()) {
tickEndCallbacks.forEach(doubleConsumer -> doubleConsumer.accept(tickTime)); final double tickEnd = (System.nanoTime() - currentTime) / 1000000D;
DoubleConsumer callback;
while ((callback = tickEndCallbacks.poll()) != null) {
callback.accept(tickEnd);
}
}
// Sleep until next tick // Sleep until next tick
long sleepTime = (tickDistance - (System.nanoTime() - currentTime)) / 1000000; final long sleepTime = Math.max(1, (tickDistance - (System.nanoTime() - currentTime)) / 1000000);
sleepTime = Math.max(1, sleepTime);
try { try {
Thread.sleep(sleepTime); Thread.sleep(sleepTime);
@ -194,20 +203,42 @@ public final class UpdateManager {
this.threadProvider.onChunkUnload(instance, chunkX, chunkZ); this.threadProvider.onChunkUnload(instance, chunkX, chunkZ);
} }
/**
* Adds a callback executed at the start of the next server tick.
*
* @param callback the tick start callback
*/
public void addTickStartCallback(Runnable callback) { public void addTickStartCallback(Runnable callback) {
tickStartCallbacks.add(callback); this.tickStartCallbacks.add(callback);
} }
/**
* Removes a tick start callback.
*
* @param callback the callback to remove
*/
public void removeTickStartCallback(Runnable callback) { public void removeTickStartCallback(Runnable callback) {
tickStartCallbacks.remove(callback); this.tickStartCallbacks.remove(callback);
} }
public void addTickEndCallback(Consumer<Double> callback) { /**
tickEndCallbacks.add(callback); * Adds a callback executed at the end of the next server tick.
* <p>
* The double in the consumer represents the duration (in ms) of the tick.
*
* @param callback the tick end callback
*/
public void addTickEndCallback(DoubleConsumer callback) {
this.tickEndCallbacks.add(callback);
} }
public void removeTickEndCallback(Consumer<Double> callback) { /**
tickEndCallbacks.remove(callback); * Removes a tick end callback.
*
* @param callback the callback to remove
*/
public void removeTickEndCallback(DoubleConsumer callback) {
this.tickEndCallbacks.remove(callback);
} }
/** /**

View File

@ -9,9 +9,9 @@ public class ClientPacketsHandler {
// Max packet id // Max packet id
private static final int SIZE = 0x30; private static final int SIZE = 0x30;
private final Supplier<? extends ClientPacket>[] supplierAccesses = new Supplier[SIZE]; private final Supplier<ClientPacket>[] supplierAccesses = new Supplier[SIZE];
public void register(int id, Supplier<? extends ClientPacket> packetSupplier) { public void register(int id, Supplier<ClientPacket> packetSupplier) {
supplierAccesses[id] = packetSupplier; supplierAccesses[id] = packetSupplier;
} }
@ -19,13 +19,13 @@ public class ClientPacketsHandler {
if (id > SIZE) if (id > SIZE)
throw new IllegalStateException("Packet ID 0x" + Integer.toHexString(id) + " has been tried to be parsed, debug needed"); throw new IllegalStateException("Packet ID 0x" + Integer.toHexString(id) + " has been tried to be parsed, debug needed");
Supplier<? extends ClientPacket> supplier = supplierAccesses[id]; Supplier<ClientPacket> supplier = supplierAccesses[id];
if (supplierAccesses[id] == null) if (supplierAccesses[id] == null)
throw new IllegalStateException("Packet id 0x" + Integer.toHexString(id) + " isn't registered!"); throw new IllegalStateException("Packet id 0x" + Integer.toHexString(id) + " isn't registered!");
ClientPacket packet = supplier.get(); //ClientPacket packet = supplier.get();
//System.out.println("RECEIVED PACKET 0x" + Integer.toHexString(id)+" : "+packet.getClass().getSimpleName()); //System.out.println("RECEIVED PACKET 0x" + Integer.toHexString(id)+" : "+packet.getClass().getSimpleName());
return packet; return supplier.get();
} }
} }

View File

@ -99,9 +99,7 @@ public abstract class ThreadProvider {
this.pool = new MinestomThread(threadCount, MinecraftServer.THREAD_NAME_TICK); this.pool = new MinestomThread(threadCount, MinecraftServer.THREAD_NAME_TICK);
} }
/** // INSTANCE UPDATE
* INSTANCE UPDATE
*/
/** /**
* Process a whole tick for a chunk. * Process a whole tick for a chunk.
@ -146,9 +144,7 @@ public abstract class ThreadProvider {
chunk.tick(time, instance); chunk.tick(time, instance);
} }
/** // ENTITY UPDATE
* ENTITY UPDATE
*/
/** /**
* Executes an entity tick (all entities type creatures/objects/players) in an instance's chunk. * Executes an entity tick (all entities type creatures/objects/players) in an instance's chunk.