mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-20 07:34:48 +01:00
272 lines
14 KiB
Diff
272 lines
14 KiB
Diff
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||
|
From: Aikar <aikar@aikar.co>
|
||
|
Date: Thu, 9 Apr 2020 00:09:26 -0400
|
||
|
Subject: [PATCH] Mid Tick Chunk Tasks - Speed up processing of chunk loads and
|
||
|
generation
|
||
|
|
||
|
Credit to Spotted for the idea
|
||
|
|
||
|
A lot of the new chunk system requires constant back and forth the main thread
|
||
|
to handle priority scheduling and ensuring conflicting tasks do not run at the
|
||
|
same time.
|
||
|
|
||
|
The issue is, these queues are only checked at either:
|
||
|
|
||
|
A) Sync Chunk Loads
|
||
|
B) End of Tick while sleeping
|
||
|
|
||
|
This results in generating chunks sitting waiting for a full tick to
|
||
|
complete before it will even start the next unit of work to do.
|
||
|
|
||
|
Additionally, this also delays loading of chunks until this same timing.
|
||
|
|
||
|
We will now periodically poll the chunk task queues throughout the tick,
|
||
|
looking for work to do.
|
||
|
We do this in a fair method that considers all worlds, not just the one being
|
||
|
ticked, so that each world can get 1 task procesed each before the next pass.
|
||
|
|
||
|
In a view distance of 15, chunk loading performance was visually faster on the client.
|
||
|
|
||
|
Flying at high speed in spectator mode was able to keep up with chunk loading (as long as they are already generated)
|
||
|
|
||
|
diff --git a/src/main/java/co/aikar/timings/MinecraftTimings.java b/src/main/java/co/aikar/timings/MinecraftTimings.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/co/aikar/timings/MinecraftTimings.java
|
||
|
+++ b/src/main/java/co/aikar/timings/MinecraftTimings.java
|
||
|
@@ -0,0 +0,0 @@ import java.util.Map;
|
||
|
public final class MinecraftTimings {
|
||
|
|
||
|
public static final Timing serverOversleep = Timings.ofSafe("Server Oversleep");
|
||
|
+ public static final Timing midTickChunkTasks = Timings.ofSafe("Mid Tick Chunk Tasks");
|
||
|
public static final Timing playerListTimer = Timings.ofSafe("Player List");
|
||
|
public static final Timing commandFunctionsTimer = Timings.ofSafe("Command Functions");
|
||
|
public static final Timing connectionTimer = Timings.ofSafe("Connection Handler");
|
||
|
diff --git a/src/main/java/com/destroystokyo/paper/PaperConfig.java b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
||
|
+++ b/src/main/java/com/destroystokyo/paper/PaperConfig.java
|
||
|
@@ -0,0 +0,0 @@ public class PaperConfig {
|
||
|
log("Async Chunks: Enabled - Chunks will be loaded much faster, without lag.");
|
||
|
}
|
||
|
}
|
||
|
+
|
||
|
+ public static int midTickChunkTasks = 1000;
|
||
|
+ private static void midTickChunkTasks() {
|
||
|
+ midTickChunkTasks = getInt("settings.chunk-tasks-per-tick", midTickChunkTasks);
|
||
|
+ }
|
||
|
}
|
||
|
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
|
||
|
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
// Paper end
|
||
|
tickSection = curTime;
|
||
|
}
|
||
|
+ midTickChunksTasksRan = 0; // Paper
|
||
|
// Spigot end
|
||
|
|
||
|
//MinecraftServer.currentTick = (int) (System.currentTimeMillis() / 50); // CraftBukkit // Paper - don't overwrite current tick time
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
|
||
|
}
|
||
|
|
||
|
- private boolean haveTime() {
|
||
|
+ public boolean haveTime() { // Paper
|
||
|
// CraftBukkit start
|
||
|
if (isOversleep) return canOversleep();// Paper - because of our changes, this logic is broken
|
||
|
return this.forceTicks || this.runningTask() || Util.getMillis() < (this.mayHaveDelayedTasks ? this.delayedTasksMaxNextTickTime : this.nextTickTime);
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
});
|
||
|
}
|
||
|
|
||
|
+ // Paper start
|
||
|
+ public int midTickChunksTasksRan = 0;
|
||
|
+ private long midTickLastRan = 0;
|
||
|
+ public void midTickLoadChunks() {
|
||
|
+ if (!isSameThread() || System.nanoTime() - midTickLastRan < 1000000) {
|
||
|
+ // only check once per 0.25ms incase this code is called in a hot method
|
||
|
+ return;
|
||
|
+ }
|
||
|
+ try (co.aikar.timings.Timing ignored = co.aikar.timings.MinecraftTimings.midTickChunkTasks.startTiming()) {
|
||
|
+ for (ServerLevel value : this.getAllLevels()) {
|
||
|
+ value.getChunkSource().mainThreadProcessor.midTickLoadChunks();
|
||
|
+ }
|
||
|
+ midTickLastRan = System.nanoTime();
|
||
|
+ }
|
||
|
+ }
|
||
|
+ // Paper end
|
||
|
+
|
||
|
@Override
|
||
|
public TickTask wrapRunnable(Runnable runnable) {
|
||
|
return new TickTask(this.tickCount, runnable);
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
// Paper start - move oversleep into full server tick
|
||
|
isOversleep = true;MinecraftTimings.serverOversleep.startTiming();
|
||
|
this.managedBlock(() -> {
|
||
|
+ midTickLoadChunks(); // will only do loads since we are still considered !canSleepForTick
|
||
|
return !this.canOversleep();
|
||
|
});
|
||
|
isOversleep = false;MinecraftTimings.serverOversleep.stopTiming();
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
}
|
||
|
|
||
|
protected void tickChildren(BooleanSupplier shouldKeepTicking) {
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
MinecraftTimings.bukkitSchedulerTimer.startTiming(); // Spigot // Paper
|
||
|
this.server.getScheduler().mainThreadHeartbeat(this.tickCount); // CraftBukkit
|
||
|
MinecraftTimings.bukkitSchedulerTimer.stopTiming(); // Spigot // Paper
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
this.profiler.push("commandFunctions");
|
||
|
MinecraftTimings.commandFunctionsTimer.startTiming(); // Spigot // Paper
|
||
|
this.getFunctions().tick();
|
||
|
MinecraftTimings.commandFunctionsTimer.stopTiming(); // Spigot // Paper
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
this.profiler.popPush("levels");
|
||
|
Iterator iterator = this.getAllLevels().iterator();
|
||
|
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
processQueue.remove().run();
|
||
|
}
|
||
|
MinecraftTimings.processQueueTimer.stopTiming(); // Spigot
|
||
|
-
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
MinecraftTimings.timeUpdateTimer.startTiming(); // Spigot // Paper
|
||
|
// Send time updates to everyone, it will get the right time from the world the player is in.
|
||
|
// Paper start - optimize time updates
|
||
|
@@ -0,0 +0,0 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
|
||
|
this.profiler.push("tick");
|
||
|
|
||
|
try {
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
worldserver.timings.doTick.startTiming(); // Spigot
|
||
|
worldserver.tick(shouldKeepTicking);
|
||
|
worldserver.timings.doTick.stopTiming(); // Spigot
|
||
|
+ midTickLoadChunks(); // Paper
|
||
|
} catch (Throwable throwable) {
|
||
|
// Spigot Start
|
||
|
CrashReport crashreport;
|
||
|
diff --git a/src/main/java/net/minecraft/server/level/ServerChunkCache.java b/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
||
|
+++ b/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
||
|
@@ -0,0 +0,0 @@ import net.minecraft.core.BlockPos;
|
||
|
import net.minecraft.core.SectionPos;
|
||
|
import net.minecraft.network.protocol.Packet;
|
||
|
import net.minecraft.server.MCUtil;
|
||
|
+import net.minecraft.server.MinecraftServer;
|
||
|
import net.minecraft.server.level.progress.ChunkProgressListener;
|
||
|
import net.minecraft.util.Mth;
|
||
|
import net.minecraft.util.profiling.ProfilerFiller;
|
||
|
@@ -0,0 +0,0 @@ public class ServerChunkCache extends ChunkSource {
|
||
|
this.level.getProfiler().push("purge");
|
||
|
this.level.timings.doChunkMap.startTiming(); // Spigot
|
||
|
this.distanceManager.purgeStaleTickets();
|
||
|
+ this.level.getServer().midTickLoadChunks(); // Paper
|
||
|
this.runDistanceManagerUpdates();
|
||
|
this.level.timings.doChunkMap.stopTiming(); // Spigot
|
||
|
this.level.getProfiler().popPush("chunks");
|
||
|
@@ -0,0 +0,0 @@ public class ServerChunkCache extends ChunkSource {
|
||
|
this.level.timings.doChunkUnload.startTiming(); // Spigot
|
||
|
this.level.getProfiler().popPush("unload");
|
||
|
this.chunkMap.tick(shouldKeepTicking);
|
||
|
+ this.level.getServer().midTickLoadChunks(); // Paper
|
||
|
this.level.timings.doChunkUnload.stopTiming(); // Spigot
|
||
|
this.level.getProfiler().pop();
|
||
|
this.clearCache();
|
||
|
@@ -0,0 +0,0 @@ public class ServerChunkCache extends ChunkSource {
|
||
|
};
|
||
|
// Paper end
|
||
|
this.level.timings.chunkTicks.startTiming(); // Paper
|
||
|
- this.chunkMap.forEachVisibleChunk((playerchunk) -> { // Paper - safe iterator incase chunk loads, also no wrapping
|
||
|
+ final int[] chunksTicked = {0}; this.chunkMap.forEachVisibleChunk((playerchunk) -> { // Paper - safe iterator incase chunk loads, also no wrapping
|
||
|
Optional<LevelChunk> optional = ((Either) playerchunk.getTickingChunkFuture().getNow(ChunkHolder.UNLOADED_LEVEL_CHUNK)).left();
|
||
|
|
||
|
if (optional.isPresent()) {
|
||
|
@@ -0,0 +0,0 @@ public class ServerChunkCache extends ChunkSource {
|
||
|
//this.world.timings.chunkTicks.startTiming(); // Spigot // Paper
|
||
|
this.level.tickChunk(chunk, k);
|
||
|
//this.world.timings.chunkTicks.stopTiming(); // Spigot // Paper
|
||
|
+ if (chunksTicked[0]++ % 10 == 0) this.level.getServer().midTickLoadChunks(); // Paper
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
@@ -0,0 +0,0 @@ public class ServerChunkCache extends ChunkSource {
|
||
|
super.doRunTask(task);
|
||
|
}
|
||
|
|
||
|
+ // Paper start
|
||
|
+ private long lastMidTickChunkTask = 0;
|
||
|
+ public boolean pollChunkLoadTasks() {
|
||
|
+ if (com.destroystokyo.paper.io.chunk.ChunkTaskManager.pollChunkWaitQueue() || ServerChunkCache.this.level.asyncChunkTaskManager.pollNextChunkTask()) {
|
||
|
+ try {
|
||
|
+ ServerChunkCache.this.runDistanceManagerUpdates();
|
||
|
+ } finally {
|
||
|
+ // from below: process pending Chunk loadCallback() and unloadCallback() after each run task
|
||
|
+ chunkMap.callbackExecutor.run();
|
||
|
+ }
|
||
|
+ return true;
|
||
|
+ }
|
||
|
+ return false;
|
||
|
+ }
|
||
|
+ public void midTickLoadChunks() {
|
||
|
+ MinecraftServer server = ServerChunkCache.this.level.getServer();
|
||
|
+ // always try to load chunks, restrain generation/other updates only. don't count these towards tick count
|
||
|
+ //noinspection StatementWithEmptyBody
|
||
|
+ while (pollChunkLoadTasks()) {}
|
||
|
+
|
||
|
+ if (System.nanoTime() - lastMidTickChunkTask < 200000) {
|
||
|
+ return;
|
||
|
+ }
|
||
|
+
|
||
|
+ for (;server.midTickChunksTasksRan < com.destroystokyo.paper.PaperConfig.midTickChunkTasks && server.haveTime();) {
|
||
|
+ if (this.pollTask()) {
|
||
|
+ server.midTickChunksTasksRan++;
|
||
|
+ lastMidTickChunkTask = System.nanoTime();
|
||
|
+ } else {
|
||
|
+ break;
|
||
|
+ }
|
||
|
+ }
|
||
|
+ }
|
||
|
+ // Paper end
|
||
|
+
|
||
|
@Override
|
||
|
protected boolean pollTask() {
|
||
|
// CraftBukkit start - process pending Chunk loadCallback() and unloadCallback() after each run task
|
||
|
diff --git a/src/main/java/net/minecraft/server/level/ServerLevel.java b/src/main/java/net/minecraft/server/level/ServerLevel.java
|
||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||
|
--- a/src/main/java/net/minecraft/server/level/ServerLevel.java
|
||
|
+++ b/src/main/java/net/minecraft/server/level/ServerLevel.java
|
||
|
@@ -0,0 +0,0 @@ public class ServerLevel extends net.minecraft.world.level.Level implements Worl
|
||
|
}
|
||
|
timings.scheduledBlocks.stopTiming(); // Paper
|
||
|
|
||
|
+ this.getServer().midTickLoadChunks(); // Paper
|
||
|
gameprofilerfiller.popPush("raid");
|
||
|
this.timings.raids.startTiming(); // Paper - timings
|
||
|
this.raids.tick();
|
||
|
@@ -0,0 +0,0 @@ public class ServerLevel extends net.minecraft.world.level.Level implements Worl
|
||
|
timings.doSounds.startTiming(); // Spigot
|
||
|
this.runBlockEvents();
|
||
|
timings.doSounds.stopTiming(); // Spigot
|
||
|
+ this.getServer().midTickLoadChunks(); // Paper
|
||
|
this.handlingTick = false;
|
||
|
gameprofilerfiller.popPush("entities");
|
||
|
boolean flag3 = true || !this.players.isEmpty() || !this.getForcedChunks().isEmpty(); // CraftBukkit - this prevents entity cleanup, other issues on servers with no players
|
||
|
@@ -0,0 +0,0 @@ public class ServerLevel extends net.minecraft.world.level.Level implements Worl
|
||
|
timings.entityTick.stopTiming(); // Spigot
|
||
|
|
||
|
this.tickingEntities = false;
|
||
|
+ this.getServer().midTickLoadChunks(); // Paper
|
||
|
|
||
|
Entity entity2;
|
||
|
|
||
|
@@ -0,0 +0,0 @@ public class ServerLevel extends net.minecraft.world.level.Level implements Worl
|
||
|
}
|
||
|
|
||
|
timings.tickEntities.stopTiming(); // Spigot
|
||
|
+ this.getServer().midTickLoadChunks(); // Paper
|
||
|
this.tickBlockEntities();
|
||
|
}
|
||
|
|