SPIGOT-6772: Use entity mailbox and re-schedule entities if they get unloaded

By: DerFrZocker <derrieple@gmail.com>
This commit is contained in:
CraftBukkit/Spigot 2021-11-01 18:54:44 +11:00
parent 1f5735df63
commit 9e3a0e859f
3 changed files with 59 additions and 24 deletions

View file

@ -1,6 +1,6 @@
--- a/net/minecraft/world/level/chunk/storage/EntityStorage.java --- a/net/minecraft/world/level/chunk/storage/EntityStorage.java
+++ b/net/minecraft/world/level/chunk/storage/EntityStorage.java +++ b/net/minecraft/world/level/chunk/storage/EntityStorage.java
@@ -33,7 +33,7 @@ @@ -33,10 +33,10 @@
private static final Logger LOGGER = LogManager.getLogger(); private static final Logger LOGGER = LogManager.getLogger();
private static final String ENTITIES_TAG = "Entities"; private static final String ENTITIES_TAG = "Entities";
private static final String POSITION_TAG = "Position"; private static final String POSITION_TAG = "Position";
@ -8,7 +8,11 @@
+ public final WorldServer level; // PAIL private -> public + public final WorldServer level; // PAIL private -> public
private final IOWorker worker; private final IOWorker worker;
private final LongSet emptyChunks = new LongOpenHashSet(); private final LongSet emptyChunks = new LongOpenHashSet();
private final ThreadedMailbox<Runnable> entityDeserializerQueue; - private final ThreadedMailbox<Runnable> entityDeserializerQueue;
+ public final ThreadedMailbox<Runnable> entityDeserializerQueue; // PAIL private -> public
protected final DataFixer fixerUpper;
public EntityStorage(WorldServer worldserver, File file, DataFixer datafixer, boolean flag, Executor executor) {
@@ -51,8 +51,8 @@ @@ -51,8 +51,8 @@
if (this.emptyChunks.contains(chunkcoordintpair.pair())) { if (this.emptyChunks.contains(chunkcoordintpair.pair())) {
return CompletableFuture.completedFuture(b(chunkcoordintpair)); return CompletableFuture.completedFuture(b(chunkcoordintpair));

View file

@ -1,6 +1,6 @@
--- a/net/minecraft/world/level/entity/PersistentEntitySectionManager.java --- a/net/minecraft/world/level/entity/PersistentEntitySectionManager.java
+++ b/net/minecraft/world/level/entity/PersistentEntitySectionManager.java +++ b/net/minecraft/world/level/entity/PersistentEntitySectionManager.java
@@ -32,6 +32,11 @@ @@ -32,12 +32,17 @@
import org.apache.logging.log4j.LogManager; import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger; import org.apache.logging.log4j.Logger;
@ -12,20 +12,31 @@
public class PersistentEntitySectionManager<T extends EntityAccess> implements AutoCloseable { public class PersistentEntitySectionManager<T extends EntityAccess> implements AutoCloseable {
static final Logger LOGGER = LogManager.getLogger(); static final Logger LOGGER = LogManager.getLogger();
@@ -55,6 +60,12 @@ final Set<UUID> knownUuids = Sets.newHashSet();
final LevelCallback<T> callbacks;
- private final EntityPersistentStorage<T> permanentStorage;
+ public final EntityPersistentStorage<T> permanentStorage; // PAIL private -> public
private final EntityLookup<T> visibleEntityStorage = new EntityLookup<>();
final EntitySectionStorage<T> sectionStorage;
private final LevelEntityGetter<T> entityGetter;
@@ -55,6 +60,16 @@
this.entityGetter = new LevelEntityGetterAdapter<>(this.visibleEntityStorage, this.sectionStorage); this.entityGetter = new LevelEntityGetterAdapter<>(this.visibleEntityStorage, this.sectionStorage);
} }
+ // CraftBukkit start - add method to get all entities in chunk + // CraftBukkit start - add method to get all entities in chunk
+ public List<Entity> getEntities(ChunkCoordIntPair chunkCoordIntPair) { + public List<Entity> getEntities(ChunkCoordIntPair chunkCoordIntPair) {
+ return sectionStorage.b(chunkCoordIntPair.pair()).flatMap(EntitySection::b).map(entiy -> (Entity) entiy).collect(Collectors.toList()); + return sectionStorage.b(chunkCoordIntPair.pair()).flatMap(EntitySection::b).map(entity -> (Entity) entity).collect(Collectors.toList());
+ }
+
+ public boolean isPending(long pair) {
+ return chunkLoadStatuses.get(pair) == b.PENDING;
+ } + }
+ // CraftBukkit end + // CraftBukkit end
+ +
void a(long i, EntitySection<T> entitysection) { void a(long i, EntitySection<T> entitysection) {
if (entitysection.a()) { if (entitysection.a()) {
this.sectionStorage.e(i); this.sectionStorage.e(i);
@@ -82,7 +93,7 @@ @@ -82,7 +97,7 @@
long i = SectionPosition.c(t0.getChunkCoordinates()); long i = SectionPosition.c(t0.getChunkCoordinates());
EntitySection<T> entitysection = this.sectionStorage.c(i); EntitySection<T> entitysection = this.sectionStorage.c(i);
@ -34,7 +45,7 @@
t0.a(new PersistentEntitySectionManager.a(t0, i, entitysection)); t0.a(new PersistentEntitySectionManager.a(t0, i, entitysection));
if (!flag) { if (!flag) {
this.callbacks.f(t0); this.callbacks.f(t0);
@@ -186,7 +197,7 @@ @@ -186,7 +201,7 @@
}); });
} }
@ -43,7 +54,7 @@
PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i); PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i);
if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.FRESH) { if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.FRESH) {
@@ -196,6 +207,12 @@ @@ -196,6 +211,12 @@
} }
private boolean a(long i, Consumer<T> consumer) { private boolean a(long i, Consumer<T> consumer) {
@ -56,7 +67,7 @@
PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i); PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i);
if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.PENDING) { if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.PENDING) {
@@ -207,6 +224,7 @@ @@ -207,6 +228,7 @@
if (list.isEmpty()) { if (list.isEmpty()) {
if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.LOADED) { if (persistententitysectionmanager_b == PersistentEntitySectionManager.b.LOADED) {
@ -64,7 +75,7 @@
this.permanentStorage.a(new ChunkEntities<>(new ChunkCoordIntPair(i), ImmutableList.of())); this.permanentStorage.a(new ChunkEntities<>(new ChunkCoordIntPair(i), ImmutableList.of()));
} }
@@ -215,6 +233,7 @@ @@ -215,6 +237,7 @@
this.c(i); this.c(i);
return false; return false;
} else { } else {
@ -72,7 +83,7 @@
this.permanentStorage.a(new ChunkEntities<>(new ChunkCoordIntPair(i), list)); this.permanentStorage.a(new ChunkEntities<>(new ChunkCoordIntPair(i), list));
list.forEach(consumer); list.forEach(consumer);
return true; return true;
@@ -238,7 +257,7 @@ @@ -238,7 +261,7 @@
private boolean d(long i) { private boolean d(long i) {
boolean flag = this.a(i, (entityaccess) -> { boolean flag = this.a(i, (entityaccess) -> {
entityaccess.cD().forEach(this::g); entityaccess.cD().forEach(this::g);
@ -81,7 +92,7 @@
if (!flag) { if (!flag) {
return false; return false;
@@ -254,19 +273,23 @@ @@ -254,19 +277,23 @@
} }
private void f() { private void f() {
@ -107,7 +118,7 @@
} }
} }
@@ -292,7 +315,7 @@ @@ -292,7 +319,7 @@
} }
public void b() { public void b() {
@ -116,7 +127,7 @@
boolean flag = this.chunkVisibility.get(i) == Visibility.HIDDEN; boolean flag = this.chunkVisibility.get(i) == Visibility.HIDDEN;
if (flag) { if (flag) {
@@ -311,7 +334,7 @@ @@ -311,7 +338,7 @@
while (!longset.isEmpty()) { while (!longset.isEmpty()) {
this.permanentStorage.a(false); this.permanentStorage.a(false);
this.g(); this.g();
@ -125,7 +136,7 @@
boolean flag = this.chunkVisibility.get(i) == Visibility.HIDDEN; boolean flag = this.chunkVisibility.get(i) == Visibility.HIDDEN;
return flag ? this.d(i) : this.a(i, (entityaccess) -> { return flag ? this.d(i) : this.a(i, (entityaccess) -> {
@@ -323,7 +346,15 @@ @@ -323,7 +350,15 @@
} }
public void close() throws IOException { public void close() throws IOException {
@ -142,7 +153,7 @@
this.permanentStorage.close(); this.permanentStorage.close();
} }
@@ -350,7 +381,7 @@ @@ -350,7 +385,7 @@
public void a(Writer writer) throws IOException { public void a(Writer writer) throws IOException {
CSVWriter csvwriter = CSVWriter.a().a("x").a("y").a("z").a("visibility").a("load_status").a("entity_count").a(writer); CSVWriter csvwriter = CSVWriter.a().a("x").a("y").a("z").a("visibility").a("load_status").a("entity_count").a(writer);
@ -151,7 +162,7 @@
PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i); PersistentEntitySectionManager.b persistententitysectionmanager_b = (PersistentEntitySectionManager.b) this.chunkLoadStatuses.get(i);
this.sectionStorage.a(i).forEach((j) -> { this.sectionStorage.a(i).forEach((j) -> {
@@ -389,7 +420,7 @@ @@ -389,7 +424,7 @@
private EntitySection<T> currentSection; private EntitySection<T> currentSection;
a(EntityAccess entityaccess, long i, EntitySection entitysection) { a(EntityAccess entityaccess, long i, EntitySection entitysection) {
@ -160,7 +171,7 @@
this.currentSectionKey = i; this.currentSectionKey = i;
this.currentSection = entitysection; this.currentSection = entitysection;
} }
@@ -409,7 +440,7 @@ @@ -409,7 +444,7 @@
PersistentEntitySectionManager.this.a(this.currentSectionKey, this.currentSection); PersistentEntitySectionManager.this.a(this.currentSectionKey, this.currentSection);
EntitySection<T> entitysection = PersistentEntitySectionManager.this.sectionStorage.c(i); EntitySection<T> entitysection = PersistentEntitySectionManager.this.sectionStorage.c(i);

View file

@ -6,6 +6,8 @@ import java.lang.ref.WeakReference;
import java.util.Arrays; import java.util.Arrays;
import java.util.Collection; import java.util.Collection;
import java.util.Objects; import java.util.Objects;
import java.util.concurrent.locks.LockSupport;
import java.util.function.BooleanSupplier;
import java.util.function.Predicate; import java.util.function.Predicate;
import net.minecraft.core.BlockPosition; import net.minecraft.core.BlockPosition;
import net.minecraft.core.IRegistry; import net.minecraft.core.IRegistry;
@ -13,6 +15,7 @@ import net.minecraft.core.SectionPosition;
import net.minecraft.nbt.GameProfileSerializer; import net.minecraft.nbt.GameProfileSerializer;
import net.minecraft.nbt.NBTTagCompound; import net.minecraft.nbt.NBTTagCompound;
import net.minecraft.server.level.WorldServer; import net.minecraft.server.level.WorldServer;
import net.minecraft.util.thread.ThreadedMailbox;
import net.minecraft.world.level.ChunkCoordIntPair; import net.minecraft.world.level.ChunkCoordIntPair;
import net.minecraft.world.level.EnumSkyBlock; import net.minecraft.world.level.EnumSkyBlock;
import net.minecraft.world.level.biome.WorldChunkManager; import net.minecraft.world.level.biome.WorldChunkManager;
@ -24,6 +27,7 @@ import net.minecraft.world.level.chunk.ChunkStatus;
import net.minecraft.world.level.chunk.DataPaletteBlock; import net.minecraft.world.level.chunk.DataPaletteBlock;
import net.minecraft.world.level.chunk.IChunkAccess; import net.minecraft.world.level.chunk.IChunkAccess;
import net.minecraft.world.level.chunk.NibbleArray; import net.minecraft.world.level.chunk.NibbleArray;
import net.minecraft.world.level.chunk.storage.EntityStorage;
import net.minecraft.world.level.entity.PersistentEntitySectionManager; import net.minecraft.world.level.entity.PersistentEntitySectionManager;
import net.minecraft.world.level.levelgen.HeightMap; import net.minecraft.world.level.levelgen.HeightMap;
import net.minecraft.world.level.levelgen.SeededRandom; import net.minecraft.world.level.levelgen.SeededRandom;
@ -133,20 +137,36 @@ public class CraftChunk implements Chunk {
entityManager.b(pair); // Start entity loading entityManager.b(pair); // Start entity loading
// now we wait until the entities are loaded, // SPIGOT-6772: Use entity mailbox and re-schedule entities if they get unloaded
// the converting from NBT to entity object is done on the main Thread which is why we wait ThreadedMailbox<Runnable> mailbox = ((EntityStorage) entityManager.permanentStorage).entityDeserializerQueue;
getCraftWorld().getHandle().getMinecraftServer().awaitTasks(() -> { BooleanSupplier supplier = () -> {
boolean status = entityManager.a(pair);
// only execute inbox if our entities are not present // only execute inbox if our entities are not present
if (status) { if (entityManager.a(pair)) {
return true; return true;
} }
if (!entityManager.isPending(pair)) {
// Our entities got unloaded, this should normally not happen.
entityManager.b(pair); // Re-start entity loading
}
// tick loading inbox, which loads the created entities to the world // tick loading inbox, which loads the created entities to the world
// (if present) // (if present)
entityManager.tick(); entityManager.tick();
// check if our entities are loaded // check if our entities are loaded
return entityManager.a(pair); return entityManager.a(pair);
}); };
// now we wait until the entities are loaded,
// the converting from NBT to entity object is done on the main Thread which is why we wait
while (!supplier.getAsBoolean()) {
if (mailbox.b() != 0) { // PAIL rename size
mailbox.run();
} else {
Thread.yield();
LockSupport.parkNanos("waiting for entity loading", 100000L);
}
}
return entityManager.getEntities(new ChunkCoordIntPair(x, z)).stream() return entityManager.getEntities(new ChunkCoordIntPair(x, z)).stream()
.map(net.minecraft.world.entity.Entity::getBukkitEntity) .map(net.minecraft.world.entity.Entity::getBukkitEntity)