2021-08-26 04:16:27 +02:00
|
|
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
|
|
|
From: Spottedleaf <spottedleaf@spottedleaf.dev>
|
|
|
|
Date: Thu, 18 Jun 2020 18:23:20 -0700
|
|
|
|
Subject: [PATCH] Prevent unload() calls removing tickets for sync loads
|
|
|
|
|
|
|
|
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/level/ServerChunkCache.java b/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
2022-01-01 04:05:42 +01:00
|
|
|
index f871b8569f9812ee93b16fe9e45ea1a1c4f8f4aa..e306dbf3487676e01c483d121c1f26e141358e3f 100644
|
2021-08-26 04:16:27 +02:00
|
|
|
--- a/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/level/ServerChunkCache.java
|
2021-11-30 06:25:11 +01:00
|
|
|
@@ -725,6 +725,8 @@ public class ServerChunkCache extends ChunkSource {
|
2021-08-26 04:16:27 +02:00
|
|
|
return completablefuture;
|
|
|
|
}
|
|
|
|
|
|
|
|
+ private long syncLoadCounter; // Paper - prevent plugin unloads from removing our ticket
|
|
|
|
+
|
2021-11-17 06:00:14 +01:00
|
|
|
private CompletableFuture<Either<ChunkAccess, ChunkHolder.ChunkLoadingFailure>> getChunkFutureMainThread(int chunkX, int chunkZ, ChunkStatus leastStatus, boolean create) {
|
2021-08-26 04:16:27 +02:00
|
|
|
// Paper start - add isUrgent - old sig left in place for dirty nms plugins
|
2021-11-17 06:00:14 +01:00
|
|
|
return getChunkFutureMainThread(chunkX, chunkZ, leastStatus, create, false);
|
2021-11-30 06:25:11 +01:00
|
|
|
@@ -743,9 +745,12 @@ public class ServerChunkCache extends ChunkSource {
|
2021-08-26 04:16:27 +02:00
|
|
|
ChunkHolder.FullChunkStatus currentChunkState = ChunkHolder.getFullChunkStatus(playerchunk.getTicketLevel());
|
|
|
|
currentlyUnloading = (oldChunkState.isOrAfter(ChunkHolder.FullChunkStatus.BORDER) && !currentChunkState.isOrAfter(ChunkHolder.FullChunkStatus.BORDER));
|
|
|
|
}
|
|
|
|
+ final Long identifier; // Paper - prevent plugin unloads from removing our ticket
|
2021-11-17 06:00:14 +01:00
|
|
|
if (create && !currentlyUnloading) {
|
2021-08-26 04:16:27 +02:00
|
|
|
// CraftBukkit end
|
|
|
|
this.distanceManager.addTicket(TicketType.UNKNOWN, chunkcoordintpair, l, chunkcoordintpair);
|
|
|
|
+ identifier = Long.valueOf(this.syncLoadCounter++); // Paper - prevent plugin unloads from removing our ticket
|
|
|
|
+ this.distanceManager.addTicketAtLevel(TicketType.REQUIRED_LOAD, chunkcoordintpair, l, identifier); // Paper - prevent plugin unloads from removing our ticket
|
|
|
|
if (isUrgent) this.distanceManager.markUrgent(chunkcoordintpair); // Paper - Chunk priority
|
|
|
|
if (this.chunkAbsent(playerchunk, l)) {
|
|
|
|
ProfilerFiller gameprofilerfiller = this.level.getProfiler();
|
2021-11-30 06:25:11 +01:00
|
|
|
@@ -756,13 +761,21 @@ public class ServerChunkCache extends ChunkSource {
|
2021-08-26 04:16:27 +02:00
|
|
|
playerchunk = this.getVisibleChunkIfPresent(k);
|
|
|
|
gameprofilerfiller.pop();
|
|
|
|
if (this.chunkAbsent(playerchunk, l)) {
|
|
|
|
+ this.distanceManager.removeTicketAtLevel(TicketType.REQUIRED_LOAD, chunkcoordintpair, l, identifier); // Paper
|
2021-11-30 06:25:11 +01:00
|
|
|
throw (IllegalStateException) Util.pauseInIde(new IllegalStateException("No chunk holder after ticket has been added"));
|
2021-08-26 04:16:27 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
- }
|
2021-11-17 06:00:14 +01:00
|
|
|
|
2021-08-26 04:16:27 +02:00
|
|
|
+ } else { identifier = null; } // Paper - prevent plugin unloads from removing our ticket
|
|
|
|
// Paper start - Chunk priority
|
2021-11-17 06:00:14 +01:00
|
|
|
CompletableFuture<Either<ChunkAccess, ChunkHolder.ChunkLoadingFailure>> future = this.chunkAbsent(playerchunk, l) ? ChunkHolder.UNLOADED_CHUNK_FUTURE : playerchunk.getOrScheduleFuture(leastStatus, this.chunkMap);
|
2021-08-26 04:16:27 +02:00
|
|
|
+ // Paper start - prevent plugin unloads from removing our ticket
|
2021-11-17 06:00:14 +01:00
|
|
|
+ if (create && !currentlyUnloading) {
|
2021-08-26 04:16:27 +02:00
|
|
|
+ future.thenAcceptAsync((either) -> {
|
|
|
|
+ ServerChunkCache.this.distanceManager.removeTicketAtLevel(TicketType.REQUIRED_LOAD, chunkcoordintpair, l, identifier);
|
|
|
|
+ }, ServerChunkCache.this.mainThreadProcessor);
|
|
|
|
+ }
|
|
|
|
+ // Paper end - prevent plugin unloads from removing our ticket
|
|
|
|
if (isUrgent) {
|
|
|
|
future.thenAccept(either -> this.distanceManager.clearUrgent(chunkcoordintpair));
|
|
|
|
}
|
|
|
|
diff --git a/src/main/java/net/minecraft/server/level/TicketType.java b/src/main/java/net/minecraft/server/level/TicketType.java
|
|
|
|
index 3c1698ba0d3bc412ab957777d9b5211dbc555208..41ddcf6775f99c56cf4b13b284420061e5dd6bdc 100644
|
|
|
|
--- a/src/main/java/net/minecraft/server/level/TicketType.java
|
|
|
|
+++ b/src/main/java/net/minecraft/server/level/TicketType.java
|
|
|
|
@@ -31,6 +31,7 @@ public class TicketType<T> {
|
|
|
|
public static final TicketType<Unit> PLUGIN = TicketType.create("plugin", (a, b) -> 0); // CraftBukkit
|
|
|
|
public static final TicketType<org.bukkit.plugin.Plugin> PLUGIN_TICKET = TicketType.create("plugin_ticket", (plugin1, plugin2) -> plugin1.getClass().getName().compareTo(plugin2.getClass().getName())); // CraftBukkit
|
|
|
|
public static final TicketType<Long> DELAY_UNLOAD = create("delay_unload", Long::compareTo, 300); // Paper
|
|
|
|
+ public static final TicketType<Long> REQUIRED_LOAD = create("required_load", Long::compareTo); // Paper - make sure getChunkAt does not fail
|
|
|
|
|
|
|
|
public static <T> TicketType<T> create(String name, Comparator<T> argumentComparator) {
|
|
|
|
return new TicketType<>(name, argumentComparator, 0L);
|