mirror of
https://github.com/Minestom/Minestom.git
synced 2025-01-03 23:17:48 +01:00
UpdateManager comments + synchronization fix
This commit is contained in:
parent
3cb880cf80
commit
419541b878
@ -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);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -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();
|
||||||
}
|
}
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@ -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.
|
||||||
|
Loading…
Reference in New Issue
Block a user