Rework of the thread provider + ConnectionManager#getPlayer for UUID

This commit is contained in:
Felix Cravic 2020-08-06 16:28:04 +02:00
parent 118ed220e3
commit a38a143bba
11 changed files with 135 additions and 162 deletions

View File

@ -149,9 +149,10 @@ public class PlayerInit {
p.teleport(player.getPosition()); p.teleport(player.getPosition());
}*/ }*/
ChickenCreature chickenCreature = new ChickenCreature(player.getPosition()); for (int i = 0; i < 100; i++) {
chickenCreature.setInstance(player.getInstance()); ChickenCreature chickenCreature = new ChickenCreature(player.getPosition());
chickenCreature.setTarget(player); chickenCreature.setInstance(player.getInstance());
}
/*EntityZombie zombie = new EntityZombie(player.getPosition()); /*EntityZombie zombie = new EntityZombie(player.getPosition());
zombie.setAttribute(Attribute.MOVEMENT_SPEED, 0.25f); zombie.setAttribute(Attribute.MOVEMENT_SPEED, 0.25f);

View File

@ -1,7 +1,7 @@
package fr.themode.demo.entity; package fr.themode.demo.entity;
import net.minestom.server.attribute.Attribute; import net.minestom.server.attribute.Attribute;
import net.minestom.server.entity.ai.goal.FollowTargetGoal; import net.minestom.server.entity.ai.goal.RandomStrollGoal;
import net.minestom.server.entity.ai.target.PlayerTarget; import net.minestom.server.entity.ai.target.PlayerTarget;
import net.minestom.server.entity.type.EntityChicken; import net.minestom.server.entity.type.EntityChicken;
import net.minestom.server.utils.Position; import net.minestom.server.utils.Position;
@ -12,8 +12,8 @@ public class ChickenCreature extends EntityChicken {
super(defaultPosition); super(defaultPosition);
//goalSelectors.add(new DoNothingGoal(this, 500, 0.1f)); //goalSelectors.add(new DoNothingGoal(this, 500, 0.1f));
//goalSelectors.add(new RandomStrollGoal(this, 2)); goalSelectors.add(new RandomStrollGoal(this, 2));
goalSelectors.add(new FollowTargetGoal(this)); //goalSelectors.add(new FollowTargetGoal(this));
targetSelectors.add(new PlayerTarget(this, 15)); targetSelectors.add(new PlayerTarget(this, 15));

View File

@ -288,6 +288,10 @@ public class MinecraftServer {
return tagManager; return tagManager;
} }
public static UpdateManager getUpdateManager() {
return updateManager;
}
public void start(String address, int port, ResponseDataConsumer responseDataConsumer) { public void start(String address, int port, ResponseDataConsumer responseDataConsumer) {
LOGGER.info("Starting Minestom server."); LOGGER.info("Starting Minestom server.");
MinecraftServer.responseDataConsumer = responseDataConsumer; MinecraftServer.responseDataConsumer = responseDataConsumer;

View File

@ -4,12 +4,11 @@ import net.minestom.server.chat.ChatColor;
import net.minestom.server.chat.ColoredText; import net.minestom.server.chat.ColoredText;
import net.minestom.server.entity.EntityManager; import net.minestom.server.entity.EntityManager;
import net.minestom.server.entity.Player; import net.minestom.server.entity.Player;
import net.minestom.server.instance.Chunk;
import net.minestom.server.instance.Instance; import net.minestom.server.instance.Instance;
import net.minestom.server.instance.InstanceManager; import net.minestom.server.instance.InstanceManager;
import net.minestom.server.network.ConnectionManager; import net.minestom.server.network.ConnectionManager;
import net.minestom.server.network.packet.server.play.KeepAlivePacket; import net.minestom.server.network.packet.server.play.KeepAlivePacket;
import net.minestom.server.thread.SingleThreadProvider; import net.minestom.server.thread.PerInstanceThreadProvider;
import net.minestom.server.thread.ThreadProvider; 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;
@ -27,8 +26,7 @@ public final class UpdateManager {
private ThreadProvider threadProvider; private ThreadProvider threadProvider;
{ {
threadProvider = new SingleThreadProvider(); threadProvider = new PerInstanceThreadProvider();
//threadProvider = new PerInstanceThreadProvider();
//threadProvider = new PerGroupChunkProvider(); //threadProvider = new PerGroupChunkProvider();
} }
@ -49,24 +47,15 @@ public final class UpdateManager {
long currentTime; long currentTime;
while (!stopRequested) { while (!stopRequested) {
currentTime = System.nanoTime(); currentTime = System.nanoTime();
final long time = System.currentTimeMillis();
// Server tick // Server tick
//long testTime = System.nanoTime(); threadProvider.update(time);
threadProvider.start();
for (Instance instance : instanceManager.getInstances()) {
for (Chunk chunk : instance.getChunks()) {
threadProvider.linkThread(instance, chunk);
}
}
threadProvider.end();
threadProvider.update();
//System.out.println("time: " + (System.nanoTime() - testTime));
// Waiting players update // Waiting players update
entityManager.updateWaitingPlayers(); entityManager.updateWaitingPlayers();
// Keep Alive Handling // Keep Alive Handling
final long time = System.currentTimeMillis();
final KeepAlivePacket keepAlivePacket = new KeepAlivePacket(time); final KeepAlivePacket keepAlivePacket = new KeepAlivePacket(time);
for (Player player : connectionManager.getOnlinePlayers()) { for (Player player : connectionManager.getOnlinePlayers()) {
final long lastKeepAlive = time - player.getLastKeepAlive(); final long lastKeepAlive = time - player.getLastKeepAlive();
@ -112,6 +101,33 @@ public final class UpdateManager {
this.threadProvider = threadProvider; this.threadProvider = threadProvider;
} }
/**
* Signal the thread provider that a chunk has been loaded
*
* @param instance the instance of the chunk
* @param chunkX the chunk X
* @param chunkZ the chunk Z
*/
public void signalChunkLoad(Instance instance, int chunkX, int chunkZ) {
if (this.threadProvider == null)
return;
this.threadProvider.onChunkLoad(instance, chunkX, chunkZ);
}
/**
* Signal the thread provider that a chunk has been unloaded
*
* @param instance the instance of the chunk
* @param chunkX the chunk X
* @param chunkZ the chunk Z
*/
public void signalChunkUnload(Instance instance, int chunkX, int chunkZ) {
if (this.threadProvider == null)
return;
this.threadProvider.onChunkUnload(instance, chunkX, chunkZ);
}
public void stop() { public void stop() {
stopRequested = true; stopRequested = true;
} }

