2020-05-06 09:48:49 +00:00
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
2019-06-09 19:22:44 +00:00
From: Shane Freeder <theboyetronic@gmail.com>
Date: Sun, 9 Jun 2019 03:53:22 +0100
Subject: [PATCH] incremental chunk saving
diff --git a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
2021-01-11 01:44:06 +00:00
index ffe9b1a63d78925e1d77b9e730aef42fed6d58fa..1278d09f70c1e97607ef20d87a178dc252c7f723 100644
2019-06-09 19:22:44 +00:00
--- a/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
+++ b/src/main/java/com/destroystokyo/paper/PaperWorldConfig.java
2021-01-11 01:44:06 +00:00
@@ -446,4 +446,19 @@ public class PaperWorldConfig {
2019-06-09 19:22:44 +00:00
keepLoadedRange = (short) (getInt("keep-spawn-loaded-range", Math.min(spigotConfig.viewDistance, 10)) * 16);
log( "Keep Spawn Loaded Range: " + (keepLoadedRange/16));
}
+
+ public int autoSavePeriod = -1;
+ private void autoSavePeriod() {
+ autoSavePeriod = getInt("auto-save-interval", -1);
+ if (autoSavePeriod > 0) {
+ log("Auto Save Interval: " +autoSavePeriod + " (" + (autoSavePeriod / 20) + "s)");
+ } else if (autoSavePeriod < 0) {
+ autoSavePeriod = net.minecraft.server.MinecraftServer.getServer().autosavePeriod;
+ }
+ }
+
+ public int maxAutoSaveChunksPerTick = 24;
+ private void maxAutoSaveChunksPerTick() {
+ maxAutoSaveChunksPerTick = getInt("max-auto-save-chunks-per-tick", 24);
+ }
}
diff --git a/src/main/java/net/minecraft/server/Chunk.java b/src/main/java/net/minecraft/server/Chunk.java
2021-02-21 20:55:01 +00:00
index b258d4a85c41c2de55a28c565766f57c3f1604ff..fa549cd2ab9d0665dc042f3d4e79e205eaf4da54 100644
2019-06-09 19:22:44 +00:00
--- a/src/main/java/net/minecraft/server/Chunk.java
+++ b/src/main/java/net/minecraft/server/Chunk.java
2020-06-25 23:38:24 +00:00
@@ -43,7 +43,7 @@ public class Chunk implements IChunkAccess {
2019-06-09 19:22:44 +00:00
private TickList<Block> o;
private TickList<FluidType> p;
private boolean q;
- private long lastSaved;
+ public long lastSaved; // Paper
private volatile boolean s;
2019-12-12 16:20:43 +00:00
private long inhabitedTime;
2019-06-09 19:22:44 +00:00
@Nullable
2019-07-28 00:38:29 +00:00
diff --git a/src/main/java/net/minecraft/server/ChunkProviderServer.java b/src/main/java/net/minecraft/server/ChunkProviderServer.java
2021-02-21 20:55:01 +00:00
index d340990000144d9500069824e7e6609400a54ece..71c7bf2fd4fdb0ad880ca81c198a7cf25cdb9e87 100644
2019-07-28 00:38:29 +00:00
--- a/src/main/java/net/minecraft/server/ChunkProviderServer.java
+++ b/src/main/java/net/minecraft/server/ChunkProviderServer.java
2021-02-21 20:55:01 +00:00
@@ -535,6 +535,15 @@ public class ChunkProviderServer extends IChunkProvider {
2019-07-28 00:38:29 +00:00
} // Paper - Timings
}
+ // Paper start - duplicate save, but call incremental
+ public void saveIncrementally() {
+ this.tickDistanceManager();
+ try (co.aikar.timings.Timing timed = world.timings.chunkSaveData.startTiming()) { // Paper - Timings
+ this.playerChunkMap.saveIncrementally();
+ } // Paper - Timings
+ }
+ // Paper end
+
@Override
public void close() throws IOException {
2019-08-05 16:35:40 +00:00
// CraftBukkit start
2019-06-09 19:22:44 +00:00
diff --git a/src/main/java/net/minecraft/server/MinecraftServer.java b/src/main/java/net/minecraft/server/MinecraftServer.java
2021-03-08 23:12:31 +00:00
index ac512c5a1bd10b119a07c11dc6f8569005348ac8..7f63e60caac65a24909566e3f9a0f5ba5aa1575a 100644
2019-06-09 19:22:44 +00:00
--- a/src/main/java/net/minecraft/server/MinecraftServer.java
+++ b/src/main/java/net/minecraft/server/MinecraftServer.java
2021-03-08 23:12:31 +00:00
@@ -154,6 +154,7 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
2019-06-09 19:22:44 +00:00
public static int currentTick = 0; // Paper - Further improve tick loop
public java.util.Queue<Runnable> processQueue = new java.util.concurrent.ConcurrentLinkedQueue<Runnable>();
public int autosavePeriod;
+ public boolean serverAutoSave = false; // Paper
public CommandDispatcher vanillaCommandDispatcher;
private boolean forceTicks;
2020-06-26 16:20:03 +00:00
// CraftBukkit end
2021-03-08 23:12:31 +00:00
@@ -1139,14 +1140,24 @@ public abstract class MinecraftServer extends IAsyncTaskHandlerReentrant<TickTas
2019-06-09 19:22:44 +00:00
this.serverPing.b().a(agameprofile);
}
- if (autosavePeriod > 0 && this.ticks % autosavePeriod == 0) { // CraftBukkit
2019-09-30 01:14:40 +00:00
- MinecraftServer.LOGGER.debug("Autosave started");
2019-06-09 19:22:44 +00:00
+ //if (autosavePeriod > 0 && this.ticks % autosavePeriod == 0) { // CraftBukkit // Paper - move down
2019-09-30 01:14:40 +00:00
+ //MinecraftServer.LOGGER.debug("Autosave started"); // Paper
2019-06-09 19:22:44 +00:00
+ serverAutoSave = (autosavePeriod > 0 && this.ticks % autosavePeriod == 0); // Paper
this.methodProfiler.enter("save");
+ if (autosavePeriod > 0 && this.ticks % autosavePeriod == 0) { // Paper
this.playerList.savePlayers();
- this.saveChunks(true, false, false);
+ }// Paper
+ // Paper start
2019-06-10 05:39:04 +00:00
+ for (WorldServer world : getWorlds()) {
+ if (world.paperConfig.autoSavePeriod > 0) {
2020-06-26 06:29:44 +00:00
+ world.saveIncrementally(serverAutoSave);
2019-06-10 05:39:04 +00:00
+ }
2019-06-09 19:22:44 +00:00
+ }
+ // Paper end
+
this.methodProfiler.exit();
2019-09-30 01:14:40 +00:00
- MinecraftServer.LOGGER.debug("Autosave finished");
2019-06-09 19:22:44 +00:00
- }
2019-09-30 01:14:40 +00:00
+ //MinecraftServer.LOGGER.debug("Autosave finished"); // Paper
2019-06-09 19:22:44 +00:00
+ //} // Paper
this.methodProfiler.enter("snooper");
if (((DedicatedServer) this).getDedicatedServerProperties().snooperEnabled && !this.snooper.d() && this.ticks > 100) { // Spigot
2020-01-28 00:16:53 +00:00
diff --git a/src/main/java/net/minecraft/server/PlayerChunk.java b/src/main/java/net/minecraft/server/PlayerChunk.java
2021-02-21 20:55:01 +00:00
index 2d749c07b159604e1ab904e026adf326eb9a732e..ad01b9333e73c8e4a7c848edc4ce252a51ea52b6 100644
2020-01-28 00:16:53 +00:00
--- a/src/main/java/net/minecraft/server/PlayerChunk.java
+++ b/src/main/java/net/minecraft/server/PlayerChunk.java
2020-08-25 02:22:08 +00:00
@@ -42,6 +42,9 @@ public class PlayerChunk {
2020-01-28 00:16:53 +00:00
private final PlayerChunkMap chunkMap; // Paper
+ long lastAutoSaveTime; // Paper - incremental autosave
+ long inactiveTimeStart; // Paper - incremental autosave
+
public PlayerChunk(ChunkCoordIntPair chunkcoordintpair, int i, LightEngine lightengine, PlayerChunk.c playerchunk_c, PlayerChunk.d playerchunk_d) {
this.statusFutures = new AtomicReferenceArray(PlayerChunk.CHUNK_STATUSES.size());
this.fullChunkFuture = PlayerChunk.UNLOADED_CHUNK_FUTURE;
2021-02-21 20:55:01 +00:00
@@ -397,7 +400,19 @@ public class PlayerChunk {
2020-01-28 00:16:53 +00:00
boolean flag2 = playerchunk_state.isAtLeast(PlayerChunk.State.BORDER);
boolean flag3 = playerchunk_state1.isAtLeast(PlayerChunk.State.BORDER);
+ boolean prevHasBeenLoaded = this.hasBeenLoaded; // Paper
this.hasBeenLoaded |= flag3;
+ // Paper start - incremental autosave
+ if (this.hasBeenLoaded & !prevHasBeenLoaded) {
+ long timeSinceAutoSave = this.inactiveTimeStart - this.lastAutoSaveTime;
+ if (timeSinceAutoSave < 0) {
+ // safest bet is to assume autosave is needed here
+ timeSinceAutoSave = this.chunkMap.world.paperConfig.autoSavePeriod;
+ }
+ this.lastAutoSaveTime = this.chunkMap.world.getTime() - timeSinceAutoSave;
+ this.chunkMap.autoSaveQueue.add(this);
+ }
+ // Paper end
if (!flag2 && flag3) {
// Paper start - cache ticking ready status
int expectCreateCount = ++this.fullChunkCreateCount;
2021-02-21 20:55:01 +00:00
@@ -517,8 +532,32 @@ public class PlayerChunk {
2020-01-28 00:16:53 +00:00
}
public void m() {
+ boolean prev = this.hasBeenLoaded; // Paper
+ this.hasBeenLoaded = getChunkState(this.ticketLevel).isAtLeast(PlayerChunk.State.BORDER);
+ // Paper start - incremental autosave
+ if (prev != this.hasBeenLoaded) {
+ if (this.hasBeenLoaded) {
+ long timeSinceAutoSave = this.inactiveTimeStart - this.lastAutoSaveTime;
+ if (timeSinceAutoSave < 0) {
+ // safest bet is to assume autosave is needed here
+ timeSinceAutoSave = this.chunkMap.world.paperConfig.autoSavePeriod;
+ }
+ this.lastAutoSaveTime = this.chunkMap.world.getTime() - timeSinceAutoSave;
+ this.chunkMap.autoSaveQueue.add(this);
+ } else {
+ this.inactiveTimeStart = this.chunkMap.world.getTime();
+ this.chunkMap.autoSaveQueue.remove(this);
+ }
+ }
+ // Paper end
+ }
+
+ // Paper start - incremental autosave
+ public boolean setHasBeenLoaded() {
this.hasBeenLoaded = getChunkState(this.ticketLevel).isAtLeast(PlayerChunk.State.BORDER);
+ return this.hasBeenLoaded;
}
+ // Paper end
public void a(ProtoChunkExtension protochunkextension) {
for (int i = 0; i < this.statusFutures.length(); ++i) {
2019-06-09 19:22:44 +00:00
diff --git a/src/main/java/net/minecraft/server/PlayerChunkMap.java b/src/main/java/net/minecraft/server/PlayerChunkMap.java
2021-03-08 23:12:31 +00:00
index 14288d7beaa9aea531b7eee16a4be4101d0380fa..abe94c9b7262a67de1a93914d94550f5d0abed28 100644
2019-06-09 19:22:44 +00:00
--- a/src/main/java/net/minecraft/server/PlayerChunkMap.java
+++ b/src/main/java/net/minecraft/server/PlayerChunkMap.java
2020-08-25 02:22:08 +00:00
@@ -47,6 +47,7 @@ import java.util.function.Supplier;
import java.util.stream.Collectors;
import java.util.stream.Stream;
import javax.annotation.Nullable;
+import it.unimi.dsi.fastutil.objects.ObjectRBTreeSet; // Paper
import org.apache.commons.lang3.mutable.MutableBoolean;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
2021-03-08 23:12:31 +00:00
@@ -334,6 +335,64 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
2019-12-17 22:39:07 +00:00
2019-07-28 00:38:29 +00:00
}
2019-06-09 19:22:44 +00:00
2020-01-28 00:16:53 +00:00
+ // Paper start - incremental autosave
2020-08-25 02:22:08 +00:00
+ final ObjectRBTreeSet<PlayerChunk> autoSaveQueue = new ObjectRBTreeSet<>((playerchunk1, playerchunk2) -> {
2020-01-28 00:16:53 +00:00
+ int timeCompare = Long.compare(playerchunk1.lastAutoSaveTime, playerchunk2.lastAutoSaveTime);
+ if (timeCompare != 0) {
+ return timeCompare;
+ }
+
+ return Long.compare(MCUtil.getCoordinateKey(playerchunk1.location), MCUtil.getCoordinateKey(playerchunk2.location));
+ });
+
2019-07-28 00:38:29 +00:00
+ protected void saveIncrementally() {
+ int savedThisTick = 0;
2020-01-28 00:16:53 +00:00
+ // optimized since we search far less chunks to hit ones that need to be saved
2020-06-25 23:38:24 +00:00
+ List<PlayerChunk> reschedule = new java.util.ArrayList<>(this.world.paperConfig.maxAutoSaveChunksPerTick);
2020-01-28 00:16:53 +00:00
+ long currentTick = this.world.getTime();
+ long maxSaveTime = currentTick - this.world.paperConfig.autoSavePeriod;
2019-07-28 00:38:29 +00:00
+
2020-01-28 00:16:53 +00:00
+ for (Iterator<PlayerChunk> iterator = this.autoSaveQueue.iterator(); iterator.hasNext();) {
+ PlayerChunk playerchunk = iterator.next();
+ if (playerchunk.lastAutoSaveTime > maxSaveTime) {
+ break;
+ }
2019-07-28 00:38:29 +00:00
+
2020-01-28 00:16:53 +00:00
+ iterator.remove();
2019-07-28 00:38:29 +00:00
+
2020-01-28 00:16:53 +00:00
+ IChunkAccess ichunkaccess = playerchunk.getChunkSave().getNow(null);
+ if (ichunkaccess instanceof Chunk) {
+ boolean shouldSave = ((Chunk)ichunkaccess).lastSaved <= maxSaveTime;
2019-06-25 01:47:58 +00:00
+
2020-01-28 00:16:53 +00:00
+ if (shouldSave && this.saveChunk(ichunkaccess)) {
+ ++savedThisTick;
2019-06-09 19:22:44 +00:00
+
2020-01-28 00:16:53 +00:00
+ if (!playerchunk.setHasBeenLoaded()) {
+ // do not fall through to reschedule logic
+ playerchunk.inactiveTimeStart = currentTick;
+ if (savedThisTick >= this.world.paperConfig.maxAutoSaveChunksPerTick) {
+ break;
+ }
+ continue;
2019-06-09 19:22:44 +00:00
+ }
2019-06-17 07:47:51 +00:00
+ }
2020-01-28 00:16:53 +00:00
+ }
2019-07-28 00:38:29 +00:00
+
2020-01-28 00:16:53 +00:00
+ reschedule.add(playerchunk);
+
+ if (savedThisTick >= this.world.paperConfig.maxAutoSaveChunksPerTick) {
+ break;
2019-07-28 00:38:29 +00:00
+ }
+ }
2020-01-28 00:16:53 +00:00
+
+ for (int i = 0, len = reschedule.size(); i < len; ++i) {
+ PlayerChunk playerchunk = reschedule.get(i);
+ playerchunk.lastAutoSaveTime = this.world.getTime();
+ this.autoSaveQueue.add(playerchunk);
+ }
2019-07-28 00:38:29 +00:00
+ }
2020-01-28 00:16:53 +00:00
+ // Paper end
2019-07-28 00:38:29 +00:00
+
protected void save(boolean flag) {
if (flag) {
List<PlayerChunk> list = (List) this.visibleChunks.values().stream().filter(PlayerChunk::hasBeenLoaded).peek(PlayerChunk::m).collect(Collectors.toList());
2021-03-08 23:12:31 +00:00
@@ -444,6 +503,7 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
2020-01-28 00:16:53 +00:00
this.world.unloadChunk(chunk);
}
+ this.autoSaveQueue.remove(playerchunk); // Paper
this.lightEngine.a(ichunkaccess.getPos());
this.lightEngine.queueUpdate();
2021-03-08 23:12:31 +00:00
@@ -636,6 +696,8 @@ public class PlayerChunkMap extends IChunkLoader implements PlayerChunk.d {
2020-01-28 00:16:53 +00:00
playerchunk.a(new ProtoChunkExtension(chunk));
}
+ chunk.setLastSaved(this.world.getTime() - 1); // Paper - avoid autosaving newly generated/loaded chunks
+
chunk.a(() -> {
return PlayerChunk.getChunkState(playerchunk.getTicketLevel());
});
2019-06-09 19:22:44 +00:00
diff --git a/src/main/java/net/minecraft/server/WorldServer.java b/src/main/java/net/minecraft/server/WorldServer.java
2021-02-21 20:55:01 +00:00
index 3a95b98b77171b4b18a1d939eb14953d24b903e0..3f97befa2742a575b103b68b19fc6415bfbd8c04 100644
2019-06-09 19:22:44 +00:00
--- a/src/main/java/net/minecraft/server/WorldServer.java
+++ b/src/main/java/net/minecraft/server/WorldServer.java
2020-11-18 03:45:18 +00:00
@@ -776,11 +776,43 @@ public class WorldServer extends World implements GeneratorAccessSeed {
2020-06-25 23:38:24 +00:00
return !this.server.a(this, blockposition, entityhuman) && this.getWorldBorder().a(blockposition);
2019-07-28 00:38:29 +00:00
}
+ // Paper start - derived from below
2020-06-25 23:38:24 +00:00
+ public void saveIncrementally(boolean doFull) {
2019-07-28 00:38:29 +00:00
+ ChunkProviderServer chunkproviderserver = this.getChunkProvider();
+
+ if (doFull) {
+ org.bukkit.Bukkit.getPluginManager().callEvent(new org.bukkit.event.world.WorldSaveEvent(getWorld()));
+ }
+
+ try (co.aikar.timings.Timing ignored = timings.worldSave.startTiming()) {
+ if (doFull) {
2019-12-13 01:18:18 +00:00
+ this.saveData();
2019-07-28 00:38:29 +00:00
+ }
+
+ timings.worldSaveChunks.startTiming(); // Paper
+ if (!this.isSavingDisabled()) chunkproviderserver.saveIncrementally();
+ timings.worldSaveChunks.stopTiming(); // Paper
+
+
2020-06-25 23:38:24 +00:00
+ // Copied from save()
2019-07-28 00:38:29 +00:00
+ // CraftBukkit start - moved from MinecraftServer.saveChunks
2020-06-25 23:38:24 +00:00
+ if (doFull) { // Paper
2019-07-28 00:38:29 +00:00
+ WorldServer worldserver1 = this;
+
2020-06-25 23:38:24 +00:00
+ worldDataServer.a(worldserver1.getWorldBorder().t());
+ worldDataServer.setCustomBossEvents(this.server.getBossBattleCustomData().save());
2020-09-10 23:56:58 +00:00
+ convertable.a(this.server.customRegistry, this.worldDataServer, this.server.getPlayerList().save());
2019-07-28 00:38:29 +00:00
+ }
2020-06-25 23:38:24 +00:00
+ // CraftBukkit end
2019-07-28 00:38:29 +00:00
+ }
+ }
+ // Paper end
+
2020-06-25 23:38:24 +00:00
public void save(@Nullable IProgressUpdate iprogressupdate, boolean flag, boolean flag1) {
2019-06-10 20:45:17 +00:00
ChunkProviderServer chunkproviderserver = this.getChunkProvider();
2019-06-09 19:22:44 +00:00
if (!flag1) {
2019-06-10 20:45:17 +00:00
- org.bukkit.Bukkit.getPluginManager().callEvent(new org.bukkit.event.world.WorldSaveEvent(getWorld())); // CraftBukkit
2020-06-25 23:38:24 +00:00
+ if (flag) org.bukkit.Bukkit.getPluginManager().callEvent(new org.bukkit.event.world.WorldSaveEvent(getWorld())); // CraftBukkit // Paper
2019-06-09 19:22:44 +00:00
try (co.aikar.timings.Timing ignored = timings.worldSave.startTiming()) { // Paper
if (iprogressupdate != null) {
2020-06-25 23:38:24 +00:00
iprogressupdate.a(new ChatMessage("menu.savingLevel"));
2020-11-18 03:45:18 +00:00
@@ -806,6 +838,7 @@ public class WorldServer extends World implements GeneratorAccessSeed {
2019-12-13 01:18:18 +00:00
// CraftBukkit end
}
2020-11-03 02:22:15 +00:00
+ private void saveData() { this.aj(); } // Paper - OBFHELPER
private void aj() {
2020-06-25 23:38:24 +00:00
if (this.dragonBattle != null) {
2020-07-16 23:07:15 +00:00
this.worldDataServer.a(this.dragonBattle.a()); // CraftBukkit