mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-07 16:57:42 +01:00
Optimize Loaded Chunks Cache Lookups
Reduces number of instructions a chunk lookup does when accessing the last chunk cache. This reduces amount of work and opcodes and allows better inlining. In lots of profiling comparisons, this optimization was able to reduce the cost of repeated chunk lookups that hit the cache pretty significantly.
This commit is contained in:
parent
442fa32443
commit
1105e1ec76
@ -2428,7 +2428,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
@Nullable
|
@Nullable
|
||||||
private Consumer<Chunk> v;
|
private Consumer<Chunk> v;
|
||||||
- private final ChunkCoordIntPair loc;
|
- private final ChunkCoordIntPair loc;
|
||||||
+ private final ChunkCoordIntPair loc; public final long coordinateKey; // Paper - cache coordinate key
|
+ private final ChunkCoordIntPair loc; public final long coordinateKey; public final int locX; public final int locZ; // Paper - cache coordinate key
|
||||||
private volatile boolean x;
|
private volatile boolean x;
|
||||||
|
|
||||||
public Chunk(World world, ChunkCoordIntPair chunkcoordintpair, BiomeStorage biomestorage) {
|
public Chunk(World world, ChunkCoordIntPair chunkcoordintpair, BiomeStorage biomestorage) {
|
||||||
@ -2437,7 +2437,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
this.entitySlices = (List[]) (new List[16]); // Spigot
|
this.entitySlices = (List[]) (new List[16]); // Spigot
|
||||||
this.world = (WorldServer) world; // CraftBukkit - type
|
this.world = (WorldServer) world; // CraftBukkit - type
|
||||||
- this.loc = chunkcoordintpair;
|
- this.loc = chunkcoordintpair;
|
||||||
+ this.loc = chunkcoordintpair; this.coordinateKey = MCUtil.getCoordinateKey(chunkcoordintpair); // Paper - cache coordinate key
|
+ this.locX = chunkcoordintpair.x; this.locZ = chunkcoordintpair.z; // Paper - reduce need for field look ups
|
||||||
|
+ this.loc = chunkcoordintpair; this.coordinateKey = ChunkCoordIntPair.pair(locX, locZ); // Paper - cache long key
|
||||||
this.i = chunkconverter;
|
this.i = chunkconverter;
|
||||||
HeightMap.Type[] aheightmap_type = HeightMap.Type.values();
|
HeightMap.Type[] aheightmap_type = HeightMap.Type.values();
|
||||||
int j = aheightmap_type.length;
|
int j = aheightmap_type.length;
|
||||||
@ -2777,13 +2778,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
+ final Long2ObjectOpenHashMap<Chunk> loadedChunkMap = new Long2ObjectOpenHashMap<>(8192, 0.5f);
|
+ final Long2ObjectOpenHashMap<Chunk> loadedChunkMap = new Long2ObjectOpenHashMap<>(8192, 0.5f);
|
||||||
+
|
+
|
||||||
+ private final Chunk[] lastLoadedChunks = new Chunk[4 * 4];
|
+ private final Chunk[] lastLoadedChunks = new Chunk[4 * 4];
|
||||||
+ private final long[] lastLoadedChunkKeys = new long[4 * 4];
|
|
||||||
+
|
+
|
||||||
+ {
|
+ private static int getChunkCacheKey(int x, int z) {
|
||||||
+ java.util.Arrays.fill(this.lastLoadedChunkKeys, MCUtil.INVALID_CHUNK_KEY);
|
|
||||||
+ }
|
|
||||||
+
|
|
||||||
+ private static int getCacheKey(int x, int z) {
|
|
||||||
+ return x & 3 | ((z & 3) << 2);
|
+ return x & 3 | ((z & 3) << 2);
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
@ -2797,12 +2793,9 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
+
|
+
|
||||||
+ // rewrite cache if we have to
|
+ // rewrite cache if we have to
|
||||||
+ // we do this since we also cache null chunks
|
+ // we do this since we also cache null chunks
|
||||||
+ int cacheKey = getCacheKey(chunk.getPos().x, chunk.getPos().z);
|
+ int cacheKey = getChunkCacheKey(chunk.locX, chunk.locZ);
|
||||||
+
|
+
|
||||||
+ long cachedKey = this.lastLoadedChunkKeys[cacheKey];
|
+ this.lastLoadedChunks[cacheKey] = chunk;
|
||||||
+ if (cachedKey == chunk.coordinateKey) {
|
|
||||||
+ this.lastLoadedChunks[cacheKey] = chunk;
|
|
||||||
+ }
|
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ void removeLoadedChunk(Chunk chunk) {
|
+ void removeLoadedChunk(Chunk chunk) {
|
||||||
@ -2815,36 +2808,35 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
+
|
+
|
||||||
+ // rewrite cache if we have to
|
+ // rewrite cache if we have to
|
||||||
+ // we do this since we also cache null chunks
|
+ // we do this since we also cache null chunks
|
||||||
+ int cacheKey = getCacheKey(chunk.getPos().x, chunk.getPos().z);
|
+ int cacheKey = getChunkCacheKey(chunk.locX, chunk.locZ);
|
||||||
+
|
+
|
||||||
+ long cachedKey = this.lastLoadedChunkKeys[cacheKey];
|
+ Chunk cachedChunk = this.lastLoadedChunks[cacheKey];
|
||||||
+ if (cachedKey == chunk.coordinateKey) {
|
+ if (cachedChunk != null && cachedChunk.coordinateKey == chunk.coordinateKey) {
|
||||||
+ this.lastLoadedChunks[cacheKey] = null;
|
+ this.lastLoadedChunks[cacheKey] = null;
|
||||||
+ }
|
+ }
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public Chunk getChunkAtIfLoadedMainThread(int x, int z) {
|
+ public final Chunk getChunkAtIfLoadedMainThread(int x, int z) {
|
||||||
+ int cacheKey = getCacheKey(x, z);
|
+ int cacheKey = getChunkCacheKey(x, z);
|
||||||
+ long chunkKey = MCUtil.getCoordinateKey(x, z);
|
|
||||||
+
|
+
|
||||||
+ long cachedKey = this.lastLoadedChunkKeys[cacheKey];
|
+ Chunk cachedChunk = this.lastLoadedChunks[cacheKey];
|
||||||
+ if (cachedKey == chunkKey) {
|
+ if (cachedChunk != null && cachedChunk.locX == x & cachedChunk.locZ == z) {
|
||||||
+ return this.lastLoadedChunks[cacheKey];
|
+ return this.lastLoadedChunks[cacheKey];
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ Chunk ret = this.loadedChunkMap.get(chunkKey);
|
+ long chunkKey = ChunkCoordIntPair.pair(x, z);
|
||||||
+
|
+
|
||||||
+ this.lastLoadedChunkKeys[cacheKey] = chunkKey;
|
+ cachedChunk = this.loadedChunkMap.get(chunkKey);
|
||||||
+ this.lastLoadedChunks[cacheKey] = ret;
|
+ // Skipping a null check to avoid extra instructions to improve inline capability
|
||||||
+
|
+ this.lastLoadedChunks[cacheKey] = cachedChunk;
|
||||||
+ return ret;
|
+ return cachedChunk;
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public Chunk getChunkAtIfLoadedMainThreadNoCache(int x, int z) {
|
+ public final Chunk getChunkAtIfLoadedMainThreadNoCache(int x, int z) {
|
||||||
+ return this.loadedChunkMap.get(MCUtil.getCoordinateKey(x, z));
|
+ return this.loadedChunkMap.get(ChunkCoordIntPair.pair(x, z));
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public Chunk getChunkAtMainThread(int x, int z) {
|
+ public final Chunk getChunkAtMainThread(int x, int z) {
|
||||||
+ Chunk ret = this.getChunkAtIfLoadedMainThread(x, z);
|
+ Chunk ret = this.getChunkAtIfLoadedMainThread(x, z);
|
||||||
+ if (ret != null) {
|
+ if (ret != null) {
|
||||||
+ return ret;
|
+ return ret;
|
||||||
@ -2889,7 +2881,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
+ throw new IllegalStateException();
|
+ throw new IllegalStateException();
|
||||||
+ }
|
+ }
|
||||||
+ ChunkCoordIntPair chunkPos = new ChunkCoordIntPair(x, z);
|
+ ChunkCoordIntPair chunkPos = new ChunkCoordIntPair(x, z);
|
||||||
+ Long identifier = Long.valueOf(this.chunkFutureAwaitCounter++);
|
+ Long identifier = this.chunkFutureAwaitCounter++;
|
||||||
+ this.chunkMapDistance.addTicketAtLevel(TicketType.FUTURE_AWAIT, chunkPos, ticketLevel, identifier);
|
+ this.chunkMapDistance.addTicketAtLevel(TicketType.FUTURE_AWAIT, chunkPos, ticketLevel, identifier);
|
||||||
+ this.tickDistanceManager();
|
+ this.tickDistanceManager();
|
||||||
+
|
+
|
||||||
@ -4487,6 +4479,15 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
|||||||
Fluid fluid = this.getFluid(blockposition);
|
Fluid fluid = this.getFluid(blockposition);
|
||||||
|
|
||||||
return this.setTypeAndData(blockposition, fluid.getBlockData(), 3 | (flag ? 64 : 0));
|
return this.setTypeAndData(blockposition, fluid.getBlockData(), 3 | (flag ? 64 : 0));
|
||||||
|
@@ -0,0 +0,0 @@ public abstract class World implements GeneratorAccess, AutoCloseable {
|
||||||
|
if (isOutsideWorld(blockposition)) {
|
||||||
|
return Blocks.VOID_AIR.getBlockData();
|
||||||
|
} else {
|
||||||
|
- Chunk chunk = this.getChunkAt(blockposition.getX() >> 4, blockposition.getZ() >> 4);
|
||||||
|
+ Chunk chunk = (Chunk) this.getChunkProvider().getChunkAt(blockposition.getX() >> 4, blockposition.getZ() >> 4, ChunkStatus.FULL, true); // Paper - manually inline to reduce hops and avoid unnecessary null check to reduce total byte code size, this should never return null and if it does we will see it the next line but the real stack trace will matter in the chunk engine
|
||||||
|
|
||||||
|
return chunk.getType(blockposition);
|
||||||
|
}
|
||||||
diff --git a/src/main/java/net/minecraft/server/WorldBorder.java b/src/main/java/net/minecraft/server/WorldBorder.java
|
diff --git a/src/main/java/net/minecraft/server/WorldBorder.java b/src/main/java/net/minecraft/server/WorldBorder.java
|
||||||
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||||||
--- a/src/main/java/net/minecraft/server/WorldBorder.java
|
--- a/src/main/java/net/minecraft/server/WorldBorder.java
|
||||||
|
Loading…
Reference in New Issue
Block a user