View File

@ -4,6 +4,7 @@ import io.netty.buffer.ByteBuf;
import it.unimi.dsi.fastutil.longs.Long2ObjectMap; import it.unimi.dsi.fastutil.longs.Long2ObjectMap;
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap; import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap;
import net.minestom.server.MinecraftServer; import net.minestom.server.MinecraftServer;
import net.minestom.server.UpdateManager;
import net.minestom.server.data.Data; import net.minestom.server.data.Data;
import net.minestom.server.data.DataContainer; import net.minestom.server.data.DataContainer;
import net.minestom.server.entity.*; import net.minestom.server.entity.*;
@ -48,6 +49,7 @@ import java.util.function.Consumer;
public abstract class Instance implements BlockModifier, EventHandler, DataContainer { public abstract class Instance implements BlockModifier, EventHandler, DataContainer {
protected static final BlockManager BLOCK_MANAGER = MinecraftServer.getBlockManager(); protected static final BlockManager BLOCK_MANAGER = MinecraftServer.getBlockManager();
protected static final UpdateManager UPDATE_MANAGER = MinecraftServer.getUpdateManager();
private DimensionType dimensionType; private DimensionType dimensionType;

View File

@ -342,6 +342,8 @@ public class InstanceContainer extends Instance {
this.chunkEntities.remove(index); this.chunkEntities.remove(index);
chunk.unload(); chunk.unload();
UPDATE_MANAGER.signalChunkUnload(this, chunkX, chunkZ);
} }
@Override @Override
@ -428,6 +430,7 @@ public class InstanceContainer extends Instance {
final boolean loaded = chunkLoader.loadChunk(this, chunkX, chunkZ, chunk -> { final boolean loaded = chunkLoader.loadChunk(this, chunkX, chunkZ, chunk -> {
cacheChunk(chunk); cacheChunk(chunk);
callChunkLoadEvent(chunkX, chunkZ); callChunkLoadEvent(chunkX, chunkZ);
UPDATE_MANAGER.signalChunkLoad(this, chunkX, chunkZ);
if (callback != null) if (callback != null)
callback.accept(chunk); callback.accept(chunk);
}); });
@ -459,6 +462,7 @@ public class InstanceContainer extends Instance {
callback.accept(chunk); callback.accept(chunk);
} }
UPDATE_MANAGER.signalChunkLoad(this, chunkX, chunkZ);
callChunkLoadEvent(chunkX, chunkZ); callChunkLoadEvent(chunkX, chunkZ);
} }

