2021-11-26 17:25:35 +01:00
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
From: Shane Freeder <theboyetronic@gmail.com>
Date: Sun, 9 Jun 2019 03:53:22 +0100
2024-07-15 11:57:56 -07:00
Subject: [PATCH] Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
2024-12-19 12:03:16 +01:00
diff --git a/net/minecraft/server/MinecraftServer.java b/net/minecraft/server/MinecraftServer.java
2025-06-21 20:44:07 +02:00
index 338ef549efe82c250c74365c1c1071986920c8c9..39581095ccc69d113d954ed835bdfa32d25b5489 100644
2024-12-19 12:03:16 +01:00
--- a/net/minecraft/server/MinecraftServer.java
+++ b/net/minecraft/server/MinecraftServer.java
2025-06-21 20:44:07 +02:00
@@ -955,7 +955,7 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
2024-12-19 12:03:16 +01:00
boolean var4;
2021-11-26 17:25:35 +01:00
try {
this.isSaving = true;
- this.getPlayerList().saveAll();
2024-01-20 23:13:41 +01:00
+ this.getPlayerList().saveAll(); // Paper - Incremental chunk and player saving; diff on change
2024-12-19 12:03:16 +01:00
var4 = this.saveAllChunks(suppressLog, flush, forced);
2021-11-26 17:25:35 +01:00
} finally {
this.isSaving = false;
2025-06-21 20:44:07 +02:00
@@ -1534,9 +1534,29 @@ public abstract class MinecraftServer extends ReentrantBlockableEventLoop<TickTa
2021-11-26 17:25:35 +01:00
}
2024-12-19 12:03:16 +01:00
this.ticksUntilAutosave--;
2024-10-27 10:26:44 +01:00
- if (this.autosavePeriod > 0 && this.ticksUntilAutosave <= 0) { // CraftBukkit
- this.autoSave();
2024-01-20 23:13:41 +01:00
+ // Paper start - Incremental chunk and player saving
2024-10-27 10:26:44 +01:00
+ final ProfilerFiller profiler = Profiler.get();
2022-06-09 01:51:45 -07:00
+ int playerSaveInterval = io.papermc.paper.configuration.GlobalConfiguration.get().playerAutoSave.rate;
2021-11-26 17:25:35 +01:00
+ if (playerSaveInterval < 0) {
+ playerSaveInterval = autosavePeriod;
2024-07-15 11:57:56 -07:00
+ }
2024-10-27 10:26:44 +01:00
+ profiler.push("save");
2021-11-28 16:02:52 +01:00
+ final boolean fullSave = autosavePeriod > 0 && this.tickCount % autosavePeriod == 0;
2021-11-26 17:25:35 +01:00
+ try {
+ this.isSaving = true;
+ if (playerSaveInterval > 0) {
+ this.playerList.saveAll(playerSaveInterval);
+ }
2024-10-27 10:26:44 +01:00
+ for (final ServerLevel level : this.getAllLevels()) {
2022-06-09 01:51:45 -07:00
+ if (level.paperConfig().chunks.autoSaveInterval.value() > 0) {
2021-11-28 16:02:52 +01:00
+ level.saveIncrementally(fullSave);
2021-11-26 17:25:35 +01:00
+ }
+ }
+ } finally {
+ this.isSaving = false;
2024-07-15 11:57:56 -07:00
}
2024-10-27 10:26:44 +01:00
+ profiler.pop();
2024-01-20 23:13:41 +01:00
+ // Paper end - Incremental chunk and player saving
2024-10-27 10:26:44 +01:00
2024-12-19 12:03:16 +01:00
ProfilerFiller profilerFiller = Profiler.get();
this.runAllTasks(); // Paper - move runAllTasks() into full server tick (previously for timings)
diff --git a/net/minecraft/server/level/ServerLevel.java b/net/minecraft/server/level/ServerLevel.java
2025-05-28 13:23:32 +02:00
index 34b7769663e235b93c6388ab0c92c00f0297e42f..dda8d38ef61672cc714d9e5a475f9b0412ed5ff9 100644
2024-12-19 12:03:16 +01:00
--- a/net/minecraft/server/level/ServerLevel.java
+++ b/net/minecraft/server/level/ServerLevel.java
2025-05-28 13:23:32 +02:00
@@ -1339,6 +1339,28 @@ public class ServerLevel extends Level implements ServerEntityGetter, WorldGenLe
2025-04-12 17:26:44 +02:00
return !(entity instanceof Player player && (this.server.isUnderSpawnProtection(this, pos, player) || !this.getWorldBorder().isWithinBounds(pos)));
2021-11-26 17:25:35 +01:00
}
2024-01-20 23:13:41 +01:00
+ // Paper start - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
+ public void saveIncrementally(boolean doFull) {
+ if (doFull) {
2024-12-19 12:03:16 +01:00
+ org.bukkit.Bukkit.getPluginManager().callEvent(new org.bukkit.event.world.WorldSaveEvent(this.getWorld()));
2021-11-26 17:25:35 +01:00
+ }
+
2024-10-27 18:11:15 +01:00
+ if (doFull) {
2025-05-07 17:34:58 -04:00
+ this.saveLevelData(false);
2021-11-26 17:25:35 +01:00
+ }
2024-10-27 18:11:15 +01:00
+ // chunk autosave is already called by the ChunkSystem during unload processing (ChunkMap#processUnloads)
+ // Copied from save()
+ // CraftBukkit start - moved from MinecraftServer.saveChunks
+ if (doFull) { // Paper
2024-12-19 12:03:16 +01:00
+ ServerLevel serverLevel1 = this;
+ this.serverLevelData.setWorldBorder(serverLevel1.getWorldBorder().createSettings());
2024-10-27 18:11:15 +01:00
+ this.serverLevelData.setCustomBossEvents(this.server.getCustomBossEvents().save(this.registryAccess()));
2024-12-20 23:18:34 +01:00
+ this.levelStorageAccess.saveDataTag(this.server.registryAccess(), this.serverLevelData, this.server.getPlayerList().getSingleplayerData());
2024-10-27 18:11:15 +01:00
+ }
+ // CraftBukkit end
2021-11-26 17:25:35 +01:00
+ }
2024-01-20 23:13:41 +01:00
+ // Paper end - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
+
2024-12-19 12:03:16 +01:00
public void save(@Nullable ProgressListener progress, boolean flush, boolean skipSave) {
2024-12-20 23:18:34 +01:00
// Paper start - add close param
this.save(progress, flush, skipSave, false);
2024-12-19 12:03:16 +01:00
diff --git a/net/minecraft/server/level/ServerPlayer.java b/net/minecraft/server/level/ServerPlayer.java
2025-05-28 13:23:32 +02:00
index c0b74d408340101bc3aac4cb4b7232c5cc78b08a..b70929df38389d789dad46c0a6d94f6c08aa7eba 100644
2024-12-19 12:03:16 +01:00
--- a/net/minecraft/server/level/ServerPlayer.java
+++ b/net/minecraft/server/level/ServerPlayer.java
2025-05-28 13:23:32 +02:00
@@ -195,6 +195,7 @@ import org.slf4j.Logger;
2021-11-26 17:25:35 +01:00
2024-12-20 23:18:34 +01:00
public class ServerPlayer extends Player implements ca.spottedleaf.moonrise.patches.chunk_system.player.ChunkSystemServerPlayer { // Paper - rewrite chunk system
2022-02-28 22:43:03 -07:00
private static final Logger LOGGER = LogUtils.getLogger();
2024-01-20 23:13:41 +01:00
+ public long lastSave = MinecraftServer.currentTick; // Paper - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
private static final int NEUTRAL_MOB_DEATH_NOTIFICATION_RADII_XZ = 32;
private static final int NEUTRAL_MOB_DEATH_NOTIFICATION_RADII_Y = 10;
2023-12-05 20:00:14 -07:00
private static final int FLY_STAT_RECORDING_SPEED = 25;
2024-12-19 12:03:16 +01:00
diff --git a/net/minecraft/server/players/PlayerList.java b/net/minecraft/server/players/PlayerList.java
2025-05-28 13:23:32 +02:00
index a3aa852b04e6deccccf995cfc57f95fa79860c04..b1524279c02cd3be82338a6bd0320cb125a134d5 100644
2024-12-19 12:03:16 +01:00
--- a/net/minecraft/server/players/PlayerList.java
+++ b/net/minecraft/server/players/PlayerList.java
2025-05-28 13:23:32 +02:00
@@ -489,6 +489,7 @@ public abstract class PlayerList {
2022-12-19 11:46:55 +01:00
2021-11-26 17:25:35 +01:00
protected void save(ServerPlayer player) {
if (!player.getBukkitEntity().isPersistent()) return; // CraftBukkit
2024-01-20 23:13:41 +01:00
+ player.lastSave = MinecraftServer.currentTick; // Paper - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
this.playerIo.save(player);
2024-12-19 12:03:16 +01:00
ServerStatsCounter serverStatsCounter = player.getStats(); // CraftBukkit
if (serverStatsCounter != null) {
2025-05-28 13:23:32 +02:00
@@ -1075,9 +1076,23 @@ public abstract class PlayerList {
2021-11-26 17:25:35 +01:00
}
public void saveAll() {
2024-01-20 23:13:41 +01:00
+ // Paper start - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
+ this.saveAll(-1);
+ }
+
2024-12-19 12:03:16 +01:00
+ public void saveAll(final int interval) {
2022-10-24 12:43:46 -07:00
io.papermc.paper.util.MCUtil.ensureMain("Save Players" , () -> { // Paper - Ensure main
2021-11-26 17:25:35 +01:00
+ int numSaved = 0;
2024-12-19 12:03:16 +01:00
+ final long now = MinecraftServer.currentTick;
for (int i = 0; i < this.players.size(); i++) {
- this.save(this.players.get(i));
2024-10-27 18:11:15 +01:00
+ final ServerPlayer player = this.players.get(i);
+ if (interval == -1 || now - player.lastSave >= interval) {
+ this.save(player);
2024-12-19 12:03:16 +01:00
+ if (interval != -1 && ++numSaved >= io.papermc.paper.configuration.GlobalConfiguration.get().playerAutoSave.maxPerTick()) {
+ break;
+ }
2021-11-26 17:25:35 +01:00
+ }
2024-01-20 23:13:41 +01:00
+ // Paper end - Incremental chunk and player saving
2021-11-26 17:25:35 +01:00
}
return null; }); // Paper - ensure main
2024-12-19 12:03:16 +01:00
}