mirror of
https://github.com/PaperMC/Paper.git
synced 2025-01-16 06:30:46 +01:00
Clean up thread pool usage (#11681)
Using an unbound LinkedBlockingQueue means you *have* to set core and max core thread pool size the same, as they will never go above the minimum pool size by just passing them through. So this fixes the async command executor pool to actually use 2 threads, and also cleans up other usage to be explicitly "fixed" thread pool sizes, and splits off one more in Minecraft's Util class
This commit is contained in:
parent
46ef70e7d1
commit
4a5ff5039c
4 changed files with 66 additions and 22 deletions
|
@ -17,19 +17,14 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
// CraftBukkit start
|
// CraftBukkit start
|
||||||
// Register Vanilla commands into builtRoot as before
|
// Register Vanilla commands into builtRoot as before
|
||||||
+ // Paper start - Perf: Async command map building
|
+ // Paper start - Perf: Async command map building
|
||||||
+ COMMAND_SENDING_POOL.execute(() -> {
|
+ COMMAND_SENDING_POOL.execute(() -> this.sendAsync(player));
|
||||||
+ this.sendAsync(player);
|
|
||||||
+ });
|
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public static final java.util.concurrent.ThreadPoolExecutor COMMAND_SENDING_POOL = new java.util.concurrent.ThreadPoolExecutor(
|
+ public static final java.util.concurrent.ExecutorService COMMAND_SENDING_POOL = java.util.concurrent.Executors.newFixedThreadPool(2,
|
||||||
+ 0, 2, 60L, java.util.concurrent.TimeUnit.SECONDS,
|
|
||||||
+ new java.util.concurrent.LinkedBlockingQueue<>(),
|
|
||||||
+ new com.google.common.util.concurrent.ThreadFactoryBuilder()
|
+ new com.google.common.util.concurrent.ThreadFactoryBuilder()
|
||||||
+ .setNameFormat("Paper Async Command Builder Thread Pool - %1$d")
|
+ .setNameFormat("Paper Async Command Builder Thread Pool - %1$d")
|
||||||
+ .setUncaughtExceptionHandler(new net.minecraft.DefaultUncaughtExceptionHandlerWithName(net.minecraft.server.MinecraftServer.LOGGER))
|
+ .setUncaughtExceptionHandler(new net.minecraft.DefaultUncaughtExceptionHandlerWithName(net.minecraft.server.MinecraftServer.LOGGER))
|
||||||
+ .build(),
|
+ .build()
|
||||||
+ new java.util.concurrent.ThreadPoolExecutor.DiscardPolicy()
|
|
||||||
+ );
|
+ );
|
||||||
+
|
+
|
||||||
+ private void sendAsync(ServerPlayer player) {
|
+ private void sendAsync(ServerPlayer player) {
|
||||||
|
|
|
@ -78,7 +78,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
- AtomicInteger atomicInteger = new AtomicInteger(1);
|
- AtomicInteger atomicInteger = new AtomicInteger(1);
|
||||||
- executorService = new ForkJoinPool(i, pool -> {
|
- executorService = new ForkJoinPool(i, pool -> {
|
||||||
- final String string2 = "Worker-" + name + "-" + atomicInteger.getAndIncrement();
|
- final String string2 = "Worker-" + name + "-" + atomicInteger.getAndIncrement();
|
||||||
+ executorService = new java.util.concurrent.ThreadPoolExecutor(i, i,0L, TimeUnit.MILLISECONDS, new java.util.concurrent.LinkedBlockingQueue<>(), target -> new io.papermc.paper.util.ServerWorkerThread(target, s, priorityModifier));
|
+ executorService = Executors.newFixedThreadPool(i, target -> new io.papermc.paper.util.ServerWorkerThread(target, s, priorityModifier));
|
||||||
+ }
|
+ }
|
||||||
+ /* final String string2 = "Worker-" + name + "-" + atomicInteger.getAndIncrement();
|
+ /* final String string2 = "Worker-" + name + "-" + atomicInteger.getAndIncrement();
|
||||||
ForkJoinWorkerThread forkJoinWorkerThread = new ForkJoinWorkerThread(pool) {
|
ForkJoinWorkerThread forkJoinWorkerThread = new ForkJoinWorkerThread(pool) {
|
||||||
|
|
|
@ -4861,9 +4861,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
+import java.util.List;
|
+import java.util.List;
|
||||||
+import java.util.concurrent.CompletableFuture;
|
+import java.util.concurrent.CompletableFuture;
|
||||||
+import java.util.concurrent.ExecutionException;
|
+import java.util.concurrent.ExecutionException;
|
||||||
+import java.util.concurrent.LinkedBlockingQueue;
|
+import java.util.concurrent.ExecutorService;
|
||||||
+import java.util.concurrent.ThreadPoolExecutor;
|
+import java.util.concurrent.Executors;
|
||||||
+import java.util.concurrent.TimeUnit;
|
|
||||||
+import java.util.function.BiConsumer;
|
+import java.util.function.BiConsumer;
|
||||||
+import java.util.function.Consumer;
|
+import java.util.function.Consumer;
|
||||||
+import java.util.function.Function;
|
+import java.util.function.Function;
|
||||||
|
@ -4888,13 +4887,10 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
+ run.run();
|
+ run.run();
|
||||||
+ }
|
+ }
|
||||||
+ };
|
+ };
|
||||||
+ public static final ThreadPoolExecutor asyncExecutor = new ThreadPoolExecutor(
|
+ public static final ExecutorService ASYNC_EXECUTOR = Executors.newFixedThreadPool(2, new ThreadFactoryBuilder()
|
||||||
+ 2, 2, 60L, TimeUnit.SECONDS,
|
+ .setNameFormat("Paper Async Task Handler Thread - %1$d")
|
||||||
+ new LinkedBlockingQueue<>(),
|
+ .setUncaughtExceptionHandler(new net.minecraft.DefaultUncaughtExceptionHandlerWithName(MinecraftServer.LOGGER))
|
||||||
+ new ThreadFactoryBuilder()
|
+ .build()
|
||||||
+ .setNameFormat("Paper Async Task Handler Thread - %1$d")
|
|
||||||
+ .setUncaughtExceptionHandler(new net.minecraft.DefaultUncaughtExceptionHandlerWithName(MinecraftServer.LOGGER))
|
|
||||||
+ .build()
|
|
||||||
+ );
|
+ );
|
||||||
+
|
+
|
||||||
+ private MCUtil() {
|
+ private MCUtil() {
|
||||||
|
@ -5029,7 +5025,7 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public static void scheduleAsyncTask(Runnable run) {
|
+ public static void scheduleAsyncTask(Runnable run) {
|
||||||
+ asyncExecutor.execute(run);
|
+ ASYNC_EXECUTOR.execute(run);
|
||||||
+ }
|
+ }
|
||||||
+
|
+
|
||||||
+ public static <T> ResourceKey<T> toResourceKey(
|
+ public static <T> ResourceKey<T> toResourceKey(
|
||||||
|
@ -5223,8 +5219,8 @@ index 0000000000000000000000000000000000000000..00000000000000000000000000000000
|
||||||
MinecraftServer.LOGGER.error("Failed to unlock level {}", this.storageSource.getLevelId(), ioexception1);
|
MinecraftServer.LOGGER.error("Failed to unlock level {}", this.storageSource.getLevelId(), ioexception1);
|
||||||
}
|
}
|
||||||
// Spigot start
|
// Spigot start
|
||||||
+ io.papermc.paper.util.MCUtil.asyncExecutor.shutdown(); // Paper
|
+ io.papermc.paper.util.MCUtil.ASYNC_EXECUTOR.shutdown(); // Paper
|
||||||
+ try { io.papermc.paper.util.MCUtil.asyncExecutor.awaitTermination(30, java.util.concurrent.TimeUnit.SECONDS); // Paper
|
+ try { io.papermc.paper.util.MCUtil.ASYNC_EXECUTOR.awaitTermination(30, java.util.concurrent.TimeUnit.SECONDS); // Paper
|
||||||
+ } catch (java.lang.InterruptedException ignored) {} // Paper
|
+ } catch (java.lang.InterruptedException ignored) {} // Paper
|
||||||
if (org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly) {
|
if (org.spigotmc.SpigotConfig.saveUserCacheOnStopOnly) {
|
||||||
MinecraftServer.LOGGER.info("Saving usercache.json");
|
MinecraftServer.LOGGER.info("Saving usercache.json");
|
||||||
|
|
53
patches/server/Separate-dimensiondata-executor.patch
Normal file
53
patches/server/Separate-dimensiondata-executor.patch
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
From 0000000000000000000000000000000000000000 Mon Sep 17 00:00:00 2001
|
||||||
|
From: Nassim Jahnke <nassim@njahnke.dev>
|
||||||
|
Date: Thu, 28 Nov 2024 10:35:58 +0100
|
||||||
|
Subject: [PATCH] Separate dimensiondata executor
|
||||||
|
|
||||||
|
|
||||||
|
diff --git a/src/main/java/net/minecraft/Util.java b/src/main/java/net/minecraft/Util.java
|
||||||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||||||
|
--- a/src/main/java/net/minecraft/Util.java
|
||||||
|
+++ b/src/main/java/net/minecraft/Util.java
|
||||||
|
@@ -0,0 +0,0 @@ public class Util {
|
||||||
|
private static final String MAX_THREADS_SYSTEM_PROPERTY = "max.bg.threads";
|
||||||
|
private static final TracingExecutor BACKGROUND_EXECUTOR = makeExecutor("Main", -1); // Paper - Perf: add priority
|
||||||
|
private static final TracingExecutor IO_POOL = makeIoExecutor("IO-Worker-", false);
|
||||||
|
+ public static final TracingExecutor DIMENSION_DATA_IO_POOL = makeExtraIoExecutor("Dimension-Data-IO-Worker-"); // Paper - Separate dimension data IO pool
|
||||||
|
private static final TracingExecutor DOWNLOAD_POOL = makeIoExecutor("Download-", true);
|
||||||
|
// Paper start - don't submit BLOCKING PROFILE LOOKUPS to the world gen thread
|
||||||
|
public static final ExecutorService PROFILE_EXECUTOR = Executors.newFixedThreadPool(2, new java.util.concurrent.ThreadFactory() {
|
||||||
|
@@ -0,0 +0,0 @@ public class Util {
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
|
||||||
|
+ // Paper start - Separate dimension data IO pool
|
||||||
|
+ private static TracingExecutor makeExtraIoExecutor(String namePrefix) {
|
||||||
|
+ AtomicInteger atomicInteger = new AtomicInteger(1);
|
||||||
|
+ return new TracingExecutor(Executors.newFixedThreadPool(4, runnable -> {
|
||||||
|
+ Thread thread = new Thread(runnable);
|
||||||
|
+ String string2 = namePrefix + atomicInteger.getAndIncrement();
|
||||||
|
+ TracyClient.setThreadName(string2, namePrefix.hashCode());
|
||||||
|
+ thread.setName(string2);
|
||||||
|
+ thread.setDaemon(false);
|
||||||
|
+ thread.setUncaughtExceptionHandler(Util::onThreadException);
|
||||||
|
+ return thread;
|
||||||
|
+ }));
|
||||||
|
+ }
|
||||||
|
+ // Paper end - Separate dimension data IO pool
|
||||||
|
+
|
||||||
|
public static void throwAsRuntime(Throwable t) {
|
||||||
|
throw t instanceof RuntimeException ? (RuntimeException)t : new RuntimeException(t);
|
||||||
|
}
|
||||||
|
diff --git a/src/main/java/net/minecraft/world/level/storage/DimensionDataStorage.java b/src/main/java/net/minecraft/world/level/storage/DimensionDataStorage.java
|
||||||
|
index 0000000000000000000000000000000000000000..0000000000000000000000000000000000000000 100644
|
||||||
|
--- a/src/main/java/net/minecraft/world/level/storage/DimensionDataStorage.java
|
||||||
|
+++ b/src/main/java/net/minecraft/world/level/storage/DimensionDataStorage.java
|
||||||
|
@@ -0,0 +0,0 @@ public class DimensionDataStorage implements AutoCloseable {
|
||||||
|
} catch (IOException var3) {
|
||||||
|
LOGGER.error("Could not save data to {}", path.getFileName(), var3);
|
||||||
|
}
|
||||||
|
- }, Util.ioPool());
|
||||||
|
+ }, Util.DIMENSION_DATA_IO_POOL); // Paper - Separate dimension data IO pool
|
||||||
|
}
|
||||||
|
|
||||||
|
public void saveAndJoin() {
|
Loading…
Reference in a new issue