mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-19 07:33:11 +01:00
Improvements to Async Chunks
If a chunk load comes in on a chunk load or gen thread, execute it synchronously on that thread instead of enqueueing it. It doesn't make sense to enqueue it as that thread is then going to future.join() it and block until it's ready anyways. This opens risk to a deadlock if every load or gen thread is waiting on a recursive chunk but it will never finish because all of the threads are waiting.
This commit is contained in:
parent
0693716984
commit
62efc6ab31
2 changed files with 34 additions and 12 deletions
|
@ -1,4 +1,4 @@
|
||||||
From 56d73274d938d16968b54d62069ba376b53d818e Mon Sep 17 00:00:00 2001
|
From 8375c097918c91e3a4ff945847903ff33c1f770f Mon Sep 17 00:00:00 2001
|
||||||
From: Aikar <aikar@aikar.co>
|
From: Aikar <aikar@aikar.co>
|
||||||
Date: Sat, 11 Aug 2018 00:49:20 -0400
|
Date: Sat, 11 Aug 2018 00:49:20 -0400
|
||||||
Subject: [PATCH] Detect and repair corrupt Region Files
|
Subject: [PATCH] Detect and repair corrupt Region Files
|
||||||
|
@ -11,7 +11,7 @@ I don't know why mojang only checks for 4096, when anything less than 8192 is a
|
||||||
But to be safe, it will attempt to back up the file.
|
But to be safe, it will attempt to back up the file.
|
||||||
|
|
||||||
diff --git a/src/main/java/net/minecraft/server/RegionFile.java b/src/main/java/net/minecraft/server/RegionFile.java
|
diff --git a/src/main/java/net/minecraft/server/RegionFile.java b/src/main/java/net/minecraft/server/RegionFile.java
|
||||||
index 5d2853b9ce..5c0d4f912a 100644
|
index 5d2853b9ce..6a2edfd1e1 100644
|
||||||
--- a/src/main/java/net/minecraft/server/RegionFile.java
|
--- a/src/main/java/net/minecraft/server/RegionFile.java
|
||||||
+++ b/src/main/java/net/minecraft/server/RegionFile.java
|
+++ b/src/main/java/net/minecraft/server/RegionFile.java
|
||||||
@@ -22,10 +22,10 @@ import javax.annotation.Nullable;
|
@@ -22,10 +22,10 @@ import javax.annotation.Nullable;
|
||||||
|
@ -58,14 +58,11 @@ index 5d2853b9ce..5c0d4f912a 100644
|
||||||
}
|
}
|
||||||
} catch (IOException ioexception) {
|
} catch (IOException ioexception) {
|
||||||
ioexception.printStackTrace();
|
ioexception.printStackTrace();
|
||||||
@@ -276,6 +276,58 @@ public class RegionFile {
|
@@ -276,6 +276,55 @@ public class RegionFile {
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
+ // Paper start
|
+ // Paper start
|
||||||
+ public void deleteChunk(int x, int z) {
|
|
||||||
+ deleteChunk(x + z * 32);
|
|
||||||
+ }
|
|
||||||
+ public synchronized void deleteChunk(int j1) {
|
+ public synchronized void deleteChunk(int j1) {
|
||||||
+ backup();
|
+ backup();
|
||||||
+ int k = offsets[j1];
|
+ int k = offsets[j1];
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
From f7cdbb488e556812e697024408c1ac7e9c57a18b Mon Sep 17 00:00:00 2001
|
From 1248921187e1c418988fe86a9e28b2ae810a092f Mon Sep 17 00:00:00 2001
|
||||||
From: Aikar <aikar@aikar.co>
|
From: Aikar <aikar@aikar.co>
|
||||||
Date: Sat, 21 Jul 2018 16:55:04 -0400
|
Date: Sat, 21 Jul 2018 16:55:04 -0400
|
||||||
Subject: [PATCH] Async Chunk Loading and Generation
|
Subject: [PATCH] Async Chunk Loading and Generation
|
||||||
|
@ -904,10 +904,10 @@ index 98d182fdb8..487d98eb1b 100644
|
||||||
|
|
||||||
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 0000000000..2dfa59b204
|
index 0000000000..4fc5fad09e
|
||||||
--- /dev/null
|
--- /dev/null
|
||||||
+++ b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
+++ b/src/main/java/net/minecraft/server/PaperAsyncChunkProvider.java
|
||||||
@@ -0,0 +1,568 @@
|
@@ -0,0 +1,593 @@
|
||||||
+/*
|
+/*
|
||||||
+ * This file is licensed under the MIT License (MIT).
|
+ * This file is licensed under the MIT License (MIT).
|
||||||
+ *
|
+ *
|
||||||
|
@ -965,6 +965,8 @@ index 0000000000..2dfa59b204
|
||||||
+ private final Long2ObjectMap<PendingChunk> pendingChunks = Long2ObjectMaps.synchronize(new Long2ObjectOpenHashMap<>());
|
+ private final Long2ObjectMap<PendingChunk> pendingChunks = Long2ObjectMaps.synchronize(new Long2ObjectOpenHashMap<>());
|
||||||
+ private final ConcurrentLinkedQueue<Runnable> mainThreadQueue = new ConcurrentLinkedQueue<>();
|
+ private final ConcurrentLinkedQueue<Runnable> mainThreadQueue = new ConcurrentLinkedQueue<>();
|
||||||
+ private final IAsyncTaskHandler asyncHandler;
|
+ private final IAsyncTaskHandler asyncHandler;
|
||||||
|
+ private final ThreadLocal<Boolean> isChunkThread = ThreadLocal.withInitial(() -> false);
|
||||||
|
+ private final ThreadLocal<Boolean> isChunkGenThread = ThreadLocal.withInitial(() -> false);
|
||||||
+
|
+
|
||||||
+ private final WorldServer world;
|
+ private final WorldServer world;
|
||||||
+ private final IChunkLoader chunkLoader;
|
+ private final IChunkLoader chunkLoader;
|
||||||
|
@ -1069,6 +1071,9 @@ index 0000000000..2dfa59b204
|
||||||
+ // Listen for when result is ready
|
+ // Listen for when result is ready
|
||||||
+ final CompletableFuture<Chunk> future = new CompletableFuture<>();
|
+ final CompletableFuture<Chunk> future = new CompletableFuture<>();
|
||||||
+ PendingChunkRequest request = pending.addListener(future, gen);
|
+ PendingChunkRequest request = pending.addListener(future, gen);
|
||||||
|
+ if (isChunkThread.get()) {
|
||||||
|
+ pending.loadTask.run();
|
||||||
|
+ }
|
||||||
+
|
+
|
||||||
+ if (isBlockingMain && pending.hasFinished) {
|
+ if (isBlockingMain && pending.hasFinished) {
|
||||||
+ processChunkLoads();
|
+ processChunkLoads();
|
||||||
|
@ -1248,6 +1253,11 @@ index 0000000000..2dfa59b204
|
||||||
+ }
|
+ }
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
|
+ private Chunk generateChunkExecutor() {
|
||||||
|
+ isChunkThread.set(true);
|
||||||
|
+ isChunkGenThread.set(true);
|
||||||
|
+ return generateChunk();
|
||||||
|
+ }
|
||||||
+ private Chunk generateChunk() {
|
+ private Chunk generateChunk() {
|
||||||
+ synchronized (this) {
|
+ synchronized (this) {
|
||||||
+ if (requests.get() <= 0) {
|
+ if (requests.get() <= 0) {
|
||||||
|
@ -1405,8 +1415,16 @@ index 0000000000..2dfa59b204
|
||||||
+ if (loadTask == null) {
|
+ if (loadTask == null) {
|
||||||
+ // Take care of a race condition in that a request could be cancelled after the synchronize
|
+ // 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.
|
+ // on pendingChunks, but before a listener is added, which would erase these pending tasks.
|
||||||
|
+ if (shouldGenSync) {
|
||||||
+ genTask = generationExecutor.createPendingTask(this::generateChunk, taskPriority);
|
+ genTask = generationExecutor.createPendingTask(this::generateChunk, taskPriority);
|
||||||
+ loadTask = EXECUTOR.submitTask(this, taskPriority);
|
+ } else {
|
||||||
|
+ genTask = generationExecutor.createPendingTask(this::generateChunkExecutor, taskPriority);
|
||||||
|
+ }
|
||||||
|
+ loadTask = EXECUTOR.createPendingTask(this, taskPriority);
|
||||||
|
+ if (!isChunkThread.get()) {
|
||||||
|
+ // We will execute it outside of the synchronized context immediately after
|
||||||
|
+ EXECUTOR.submitTask(loadTask);
|
||||||
|
+ }
|
||||||
+ }
|
+ }
|
||||||
+ return new PendingChunkRequest(this, gen);
|
+ return new PendingChunkRequest(this, gen);
|
||||||
+ }
|
+ }
|
||||||
|
@ -1414,6 +1432,7 @@ index 0000000000..2dfa59b204
|
||||||
+
|
+
|
||||||
+ @Override
|
+ @Override
|
||||||
+ public void run() {
|
+ public void run() {
|
||||||
|
+ isChunkThread.set(true);
|
||||||
+ try {
|
+ try {
|
||||||
+ if (!loadFinished(loadChunk(x, z))) {
|
+ if (!loadFinished(loadChunk(x, z))) {
|
||||||
+ return;
|
+ return;
|
||||||
|
@ -1435,9 +1454,15 @@ index 0000000000..2dfa59b204
|
||||||
+ mainThreadQueue.notify();
|
+ mainThreadQueue.notify();
|
||||||
+ }
|
+ }
|
||||||
+ } else {
|
+ } else {
|
||||||
|
+ if (isChunkGenThread.get()) {
|
||||||
|
+ // ideally we should never run into 1 chunk generating another chunk...
|
||||||
|
+ // but if we do, let's apply same solution
|
||||||
|
+ genTask.run();
|
||||||
|
+ } else {
|
||||||
+ generationExecutor.submitTask(genTask);
|
+ generationExecutor.submitTask(genTask);
|
||||||
+ }
|
+ }
|
||||||
+ }
|
+ }
|
||||||
|
+ }
|
||||||
+
|
+
|
||||||
+ void bumpPriority() {
|
+ void bumpPriority() {
|
||||||
+ this.taskPriority = Priority.HIGH;
|
+ this.taskPriority = Priority.HIGH;
|
||||||
|
|
Loading…
Reference in a new issue