geforkt von Mirrors/Paper
157 Zeilen
7.7 KiB
Diff
157 Zeilen
7.7 KiB
Diff
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||
|
From: Aikar <aikar@aikar.co>
|
||
|
Date: Sat, 12 Sep 2020 22:20:55 -0400
|
||
|
Subject: [PATCH] Only consider chunks Loaded if at BORDER status
|
||
|
|
||
|
This greatly improves performance as it drastically reduces the amount
|
||
|
of Entities and Tile Entities that are "registered" into the world, as
|
||
|
purely "cached" chunks will no longer have their entities hanging out in the world.
|
||
|
|
||
|
Additionally this fixes our Entity Add To World and Entity Remove From World events
|
||
|
|
||
|
Those events have not been firing correctly since MC changed how chunks work here.
|
||
|
|
||
|
Now the server will only consider a chunk "loaded" if it's got a ticket putting
|
||
|
it at level 33 or lower, which matches the public Bukkit API.
|
||
|
|
||
|
diff --git a/src/main/java/net/minecraft/server/Chunk.java b/src/main/java/net/minecraft/server/Chunk.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/Chunk.java
|
||
|
+++ b/src/main/java/net/minecraft/server/Chunk.java
|
||
|
@@ -0,0 +0,0 @@ public class Chunk implements IChunkAccess {
|
||
|
int chunkX = this.loc.x;
|
||
|
int chunkZ = this.loc.z;
|
||
|
ChunkProviderServer chunkProvider = ((WorldServer)this.world).getChunkProvider();
|
||
|
+ chunkProvider.playerChunkMap.loadChunk(this); // Paper - move load logic from the entering full status to when it enters border status instead
|
||
|
for (int dx = -NEIGHBOUR_CACHE_RADIUS; dx <= NEIGHBOUR_CACHE_RADIUS; ++dx) {
|
||
|
for (int dz = -NEIGHBOUR_CACHE_RADIUS; dz <= NEIGHBOUR_CACHE_RADIUS; ++dz) {
|
||
|
Chunk neighbour = chunkProvider.getChunkAtIfLoadedMainThreadNoCache(chunkX + dx, chunkZ + dz);
|
||
|
@@ -0,0 +0,0 @@ public class Chunk implements IChunkAccess {
|
||
|
this.loadedTicketLevel = false;
|
||
|
this.resetNeighbours();
|
||
|
// Paper end
|
||
|
+ // Paper start - move unload logic from the actual full unload to be when it leaves border status
|
||
|
+ chunkProvider.playerChunkMap.loadedChunks.remove(loc.longKey);
|
||
|
+ setLoaded(false);
|
||
|
+ this.world.unloadChunk(this);
|
||
|
+ // Paper end
|
||
|
}
|
||
|
// CraftBukkit end
|
||
|
|
||
|
diff --git a/src/main/java/net/minecraft/server/ChunkProviderServer.java b/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
||
|
+++ b/src/main/java/net/minecraft/server/ChunkProviderServer.java
|
||
|
@@ -0,0 +0,0 @@ import java.util.function.Supplier;
|
||
|
import javax.annotation.Nullable;
|
||
|
import com.destroystokyo.paper.exception.ServerInternalException;
|
||
|
import it.unimi.dsi.fastutil.longs.Long2ObjectOpenHashMap; // Paper
|
||
|
+import it.unimi.dsi.fastutil.longs.LongIterator; // Paper
|
||
|
import org.apache.logging.log4j.LogManager;
|
||
|
import org.apache.logging.log4j.Logger;
|
||
|
|
||
|
@@ -0,0 +0,0 @@ public class ChunkProviderServer extends IChunkProvider {
|
||
|
//List<PlayerChunk> list = Lists.newArrayList(this.playerChunkMap.f()); // Paper
|
||
|
//Collections.shuffle(list); // Paper
|
||
|
// Paper - moved up
|
||
|
- final int[] chunksTicked = {0}; this.playerChunkMap.forEachVisibleChunk((playerchunk) -> { // Paper - safe iterator incase chunk loads, also no wrapping
|
||
|
+ final int[] chunksTicked = {0};
|
||
|
+ this.playerChunkMap.forEachVisibleChunk((playerchunk) -> {
|
||
|
+ //for (LongIterator iterator = this.playerChunkMap.loadedChunks.iterator() ; iterator.hasNext() ; ) { // Paper - iterate only loaded chunks
|
||
|
+ //PlayerChunk playerchunk = this.playerChunkMap.getVisibleChunk(iterator.nextLong());// Paper - iterate only loaded chunks
|
||
|
Optional<Chunk> optional = ((Either) playerchunk.a().getNow(PlayerChunk.UNLOADED_CHUNK)).left();
|
||
|
|
||
|
if (optional.isPresent()) {
|
||
|
@@ -0,0 +0,0 @@ public class ChunkProviderServer extends IChunkProvider {
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
- });
|
||
|
+ });// Paper - use for instead of forEachVisibleChunk
|
||
|
this.world.getMethodProfiler().enter("customSpawners");
|
||
|
if (flag1) {
|
||
|
try (co.aikar.timings.Timing ignored = this.world.timings.miscMobSpawning.startTiming()) { // Paper - timings
|
||
|
diff --git a/src/main/java/net/minecraft/server/PlayerChunkMap.java b/src/main/java/net/minecraft/server/PlayerChunkMap.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/PlayerChunkMap.java
|
||
|
+++ b/src/main/java/net/minecraft/server/PlayerChunkMap.java
|
||
|
@@ -0,0 +0,0 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
||
|
this.a(i, playerchunk);
|
||
|
} else {
|
||
|
if (this.pendingUnload.remove(i, playerchunk) && ichunkaccess != null) {
|
||
|
+ // Paper start - coment out and move to ChunkUnloadEvent
|
||
|
if (ichunkaccess instanceof Chunk) {
|
||
|
- ((Chunk) ichunkaccess).setLoaded(false);
|
||
|
+ //((Chunk) ichunkaccess).setLoaded(false);
|
||
|
}
|
||
|
|
||
|
//this.saveChunk(ichunkaccess);// Paper - delay
|
||
|
- if (this.loadedChunks.remove(i) && ichunkaccess instanceof Chunk) {
|
||
|
- Chunk chunk = (Chunk) ichunkaccess;
|
||
|
-
|
||
|
- this.world.unloadChunk(chunk);
|
||
|
- }
|
||
|
+// if (this.loadedChunks.remove(i) && ichunkaccess instanceof Chunk) {
|
||
|
+// Chunk chunk = (Chunk) ichunkaccess;
|
||
|
+//
|
||
|
+// this.world.unloadChunk(chunk);
|
||
|
+// }
|
||
|
+ // Paper end
|
||
|
this.autoSaveQueue.remove(playerchunk); // Paper
|
||
|
|
||
|
try {
|
||
|
@@ -0,0 +0,0 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
||
|
return PlayerChunk.getChunkState(playerchunk.getTicketLevel());
|
||
|
});
|
||
|
chunk.addEntities();
|
||
|
+ // Paper start
|
||
|
+ return chunk;
|
||
|
+ } // Paper
|
||
|
+ });
|
||
|
+ }, (runnable) -> {
|
||
|
+
|
||
|
+ this.mailboxMain.a(ChunkTaskQueueSorter.a(runnable, playerchunk.i().pair(), () -> 1)); // Paper - final loads are always urgent!
|
||
|
+ });
|
||
|
+ }
|
||
|
+ public void loadChunk(Chunk chunk) {
|
||
|
+ if (true) {
|
||
|
+ if (true) {
|
||
|
+ ChunkCoordIntPair chunkcoordintpair = chunk.getPos();
|
||
|
+ // Paper end
|
||
|
if (this.loadedChunks.add(chunkcoordintpair.pair())) {
|
||
|
chunk.setLoaded(true);
|
||
|
this.world.a(chunk.getTileEntities().values());
|
||
|
@@ -0,0 +0,0 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
|
||
|
list.forEach(chunk::b);
|
||
|
}
|
||
|
}
|
||
|
-
|
||
|
- return chunk;
|
||
|
- } // Paper
|
||
|
- });
|
||
|
- }, (runnable) -> {
|
||
|
- Mailbox mailbox = this.mailboxMain;
|
||
|
- long i = playerchunk.i().pair();
|
||
|
-
|
||
|
- playerchunk.getClass();
|
||
|
- mailbox.a(ChunkTaskQueueSorter.a(runnable, i, () -> 1)); // Paper - final loads are always urgent!
|
||
|
- });
|
||
|
+ // Paper start
|
||
|
+ }
|
||
|
+ }
|
||
|
+ // Move code up
|
||
|
+// return chunk;
|
||
|
+// } // Paper
|
||
|
+// });
|
||
|
+// }, (runnable) -> {
|
||
|
+// Mailbox mailbox = this.mailboxMain;
|
||
|
+// long i = playerchunk.i().pair();
|
||
|
+//
|
||
|
+// playerchunk.getClass();
|
||
|
+// mailbox.a(ChunkTaskQueueSorter.a(runnable, i, () -> 1)); // Paper - final loads are always urgent!
|
||
|
+// });
|
||
|
+ // Paper end
|
||
|
}
|
||
|
|
||
|
// Paper start
|