View File

@ -60,6 +60,22 @@ public final class ConnectionManager {
return null; return null;
} }
/**
* Get the first player which validate {@link UUID#equals(Object)}
* <p>
* This can cause issue if two or more players have the same UUID
*
* @param uuid the player UUID
* @return the first player who validate the UUID condition
*/
public Player getPlayer(UUID uuid) {
for (Player player : getOnlinePlayers()) {
if (player.getUuid().equals(uuid))
return player;
}
return null;
}
/** /**
* Send a rich message to all online players who validate the condition {@code condition} * Send a rich message to all online players who validate the condition {@code condition}
* *

View File

@ -2,6 +2,7 @@ package net.minestom.server.thread;
import net.minestom.server.instance.Chunk; import net.minestom.server.instance.Chunk;
import net.minestom.server.instance.Instance; import net.minestom.server.instance.Instance;
import net.minestom.server.utils.chunk.ChunkUtils;
import java.util.HashMap; import java.util.HashMap;
import java.util.HashSet; import java.util.HashSet;
@ -13,66 +14,66 @@ import java.util.Set;
* <p> * <p>
* (1 chunks group = 1 thread execution) * (1 chunks group = 1 thread execution)
*/ */
// FIXME: unusable at the moment, too much overhead because groups need to be created every tick // TODO
// Should have a callback for when a chunk is loaded and unloaded, so groups are updated only once
public class PerGroupChunkProvider extends ThreadProvider { public class PerGroupChunkProvider extends ThreadProvider {
/** /**
* Here are stored all cached chunks waiting for a ChunkGroup * Here are stored all cached chunks waiting for a ChunkGroup
*/ */
private Map<Chunk, Set<Chunk>> cachedChunks = new HashMap<>(); private Map<Chunk, Set<ChunkCoordinate>> cachedChunks = new HashMap<>();
/** /**
* Used to know to which instance is linked a Set of chunks * Used to know to which instance is linked a Set of chunks
*/ */
private Map<Set<Chunk>, Instance> instanceMap = new HashMap<>(); private Map<Set<ChunkCoordinate>, Instance> instanceMap = new HashMap<>();
@Override @Override
public void start() { public void onChunkLoad(Instance instance, int chunkX, int chunkZ) {
this.cachedChunks.clear();
this.instanceMap.clear();
}
@Override
public void linkThread(Instance instance, Chunk chunk) {
startChunkQuery(instance, chunk.getChunkX(), chunk.getChunkZ());
}
@Override
public void end() {
} }
@Override @Override
public void update() { public void onChunkUnload(Instance instance, int chunkX, int chunkZ) {
// The time of the tick
final long time = System.currentTimeMillis(); }
@Override
public void update(long time) {
// Set of already-updated instances // Set of already-updated instances
final Set<Instance> updatedInstance = new HashSet<>(); final Set<Instance> updatedInstance = new HashSet<>();
// Update all the chunks // Update all the chunks
for (Map.Entry<Set<Chunk>, Instance> entry : instanceMap.entrySet()) { for (Map.Entry<Set<ChunkCoordinate>, Instance> entry : instanceMap.entrySet()) {
Set<Chunk> chunks = entry.getKey(); Set<ChunkCoordinate> chunks = entry.getKey();
Instance instance = entry.getValue(); Instance instance = entry.getValue();
final boolean updateInstance = updatedInstance.add(instance); final boolean updateInstance = updatedInstance.add(instance);
pool.execute(() -> { pool.execute(() -> {
/*if (updateInstance) { if (updateInstance) {
updateInstance(instance, time); updateInstance(instance, time);
} }
for (Chunk chunk : chunks) { for (ChunkCoordinate chunkCoordinate : chunks) {
final Chunk chunk = instance.getChunk(chunkCoordinate.chunkX, chunkCoordinate.chunkZ);
if (ChunkUtils.isChunkUnloaded(chunk)) {
continue;
}
updateChunk(instance, chunk, time); updateChunk(instance, chunk, time);
updateEntities(instance, chunk, time); updateEntities(instance, chunk, time);
}*/ }
}); });
} }
} }
/*@Override
public void linkThread(Instance instance, Chunk chunk) {
startChunkQuery(instance, chunk.getChunkX(), chunk.getChunkZ());
}*/
/** /**
* Check the four chunk neighbors (up/down/left/right) * Check the four chunk neighbors (up/down/left/right)
* and add them to the cache list * and add them to the cache list
@ -81,7 +82,7 @@ public class PerGroupChunkProvider extends ThreadProvider {
* @param chunkX the chunk X * @param chunkX the chunk X
* @param chunkZ the chunk Z * @param chunkZ the chunk Z
*/ */
private void startChunkQuery(Instance instance, int chunkX, int chunkZ) { /*private void startChunkQuery(Instance instance, int chunkX, int chunkZ) {
// Constants used to loop through the neighbors // Constants used to loop through the neighbors
final int[] posX = {1, 0, -1}; final int[] posX = {1, 0, -1};
final int[] posZ = {1, 0, -1}; final int[] posZ = {1, 0, -1};
@ -124,6 +125,5 @@ public class PerGroupChunkProvider extends ThreadProvider {
this.cachedChunks.put(cachedChunk, cache); this.cachedChunks.put(cachedChunk, cache);
} }
this.instanceMap.put(cache, instance); this.instanceMap.put(cache, instance);
} }*/
} }

