mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-21 07:41:30 +01:00
Fix logic error in async chunk loading
if a chunk load was cancelled after generating stage started it would short circuit return with a null. however this skipped the creation of the loadTask, which some code would then invoke in requestChunk and trigger an NPE. This then likely left an incomplete corrupt request in the chunk map which then crashes servers. It should fix these isseues Fixes #1775 Fixes #1743
This commit is contained in:
parent
9bca846f08
commit
fa9c2a0cdd
@ -1070,7 +1070,7 @@ index 68743a6b7..1dbb03a9a 100644
|
|||||||
if (true || worldserver.worldProvider.getDimensionManager() == DimensionManager.OVERWORLD || this.getAllowNether()) { // CraftBukkit
|
if (true || worldserver.worldProvider.getDimensionManager() == DimensionManager.OVERWORLD || this.getAllowNether()) { // CraftBukkit
|
||||||
diff --git a/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
diff --git a/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
||||||
new file mode 100644
|
new file mode 100644
|
||||||
index 000000000..e9a38f9d9
|
index 000000000..896fc94a6
|
||||||
--- /dev/null
|
--- /dev/null
|
||||||
+++ b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
+++ b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
||||||
@@ -0,0 +0,0 @@
|
@@ -0,0 +0,0 @@
|
||||||
@ -1617,6 +1617,18 @@ index 000000000..e9a38f9d9
|
|||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ synchronized PendingChunkRequest addListener(CompletableFuture<Chunk> future, boolean gen, boolean autoSubmit) {
|
+ synchronized PendingChunkRequest addListener(CompletableFuture<Chunk> future, boolean gen, boolean autoSubmit) {
|
||||||
|
+ requests.incrementAndGet();
|
||||||
|
+ if (loadTask == null) {
|
||||||
|
+ // Take care of a race condition in that a request could be cancelled after the synchronize
|
||||||
|
+ // on pendingChunks, but before a listener is added, which would erase these pending tasks.
|
||||||
|
+ genTask = generationExecutor.createPendingTask(this::generateChunk, taskPriority);
|
||||||
|
+ loadTask = EXECUTOR.createPendingTask(this, taskPriority);
|
||||||
|
+ if (autoSubmit) {
|
||||||
|
+ // We will execute it outside of the synchronized context immediately after
|
||||||
|
+ loadTask.submit();
|
||||||
|
+ }
|
||||||
|
+ }
|
||||||
|
+
|
||||||
+ if (hasFinished) {
|
+ if (hasFinished) {
|
||||||
+ future.complete(chunk);
|
+ future.complete(chunk);
|
||||||
+ return new PendingChunkRequest(this);
|
+ return new PendingChunkRequest(this);
|
||||||
@ -1632,17 +1644,6 @@ index 000000000..e9a38f9d9
|
|||||||
+ }
|
+ }
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ requests.incrementAndGet();
|
|
||||||
+ if (loadTask == null) {
|
|
||||||
+ // Take care of a race condition in that a request could be cancelled after the synchronize
|
|
||||||
+ // on pendingChunks, but before a listener is added, which would erase these pending tasks.
|
|
||||||
+ genTask = generationExecutor.createPendingTask(this::generateChunk, taskPriority);
|
|
||||||
+ loadTask = EXECUTOR.createPendingTask(this, taskPriority);
|
|
||||||
+ if (autoSubmit) {
|
|
||||||
+ // We will execute it outside of the synchronized context immediately after
|
|
||||||
+ loadTask.submit();
|
|
||||||
+ }
|
|
||||||
+ }
|
|
||||||
+ return new PendingChunkRequest(this, gen);
|
+ return new PendingChunkRequest(this, gen);
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
@ -2315,7 +2316,7 @@ index a9fffa544..19ce77c4a 100644
|
|||||||
}
|
}
|
||||||
|
|
||||||
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
diff --git a/src/main/java/org/bukkit/craftbukkit/CraftServer.java b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
||||||
index 5fca9e4bd..78adc8714 100644
|
index 8dccf9498..75c4592c2 100644
|
||||||
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
--- a/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
||||||
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
+++ b/src/main/java/org/bukkit/craftbukkit/CraftServer.java
|
||||||
@@ -0,0 +0,0 @@ public final class CraftServer implements Server {
|
@@ -0,0 +0,0 @@ public final class CraftServer implements Server {
|
||||||
|
@ -7,28 +7,28 @@ Mojang was sleeping even if we had no more requests to go after
|
|||||||
the current one finished, resulting in 100ms lost per profile lookup
|
the current one finished, resulting in 100ms lost per profile lookup
|
||||||
|
|
||||||
diff --git a/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java b/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
diff --git a/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java b/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
||||||
index 26a743722..6ed3199c3 100644
|
index 71e48e87b..23f1447cf 100644
|
||||||
--- a/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
--- a/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
||||||
+++ b/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
+++ b/src/main/java/com/mojang/authlib/yggdrasil/YggdrasilGameProfileRepository.java
|
||||||
@@ -0,0 +0,0 @@ public class YggdrasilGameProfileRepository implements GameProfileRepository {
|
@@ -0,0 +0,0 @@ public class YggdrasilGameProfileRepository implements GameProfileRepository {
|
||||||
}
|
}
|
||||||
|
|
||||||
final int page = 0;
|
final int page = 0;
|
||||||
+ boolean hasRequested = false; // Paper
|
+ boolean hasRequested = false; // Paper
|
||||||
|
|
||||||
for (final List<String> request : Iterables.partition(criteria, ENTRIES_PER_PAGE)) {
|
for (final List<String> request : Iterables.partition(criteria, ENTRIES_PER_PAGE)) {
|
||||||
int failCount = 0;
|
int failCount = 0;
|
||||||
@@ -0,0 +0,0 @@ public class YggdrasilGameProfileRepository implements GameProfileRepository {
|
@@ -0,0 +0,0 @@ public class YggdrasilGameProfileRepository implements GameProfileRepository {
|
||||||
LOGGER.debug("Couldn't find profile {}", name);
|
LOGGER.debug("Couldn't find profile {}", name);
|
||||||
callback.onProfileLookupFailed(new GameProfile(null, name), new ProfileNotFoundException("Server did not find the requested profile"));
|
callback.onProfileLookupFailed(new GameProfile(null, name), new ProfileNotFoundException("Server did not find the requested profile"));
|
||||||
}
|
}
|
||||||
+ // Paper start
|
+ // Paper start
|
||||||
+ if (!hasRequested) {
|
+ if (!hasRequested) {
|
||||||
+ hasRequested = true;
|
+ hasRequested = true;
|
||||||
+ continue;
|
+ continue;
|
||||||
+ }
|
+ }
|
||||||
+ // Paper end
|
+ // Paper end
|
||||||
|
|
||||||
try {
|
try {
|
||||||
Thread.sleep(DELAY_BETWEEN_PAGES);
|
Thread.sleep(DELAY_BETWEEN_PAGES);
|
||||||
--
|
--
|
Loading…
Reference in New Issue
Block a user