View File

@ -2,96 +2,56 @@ package net.minestom.server.thread;
import net.minestom.server.instance.Chunk; import net.minestom.server.instance.Chunk;
import net.minestom.server.instance.Instance; import net.minestom.server.instance.Instance;
import net.minestom.server.utils.chunk.ChunkUtils;
import java.util.*; import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
/** /**
* Separate work between instance (1 instance = 1 thread execution) * Separate work between instance (1 instance = 1 thread execution)
*/ */
public class PerInstanceThreadProvider extends ThreadProvider { public class PerInstanceThreadProvider extends ThreadProvider {
private Map<Instance, GroupedInstanceChunk> groupMap = new HashMap<>(); private Map<Instance, Set<ChunkCoordinate>> instanceChunkMap = new HashMap<>();
@Override @Override
public void start() { public void onChunkLoad(Instance instance, int chunkX, int chunkZ) {
this.groupMap.clear(); Set<ChunkCoordinate> chunkCoordinates = instanceChunkMap.computeIfAbsent(instance, inst -> new HashSet<>());
chunkCoordinates.add(new ChunkCoordinate(chunkX, chunkZ));
} }
@Override @Override
public void linkThread(Instance instance, Chunk chunk) { public void onChunkUnload(Instance instance, int chunkX, int chunkZ) {
InstanceChunk instanceChunk = new InstanceChunk(instance, chunk); Set<ChunkCoordinate> chunkCoordinates = instanceChunkMap.computeIfAbsent(instance, inst -> new HashSet<>());
GroupedInstanceChunk groupedInstanceChunk = groupMap.computeIfAbsent(instance, inst -> new GroupedInstanceChunk()); chunkCoordinates.removeIf(chunkCoordinate -> chunkCoordinate.chunkX == chunkX &&
groupedInstanceChunk.instanceChunks.add(instanceChunk); chunkCoordinate.chunkZ == chunkZ);
}
@Override
public void end() {
} }
@Override @Override
public void update() { public void update(long time) {
final long time = System.currentTimeMillis(); for (Map.Entry<Instance, Set<ChunkCoordinate>> entry : instanceChunkMap.entrySet()) {
for (Map.Entry<Instance, GroupedInstanceChunk> entry : groupMap.entrySet()) {
final Instance instance = entry.getKey(); final Instance instance = entry.getKey();
final GroupedInstanceChunk groupedInstanceChunk = entry.getValue(); final Set<ChunkCoordinate> chunkCoordinates = entry.getValue();
pool.execute(() -> { pool.execute(() -> {
updateInstance(instance, time); updateInstance(instance, time);
for (InstanceChunk instanceChunk : groupedInstanceChunk.instanceChunks) { for (ChunkCoordinate chunkCoordinate : chunkCoordinates) {
Chunk chunk = instanceChunk.getChunk(); final Chunk chunk = instance.getChunk(chunkCoordinate.chunkX, chunkCoordinate.chunkZ);
if (ChunkUtils.isChunkUnloaded(chunk))
continue;
updateChunk(instance, chunk, time); updateChunk(instance, chunk, time);
updateEntities(instance, chunk, time); updateEntities(instance, chunk, time);
} }
}); });
}
}
/**
* Contains a list of {@link InstanceChunk}
*/
private static class GroupedInstanceChunk {
private List<InstanceChunk> instanceChunks = new ArrayList<>();
}
/**
* Contains both a chunk and its instance
*/
private static class InstanceChunk {
private Instance instance;
private Chunk chunk;
protected InstanceChunk(Instance instance, Chunk chunk) {
this.instance = instance;
this.chunk = chunk;
}
public Instance getInstance() {
return instance;
}
public Chunk getChunk() {
return chunk;
}
@Override
public boolean equals(Object o) {
if (this == o) return true;
if (o == null || getClass() != o.getClass()) return false;
InstanceChunk that = (InstanceChunk) o;
return Objects.equals(instance, that.instance) &&
Objects.equals(chunk, that.chunk);
}
@Override
public int hashCode() {
return Objects.hash(instance, chunk);
} }
} }

View File

@ -1,41 +0,0 @@
package net.minestom.server.thread;
import net.minestom.server.instance.Chunk;
import net.minestom.server.instance.Instance;
import java.util.HashSet;
import java.util.Set;
public class SingleThreadProvider extends ThreadProvider {
private Set<Instance> instances = new HashSet<>();
private long time;
@Override
public void start() {
this.instances.clear();
this.time = System.currentTimeMillis();
}
@Override
public void linkThread(Instance instance, Chunk chunk) {
if (instances.add(instance)) {
updateInstance(instance, time);
}
updateChunk(instance, chunk, time);
updateEntities(instance, chunk, time);
}
@Override
public void end() {
}
@Override
public void update() {
}
}

View File

@ -31,27 +31,29 @@ public abstract class ThreadProvider {
} }
/** /**
* Called to prepare the thread provider, to provide threads for the next server tick * Called when a chunk is loaded
*/
public abstract void start();
/**
* Assign a thread to a chunk, create one if none is defined
* *
* @param instance the instance where the chunk is * @param instance the instance of the chunk
* @param chunk the chunk which should get an assigned thread * @param chunkX the chunk X
* @param chunkZ the chunk Z
*/ */
public abstract void linkThread(Instance instance, Chunk chunk); public abstract void onChunkLoad(Instance instance, int chunkX, int chunkZ);
/** /**
* Inform the server that all chunks have been assigned to a thread * Called when a chunk is unloaded
*
* @param instance the instance of the chunk
* @param chunkX the chunk X
* @param chunkZ the chunk Z
*/ */
public abstract void end(); public abstract void onChunkUnload(Instance instance, int chunkX, int chunkZ);
/** /**
* Perform a server tick for all chunks based on their linked thread * Perform a server tick for all chunks based on their linked thread
*
* @param time the update time in milliseconds
*/ */
public abstract void update(); public abstract void update(long time);
/** /**
* Get the current size of the thread pool * Get the current size of the thread pool
@ -180,4 +182,13 @@ public abstract class ThreadProvider {
} }
} }
protected static class ChunkCoordinate {
public int chunkX, chunkZ;
public ChunkCoordinate(int chunkX, int chunkZ) {
this.chunkX = chunkX;
this.chunkZ = chunkZ;
}
}
} }